Making Bottomsheet responsive using Custom Gesture Detector in PSLab Android App
In the previous blog Creating Instruction Guide using Bottomsheet, I have created the Bottom Sheet guide in instrument activities in PSLab Android app. But simply adding the Bottom Sheet in the layout is not enough as it could lead to some UI issues like no proper way to show or hide the Bottom Sheet, therefore, he/she will find it difficult to work with Bottom Sheet that could degrade User Experience.
We need to make the Bottom Sheet responsive and interactive which we can do by capturing swipe gestures done by the user and overriding their functionality i.e. when the user slides up with the finger then the Bottom Sheet will reveal itself and when the user slides the finger down the Bottom Sheet will hide.
For this Android provides a class GestureDetector which is used with another class SimpleOnGestureListener which acts as a listener to capture Gesture events like swipe, pinch, scroll, long press etc.
In this blog, I will create a custom gesture listener that will listen to the swipe events and according to the gestures it will show/hide the Bottom Sheet.
I will start by creating a gesture listener class called “SwipeGestureListener” extending the class ‘GestureDetector.SimpleOnGestureListener’ and also as I need swipe gestures to control the Bottom Sheet, so I will pass the reference of the Bottom Sheet as a parameter in the constructor.
public class SwipeGestureListener extends GestureDetector.SimpleOnGestureListener{
private BottomSheetBehavior bottomSheet;
public SwipeGestureDetector(BottomSheetBehavior bt) {
bottomSheet = bt;
}
}
Now in this listener class as we are concerned with the swipe events so will only override the below method provided by ‘GestureDetector.SimpleOnGestureListener’ interface
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY)
This method is called whenever the user swipes its finger in any direction.
In the above code, we can see that the method provides with object e1 and e2 of type MotionEvent. The MotionEvent class is used to report movements in terms of Action Codes like ACTION_DOWN, ACTION_UP and also contains other information about the touch like the pressure of the touch, x and y coordinate, orientation of the contact area etc.
The e1 object will have the attribute values relating to the point when the swipe started and the e2 object will have attribute values relating to the point when the swipe has ended.
Now, the main thing we need to determine if the direction of the swipe which is not directly available using the MotionEvent object.
So, to determine the direction of the swipe I will fetch the coordinates of the initial point and terminal point of the swipe using the objects initial and final point i.e., e1 and e2.
//Initial Point
float x1 = e1.getX(), y1 = e1.getY();
//Final Point
float x2 = e2.getX(), y2 = e2.getY();
Then, using these coordinates to calculate the angle of the swipe and based on the angle I will return the direction of the swipe as shown in the code below
private Direction getDirection(float x1, float y1, float x2, float y2) {
Double angle = Math.toDegrees(Math.atan2(y1 - y2, x2 - x1));
if (angle > 45 && angle <= 135)
return Direction.TOP;
if (angle >= 135 && angle < 180 || angle < -135 && angle > -180)
return Direction.LEFT;
if (angle < -45 && angle>= -135)
return Direction.DOWN;
if (angle > -45 && angle <= 45)
return Direction.RIGHT;
return null; // required by java to avoid error
}
As of now, I have the direction of the swipe so I will apply switch case and handle the swipe up and swipe down gesture as below:
- When the user slides up:- Show the Bottom Sheet by changing the state of the Bottom Sheet from STATE_HIDDEN to STATE_COLLAPSED(partially viewable).
- When the user slides down: – Hide the Bottom Sheet by changing the state of the Bottom Sheet to STATE_HIDDEN.
For doing this, we will modify the ‘onFIing()’ method as shown below
@Override
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
switch (getDirection(e1.getX(), e1.getY(), e2.getX(), e2.getY())) {
case TOP:
bottomSheet.setState(BottomSheetBehavior.STATE_COLLAPSED);
return true;
case LEFT:
return true;
case DOWN:
if(bottomSheet.getState()==BottomSheetBehavior.STATE_COLLAPSED){
bottomSheet.setState(BottomSheetBehavior.STATE_HIDDEN);
}
return true;
case RIGHT:
return true;
default:
return false;
}
}
Now, the custom gesture listener is implemented but it cannot start listening to the touch event on its own, so we need to resolve this by performing the following steps:
- Firstly, we need to create an object of class GestureDetector and pass the current activity context and the object of class ‘SwipeGestureListener’ as parameters. Also while creating the listener for ‘SwipeGestureListener’ we need to pass the object of the Bottom Sheet in it as a parameter.
GestureDetector gestureDetector = new GestureDetector(this, new SwipeGestureListener(bottomSheetBehavior));
- Then we need to override the ‘onTouchEvent()’ method of our Activity and pass the event which is received as a parameter to the GestureDetector.
Doing this will pass the touch event that it received to the GestureDetector for it to handle.@Override public boolean onTouchEvent(MotionEvent event) { gestureDetector.onTouchEvent(event); return super.onTouchEvent(event); }
The Bottom Sheet is now responsive to the gestures on the screen and this will improve the User Experience.
Resources
- Detect Common Gestures – Android Developer Article – Android documentation
- Choreographic animations with Android’s Bottom Sheet – Blog by Orkhan Gasimli