name="AppTheme"
parent="
Theme.AppCompat.Light.DarkActionBar
">
name="colorPrimary">@color/colorPrimary
name="colorPrimaryDark">@color/colorPrimaryDark
name="colorAccent">@color/colorAccent
name="AppTheme"
parent="Theme.AppCompat.Light.DarkActionBar">
name="StrongBeatBoxButton"
parent="@style/BeatBoxButton">
- bold
name="AppTheme"
parent="Theme.AppCompat.Light.DarkActionBar">
...
name="AppTheme"
parent="Theme.AppCompat
.Light.DarkActionBar
">
...
name="AppTheme"
parent="Theme.AppCompat">
name="colorPrimary">@color/
colorPrimaryred<
/item>
name="colorPrimaryDark">@color/
colorPrimary
Darkdark_red
name="colorAccent">@color/
colorAccentgray
item>
name="Theme.AppCompat"
parent="Base.Theme.AppCompat" />
Тема
Theme.AppCompat
наследует
атрибуты
от
Base.Theme.AppCompat
. Интересно, что
Theme.AppCompat
не
переопределяет никакие атрибуты, а только содержит ссылку
на своего родителя.
Щелкните мышью по
Base.Theme.AppCompat
с нажатой
клавишей (
Ctrl
). Android Studio сообщит, что тема уточняется
по ресурсам. Существует несколько разных версий этой темы в
зависимости от используемой версии Android.
name="Base.Theme.AppCompat"
parent="Base.V7.Theme.AppCompat">
name="Base.V7.Theme.AppCompat"
parent="Platform.AppCompat">
-
androidx.appcompat.app.AppCompatVie
wInflater
- false
- true
name="windowActionBarOverlay">false
name="Platform.AppCompat"
parent="android:Theme.Holo">
name="android:windowNoTitle">true
name="android:windowActionBar">false
- ?
attr/buttonBarStyle
- ?
attr/buttonBarButtonStyle
name="android:borderlessButtonStyle">?
attr/borderlessButtonStyle
...
name="Platform.AppCompat"
parent="android:Theme.Holo">
...
...
name="android:windowBackground">@color/backgrou
nd_material_dark
name="AppTheme"
parent="Theme.AppCompat">
- @color/red
name="colorPrimaryDark">@color/dark_red
- @color/gray
-
name="android:windowBackground">@color/soothing
_blue
name="Base.V7.Theme.AppCompat"
parent="Platform.AppCompat">
...
name="buttonStyle">@style/Widget.AppCompat.Butt
on
name="buttonStyleSmall">@style/Widget.AppCompat
.Button.Small
...
name="Widget.AppCompat.Button"
parent="Base.Widget.AppCompat.Button"/>
Стиль
Widget.AppCompat.Button
самостоятельно никакие
атрибуты не определяет. Чтобы просмотреть его содержимое,
перейдите к его родителю. Вы увидите, что базовый стиль
существует
в
двух
версиях.
Выберите
версию
values/values.xml
.
name="Base.Widget.AppCompat.Button"
parent="android:Widget">
name="android:background">@drawable/abc_btn_def
ault_mtrl_shape
- ?
android:attr/textAppearanceButton
- 48dip
- 88dip
- true
- true
name="android:gravity">center_vertical|center_h
orizontal
name="AppTheme"
parent="Theme.AppCompat">
name="colorPrimary">@color/red
name="AppTheme"
parent="Theme.AppCompat">
name="colorPrimary">@color/red
name="colorPrimaryDark">@color/dark_red
name="colorAccent">@color/gray
name="android:windowBackground">@color/soothing
_blue
name="BeatBoxButton"
parent="Widget.AppCompat.Button">
name="android:background">@color/dark_bluem>
name="Platform.AppCompat"
parent="android:Theme.Holo">
...
name="AppTheme"
parent="Theme.AppCompat">
...
name="BeatBoxButton"
parent="Widget.AppCompat.Button">
name="AppTheme"
parent="Theme.AppCompat">
...
name="BeatBoxButton"
parent="Widget.AppCompat.Button">
name="android:background">@drawable/button_beat
_box_normal
-
name="android:background">@drawable/button_beat
_box
Листинг 26.11. Добавление класса для работы с хранимым
запросом (QueryPreferences.kt)
private
const
val
PREF_SEARCH_QUERY
=
"searchQuery"
object QueryPreferences {
fun getStoredQuery(context: Context):
String {
val
prefs
=
PreferenceManager.getDefaultSharedPreferences(c
ontext)
return
prefs.getString(PREF_SEARCH_QUERY, "")!!
}
fun setStoredQuery(context: Context, query:
String) {
PreferenceManager.getDefaultSharedPrefe
rences(context)
.edit()
.putString(PREF_SEARCH_QUERY,
query)
.apply()
}
}
Вашему приложению нужен только один экземпляр
QueryPreferences
, который может использоваться всеми
другими компонентами. Из-за этого мы используем ключевое
слово
object
(вместо
class
), чтобы указать, что
QueryPreferences
— это синглтон. Так тем более в
приложении будет создан всего один экземпляр. Это также
позволяет вам получить доступ к функциям в объекте,
используя синтаксис
ClassName.functionName(...)
, как вы
увидите вскоре.
Значение
PREF_SEARCH_QUERY
используется в качестве
ключа для хранения запроса. Этот ключ применяется во всех
операциях чтения или записи запроса.
Функция
PreferencesManager.getDefaultSharedPreferences(Con
text)
возвращает экземпляр с именем по умолчанию и
приватными разрешениями (так что предпочтения доступны
только изнутри вашего приложения). Чтобы получить
конкретный экземпляр
SharedPreferences
, вы можете
использовать
функцию
Context.getSharedPreferences(String,Int)
. Однако на
практике нас в большей степени волнует, чтобы экземпляр был
общим для всего приложения.
Функция
getStoredQuery(Context)
возвращает значение
запроса, хранящееся в общих настройках. Для этого функция
сначала получает объект
SharedPreferences
по умолчанию
для заданного контекста. (Так как
QueryPreferences
не имеет
собственного контекста, вызывающий компонент должен
передать свой контекст как входной параметр.)
Получение ранее сохраненного значения сводится к
простому вызову
Shared
Preferences.getString(...)
,
SharedPreferences.getInt(...)
или другой функции,
соответствующей
типу
данных.
Второй
параметр
SharedPreferences.getString
(String,String)
определяет возвращаемое значение по умолчанию, которое
должно возвращаться при отсутствии записи с ключом
PREF_SEARCH_QUERY
.
Возвращаемый тип
SharedPreferences.getString(...)
определен как тип
String
, допускающий значение null, так как
компилятор не может гарантировать, что значение, связанное с
PREF_SEARCH_QUERY
, существует и что оно не является null. Но
вы знаете, что такой ситуации не бывает, поэтому
предоставляете пустую строку в качестве значения по
умолчанию
в
тех
случаях,
когда
функция
setStoredQuery(context:Context,query:String)
еще не
вызывалась. Поэтому здесь безопасно использовать оператор
ненулевого утверждения (
!!
) без блока
try
/
catch
.
Функция
setStoredQuery(Context)
записывает запрос в
хранилище общих настроек для заданного контекста. В
приведенном
выше
коде
QueryPreferences
вызов
SharedPreferences.edit()
используется для получения
экземпляра
Shared
Preferences.Editor
.
Этот
класс
используется для сохранения значений в
Shared
Preferences
.
Он позволяет объединять изменения в транзакции по аналогии
с тем, как это делается во
FragmentTransaction
.
Множественные изменения могут быть сгруппированы в одну
операцию записи в хранилище.
После того как все изменения будут внесены, вызовите
apply()
для объекта
Editor
, чтобы эти изменения стали
видимыми для всех пользователей файла
SharedPreferences
.
Функция
apply()
вносит изменения в память немедленно, а
непосредственная запись в файл осуществляется в фоновом
потоке.
QueryPreferences
предоставляет всю функциональность
долгосрочного хранения данных для PhotoGallery.
Теперь, когда у вас есть механизм простого сохранения и
выборки последнего запроса пользователя, можно переходить к
обновлению
PhotoGalleryFragment
для чтения и записи
запроса из общих настроек. Мы считываем запрос при первом
создании
ViewModel
и
используем
значение
для
инициализации
mutableSearchTerm
. Запишем запрос при
каждом изменении
mutableSearchTerm
.
Do'stlaringiz bilan baham: |