Событие onclick в настройках Android

В моем файле preferences.xml у меня есть такой элемент предпочтения:

<Preference android:title="About" />

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


person Badr Hari    schedule 16.03.2011    source источник


Ответы (5)


Вам нужно установить android:key для элемента:

<Preference
    android:key="myKey"
    android:title="About" />

Затем вы можете сделать следующее в своем коде:

Preference myPref = (Preference) findPreference("myKey");
myPref.setOnPreferenceClickListener(new OnPreferenceClickListener() {
    public boolean onPreferenceClick(Preference preference) {
        // open browser or intent here
        return true;
    }
});
person Will Tate    schedule 16.03.2011
comment
Где мне разместить это? в моем SettingsActivity в onPostCreate не будет работать, потому что это слишком рано (Nullpointer-Exception). Любая помощь приветствуется - person Wandang; 21.02.2013
comment
findPreference устарел. - person zackygaurav; 09.09.2015
comment
PreferenceActivity.findPreference устарел. Однако в документации говорится, что [T] его функциональность теперь должна быть найдена в новом классе PreferenceFragment. Если вы используете PreferenceActivity в его старом режиме, документация здесь относится к устаревшим API, а PreferenceFragment.findPreference не устарела (по состоянию на уровень API 23, текущий на момент написания). Итак, сам findPreference не устарел; это прямое использование PreferenceActivity, которое устарело. Google хочет, чтобы мы вместо этого перешли на PreferenceFragment, а не отказывались от findPreference. - person Mike Playle; 19.03.2016
comment
В onPreferenceClick вернуть true, если щелчок был обработан иначе false. - person Thupten; 10.10.2017
comment
Это работает для меня с PreferenceFragmentCompatDividers. - person Joseph Lam; 23.11.2017

для запуска сайта:

<PreferenceScreen android:title="website">
    <intent
        android:action="android.intent.action.VIEW"
        android:data="http://www.example.com"
        />
</PreferenceScreen>

для запуска определенного действия:

<PreferenceScreen android:title="something">
    <intent
        android:action="android.intent.action.MAIN"
        android:targetPackage="com.example.foo"
        android:targetClass="com.example.foo.SomeActivity"
        />
</PreferenceScreen>

вы также можете использовать "android:mimetype" для установки mimetype.

