My aim: have a Page slide onto the screen when the user swipes in from the left side of any screen.
My current solution:
- I start my Service (GlobalTouchService)
- which creates a narrow transparent view (floatingView) with an OnTouchListener
- this view is then added to side of the screen using WindowManager.addView
- on MotionEvent.ACTION_DOWN I add the inflated layout to this View (I call it innerView)
- on MotionEvent.ACTION_MOVE I adjust the width of the window and the left and right margins of the innerView by the event.getRawX() coordinate
Problem: It works, but its movement is very jerky (slow). I am sure there is a much better way of doing this to make it as smooth as an "official" navigation drawer.
Source file snipets
AndroidManifest.xml (permission necessary for system wide touch detection)
<service android:name=".GlobalTouchService" />
</application>
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
panel_layout.xml (this is the contents of the window that's to be dragged in)
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#9900cc"
android:orientation="vertical"
android:gravity="right"
tools:context=".GlobalTouchService" >
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/dummy_content2" />
</LinearLayout>
GlobalTouchService.java (I include the whole service here)
public class GlobalTouchService extends Service implements OnTouchListener{
private String TAG = this.getClass().getSimpleName();
private WindowManager mWindowManager;
private LinearLayout floatingView;
private LinearLayout innerView;
private LinearLayout.LayoutParams floatingParams;
private LinearLayout.LayoutParams innerParams;
private WindowManager.LayoutParams windowParams;
private int dw,dh;
@Override
public IBinder onBind(Intent arg0) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
// get screen dimensions
mWindowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
dw=mWindowManager.getDefaultDisplay().getWidth();
dh=mWindowManager.getDefaultDisplay().getHeight();
// prepare contents to show when ACTION_DOWN is detected
innerView =new LinearLayout(this);
innerParams =new LinearLayout.LayoutParams(dw,dh);
innerParams.gravity=Gravity.RIGHT | Gravity.TOP;
innerView.setLayoutParams(innerParams);
View view=View.inflate(this, R.layout.panel_layout, new FrameLayout(this));
innerView.addView(view);
// create blank view to catch touches
floatingView = new LinearLayout(this);
floatingParams = new LinearLayout.LayoutParams(dw, dh);
floatingView.setLayoutParams(floatingParams);
floatingView.setOnTouchListener(this);
// add view to the windowmanager
windowParams = new WindowManager.LayoutParams(
dw/30, // narrow stripe on the left
dh, // full height of the screen
WindowManager.LayoutParams.TYPE_PHONE,
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
| WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT);
windowParams.gravity = Gravity.LEFT | Gravity.TOP;
mWindowManager.addView(floatingView, windowParams);
}
@Override
public void onDestroy() {
if(mWindowManager != null) {
// remove window when service is finished
if(floatingView != null) mWindowManager.removeView(floatingView);
}
super.onDestroy();
}
@Override
public boolean onTouch(View v, MotionEvent event) {
if(event.getAction() == MotionEvent.ACTION_DOWN && windowParams.width<dw/3 ) {
// add the inflated contents
floatingView.addView(innerView);
return true;
} else if(event.getAction() == MotionEvent.ACTION_UP && event.getRawX()<dw/3) {
// if released early collapses back to the left edge
windowParams.width=dw/30;
floatingView.removeView(innerView);
mWindowManager.updateViewLayout(floatingView, windowParams);
return true;
} if(event.getAction()==MotionEvent.ACTION_MOVE) {
windowParams.width=Math.max((int)event.getRawX(),dw/30);
innerParams.leftMargin=-(dw-(int)event.getRawX());
innerParams.rightMargin=(dw-(int)event.getRawX());
mWindowManager.updateViewLayout(floatingView, windowParams);
return true;
}
return false;
}
}