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

111

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

<Preference android:title="About" />

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

Бадр Хари
источник

Ответы:

230

Бадр,

Вам нужно установить android:keyдля элемента, тогда в вашем коде вы можете сделать ...

Предполагая, что вы используете в своем XML следующее:

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

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

Preference myPref = (Preference) findPreference("myKey");
myPref.setOnPreferenceClickListener(new OnPreferenceClickListener() {
             public boolean onPreferenceClick(Preference preference) {
                 //open browser or intent here
                 return true;
             }
         });
Уилл Тейт
источник
3
Где мне это разместить? в моем SettingsActivity не onPostCreateбудет работать, потому что это слишком рано (Nullpointer-Exception). Любая помощь приветствуется
Wandang
2
findPreferenceустарела.
zackygaurav
6
PreferenceActivity.findPreferenceустарела. Однако в документации говорится: «[T] его функциональные возможности теперь должны быть найдены в новом классе PreferenceFragment. Если вы используете PreferenceActivity в его старом режиме, там документация применяется к устаревшим API здесь.» И PreferenceFragment.findPreferenceне является устаревшим (в отношении API уровень 23, актуален на момент написания). Итак, findPreferenceсам по себе не является устаревшим; это прямое использование PreferenceActivityне рекомендуется. Google хочет, чтобы мы PreferenceFragmentвместо этого перешли, а не бросили findPreference.
Майк Плейл
В onPreferenceClick, верните, trueесли щелчок был обработан иначе false.
Thupten
У меня это работает с PreferenceFragmentCompatDividers.
Джозеф Лам
59

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

<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.

Джейсон Гилберт
источник
1
Я согласен, этот подход лучше, особенно потому, что он не использует устаревший метод findPreference (String key) и потому, что он в целом чище.
lyallcooper
Очень хорошо, но можно ли таким образом создать IntentChooser?
Peterdk
@jasongilbert Можно ли отправить широковещательную рассылку с помощью ACTION вместо того, чтобы начинать действие с щелчка предпочтения?
Sazzad Hissain Khan
1
@jasongilbert Должен ли я указать что-либо intent-filterв манифесте для SomeActivity, потому что я получаю ActivityNotFoundException:(
theapache64
11

Вам нужно использовать событие 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; 
        } 
    } 
Pentium10
источник
Ссылка 404
JZAU
6

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

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;
        }
    });
}
akelec
источник
В чем разница между onPreferenceClickи onPreferenceTreeClick ?
Юша Алеайуб
3

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

Я использую targetSdkVersion 25 , и его ответ не сработал, мне пришлось обернуть тег намерения тегом предпочтений. Пример:

<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>
Малек Хиджази
источник