Как добавить кнопку на PreferenceScreen?

135

Я довольно новичок в разработке Android и только что наткнулся на Настройки. Я нашел PreferenceScreenи хотел создать функциональность входа с ним. Единственная проблема, с которой я столкнулся - это то, что я не знаю, как добавить кнопку «Войти» в PreferenceScreen.

Вот как PreferenceScreenвыглядит моя :

<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
...
    <PreferenceScreen android:title="@string/login" android:key="Login">
        <EditTextPreference android:persistent="true" android:title="@string/username" android:key="Username"></EditTextPreference>
        <EditTextPreference android:title="@string/password" android:persistent="true" android:password="true" android:key="Password"></EditTextPreference>
    </PreferenceScreen>
...
</PreferenceScreen>

Кнопка должна быть прямо под двумя EditTextPreferenceс.

Есть ли простое решение этой проблемы? Единственное решение, которое я нашел, не работало, потому что я использую sub PreferenceScreens.

Обновить:

Я понял, что могу добавить кнопки таким образом:

<PreferenceScreen android:title="@string/login" android:key="Login">
        <EditTextPreference android:persistent="true" android:title="@string/username" android:key="Username"></EditTextPreference>
        <EditTextPreference android:title="@string/password" android:persistent="true" android:password="true" android:key="Password"></EditTextPreference>
        <Preference android:layout="@layout/loginButtons" android:key="loginButtons"></Preference>
</PreferenceScreen>

и файл макета ( loginButtons.xml) выглядит так:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent"
    android:weightSum="10" 
    android:baselineAligned="false" android:orientation="horizontal">
    <Button android:text="Login" android:layout_width="fill_parent"
        android:layout_weight="5" android:layout_height="wrap_content"
        android:id="@+id/loginButton" android:layout_gravity="left"></Button>
    <Button android:text="Password?" android:layout_width="fill_parent"
        android:layout_weight="5" android:layout_height="wrap_content"
        android:id="@+id/forgottenPasswordButton"></Button>
</LinearLayout>

Так что теперь кнопки появляются, но я не могу получить к ним доступ в коде. Я попробовал это с, findViewById()но это возвращает ноль. Любые идеи, как я мог получить доступ к этим кнопкам?

Ван Наактгеборен
источник
Посмотрите на этот вопрос stackoverflow.com/questions/2697233/…
Юозас Контвайнис
2
Кстати, ответ @neelabh наиболее прост - вы можете достичь требуемого поведения, указав обработчик событий в xml-макете: просто добавьте android:onClick="method"к каждой кнопке, где метод определен в действии как public void method(View v).
Стэн

Ответы:

304

Для XML:

<Preference android:title="Acts like a button"
                android:key="@string/myCoolButton"
                android:summary="This is a cool button"/>

Тогда для Java в вашем onCreate()

Preference button = findPreference(getString(R.string.myCoolButton));
button.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
                @Override
                public boolean onPreferenceClick(Preference preference) {   
                    //code for what you want it to do   
                    return true;
                }
            });

Это будет выглядеть как обычное предпочтение, только с заголовком и сводкой, поэтому оно будет выглядеть так, как будто оно принадлежит.

Изменить: я перешел на использование @string/myCoolButtonи getString(R.string.myCoolButton)потому что лучше использовать ресурсы для помощи компилятора, языкового перевода и простоты изменений String.

камыш
источник
7
Многие из предварительно загруженных приложений Google используют этот подход, и поэтому стоит следовать ему! Я обнаружил, что стандартная серая кнопка выглядит неуместно на экране настроек, и благодаря вам я нашел хороший способ сделать ее стандартизированной :)
Tom
5
Убедитесь, что android:key="key"и findPreference("key");используете тот же ключ.
Рид
4
на самом деле это вообще не отображает кнопку, которая сбивает с толку пользователя, а не то, о чем спрашивал ОП. Правильный ответ, чтобы на самом деле показать кнопку Николаса.
mutable2112
7
Я только что заметил, что опубликовал это в 2011 году в 1:11. Ха - ха.
Рид
3
Должен был сделать это через 1 месяц и 11 дней: 11/1 / '11 - 1:11 :)
P Kuijpers
43

Я полагаю, это слишком поздно. Это то, что я сделал для предпочтения кнопки.

Предпочтение в файле preference.xml в папке xml