person jason gilbert    schedule 31.08.2011
comment
Я согласен, этот подход лучше, особенно потому, что он не использует устаревший метод findPreference(String key) и потому что в целом он чище. - person lyallcooper; 04.06.2013
comment
Очень красиво, но можно ли таким образом создать IntentChooser? - person Peterdk; 16.05.2014
comment
@jasongilbert Можно ли отправить трансляцию с помощью ДЕЙСТВИЯ, а не начинать действие с щелчка по предпочтениям? - person Sazzad Hissain Khan; 30.11.2015
comment
@jasongilbert Должен ли я указать intent-filter в манифесте для SomeActivity, потому что я получаю ActivityNotFoundException :( - person theapache64; 08.02.2016

Вам нужно использовать событие onPreferenceTreeClick.

Например, см. http://www.javased.com/index.php?source_dir=platform_packages_apps_phone/src/com/android/phone/MobileNetworkSettings.java

 @Override 
    public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference) { 
        /** TODO: Refactor and get rid of the if's using subclasses */ 
        if (mGsmUmtsOptions != null && 
                mGsmUmtsOptions.preferenceTreeClick(preference) == true) { 
            return true; 
        } else if (mCdmaOptions != null && 
                   mCdmaOptions.preferenceTreeClick(preference) == true) { 
            if (Boolean.parseBoolean( 
                    SystemProperties.get(TelephonyProperties.PROPERTY_INECM_MODE))) { 

                mClickedPreference = preference; 

                // In ECM mode launch ECM app dialog 
                startActivityForResult( 
                    new Intent(TelephonyIntents.ACTION_SHOW_NOTICE_ECM_BLOCK_OTHERS, null), 
                    REQUEST_CODE_EXIT_ECM); 
            } 
            return true; 
        } else if (preference == mButtonPreferredNetworkMode) { 
            //displays the value taken from the Settings.System 
            int settingsNetworkMode = android.provider.Settings.Secure.getInt(mPhone.getContext(). 
                    getContentResolver(), android.provider.Settings.Secure.PREFERRED_NETWORK_MODE, 
                    preferredNetworkMode); 
            mButtonPreferredNetworkMode.setValue(Integer.toString(settingsNetworkMode)); 
            return true; 
        } else if (preference == mButtonDataRoam) { 
            if (DBG) log("onPreferenceTreeClick: preference == mButtonDataRoam."); 

            //normally called on the toggle click 
            if (mButtonDataRoam.isChecked()) { 
                // First confirm with a warning dialog about charges 
                mOkClicked = false; 
                new AlertDialog.Builder(this).setMessage( 
                        getResources().getString(R.string.roaming_warning)) 
                        .setTitle(android.R.string.dialog_alert_title) 
                        .setIconAttribute(android.R.attr.alertDialogIcon) 
                        .setPositiveButton(android.R.string.yes, this) 
                        .setNegativeButton(android.R.string.no, this) 
                        .show() 
                        .setOnDismissListener(this); 
            } else { 
                mPhone.setDataRoamingEnabled(false); 
            } 
            return true; 
        } else if (preference == mButtonDataEnabled) { 
            if (DBG) log("onPreferenceTreeClick: preference == mButtonDataEnabled."); 
            ConnectivityManager cm = 
                    (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE); 

            cm.setMobileDataEnabled(mButtonDataEnabled.isChecked()); 
            return true; 
        } else if (preference == mLteDataServicePref) { 
            String tmpl = android.provider.Settings.Secure.getString(getContentResolver(), 
                        android.provider.Settings.Secure.SETUP_PREPAID_DATA_SERVICE_URL); 
            if (!TextUtils.isEmpty(tmpl)) { 
                TelephonyManager tm = (TelephonyManager) getSystemService( 
                        Context.TELEPHONY_SERVICE); 
                String imsi = tm.getSubscriberId(); 
                if (imsi == null) { 
                    imsi = ""; 
                } 
                final String url = TextUtils.isEmpty(tmpl) ? null 
                        : TextUtils.expandTemplate(tmpl, imsi).toString(); 
                Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url)); 
                startActivity(intent); 
            } else { 
                android.util.Log.e(LOG_TAG, "Missing SETUP_PREPAID_DATA_SERVICE_URL"); 
            } 
            return true; 
        } else { 
            // if the button is anything but the simple toggle preference, 
            // we'll need to disable all preferences to reject all click 
            // events until the sub-activity's UI comes up. 
            preferenceScreen.setEnabled(false); 
            // Let the intents be launched by the Preference manager 
            return false; 
        } 
    } 
person Pentium10    schedule 16.03.2011
comment
ссылка 404 - person JZAU; 24.05.2018

ОБНОВЛЕНИЕ 2018+ Сегодня метод findPreference устарел. Итак, чтобы добиться этого, просто переопределите метод onPreferenceTreeClick в вашем фрагменте Preference. Например:

public class MySettingsFragment extends PreferenceFragment {

    @Override
    public boolean onPreferenceTreeClick (PreferenceScreen preferenceScreen,
                                          Preference preference)
    {
        String key = preference.getKey();
        if(key.equals("someKey")){
            // do your work
            return true;
        }
        return false;
    }
}

Кроме того, если вам нужно обработать клик внутри определенного элемента предпочтения (например, ListPreference), вы должны зарегистрировать setOnPreferenceChangeListener внутри метода onCreate элемента MySettingsFragment:

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    // Load the preferences from an XML resource
    addPreferencesFromResource(R.xml.preferences);

    // register listener
    final Preference prefList = findPreference("key");
    prefList.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener() {
        public boolean onPreferenceChange(Preference preference, Object value) {
            System.out.println("Selected: " + value);
            return true;
        }
    });
}
person akelec    schedule 22.10.2018
comment
В чем разница между onPreferenceClick и onPreferenceTreeClick ? - person Yousha Aleayoub; 07.01.2019

Продолжение ответа @jason gilbert

У меня targetSdkVersion 25, и его ответ не сработал, мне пришлось обернуть тег намерения тегом Preference. Пример:

<PreferenceScreen android:title="something">
    <Preference title="Title">
       <intent
       android:action="android.intent.action.MAIN"
       android:targetPackage="com.example.foo"
       android:targetClass="com.example.foo.SomeActivity"
       />
    </Preference>
</PreferenceScreen>
person Malek Hijazi    schedule 17.05.2017