The Fan Controller app displays a circular UI element that resembles a physical fan control, to control the fan speed. It demonstrates how to create custom views.
data:image/s3,"s3://crabby-images/cac5c/cac5cc55c53fc9cecd94831579c7d6eebccea707" alt="App Preview 1"
data:image/s3,"s3://crabby-images/33d7a/33d7afd90d272bc6d5484c4b630aad03d68d31ad" alt="App Preview 2"
data:image/s3,"s3://crabby-images/d3ddf/d3ddf1cde5f639cae0ad6cdc599af67e4fadfae7" alt="App Preview 3"
data:image/s3,"s3://crabby-images/f6226/f6226405bb3a34370c5f7da7b8c2af51148efa82" alt="App Preview 4"
The View class provides many subclasses called UI widgets for common use cases, like TextView, ImageView, Button and EditText. We can create custom views by subclassing these views or directly extend the View class to start from scratch. To save efforts we must extend the closest possible existing view available. Steps for creating custom views :
- Extend View, or extend a View subclass.
- If a subclass is extended only override the behaviour or aspects that are needed to be changed.
- If the View class is extended, override
onDraw()
,onMeasure()
and other methods to draw its shape and control its appearance. - Add code to respond to user interaction and redraw if necessary.
- Use custom view as UI widget in layout.
- Define custom attributes for the view to provide customisation in different layouts.
When we extend the view subclass such as EditText, that subclass defines view's appearance and attributes and draws itself. We don't have to write the code to draw the view because we can override methods of the parent to customize the view when required. But, if we create view from scratch by extending view directly, we have to draw the entire view everytime the screen refreshes. We need to override view methods to handle drawing. Following are the things to be done to draw custom views:
-
Override
onSizeChanged()
to calculate the view's size when it appears first time and each time its size changes. It includes calculation for positions, dimensions, or any other value related to the custom view's size. -
Override
onDraw()
to draw the custom view, using a Canvas object styled by a Paint object. -
Call
invalidate()
when responding to a user click that changes how the view is drawn, forcing a call toonDraw()
to redraw the view. -
Override
onMeasure()
to accurately define how the custom view is aligned by parent view and fits into layout.
To enable custom view to be clickable following needs to be done:
-
Set view's
isClickable
property to true. -
Implement
performClick()
to handle the click. -
Call
invalidate()
to redraw the view.
For standard view, we implement onClickListener()
to handle the view clicks. For custom views we
instead implement performClick()
method. The super class method also calls the click listener, so
we add default functions to perform click, and leave on-click listener available for further
customisation.
We can make the custom view to have custom attributes. Following are the steps to achieve it.
-
Create
res/values/attrs.xml
file, if it doesn't exist already. -
Declare a styleable and define the required attributes and their value types.
<declare-styleable name="DialView"> <attr name="fanColor" format="color" /> </declare-styleable>
-
Set the values to these attributes in the custom view, using
app
namespace.<com.example.fancontroller.DialView android:id="@+id/dial_view" app:fanColor="#FFEB3B"/>
-
In order to use the attributes, we need to retrieve them. They are stored in an AttributeSet. We use
withStyledAttributes
extension function to initialise the attributes in our custom view class.context.withStyledAttributes(attrs, R.styleable.DialView) { fanSpeedColor = getColor(R.styleable.DialView_fanColor, 0) }
-
onDraw()
method is called every time the screen refreshes, which can be many times a second. To avoid visual glitches, and better performance, we should do as little work as possible inonDraw()
. One way to do this is by avoiding allocations inonDraw()
. -
The
performClick()
method canonClickListener()
. To do so we need to call super class method first, which enables accessibility events and calls onClickListener. -
If something in the custom view changes for any reason, including user interaction, and the change needs to be displayed, we must call
invalidate()
, to invalidate the current view and to re-draw the view.