....
    <Preference
        android:key="resetBD"
        android:title="@string/ajustes_almacenamiento"
        android:summary="@string/ajustes_almacenamiento_desc"
        android:widgetLayout="@layout/pref_reset_bd_button" 
    ></Preference>
  ...

и pref_reset_bd_button.xml в папке макета

 <?xml version="1.0" encoding="utf-8"?>
   <Button
       xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/resetButton" 
        android:text="@string/ajustes_almacenamiento_bt" 
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="resetearBD">

   </Button>
никола
источник
android:widgetLayoutявляется идеальной связью между фрагментами и предпочтениями (один или оба определены в XML). Большое спасибо. Здесь
peter_the_oak
Привет, Николас. Ваш код работает, но один вопрос ... Как вы можете сделать так, чтобы кнопка работала в классе, который расширяет PreferenceActivity
SP
4
Подскажите пожалуйста, как найти эту кнопку с идентификатором "resetButton" спасибо!
Бен Джима
Я нашел с помощью android: макет превосходит Android: widgetLayout. Используя widgetLayout, моя кнопка стала тем, что я могу описать только как layout_alignParentRight = "true", но это поведение
исчезло
1
И я не могу заставить это запустить обработчик события onPreferenceChange или Click
JoeHz
38

Я хотел добавить ссылку Exit в настройки и смог изменить код Jakar, чтобы он работал следующим образом:

<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" >   
   <PreferenceCategory android:title="Settings">
       <Preference android:title="Click to exit" android:key="exitlink"/>       
   </PreferenceCategory>    
</PreferenceScreen>

Первоначально «Preference» был «EditTextPreference», который я отредактировал вручную.

Тогда в классе:

public class MyPreferences extends PreferenceActivity {
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    addPreferencesFromResource(R.xml.mypreferences);

    Preference button = (Preference)getPreferenceManager().findPreference("exitlink");      
    if (button != null) {
        button.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
            @Override
            public boolean onPreferenceClick(Preference arg0) {
                finish();   
                return true;
            }
        });     
    }
}
}
Dunfield
источник
3
Прекрасно работает. Это блестяще !
RRTW
8

Добавить

setContentView(R.layout.buttonLayout);

Ниже

addPreferencesFromResource(R.xml.yourPreference);

buttonLayout:

<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

<RelativeLayout
        android:id="@+id/top_control_bar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
</RelativeLayout>

<LinearLayout
        android:id="@+id/bottom_control_bar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true">

    <Button
            android:id="@+id/button"
            android:layout_width="match_parent"
            android:layout_height="70dp"
            android:text="@string/saveAlarm"/>
</LinearLayout>

<ListView
        android:id="@android:id/list"
        android:layout_width="fill_parent"
        android:layout_height="0dip"
        android:layout_above="@id/bottom_control_bar"
        android:layout_below="@id/top_control_bar"
        android:choiceMode="multipleChoice">
</ListView>

Кнопка доступа по:

Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View v) {
        //Your event
    }
});

Вы можете получить кнопку сверху или снизу экрана, поместив кнопку в RelativeLayout:

  • top_control_bar
  • bottom_control_bar

bottom_control_bar

Это сработало для меня. Я надеюсь, что смогу помочь кому-нибудь с этим фрагментом кода.

Тийс Лиммен
источник
Я использовал это в своем приложении, оно удерживает кнопку в верхней части списка, которая выглядит лучше и удобнее для кнопок OK и Отмена. Спасибо.
Grux
1

Я не думаю, что есть простой способ просто добавить кнопку на экран настроек. Предпочтения ограничены:

EditTextPreference
ListPreference
RingtonePreference
CheckboxPreference

При использовании экрана настроек вы ограничены использованием этих опций, и нет единой «кнопки-предпочтения», которую можно было бы легко использовать для ваших нужд.

Я искал похожую функциональность с добавлением кнопок на экран настроек, но, похоже, вам нужно создать собственный экран для этого, управляя изменением настроек в вашей собственной реализации.

