소프트 키보드 노출에 따른 화면 레이아웃

Mobile Programming/Android Tip 2013. 11. 24. 22:14

구글 개발자 사이트에 잘 기술되어있음.

까먹어서 다시 검색한 김에 정리해 둔다.

나중에 번역해서 다시 글 수정할 예정.. 


android:windowSoftInputMode
How the main window of the activity interacts with the window containing the on-screen soft keyboard. The setting for this attribute affects two things:
  • The state of the soft keyboard — whether it is hidden or visible — when the activity becomes the focus of user attention.
  • The adjustment made to the activity's main window — whether it is resized smaller to make room for the soft keyboard or whether its contents pan to make the current focus visible when part of the window is covered by the soft keyboard.

The setting must be one of the values listed in the following table, or a combination of one "state..." value plus one "adjust..." value. Setting multiple values in either group — multiple "state..." values, for example — has undefined results. Individual values are separated by a vertical bar (|). For example:

<activity android:windowSoftInputMode="stateVisible|adjustResize" . . . >

Values set here (other than "stateUnspecified" and "adjustUnspecified") override values set in the theme.

ValueDescription
"stateUnspecified"The state of the soft keyboard (whether it is hidden or visible) is not specified. The system will choose an appropriate state or rely on the setting in the theme.

This is the default setting for the behavior of the soft keyboard.

"stateUnchanged"The soft keyboard is kept in whatever state it was last in, whether visible or hidden, when the activity comes to the fore.
"stateHidden"The soft keyboard is hidden when the user chooses the activity — that is, when the user affirmatively navigates forward to the activity, rather than backs into it because of leaving another activity.
"stateAlwaysHidden"The soft keyboard is always hidden when the activity's main window has input focus.
"stateVisible"The soft keyboard is visible when that's normally appropriate (when the user is navigating forward to the activity's main window).
"stateAlwaysVisible"The soft keyboard is made visible when the user chooses the activity — that is, when the user affirmatively navigates forward to the activity, rather than backs into it because of leaving another activity.
"adjustUnspecified"It is unspecified whether the activity's main window resizes to make room for the soft keyboard, or whether the contents of the window pan to make the current focus visible on-screen. The system will automatically select one of these modes depending on whether the content of the window has any layout views that can scroll their contents. If there is such a view, the window will be resized, on the assumption that scrolling can make all of the window's contents visible within a smaller area.

This is the default setting for the behavior of the main window.

"adjustResize"The activity's main window is always resized to make room for the soft keyboard on screen.
"adjustPan"The activity's main window is not resized to make room for the soft keyboard. Rather, the contents of the window are automatically panned so that the current focus is never obscured by the keyboard and users can always see what they are typing. This is generally less desirable than resizing, because the user may need to close the soft keyboard to get at and interact with obscured parts of the window.

This attribute was introduced in API Level 3.


AIDL을 이용한 RPC 서비스를 구현해보자!

Mobile Programming/Android 2011. 1. 19. 12:07

앞선 포스팅에서 로컬 서비스를 구현해 보았다.
RPC 서비스는 Remote Procedure Call 의 Abbreviation(?) 로서 , 다른 프로세스들이 소비할 수 있는 서비스임을 뜻한다.

다른 플랫폼들과 마찬가지로 클라이언트에 공개될 인터페이스 정의를 위해 IDL (interface definition language) 을 사용한다.

안드로이드에서는 이를 Android IDL , 즉 AIDL 이라고 지칭한다 .

서론은 이쯤하고 , RPC 서비스를 구현해보자.

1. AIDL 파일 만들기. 인터페이스를 만들자. (구현할 필요는 없다.)
2.이 파일을 src 폴더안에 넣기.
3.컴파일하기.
4.res아래에 웬 자바파일이 생겼다!
5.service를 상속한 액티비티를 하나 만들자.
6.그 안에 1번의 인터페이스를 구현할 클래스를 만들자. 상속하는 클래스는 ~.Stub
7.6에 메서드들을 구현한다.
8.로컬서비스에서 바인더클래스를 만드는대신에 6의 클래스를 만들어준것이다. 이 클래스를 onBind()에서 리턴해주자.
9.나머지는 로컬과 같다!

소스는 다음과 같다.


public class LocalService extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        ServiceConnection tserviceConnection = new ServiceConnection(){

			public void onServiceConnected(ComponentName name, IBinder service) {
				// TODO Auto-generated method stub
				Log.d("aidl","connected");

				try {
					((Service2.Service2impl)service).sayHello2();
				} catch (RemoteException e) {
					// TODO Auto-generated catch block
					Log.d("aidl",e.getStackTrace().toString());
				}
				
				
			}

			public void onServiceDisconnected(ComponentName name) {
				// TODO Auto-generated method stub
				
        	
			}
        };
        
        startService(new Intent(LocalService.this, Service2.class));
        
        bindService(new Intent(LocalService.this, Service2.class),tserviceConnection,0);
        
        Button btn_dis = (Button)findViewById(R.id.discon);
        
        btn_dis.setOnClickListener(new OnClickListener(){

			public void onClick(View v) {
				// TODO Auto-generated method stub
				stopService(new Intent(LocalService.this,Service2.class));
			}
        	
        });
        
    }
}
다음은 구현부이다.
public class Service2 extends Service {
	
	public class Service2impl extends aidlService.Stub{

		public void sayHello2() throws RemoteException {
			// TODO Auto-generated method stub
			Log.d("aidl","service2, sayHello2!");
		}
		
	}
	
