Naming. Follow the convention of prefixing the type, as in type_foo_bar.xml
. Examples: fragment_contact_details.xml
, view_primary_button.xml
, activity_main.xml
.
Organizing layout XMLs. If you're unsure how to format a layout XML, the following convention may help.
- One attribute per line, indented by 4 spaces
android:id
as the first attribute alwaysandroid:layout_****
attributes at the topstyle
attribute at the bottom- Tag closer
/>
on its own line, to facilitate ordering and adding attributes. - Rather than hard coding
android:text
, consider using Designtime attributes available for Android Studio.
<?xml version="1.0" encoding="utf-8"?>
<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:orientation="vertical"
>
<TextView
android:id="@+id/name"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:text="@string/name"
style="@style/FancyText"
/>
<include layout="@layout/reusable_part" />
</LinearLayout>
As a rule of thumb, attributes android:layout_****
should be defined in the layout XML, while other attributes android:****
should stay in a style XML. This rule has exceptions, but in general works fine. The idea is to keep only layout (positioning, margin, sizing) and content attributes in the layout files, while keeping all appearance details (colors, padding, font) in styles files.
The exceptions are:
android:id
should obviously be in the layout filesandroid:orientation
for aLinearLayout
normally makes more sense in layout filesandroid:text
should be in layout files because it defines content- Sometimes it will make sense to make a generic style defining
android:layout_width
andandroid:layout_height
but by default these should appear in the layout files
Use styles. Almost every project needs to properly use styles, because it is very common to have a repeated appearance for a view. At least you should have a common style for most text content in the application, for example:
<style name="ContentText">
<item name="android:textSize">@dimen/font_normal</item>
<item name="android:textColor">@color/basic_black</item>
</style>
Applied to TextViews:
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/price"
style="@style/ContentText"
/>
You probably will need to do the same for buttons, but don't stop there yet. Go beyond and move a group of related and repeated android:****
attributes to a common style.
Split a large style file into other files. You don't need to have a single styles.xml
file. Android SDK supports other files out of the box, there is nothing magical about the name styles
, what matters are the XML tags <style>
inside the file. Hence you can have files styles.xml
, styles_home.xml
, styles_item_details.xml
, styles_forms.xml
. Unlike resource directory names which carry some meaning for the build system, filenames in res/values
can be arbitrary.
colors.xml
is a color palette. There should be nothing in your colors.xml
other than a mapping from a color name to an RGBA value. This helps avoid repeating RGBA values and as such will make it easy to change or refactor colors, and also will make it explicit how many different colors are being used. Normally for a aesthetic UI, it is important to reduce the variety of colors being used.
So, don't define your colors.xml like this:
<resources>
<color name="button_foreground">#FFFFFF</color>
<color name="button_background">#2A91BD</color>
</resources>
Instead, do this:
<resources>
<!-- grayscale -->
<color name="white">#FFFFFF</color>
<!-- basic colors -->
<color name="blue">#2A91BD</color>
</resources>
Ask the designer of the application for this palette. The names do not need to be plain color names as "green", "blue", etc. Names such as "brand_primary", "brand_secondary", "brand_negative" are totally acceptable as well.
By referencing the color palette from your styles allows you to abstract the underlying colors from their usage in the app, as per:
colors.xml
- defines only the color palette.styles.xml
- defines styles which reference the color palette and reflects the color usage. (e.g. the button foreground is white).activity_main.xml
- references the appropriate style instyles.xml
to color the button.
If needed, even further separation between underlying colors and style usage can be achieved by defined an additional color resource file which references the color palette. As per:
<color name="button_foreground">@color/white</color>
<color name="button_background">@color/blue</color>
Then in styles.xml:
<style name="AcceptButton">
<item name="android:foreground">@color/button_foreground</item>
<item name="android:background">@color/button_background</item>
</style>
This approach offers improved color refactoring and more stable style definitions when multiple related styles share similar color and usage properties. However, it comes at the cost of maintaining another set of color mappings.
Treat dimens.xml like colors.xml. You should also define a "palette" of typical spacing and font sizes, for basically the same purposes as for colors. A good example of a dimens file:
<resources>
<!-- font sizes -->
<dimen name="font_larger">22sp</dimen>
<dimen name="font_large">18sp</dimen>
<dimen name="font_normal">15sp</dimen>
<dimen name="font_small">12sp</dimen>
<!-- typical spacing between two views -->
<dimen name="spacing_huge">40dp</dimen>
<dimen name="spacing_large">24dp</dimen>
<dimen name="spacing_normal">14dp</dimen>
<dimen name="spacing_small">10dp</dimen>
<dimen name="spacing_tiny">4dp</dimen>
<!-- typical sizes of views -->
<dimen name="button_height_tall">60dp</dimen>
<dimen name="button_height_normal">40dp</dimen>
<dimen name="button_height_short">32dp</dimen>
</resources>
You should use the spacing_****
dimensions for layouting, in margins and paddings, instead of hard-coded values, much like strings are normally treated. This will give a consistent look-and-feel, while making it easier to organize and change styles and layouts.
strings.xml
Name your strings with keys that resemble namespaces, and don't be afraid of repeating a value for two or more keys. Languages are complex, so namespaces are necessary to bring context and break ambiguity.
Bad
<string name="network_error">Network error</string>
<string name="call_failed">Call failed</string>
<string name="map_failed">Map loading failed</string>
Good
<string name="error_message_network">Network error</string>
<string name="error_message_call">Call failed</string>
<string name="error_message_map">Map loading failed</string>
Don't write string values in all uppercase. Stick to normal text conventions (e.g., capitalize first character). If you need to display the string in all caps, then do that using for instance the attribute textAllCaps
on a TextView.
Bad
<string name="error_message_call">CALL FAILED</string>
Good
<string name="error_message_call">Call failed</string>