Эрик Нордвик
источник
Возникла такая же проблема: если вам нужна кнопка в ваших предпочтениях, вы должны использовать самодельный преф экран :(
Janusz
1

Вы можете сделать это так, чтобы получить доступ к кнопке!

 View footerView = ((LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE)).inflate(R.layout.layoutButtons, null, false);

Не забудьте добавить android:idк LinearLayout, который содержит кнопку в layoutButtons.xml , т.е. android:id="@+id/mylayout"

 LinearLayout mLayout = (LinearLayout) footerView.findViewById(R.id.mylayout);
 Button login = (Button) footerView.findViewById(R.id.loginButton);
 login.setOnClickListener(this);
 ListView lv = (ListView) findViewById(android.R.id.list);
 lv.addFooterView(footerView);

 // Lines 2 and 3 can be used in the same way for a second Button!
шах
источник
не забудьте пометить этот вопрос как истинный, если вы чувствуете, что я помог вам в этом?
Шах
0

Вы также можете настроить макет предпочтения путем переопределения Preference.onCreateView(parent). В приведенном ниже примере используется анонимный внутренний класс для настройки красного цвета.

    screen.addPreference(
        new Preference(context) {
            @Override
            protected View onCreateView(ViewGroup parent) {
                View view = super.onCreateView(parent);
                view.setBackgroundColor(Color.RED);
                return view;
            }
        });

Вы можете использовать эту технику, чтобы добавить кнопку в представление по умолчанию.

Роджер Кис
источник
0

Если кто-то попросит пользователя щелкнуть один из пунктов «Предпочтения» и обработает этот щелчок, и ваше решение будет основано PreferenceFragmentCompat, то вы можете сделать это:

<PreferenceScreen
...
>
...
<Preference
    android:key="@string/pref_key_clickable_item"
    android:persistent="false"
    android:title="Can be clicked"
    android:summary="Click this item so stuff will happen"/>
...
</PreferenceScreen>

Ява:

@Override
onPreferenceTreeClick(Preference preference) {
    if (preference.getKey().equals(getContext().getString(R.string.pref_key_clickable_item))) {
       // user clicked the item
       // return "true" to indicate you handled the click
       return true;
    }
    return false;
}

Котлин:

override fun onPreferenceTreeClick(preference: Preference): Boolean {
    if (preference.key == context?.getString(R.string.pref_key_clickable_ite)) {
       // user clicked the item
       // return "true" to indicate you handled the click
       return true
    }
    return false
}
Чаба Тот
источник
0

Создайте собственный макет для SettingsFragment.javaielayout_settings.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".ui.settings.SettingsFragment"
    tools:ignore="NewApi">

    <com.google.android.material.appbar.AppBarLayout
        android:id="@+id/appBarSettings"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:theme="@style/AppBarOverlay">

        <com.google.android.material.appbar.MaterialToolbar
            android:id="@+id/toolbarSettings"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:minHeight="?attr/actionBarSize"
            app:title="@string/fragment_title_settings" />
    </com.google.android.material.appbar.AppBarLayout>

    <!--this works as the container for the SettingsFragment.java
    put the code for the Button above or below this FrameLayout
    according to your requirement-->

    <FrameLayout
        android:id="@android:id/list_container"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
      app:layout_behavior="com.google.android.material.appbar.AppBarLayout$ScrollingViewBehavior" />


    <com.google.android.material.button.MaterialButton
        android:id="@+id/btnSample"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom|center"
        android:text="@string/button_sample_text" />

</androidx.coordinatorlayout.widget.CoordinatorLayout>

Затем обратитесь к файлу макета в вашем styles.xml:

<style name="AppTheme" parent="....">
      ...............
      ...............
      <item name="preferenceTheme">@style/MyPreferenceThemeOverlay</item>
</style>

<style name="MyPreferenceThemeOverlay" parent="PreferenceThemeOverlay">
      <item name="android:layout>@layout/layout_settings</item>
</style>

И затем, в onViewCreated()методе вашего SettingsFragment.java, вы можете использовать кнопку, как это:

    @Override
    public void onViewCreated(@NonNull View view, @Nullable Bundle 
                             savedInstanceState) {

        MaterialButton btnSample = view.findViewById(R.id.btnSample);
        btnSample.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(requireContext(), "Sample Button 
                   clicked!!", Toast.LENGTH_LONG).show();
            }
        });
    }

Пример экрана настроек с кнопкой

cgb_pandey
источник
-1

попробуйте android: onClick = "myMethod" работает как брелок для простых событий onclick

neelabh
источник
4
Preferenceне имеет onClickсобственности
ercan
Ответ подразумевается android:onClickв прикрепленном виде кнопки, а не в самом предпочтении.
Стэн