It is quite likely that by this point in time, you are “chomping at the bit” to get into actually writing some code. This is understandable. That being said, before we dive into the Java source code for our stub project, we really should chat briefly about resources.
Resources are static bits of information held outside the Java source code.
As we discussed previously, resources are stored as files under the res/
directory in your source set (e.g., app/src/main/res/
).
Here is where you will find all your icons and other images,
your externalized strings for internationalization, and more.
These are separate from the Java source code not only because they are different in format. They are separate because you can have multiple definitions of a resource, to use in different circumstances. For example, with internationalization, you will have strings for different languages. Your Java code will be able to remain largely oblivious to this, as Android will choose the right resource to use, from all candidates, in a given circumstance (e.g., choose the Spanish string if the device’s locale is set to Spanish).
We will cover all the details of these resource sets later in the book. Right now, we need to discuss the resources in use by our stub project, plus one more.
This chapter will refer to the res/
directory as a shorthand for the
app/src/main/res/
directory of the project.
Keeping your labels and other bits of text outside the main source code of your application is generally considered to be a very good idea. In particular, it helps with internationalization (I18N) and localization (L10N). Even if you are not going to translate your strings to other languages, it is easier to make corrections if all the strings are in one spot instead of scattered throughout your source code.
Generally speaking, all you need to do is have an XML file in the res/values
directory (typically named res/values/strings.xml
), with a resources
root
element, and one child string
element for each string you wish to encode as a
resource. The string
element takes a name
attribute, which is the unique
name for this string, and a single text element containing the text of the
string:
<resources>
<string name="quick">The quick brown fox...</string>
<string name="laughs">He who laughs last...</string>
</resources>
One tricky part is if the string value contains a quote or an
apostrophe. In those cases, you will want to escape those values, by
preceding them with a backslash (e.g., These are the times that try men\'s souls
).
Or, if it is just an apostrophe, you could enclose the value in quotes
(e.g., "These are the times that try men's souls."
).
For example, a project’s strings.xml
file could look like this:
<resources>
<string name="app_name">EmPubLite</string>
</resources>
We can reference these string resources from various locations, in our Java
source code and elsewhere. For example, the app_name
string resource often is used
in the AndroidManifest.xml
file:
<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.commonsware.empublite"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<supports-screens
android:largeScreens="true"
android:normalScreens="true"
android:smallScreens="false"
android:xlargeScreens="true" />
<application
android:allowBackup="false"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme"
tools:ignore="GoogleAppIndexingWarning">
<activity android:name=".EmPubLiteActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
Here, the android:label
attribute of the <application>
element
refers to the app_name
string resource. This will appear in a few places in
the application, notably in the list of installed applications in Settings. So, if you wish to change
how your application’s name appears in these places, simply adjust the
app_name
string resource to suit.
The syntax @string/app_name
tells Android “find the string resource named
app_name
”. This causes Android to scan the appropriate strings.xml
file
(or any other file containing string resources in your res/values/
directory)
to try to find app_name
.
When you view a manifest like that in Android Studio, it may appear as though
you are not using resources, as you may not see @string/...
references:
Figure 50: AndroidManifest.xml, As Initially Viewed in Android Studio
Here, android:label
looks as though it is the hard-coded value “EmPubLite”.
However, notice that the attribute value is formatted differently than the others. The rest are green text with a white background, while this one is gray text with a shaded background.
That is because Android Studio is lying to you.
If you hover your mouse over the value, you will see the real attribute appear just below it:
Figure 51: AndroidManifest.xml, With Mouse Hovering Over “EmPubLite”
And, if you click on the fake value, you will see the real XML, with the real string resource value.
What is happening is that Android Studio, by default, will substitute a candidate value for the resource in its presentation of the manifest, other resources that refer to resources, and even Java code. Any time you see that gray-on-light-blue formatting, remember that this is not the real value, and that you have to uncover the real value via hovering over it or clicking on it.
Many things in Android can display rich text, where the text has been formatted
using some lightweight HTML markup: <b>
, <i>
, and <u>
. Your
string resources support this, simply by using the HTML tags as you would in
a Web page:
<resources>
<string name="b">This has <b>bold</b> in it.</string>
<string name="i">Whereas this has <i>italics</i>!</string>
</resources>
Since a strings resource XML file is an XML file, if your
message contains [
, ]
, or &
characters (other than the formatting
tags listed above), you will need to use
a CDATA
section:
<string name="report_body">
<![CDATA[
<html>
<body>
<h1>TPS Report for: {{reportDate}}</h1>
<p>Here are the contents of the TPS report:</p>
<p>{{message}}</p>
<p>If you have any questions regarding this report, please
do <b>not</b> ask Mark Murphy.</p>
</body>
</html>
]]>
</string>
Our string resources in our stub project are in the res/values/strings.xml
file. Since this directory name (values
) has no suffixes,
the string resources in that
directory will be valid for any sort of situation, including any locale for
the device. We will need additional directories, with distinct strings.xml
files, to support other languages. We will cover how to do that
later in this book.
If you double-click on a string resource file, like res/values/strings.xml
,
in Android Studio, you are
presented the XML and edit it that way. There is an option for entering
a dedicated string translation view, covered
later in this book.
Android 7.0+ users can indicate that they support more than one language:
Figure 52: Android 7.0 Language Settings
The user can choose the relative priorities of these languages, by grabbing the handle on the right side of the row and dragging the language higher or lower in the list.
This has impacts on resource resolution for any locale-dependent
resources, such as strings. Now Android will check multiple languages
for resource matches, before falling back to the default language
(e.g., whatever you have in res/values/strings.xml
). Hence, it is
important that you ensure that you have a complete set of strings for
every language that you support, lest the user perhaps wind up with
a mixed set of languages in the UI.
You can find out what languages the user has requested via a LocaleList
class and its getDefault()
static
method. This, as the name suggests,
has a list of Locale
objects representing the user’s preferred languages.
If you had previously been using Locale
alone for this (e.g., for
specialized in-app language assistance beyond resources), you will want
to switch to LocaleList
for Android 7.0 and beyond.
All Android versions support images in the PNG, JPEG, and GIF formats. GIF is officially discouraged, however; PNG is the overall preferred format. Android also supports some proprietary XML-based image formats, though we will not discuss those at length until later in the book. Many newer versions of Android also support Google’s WebP image format, though this is not especially popular.
There are two types of resources that use images like these: drawables and mipmaps. In truth, they are nearly identical. Mipmaps are used mostly for “launcher icons” — the icons seen in home screen launchers that identify activities that the user can start. Drawables hold everything else.
(if you are a seasoned Android developer and are reading this section:
while drawable resources might be removed when packaging an APK, such as
for the Android Gradle Plugin split
system for making density-specific
editions of an app, mipmap resources are left alone, apparently)
It is possible to have res/drawable/
and res/mipmap/
directories
in an Android module. However, you will not find bitmaps there usually.
Instead, those reside in directories like res/drawable-mdpi/
and
res/drawable-hdpi/
.
These refer to distinct resource sets. The suffixes (e.g., -mdpi
,
-hdpi
) are filters, indicating under what circumstances the images
stored in those directories should be used. Specifically, -ldpi
indicates images
that should be used on devices with low-density screens (around 120 dots-per-inch,
or “dpi”). The -mdpi
suffix indicates resources for medium-density
screens (around 160dpi), -hdpi
indicates resources for high-density
screens (around 240dpi). -xhdpi
indicates resources for
extra-high-density screens (around 320dpi), -xxhdpi
indicates
extra-extra-high-density screens (around 480dpi), -xxxhdpi
indicates
extra-extra-extra-high-density screens (around 640dpi), and so on.
In the EmPubLite
tutorial project, you will find a series of mipmap
directories with the same sorts of suffixes (e.g., res/mipmap-hdpi
).
Inside each of those directories, you will see an ic_launcher.png
file.
This is the stock icon that will be used for your application in the home
screen launcher. Each of the images is of the same icon, but the higher-density
icons have more pixels. The objective is for the image to be roughly the same
physical size on every device, using higher densities to have more detailed
images.
Our AndroidManifest.xml
file then references our ic_launcher
icon
in the <application>
element:
<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.commonsware.empublite"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<supports-screens
android:largeScreens="true"
android:normalScreens="true"
android:smallScreens="false"
android:xlargeScreens="true" />
<application
android:allowBackup="false"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme"
tools:ignore="GoogleAppIndexingWarning">
<activity android:name=".EmPubLiteActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
Note that the manifest simply refers to @mipmap/ic_launcher
, telling Android
to find a mipmap resource named ic_launcher
. The resource reference does not
indicate the file type of the resource — there is no .png
in the resource
identifier. This means you cannot have ic_launcher.png
and ic_launcher.jpg
in the same project, as they would both be identified by the same identifier.
You will need to keep the “base name” (filename sans extension) distinct for
all of your images.
Also, the @mipmap/ic_launcher
reference does not mention what screen density
to use. That is because Android will choose the right screen density to use,
based upon the device that is running your app. You do not have to worry about
it explicitly, beyond having multiple copies of your icon.
If Android detects that the device has a screen density for which you lack
an icon, Android will take the next-closest one and scale it.
Android 4.0 added partial support for Google’s WebP image format, and Android 4.3 devices support the previously-missing features (lossless compression and transparency). WebP serves as a replacement for both PNG and JPEG, and in some circumstances it can result in smaller on-disk sizes for near-equivalent image quality.
Android Studio, starting with version 2.3, has special support to help you convert drawable resources and other images from JPEG and PNG to WebP. Simply right-click over the image in the project tree and choose “Convert to WebP” from the context menu.
Initially, you are given a window for controlling the quality and output:
Figure 53: WebP Converter in Android Studio
“Lossy encoding” refers to the type performed by JPEG, taking into account that humans have limited ability to distinguish similar colors to achieve tighter compression. “Lossless encoding” refers to the type performed by PNG, where the compressed image is identical to the original, just as a ZIP file’s contents are identical to the files before they were ZIPped. For lossy encoding, you can choose a quality percentage, where higher quality images will not compress as well.
You can also:
minSdkVersion
will not support
such imagesIf you choose lossy compression and leave the “preview” checkbox checked, you are then presented with a window showing the results of the conversion at your requested quality level:
Figure 54: WebP Conversion Preview
You can adjust the quality slider below the images to see how the image changes with different quality levels and how much additional disk savings you will get from the WebP conversion.
When you are done, the WebP converter will replace your old PNG or JPEG file with the converted WebP image.
Dimensions are used in several places in Android to describe distances, such as a widget’s size. There are several different units of measurement available to you:
px
means hardware pixels, whose size will vary by device, since not all
devices have the same screen densityin
and mm
for inches and millimeters, respectively, based on the actual
size of the screenpt
for points, which in publishing terms is 1/72nd of an inch (again,
based on the actual physical size of the screen)dip
(or dp
) for density-independent pixels — one dip
equals one hardware pixel
for a ~160dpi resolution screen, but one dip
equals two hardware pixels on
a ~320dpi screensp
for scaled pixels, where one sp
equals one dip
for normal
font scale levels, increasing and decreasing as needed based upon
the user’s chosen font scale level in SettingsDimension resources, by default, are held in a dimens.xml
file in the
res/values/
directory that also holds your strings.
To encode a dimension as a resource, add a dimen
element to dimens.xml
,
with a name
attribute for your unique name for this resource, and a single child text
element representing the value:
<resources>
<dimen name="thin">10dip</dimen>
<dimen name="fat">1in</dimen>
</resources>
In a layout, you can reference dimensions as @dimen/...
, where the ellipsis
is a placeholder for your unique name for the resource (e.g., thin
and fat
from the sample above). In Java, you reference dimension resources by the
unique name prefixed with R.dimen.
(e.g., Resources.getDimension(R.dimen.thin)
).
While our stub project does not use dimension resources, we will be seeing them soon enough.
As with most types of XML resources, Android Studio just has you edit the XML directly, when you double-click on the resource in the project explorer.
Your stub project also has a res/layout/
directory, in addition to the ones
described above. That is for UI layouts, describing what your user interface
should look like. We will get into the details of that type of resource as we
start examining our user interfaces in
an upcoming chapter.