아래 예제는 Android studio 1.2를 기반으로 작성한 코드입니다.
안드로이드 WindowManagert의 onTouch event 처리 방법을 소개합니다. WindowManager은 Activity를 벗어나서 처리하는
부분으로 앱을 벗어나서 자유롭게 사용할 수 있는 Layout 입니다. 이 레이아웃을 사용한 앱으로는 페이스북의 Message 앱이
가장 유명하며, 구현이 가장 잘 되어 있습니다. 우선 최종적으로 2개의 예제를 작성하였습니다.
이번 글은 가장 기본적인 좌표 이동 방법을 소개하고 다음 글에서는 좀 더 Smooth한 이동을 위한 애니메이션 처리 방법을
살펴보겠습니다.
목차
- WindowManager의 onTouch Event 처리하기
- WindowManager의 Animation을 이용한 Smooth한 onTouch Event 처리
예제 영상
Smooth 보다는 자연스럽게 따라오지는 않지만 아래의 화면만 가지고는 확인할 수는 없습니다.
사용한 API
- WindowManager : http://developer.android.com/reference/android/view/WindowManager.html
예제코드
- windowManager init.
WindowManager를 이용하여 아래와 같이 Layout을 초기화 합니다.
WindowManager는 Context.getSystemService(Context.WINDOW_SERVICE)을 이용하여 초기화 후 사용합니다.
windowManager에 addView를 하기위해서는 Layout과 LayoutParams를 각각 초기화 하고, addView를 진행합니다.
LayoutParams에는 width, heigh를 지정하고, x, y 좌표와 Type, flag, format을 각각 지정합니다.
- type 지정자는 아래와 같습니다.
Type은 해당 레이아웃이 어느 위치에 보여질지를 지정하는 것입니다.
Notification 위에 올릴것인지 Notification 아래에 올릴것인지 등 아래와 같은 설정값을 가집니다.
자세한 내용은 API 문서를 참고하시면 됩니다.
TYPE_APPLICATION_ATTACHED_DIALOG
TYPE_TOAST // 아래 예제는 TOAST를 사용하였습니다.
- flag 지정자는 아래와 같습니다.
flag는 해당 버튼을 누를 수 있는 버튼으로 만들것인지, 그냥 단순히 보여주는 형태로 제공될것인지 등에 대하여
설정할 수 있습니다.
FLAG_ALLOW_LOCK_WHILE_SCREEN_ON
FLAG_LAYOUT_IN_SCREEN // 화면을 벗어나지 않도록 설정합니다.
FLAG_LAYOUT_NO_LIMITS // 화면을 벗어날 수 있도록 설정합니다.(단, 위의 TYPE 설정에 따라서 동작이 달라질 수 있습니다.)
- gravity는 왼쪽 상단을 0, 0의 위치로 설정하였습니다.
예를 들어 TOP, CENTER_HORIZONTAL를 기준으로 하게된다면 상단 가운데 부분이 0, 0이 되며
가운데 이전은 - 값을 가지게 됩니다.
/** * Window View 를 초기화 한다. X, Y 좌표는 0, 0으로 지정한다. */ private void initWindowLayout() { windowView = (LinearLayout) layoutInflater.inflate(R.layout.window_layout, null); windowViewLayoutParams = new WindowManager.LayoutParams( WindowManager.LayoutParams.WRAP_CONTENT, WindowManager.LayoutParams.WRAP_CONTENT, 0, 0, // X, Y 좌표 WindowManager.LayoutParams.TYPE_TOAST, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS, PixelFormat.TRANSLUCENT); windowViewLayoutParams.gravity = Gravity.TOP | Gravity.LEFT; windowManager.addView(windowView, windowViewLayoutParams); windowView.setOnTouchListener(this); }
예제 코드 - onTouch 부분
private float prevX; private float prevY; @Override public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: // 처음 위치를 기억해둔다. prevX = event.getRawX(); prevY = event.getRawY(); break; case MotionEvent.ACTION_MOVE: float rawX = event.getRawX(); // 절대 X 좌표 값을 가져온다. float rawY = event.getRawY(); // 절대 Y 좌표값을 가져온다. // 이동한 위치에서 처음 위치를 빼서 이동한 거리를 구한다. float x = rawX - prevX; float y = rawY - prevY; setCoordinateUpdate(x, y); prevX = rawX; prevY = rawY; break; } return false; } /** * 이동한 거리를 x, y를 넘겨 LayoutParams에 갱신한다. * @param x * @param y */ private void setCoordinateUpdate(float x, float y) { if (windowViewLayoutParams != null) { windowViewLayoutParams.x += (int) x; windowViewLayoutParams.y += (int) y; windowManager.updateViewLayout(windowView, windowViewLayoutParams); } }
마무리
위와 같은 간단한 코드를 통해 WindowManager의 ViewLayout을 이동할 수 있습니다.
다음 글에서는 여기에 Animation을 추가하여 좀 더 Smooth한 이동이 가능한 Window Layout을 구현해보도록 하겠습니다.
'IT_Programming > Android_Java' 카테고리의 다른 글
안드로이드 프로가드 설정 기본 포맷! (0) | 2015.03.18 |
---|---|
Android Studio Gradle 관련 빈번히 마주할 수 있는 오류사항 정리! (0) | 2015.03.18 |
[ Android ] 디바이스 및 킷캣(Kitkat) 업데이트에 따른 갤러리 경로 호출 문제 (0) | 2015.02.25 |
안드로이드 어플리케이션이 회전될 때 주의해야할 점 (0) | 2015.02.10 |
Scheduling Repeating Alarms (0) | 2015.01.20 |