Zdarzenie onclick preferencji Androida

111

W moich preferencjach.xml mam taki element preferencji:

<Preference android:title="About" />

Chcę przypisać onClickzdarzenie, więc gdyby użytkownik je kliknął, mógłbym otworzyć nową Intentlub przeglądarkę. Próbowałem to zrobić tak, jak robię to z przyciskami, ale to nie działa.

Badr Hari
źródło

Odpowiedzi:

230

Badr,

Musisz ustawić android:keydla przedmiotu, a następnie w swoim kodzie możesz zrobić ...

Zakładając, że używasz w swoim XML:

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

Następnie możesz wykonać następujące czynności w swoim kodzie:

Preference myPref = (Preference) findPreference("myKey");
myPref.setOnPreferenceClickListener(new OnPreferenceClickListener() {
             public boolean onPreferenceClick(Preference preference) {
                 //open browser or intent here
                 return true;
             }
         });
Will Tate
źródło
3
Gdzie mam to umieścić? w moich ustawieniach onPostCreateAktywność nie zadziała, ponieważ jest za wcześnie (wyjątek zerowy). Każda pomoc jest mile
widziana
2
findPreferencejest przestarzałe.
zackygaurav
6
PreferenceActivity.findPreferencejest przestarzałe. Jednak dokumentacja mówi, że „[T] jego funkcjonalność powinna być teraz znaleziona w nowej klasie PreferenceFragment. Jeśli używasz PreferenceActivity w starym trybie, dokumentacja tam odnosi się do przestarzałych interfejsów API tutaj” i PreferenceFragment.findPreferencenie jest przestarzała (od API poziom 23, aktualny w momencie pisania). Więc findPreferencesamo w sobie nie jest przestarzałe; jest to bezpośrednie użycie, PreferenceActivityktóre jest przestarzałe. Google chce, żebyśmy PreferenceFragmentzamiast tego przenieśli się, a nie porzucili findPreference.
Mike Playle
W onPreferenceClick, zwróć, truejeśli kliknięcie zostało obsłużone w inny sposób false.
Thupten
To działa dla mnie z PreferenceFragmentCompatDividers.
Joseph Lam
59

aby uruchomić stronę internetową:

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

aby rozpocząć określone działanie:

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

możesz także użyć „android: mimetype”, aby ustawić typ MIME.

jason gilbert
źródło
1
Zgadzam się, to podejście jest lepsze, zwłaszcza, że ​​nie używa obecnie przestarzałej metody findPreference (klucz String) i ponieważ jest ogólnie czystsza.
lyallcooper
Bardzo fajnie, ale czy w ten sposób można też stworzyć IntentChooser?
Peterdk
@jasongilbert Czy można wysłać transmisję za pomocą AKCJI zamiast rozpoczynać aktywność od kliknięcia preferencji?
Sazzad Hissain Khan
1
@jasongilbert Czy powinienem określić jakiekolwiek intent-filterw manifeście dla SomeActivity, bo otrzymuję ActivityNotFoundException:(
theapache64
11

Musisz użyć zdarzenia onPreferenceTreeClick.

Na przykład zobacz 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
źródło
Link to 404
JZAU
6

AKTUALIZACJA 2018+ Dziś findPreferencemetoda jest pozbawiona funkcjonalności. Aby to osiągnąć, po prostu zastąp onPreferenceTreeClickmetodę w swoim fragmencie Preference. Na przykład:

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;
    }
}

Ponadto, jeśli chcesz obsłużyć kliknięcie wewnątrz określonego elementu preferencji (np. ListPreference), Powinieneś zarejestrować setOnPreferenceChangeListenerwewnątrz onCreatemetodę 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
źródło
Jaka jest różnica między onPreferenceClicki onPreferenceTreeClick ?
Yousha Aleayoub
3

Kontynuacja odpowiedzi @jason gilbert

Jestem na targetSdkVersion 25 , a jego odpowiedź nie zadziałała, musiałem owinąć tag intencji tagiem Preference. Przykład:

<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>
Malek Hijazi
źródło