Basic
UI design, Styles & Themes
User Interface
In an Android application, the user interface is built using
View objects are the basic units of user interface expression on the Android platform. The View class serves as the base for subclasses called "widgets," which offer fully implemented UI objects, like text fields and buttons. The ViewGroup class serves as the base for subclasses called "layouts," which offer different kinds of layout architecture, like linear, tabular and relative.
A View object is a data structure whose properties store the layout parameters and content for a specific rectangular area of the screen. A View object handles its own measurement, layout, drawing, focus change, scrolling, and key/gesture interactions for the rectangular area of the screen in which it resides. As an object in the user interface, a View is also a point of interaction for the user and the receiver of the interaction events.
In order to attach the view hierarchy tree to the screen for rendering, your Activity must call the
The name of an XML element is respective to the Java class that it represents. So a
For example, a simple vertical layout with a text view and a button looks like this:
For more on building a UI layout, read XML Layouts.
Tip: You can also draw View and ViewGroups objects in Java code, using the
There are a variety of ways in which you can layout your views. Using more and different kinds of view groups, you can structure child views and view groups in an infinite number of ways. Some pre-defined view groups offered by Android (called layouts) include LinearLayout, RelativeLayout, TableLayout, GridLayout and others. Each offers a unique set of layout parameters that are used to define the positions of child views and layout structure.
To learn about some of the different kinds of view groups used for a layout, read Common Layout Objects.
Menus are also structured using a View hierarchy, but you don't define this structure yourself. Instead, you define the
Menus also handle their own events, so there's no need to register event listeners on the items in your menu. When an item in your menu is selected, the
And just like your application layout, you have the option to declare the items for you menu in an XML file.
The AdapterView object is an implementation of ViewGroup that determines its child views based on a given Adapter object. The Adapter acts like a courier between your data source (perhaps an array of external strings) and the AdapterView, which displays it. There are several implementations of the Adapter class, for specific tasks, such as the CursorAdapter for reading database data from a Cursor, or an ArrayAdapter for reading from an arbitrary array.
To learn more about using an Adapter to populate your views, read Binding to Data with AdapterView.
Learn more about using styles and themes in the Styles and Themes document.
The Android documentation for supporting multiple screen sizes says:
Notes: Density
does not exactly follow the real screen size, pixels are not
necessarily squares, there
is no strict definition for what a “pixel” means. This may
be confusing but not as much as CSS
Units (em, ex, pt, etcetera).
A style is a collection of properties that
specify the look and format for a
Styles in Android share a similar philosophy to cascading stylesheets in web design—they allow you to separate the design from the content.
For example, by using a style, you can take this layout XML:
A theme is a style applied to an entire
The root node of the XML file must be
For each style you want to create, add a
The
Remember, a style that you want to use as an Activity or application theme is defined in XML exactly the same as a style for a View. A style such as the one defined above can be applied as a style for a single View or as a theme for an entire Activity or application. How to apply a style for a single View or as an application theme is discussed later.
You can continue inheriting like this as many times as you'd like, by chaining names with periods. For example, you can extend
Note: This technique for inheritance by chaining together names only works for styles defined by your own resources. You can't inherit Android built-in styles this way. To reference a built-in style, such as
The best place to find properties that apply to a specific
For a reference of all available style properties, see the
Some style properties, however, are not supported by any View element and can only be applied as a theme. These style properties apply to the entire window and not to any type of View. For example, style properties for a theme can hide the application title, hide the status bar, or change the window's background. These kind of style properties do not belong to any View object. To discover these theme-only style properties, look at the
Note: Don't forget to prefix the property names in each
To apply a style definition as a theme, you must apply the style to an
Note: The
Just as A ndroid provides other built-in resources, there are many pre-defined themes that you can use, to avoid writing them yourself. For example, you can use the
Now use
For example, here is the declaration for a custom theme which is simply the standard platforms default light theme. It would go in an XML file under
A list of the standard attributes that you can use in themes can be found at
For more information about providing alternative resources, such as themes and layouts, based on the platform version or other device configurations, see the Providing Resources document.
The
These files will help you learn through example. For instance, in the Android themes source code, you'll find a declaration for
For more information about the syntax used to create styles in XML, see Available Resource Types: Style and Themes.
For a reference of available style attributes that you can use to define a style or theme (e.g., "windowBackground" or "textAppearance"), see
View
and ViewGroup
objects. There are many types of views and view groups, each of
which is a descendant of the View
class.View objects are the basic units of user interface expression on the Android platform. The View class serves as the base for subclasses called "widgets," which offer fully implemented UI objects, like text fields and buttons. The ViewGroup class serves as the base for subclasses called "layouts," which offer different kinds of layout architecture, like linear, tabular and relative.
A View object is a data structure whose properties store the layout parameters and content for a specific rectangular area of the screen. A View object handles its own measurement, layout, drawing, focus change, scrolling, and key/gesture interactions for the rectangular area of the screen in which it resides. As an object in the user interface, a View is also a point of interaction for the user and the receiver of the interaction events.
View Hierarchy
On the Android platform, you define an Activity's UI using a hierarchy of View and ViewGroup nodes, as shown in the diagram below. This hierarchy tree can be as simple or complex as you need it to be, and you can build it up using Android's set of predefined widgets and layouts, or with custom Views that you create yourself.In order to attach the view hierarchy tree to the screen for rendering, your Activity must call the
setContentView()
method and pass a reference to the root node object. The Android
system receives this reference and uses it to invalidate, measure,
and draw the tree. The root node of the hierarchy requests that its
child nodes draw themselves — in turn, each view group node is
responsible for calling upon each of its own child views to draw
themselves. The children may request a size and location within the
parent, but the parent object has the final decision on where how
big each child can be. Android parses the elements of your layout
in-order (from the top of the hierarchy tree), instantiating the
Views and adding them to their parent(s). Because these are drawn
in-order, if there are elements that overlap positions, the last one
to be drawn will lie on top of others previously drawn to that
space.Layout
The most common way to define your layout and express the view hierarchy is with an XML layout file. XML offers a human-readable structure for the layout, much like HTML. Each element in XML is either a View or ViewGroup object (or descendant thereof). View objects are leaves in the tree, ViewGroup objects are branches in the tree (see the View Hierarchy figure above).The name of an XML element is respective to the Java class that it represents. So a
<TextView>
element creates a TextView
in your UI, and a <LinearLayout>
element creates a LinearLayout
view group. When you load a layout resource, the Android system
initializes these run-time objects, corresponding to the elements in
your layout.For example, a simple vertical layout with a text view and a button looks like this:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <TextView android:id="@+id/text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Hello, I am a TextView" /> <Button android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Hello, I am a Button" /> </LinearLayout>Notice that the LinearLayout element contains both the TextView and the Button. You can nest another LinearLayout (or other type of view group) inside here, to lengthen the view hierarchy and create a more complex layout.
For more on building a UI layout, read XML Layouts.
Tip: You can also draw View and ViewGroups objects in Java code, using the
addView(View)
methods to dynamically insert new View and ViewGroup objects.There are a variety of ways in which you can layout your views. Using more and different kinds of view groups, you can structure child views and view groups in an infinite number of ways. Some pre-defined view groups offered by Android (called layouts) include LinearLayout, RelativeLayout, TableLayout, GridLayout and others. Each offers a unique set of layout parameters that are used to define the positions of child views and layout structure.
To learn about some of the different kinds of view groups used for a layout, read Common Layout Objects.
Widgets
A widget is a View object that serves as an interface for interaction with the user. Android provides a set of fully implemented widgets, like buttons, checkboxes, and text-entry fields, so you can quickly build your UI. Some widgets provided by Android are more complex, like a date picker, a clock, and zoom controls. But you're not limited to the kinds of widgets provided by the Android platform. If you'd like to do something more customized and create your own actionable elements, you can, by defining your own View object or by extending and combining existing widgets.Input Events
Once you've added some Views/widgets to the UI, you probably want to know about the user's interaction with them, so you can perform actions. To be informed of user input events, you need to do one of two things:- Define an event listener and register it with the View. More often than not, this is how you'll listen for events. The View class contains a collection of nested interfaces named On<something>Listener, each with a callback method called
On<something>()
. For example,View.OnClickListener
(for handling "clicks" on a View),View.OnTouchListener
(for handling touch screen events in a View), andView.OnKeyListener
(for handling device key presses within a View). So if you want your View to be notified when it is "clicked" (such as when a button is selected), implement OnClickListener and define itsonClick()
callback method (where you perform the action upon click), and register it to the View withsetOnClickListener()
. - Override an existing callback method for the View.
This is what you should do when you've implemented your own View
class and want to listen for specific events that occur within it.
Example events you can handle include when the screen is touched
(
onTouchEvent()
), when the trackball is moved (onTrackballEvent()
), or when a key on the device is pressed (onKeyDown()
). This allows you to define the default behavior for each event inside your custom View and determine whether the event should be passed on to some other child View. Again, these are callbacks to the View class, so your only chance to define them is when you build a custom component.
Menus
Application menus are another important part of an application's UI. Menus offers a reliable interface that reveals application functions and settings. The most common application menu is revealed by pressing the Menu button on the device. However, you can also add Context Menus, which may be revealed when the user presses and holds down on an item.Menus are also structured using a View hierarchy, but you don't define this structure yourself. Instead, you define the
onCreateOptionsMenu()
or onCreateContextMenu()
callback methods for your Activity and declare the items that you
want to include in your menu. At the appropriate time, Android will
automatically create the necessary View hierarchy for the menu and
draw each of your menu items in it.Menus also handle their own events, so there's no need to register event listeners on the items in your menu. When an item in your menu is selected, the
onOptionsItemSelected()
or onContextItemSelected()
method will be called by the framework.And just like your application layout, you have the option to declare the items for you menu in an XML file.
Advanced Topics
Once you've grappled the fundamentals of creating a user interface, you can explore some advanced features for creating a more complex application interface.Adapters
Sometimes you'll want to populate a view group with some information that can't be hard-coded, instead, you want to bind your view to an external source of data. To do this, you use an AdapterView as your view group and each child View is initialized and populated with data from the Adapter.The AdapterView object is an implementation of ViewGroup that determines its child views based on a given Adapter object. The Adapter acts like a courier between your data source (perhaps an array of external strings) and the AdapterView, which displays it. There are several implementations of the Adapter class, for specific tasks, such as the CursorAdapter for reading database data from a Cursor, or an ArrayAdapter for reading from an arbitrary array.
To learn more about using an Adapter to populate your views, read Binding to Data with AdapterView.
Styles and Themes
Perhaps you're not satisfied with the look of the standard widgets. To revise them, you can create some of your own styles and themes.- A style is a set of one or more formatting attributes that you can apply as a unit to individual elements in your layout. For example, you could define a style that specifies a certain text size and color, then apply it to only specific View elements.
- A theme is a set of one or more formatting attributes that
you can apply as a unit to all activities in an application, or
just a single activity. For example, you could define a theme that
sets specific colors for the window frame and the panel background,
and sets text sizes and colors for menus. This theme can then be
applied to specific activities or the entire application.
Learn more about using styles and themes in the Styles and Themes document.
Android units – pixels, density, dpi, dip, dp, dps, sp, sip
The Android documentation for supporting multiple screen sizes says:
The density-independent pixel is equivalent to one physical pixel on a 160 dpi screen, the baseline density assumed by the platform (as described later in this document). At run time, the platform transparently handles any scaling of the dp units needed, based on the actual density of the screen in use. The conversion of dp units to screen pixels is simple: pixels = dps * (density / 160). For example, on 240 dpi screen, 1 dp would equal 1.5 physical pixels. Using dp units to define your application’s UI is highly recommended, as a way of ensuring proper display of your UI on different screens.Confused? Try this. Take the example of two screens that are the same size but one has a resolution of 160 dpi (dots per inch, i.e. pixels per inch) and the other is 240 dpi.
Lower resolution screen | Higher resolution, same size | |
Physical Width | 1.5 inches | 1.5 inches |
Dots Per Inch (“dpi”) | 160 | 240 |
Pixels (=width*dpi) | 240 | 360 |
Density (factor of baseline 160) | 1.0 | 1.5 |
Density-independent Pixels (“dip” or “dp” or “dps”) | 240 | 240 |
Scale-independent pixels (“sip” or “sp”) | Depends on user font size settings | same |
Styles and Themes
View
or window. A style can specify properties such as height, padding,
font color, font size, background color, and much more. A style is
defined in an XML resource that is separate from the XML that
specifies the layout.Styles in Android share a similar philosophy to cascading stylesheets in web design—they allow you to separate the design from the content.
For example, by using a style, you can take this layout XML:
<TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:textColor="#00FF00" android:typeface="monospace" android:text="@string/hello" />And turn it into this:
<TextView style="@style/CodeFont" android:text="@string/hello" />All of the attributes related to style have been removed from the layout XML and put into a style definition called
CodeFont
,
which is then applied with the style
attribute. You'll see the definition for this style in the
following section.A theme is a style applied to an entire
Activity
or application, rather than an individual View
(as in the example above). When a style is applied as a theme,
every View in the Activity or application will apply each style
property that it supports. For example, you can apply the same
CodeFont
style as a theme for an
Activity and then all text inside that Activity will have green
monospace font.Defining Styles
To create a set of styles, save an XML file in theres/values/
directory of your project. The name of the XML file is arbitrary,
but it must use the .xml
extension and
be saved in the res/values/
folder.The root node of the XML file must be
<resources>
.For each style you want to create, add a
<style>
element to the file with a name
that
uniquely identifies the style (this attribute is required). Then
add an <item>
element for each
property of that style, with a name
that declares the style property and a value to go with it (this
attribute is required). The value for the <item>
can be a keyword string, a hex color, a reference to another
resource type, or other value depending on the style property.
Here's an example file with a single style:<?xml version="1.0" encoding="utf-8"?> <resources> <style name="CodeFont" parent="@android:style/TextAppearance.Medium"> <item name="android:layout_width">fill_parent</item> <item name="android:layout_height">wrap_content</item> <item name="android:textColor">#00FF00</item> <item name="android:typeface">monospace</item> </style> </resources>Each child of the
<resources>
element is converted into an application resource object at
compile-time, which can be referenced by the value in the <style>
element's name
attribute. This example
style can be referenced from an XML layout as @style/CodeFont
(as demonstrated in the introduction above).The
parent
attribute in the <style>
element is optional and specifies the resource ID of another style
from which this style should inherit properties. You can then
override the inherited style properties if you want to.Remember, a style that you want to use as an Activity or application theme is defined in XML exactly the same as a style for a View. A style such as the one defined above can be applied as a style for a single View or as a theme for an entire Activity or application. How to apply a style for a single View or as an application theme is discussed later.
Inheritance
Theparent
attribute in the <style>
element lets you specify a style from which your style should
inherit properties. You can use this to inherit properties from an
existing style and then define only the properties that you want to
change or add. You can inherit from styles that you've created
yourself or from styles that are built into the platform. (See
Using
Platform Styles and Themes, below, for information about
inheriting from styles defined by the Android platform.) For
example, you can inherit the Android platform's default text
appearance and then modify it:<style name="GreenText" parent="@android:style/TextAppearance"> <item name="android:textColor">#00FF00</item> </style>If you want to inherit from styles that you've defined yourself, you do not have to use the
parent
attribute. Instead, just prefix the name of the style you want to
inherit to the name of your new style, separated by a period. For
example, to create a new style that inherits the CodeFont
style defined above, but make the color red, you can author the new
style like this:<style name="CodeFont.Red"> <item name="android:textColor">#FF0000</item> </style>Notice that there is no
parent
attribute in the <style>
tag,
but because the name
attribute begins
with the CodeFont
style name (which is
a style that you have created), this style inherits all style
properties from that style. This style then overrides the
android:textColor
property to make the
text red. You can reference this new style as @style/CodeFont.Red
.You can continue inheriting like this as many times as you'd like, by chaining names with periods. For example, you can extend
CodeFont.Red
to be bigger, with:<style name="CodeFont.Red.Big"> <item name="android:textSize">30sp</item> </style>This inherits from both
CodeFont
and
CodeFont.Red
styles, then adds the
android:textSize
property.Note: This technique for inheritance by chaining together names only works for styles defined by your own resources. You can't inherit Android built-in styles this way. To reference a built-in style, such as
TextAppearance
,
you must use the parent
attribute.Style Properties
Now that you understand how a style is defined, you need to learn what kind of style properties—defined by the<item>
element—are available. You're probably familiar with some
already, such as layout_width
and textColor
.
Of course, there are many more style properties you can use.The best place to find properties that apply to a specific
View
is the corresponding class reference, which lists all of the
supported XML attributes. For example, all of the attributes listed
in the table of TextView
XML attributes can be used in a style definition for a TextView
element (or one of its subclasses). One of the attributes listed in
the reference is android:inputType
,
so where you might normally place the android:inputType
attribute in an <EditText>
element, like this:<EditText android:inputType="number" ... />You can instead create a style for the
EditText
element that includes this property:<style name="Numbers"> <item name="android:inputType">number</item> ... </style>So your XML for the layout can now implement this style:
<EditText style="@style/Numbers" ... />This simple example may look like more work, but when you add more style properties and factor-in the ability to re-use the style in various places, the pay-off can be huge.
For a reference of all available style properties, see the
R.attr
reference. Keep in mind that all View objects don't accept all the
same style attributes, so you should normally refer to the specific
View
class for supported style properties. However, if you apply a style
to a View that does not support all of the style properties, the
View will apply only those properties that are supported and simply
ignore the others.Some style properties, however, are not supported by any View element and can only be applied as a theme. These style properties apply to the entire window and not to any type of View. For example, style properties for a theme can hide the application title, hide the status bar, or change the window's background. These kind of style properties do not belong to any View object. To discover these theme-only style properties, look at the
R.attr
reference for attributes that begin with window
.
For instance, windowNoTitle
and
windowBackground
are style properties
that are effective only when the style is applied as a theme to an
Activity or application. See the next section for information about
applying a style as a theme.Note: Don't forget to prefix the property names in each
<item>
element with the
android:
namespace. For example: <item
name="android:inputType">
.Applying Styles and Themes to the UI
There are two ways to set a style:- To an individual View, by adding the
style
attribute to a View element in the XML for your layout. - Or, to an entire Activity or application, by adding the
android:theme
attribute to the<activity>
or<application>
element in the Android manifest.
View
in the layout, the properties defined by the style are applied only
to that View
.
If a style is applied to a ViewGroup
,
the child View
elements will not inherit the style
properties—only the element to which you directly apply the style
will apply its properties. However, you can apply a style
so that it applies to all View
elements—by applying the style as a theme.To apply a style definition as a theme, you must apply the style to an
Activity
or application in the Android manifest. When you do so, every View
within the Activity or application will apply each property that it
supports. For example, if you apply the CodeFont
style from the previous examples to an Activity, then all View
elements that support the text style properties will apply them.
Any View that does not support the properties will ignore them. If
a View supports only some of the properties, then it will apply
only those properties.Apply a style to a View
Here's how to set a style for a View in the XML layout:<TextView style="@style/CodeFont" android:text="@string/hello" />Now this TextView will be styled as defined by the style named
CodeFont
. (See the sample above, in
Defining
Styles.)Note: The
style
attribute does not use the android:
namespace prefix.Apply a theme to an Activity or application
To set a theme for all the activities of your application, open theAndroidManifest.xml
file and edit
the <application>
tag to include
the android:theme
attribute with the
style name. For example:<application android:theme="@style/CustomTheme">If you want a theme applied to just one Activity in your application, then add the
android:theme
attribute to the <activity>
tag
instead.Just as A ndroid provides other built-in resources, there are many pre-defined themes that you can use, to avoid writing them yourself. For example, you can use the
Dialog
theme and make your Activity appear like a dialog box:<activity android:theme="@android:style/Theme.Dialog">Or if you want the background to be transparent, use the Translucent theme:
<activity android:theme="@android:style/Theme.Translucent">If you like a theme, but want to tweak it, just add the theme as the
parent
of your custom theme. For
example, you can modify the traditional light theme to use your own
color like this:<color name="custom_theme_color">#b0b0ff</color> <style name="CustomTheme" parent="android:Theme.Light"> <item name="android:windowBackground">@color/custom_theme_color</item> <item name="android:colorBackground">@color/custom_theme_color</item> </style>(Note that the color needs to supplied as a separate resource here because the
android:windowBackground
attribute only supports a reference to another resource; unlike
android:colorBackground
, it can not be
given a color literal.)Now use
CustomTheme
instead of
Theme.Light
inside the Android
Manifest:<activity android:theme="@style/CustomTheme">
Select a theme based on platform version
Newer versions of Android have additional themes available to applications, and you might want to use these while running on those platforms while still being compatible with older versions. You can accomplish this through a custom theme that uses resource selection to switch between different parent themes, based on the platform version.For example, here is the declaration for a custom theme which is simply the standard platforms default light theme. It would go in an XML file under
res/values
(typically res/values/styles.xml
):
<style name="LightThemeSelector" parent="android:Theme.Light"> ... </style>To have this theme use the newer holographic theme when the application is running on Android 3.0 (API Level 11) or higher, you can place an alternative declaration for the theme in an XML file in
res/values-v11
, but make the parent
theme the holographic theme:<style name="LightThemeSelector" parent="android:Theme.Holo.Light"> ... </style>Now use this theme like you would any other, and your application will automatically switch to the holographic theme if running on Android 3.0 or higher.
A list of the standard attributes that you can use in themes can be found at
R.styleable.Theme
.For more information about providing alternative resources, such as themes and layouts, based on the platform version or other device configurations, see the Providing Resources document.
Using Platform Styles and Themes
The Android platform provides a large collection of styles and themes that you can use in your applications. You can find a reference of all available styles in theR.style
class. To use the styles listed here, replace all underscores in
the style name with a period. For example, you can apply the
Theme_NoTitleBar
theme with "@android:style/Theme.NoTitleBar"
.The
R.style
reference, however, is not well documented and does not thoroughly
describe the styles, so viewing the actual source code for these
styles and themes will give you a better understanding of what
style properties each one provides. For a better reference to the
Android styles and themes, see the following source code:These files will help you learn through example. For instance, in the Android themes source code, you'll find a declaration for
<style name="Theme.Dialog">
.
In this definition, you'll see all of the properties that are used
to style dialogs that are used by the Android framework.For more information about the syntax used to create styles in XML, see Available Resource Types: Style and Themes.
For a reference of available style attributes that you can use to define a style or theme (e.g., "windowBackground" or "textAppearance"), see
R.attr
or the respective View class for which you are creating a style.
No comments:
Post a Comment