	Service2impl tb = new Service2impl();
	
	@Override
	public IBinder onBind(Intent intent) {
		// TODO Auto-generated method stub
		return tb;
	}

}


여기까지 적고 메니페스트에 서비스에 대한 정보 한줄만을 적어도 로컬에서는 잘 돌아가지만, 이 서비스를 외부에서 이용할 수 있도록 인텐트 필터를 더 넣어줄 필요가 있다.


        	
        		
        	



여기까지하면 이제 외부에서 com.android.servicelocal 패키지안에 있는 aidlService를 접근할 수 있게된다 .

이제 새로운 프로젝트를 만들어서 이 서비스에 접근해보도록 하자 . 
역시 이름 명명법이 좀 헷갈리긴 하지만..  양해바랍니다.

TestService가 외부 액티비티로서 이 클래스는 com.android.testService 패키지 안에 존재한다.
앞서 aidl 파일을 이미 만들어 주었지만, 서비스에 접근할 이를테면, Client 프로세스에게도 일종의 계약서 같은 것이 필요하다.
따라서 생성해두었던 aidl 파일을 서비스가 존재하는 패키지경로로 패키지를 만들고 그안에 복사해 두도록 하자.


외부 서비스를 호출할때는 startService 대신에 bindService를 이용한다.
bindService의 첫번째 인자 intent는 이전 로컬에서 생성할때와는 다르게 인텐트 필터를 통해 서비스를 찾아갈수 있도록 
getName() 을 호출해주어야한다 .  (액션을 다른 스트링으로 주고 , 그 액션 인텐트를 넘겨주어도 OK!)

호출하는 외부 액티비티는 논거할 다른 기술은 없고 , 
bind할때 인텐트에 주의하고 , 그 서비스를 조작하기위해 서비스를 얻어올때 로컬과 같이 캐스팅을 통하지않고 
자동으로 생성된 java 파일의 클래스 객체.Stub.asInterface(IBinder) 임을 확인하자.



public class TestService extends Activity {

	public void onCreate(Bundle savedInstanceState){
		super.onCreate(savedInstanceState);
		
		setContentView(R.layout.main);
		
		final ServiceConnection serConn = new ServiceConnection(){

			@Override
			public void onServiceConnected(ComponentName name, IBinder service) {
				// TODO Auto-generated method stub
				Log.d("client","serviceConnected!");
				aidlService aservice;
				
				aservice = aidlService.Stub.asInterface(service);
				
				try {
					aservice.sayHello2();
				} catch (RemoteException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}

			@Override
			public void onServiceDisconnected(ComponentName name) {
				// TODO Auto-generated method stub
				Log.d("client","disConnect");
			}
			
		};
		
		Button btn_bind = (Button)findViewById(R.id.btn_bind);
		Button btn_stop = (Button)findViewById(R.id.btn_stop);
		
		btn_bind.setOnClickListener(new OnClickListener(){

			@Override
			public void onClick(View v) {
				// TODO Auto-generated method stub
				bindService(new Intent("com.neo" ),serConn,Context.BIND_AUTO_CREATE);
			}
			
		});
		
		btn_stop.setOnClickListener(new OnClickListener(){

			@Override
			public void onClick(View v) {
				// TODO Auto-generated method stub
				unbindService(serConn);
			}
			
		});
		
		
	}
	
}

Local Service를 이용해보자!

Mobile Programming/Android 2011. 1. 18. 18:46
서비스는 백그라운드에서 네트워크를 통해 어떤 파일을 다운받는다거나 ,  음악을 플레이 한다거나등의 작업을 하는 컴포넌트이다.

먼저 액티비티와 같이 Manifest 에 정의해주고 , 

1 . 액티비티에서 서비스 클래스를 실행시키고 ( startService )
2 . bindService 의 두번째 인자에 해당하는 ServiceConnection의 객체를 생성하고 onServiceConnected에 필요한 작업을 구현한다.
3 . 바인드 요청을 위해 bindService 메서드를 호출한다.
4 . 바인드 요청을 받을 서비스 클래스는 onBind 를 구현한다.
5 .  onBind의 리턴값은 액티비티와 통신을 할 IBinder를 상속한 클래스의 객체이다 . 따라서 그 객체를 구현해 줄 필요가 있다.
6 . 객체를 리턴한다.

코드로 확인해보기 바란다.

먼저 액티비티이다. (클래스 명이 헷갈리게 명명된 점 양해부탁드립니다.)


 
public class LocalService extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        ServiceConnection tserviceConnection = new ServiceConnection(){

			public void onServiceConnected(ComponentName name, IBinder service) {
				// TODO Auto-generated method stub
				Log.d("localservice","connected");
				((Service.TempBinder)service).sayHello();
			}

			public void onServiceDisconnected(ComponentName name) {
				// TODO Auto-generated method stub
				
			}
        	
        };
        
        startService(new Intent(LocalService.this, Service.class));
        
        bindService(new Intent(LocalService.this, Service.class),tserviceConnection,0);
    }
}
다음은 서비스클래스이다.
public class Service extends android.app.Service {

	class TempBinder extends Binder{
		public void sayHello(){
			Log.d("localservice","hello");
		}
	}
	
	TempBinder tb = new TempBinder();
	@Override
	public IBinder onBind(Intent intent) {
		// TODO Auto-generated method stub
		return tb;
	}

}