アクションバーの互換性がサポートライブラリ、リビジョン18に追加されました。Androidの古いバージョンでアクションバーを使用してアクティビティを作成するための ActionBarActivity
クラスが追加されました。
サポートライブラリのアクションバーをPreferenceActivity
に追加する方法はありますか?
以前は ActionBarSherlock を使用し、SherlockPreferenceActivity
を使用していました。
編集:appcompat-v7 22.1.0では、AppCompatのサポートを任意のアクティビティに拡張するために使用できるデリゲートとして、AppCompatDelegate抽象クラスが追加されました。
次のように使用します。
...
import Android.support.v7.app.ActionBar;
import Android.support.v7.app.AppCompatDelegate;
import Android.support.v7.widget.Toolbar;
...
public class SettingsActivity extends PreferenceActivity {
private AppCompatDelegate mDelegate;
@Override
protected void onCreate(Bundle savedInstanceState) {
getDelegate().installViewFactory();
getDelegate().onCreate(savedInstanceState);
super.onCreate(savedInstanceState);
}
@Override
protected void onPostCreate(Bundle savedInstanceState) {
super.onPostCreate(savedInstanceState);
getDelegate().onPostCreate(savedInstanceState);
}
public ActionBar getSupportActionBar() {
return getDelegate().getSupportActionBar();
}
public void setSupportActionBar(@Nullable Toolbar toolbar) {
getDelegate().setSupportActionBar(toolbar);
}
@Override
public MenuInflater getMenuInflater() {
return getDelegate().getMenuInflater();
}
@Override
public void setContentView(@LayoutRes int layoutResID) {
getDelegate().setContentView(layoutResID);
}
@Override
public void setContentView(View view) {
getDelegate().setContentView(view);
}
@Override
public void setContentView(View view, ViewGroup.LayoutParams params) {
getDelegate().setContentView(view, params);
}
@Override
public void addContentView(View view, ViewGroup.LayoutParams params) {
getDelegate().addContentView(view, params);
}
@Override
protected void onPostResume() {
super.onPostResume();
getDelegate().onPostResume();
}
@Override
protected void onTitleChanged(CharSequence title, int color) {
super.onTitleChanged(title, color);
getDelegate().setTitle(title);
}
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
getDelegate().onConfigurationChanged(newConfig);
}
@Override
protected void onStop() {
super.onStop();
getDelegate().onStop();
}
@Override
protected void onDestroy() {
super.onDestroy();
getDelegate().onDestroy();
}
public void invalidateOptionsMenu() {
getDelegate().invalidateOptionsMenu();
}
private AppCompatDelegate getDelegate() {
if (mDelegate == null) {
mDelegate = AppCompatDelegate.create(this, null);
}
return mDelegate;
}
}
ハッキングはもうありません。 AppCompatPreferenceActivity.Java から取得したコード。
現在、AppCompatで達成する方法はありません。内部的にバグをオープンしました。
Google Playストアで使用しているものと同様の回避策を作成できました。 元の回答へのリンク
独自のコードに非常に似ていますが、設定されたタイトルを許可するxmlを追加しました。
PreferenceActivity
の使用を継続:
settings_toolbar.xml :
<?xml version="1.0" encoding="utf-8"?>
<Android.support.v7.widget.Toolbar
xmlns:Android="http://schemas.Android.com/apk/res/Android"
xmlns:app="http://schemas.Android.com/apk/res-auto"
Android:id="@+id/toolbar"
app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
Android:minHeight="?attr/actionBarSize"
app:navigationContentDescription="@string/abc_action_bar_up_description"
Android:background="?attr/colorPrimary"
app:navigationIcon="?attr/homeAsUpIndicator"
app:title="@string/action_settings"
/>
SettingsActivity.Java :
public class SettingsActivity extends PreferenceActivity {
@Override
protected void onPostCreate(Bundle savedInstanceState) {
super.onPostCreate(savedInstanceState);
LinearLayout root = (LinearLayout)findViewById(Android.R.id.list).getParent().getParent().getParent();
Toolbar bar = (Toolbar) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false);
root.addView(bar, 0); // insert at top
bar.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
}
}
Result :
here で指摘したように、Gingerbread Devicesはこの行でNullPointerExceptionを返しています。
LinearLayout root = (LinearLayout)findViewById(Android.R.id.list).getParent().getParent().getParent();
SettingsActivity.Java :
public class SettingsActivity extends PreferenceActivity {
@Override
protected void onPostCreate(Bundle savedInstanceState) {
super.onPostCreate(savedInstanceState);
Toolbar bar;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
LinearLayout root = (LinearLayout) findViewById(Android.R.id.list).getParent().getParent().getParent();
bar = (Toolbar) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false);
root.addView(bar, 0); // insert at top
} else {
ViewGroup root = (ViewGroup) findViewById(Android.R.id.content);
ListView content = (ListView) root.getChildAt(0);
root.removeAllViews();
bar = (Toolbar) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false);
int height;
TypedValue tv = new TypedValue();
if (getTheme().resolveAttribute(R.attr.actionBarSize, tv, true)) {
height = TypedValue.complexToDimensionPixelSize(tv.data, getResources().getDisplayMetrics());
}else{
height = bar.getHeight();
}
content.setPadding(0, height, 0, 0);
root.addView(content);
root.addView(bar);
}
bar.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
}
}
上記の問題があればお知らせください!
更新2:回避策の適用
多くの開発ノートで指摘されているように、PreferenceActivity
は要素の色付けをサポートしていませんが、いくつかの内部クラスを利用することでこれを実現できます。これらのクラスが削除されるまでです。 (appCompat support-v7 v21.0.3を使用して動作します)。
次のインポートを追加します。
import Android.support.v7.internal.widget.TintCheckBox;
import Android.support.v7.internal.widget.TintCheckedTextView;
import Android.support.v7.internal.widget.TintEditText;
import Android.support.v7.internal.widget.TintRadioButton;
import Android.support.v7.internal.widget.TintSpinner;
次に、onCreateView
メソッドをオーバーライドします。
@Override
public View onCreateView(String name, Context context, AttributeSet attrs) {
// Allow super to try and create a view first
final View result = super.onCreateView(name, context, attrs);
if (result != null) {
return result;
}
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Lollipop) {
// If we're running pre-L, we need to 'inject' our tint aware Views in place of the
// standard framework versions
switch (name) {
case "EditText":
return new TintEditText(this, attrs);
case "Spinner":
return new TintSpinner(this, attrs);
case "CheckBox":
return new TintCheckBox(this, attrs);
case "RadioButton":
return new TintRadioButton(this, attrs);
case "CheckedTextView":
return new TintCheckedTextView(this, attrs);
}
}
return null;
}
Result:
AppCompat 22.1では、新しい色付きの要素が導入されました。つまり、内部クラスを使用して前回の更新と同じ効果を達成する必要がなくなりました。代わりに、これに従います(onCreateView
を引き続きオーバーライドします):
@Override
public View onCreateView(String name, Context context, AttributeSet attrs) {
// Allow super to try and create a view first
final View result = super.onCreateView(name, context, attrs);
if (result != null) {
return result;
}
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Lollipop) {
// If we're running pre-L, we need to 'inject' our tint aware Views in place of the
// standard framework versions
switch (name) {
case "EditText":
return new AppCompatEditText(this, attrs);
case "Spinner":
return new AppCompatSpinner(this, attrs);
case "CheckBox":
return new AppCompatCheckBox(this, attrs);
case "RadioButton":
return new AppCompatRadioButton(this, attrs);
case "CheckedTextView":
return new AppCompatCheckedTextView(this, attrs);
}
}
return null;
}
ネストされた優先画面
ネストされた<PreferenceScreen />
sにツールバーを含めることで多くの人が問題を経験していますが、解決策を見つけました!! -多くの試行錯誤の後!
以下をSettingsActivity
に追加します。
@SuppressWarnings("deprecation")
@Override
public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference) {
super.onPreferenceTreeClick(preferenceScreen, preference);
// If the user has clicked on a preference screen, set up the screen
if (preference instanceof PreferenceScreen) {
setUpNestedScreen((PreferenceScreen) preference);
}
return false;
}
public void setUpNestedScreen(PreferenceScreen preferenceScreen) {
final Dialog dialog = preferenceScreen.getDialog();
Toolbar bar;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
LinearLayout root = (LinearLayout) dialog.findViewById(Android.R.id.list).getParent();
bar = (Toolbar) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false);
root.addView(bar, 0); // insert at top
} else {
ViewGroup root = (ViewGroup) dialog.findViewById(Android.R.id.content);
ListView content = (ListView) root.getChildAt(0);
root.removeAllViews();
bar = (Toolbar) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false);
int height;
TypedValue tv = new TypedValue();
if (getTheme().resolveAttribute(R.attr.actionBarSize, tv, true)) {
height = TypedValue.complexToDimensionPixelSize(tv.data, getResources().getDisplayMetrics());
}else{
height = bar.getHeight();
}
content.setPadding(0, height, 0, 0);
root.addView(content);
root.addView(bar);
}
bar.setTitle(preferenceScreen.getTitle());
bar.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
dialog.dismiss();
}
});
}
PreferenceScreen
がそんなに苦痛なのは、それらがラッパーダイアログに基づいているためです。そのため、ダイアログレイアウトをキャプチャしてツールバーを追加する必要があります。
Toolbar
を設計的にインポートすると、v21より前のデバイスではエレベーションとシャドウイングが許可されないため、Toolbar
でエレベーションを使用する場合は、AppBarLayout
でラップする必要があります。
`settings_toolbar.xml:
<Android.support.design.widget.AppBarLayout
Android:layout_width="match_parent"
Android:layout_height="wrap_content">
<Android.support.v7.widget.Toolbar
.../>
</Android.support.design.widget.AppBarLayout>
build.gradle
ファイルの依存関係としてデザインサポートライブラリを追加することを忘れないでください:
compile 'com.Android.support:support-v4:22.2.0'
compile 'com.Android.support:appcompat-v7:22.2.0'
compile 'com.Android.support:design:22.2.0'
報告された重複する問題を調査しましたが、問題を再現できません。
上記のように使用中の完全なコードは以下を生成します。
何か不足している場合は、 this repo でお知らせください。調査します。
Support-v4 Fragmentに基づいたPreferenceFragment実装が見つかりました:
https://github.com/kolavar/Android-support-v4-preferencefragment
編集:私はちょうどそれをテストし、その動作は素晴らしいです!
PreferenceActivity
をABCと統合することは、少なくとも私にとっては不可能です。私は見つけることができる2つの可能性を試しましたが、どれもうまくいきませんでした:
ActionBarPreferenceActivity
はPreferenceActivity
を拡張します。これを行うと、ActionBarActivityDelegate.createDelegate(ActionBarActivity activity)
によって制限されます。また、アクセスできないActionBar.Callbacks
を実装する必要があります
ActionBarPreferenceActivity
はActionBarActivity
を拡張します。このアプローチでは、まったく新しいPreferenceActivity
、PreferenceManager
を書き換える必要があり、PreferenceFragment
になる場合があります。つまり、com.Android.internal.util.XmlUtils
のような隠しクラスにアクセスする必要があります。任意のアクティビティに追加できるActionBarWrapper
を実装します。
好みのアクティビティが本当に必要な場合、今のところ私のアドバイスはActionBarSherlock
です。
しかし、私はそれをどうにか実装しました here 。
Androidのサポートライブラリにはバグがあり、これが発生しないため、OPはMenuItem
をActionBar
of PreferenceActivity
に入れる方法を知りたいと考えています。
目標を達成するために、すでに提案されているよりもずっときれいな方法を見つけました(そして、それを Android Docs で見つけました):
Android:parentActivityName
アクティビティの論理的な親のクラス名。ここでの名前は、対応する要素のAndroid:name属性に指定されたクラス名と一致する必要があります。
システムはこの属性を読み取り、ユーザーがアクションバーのUpボタンを押したときに開始するアクティビティを決定します。システムはこの情報を使用して、TaskStackBuilderでアクティビティのバックスタックを合成することもできます。
APIレベル4〜16をサポートするために、「Android.support.PARENT_ACTIVITY」の値を指定する要素で親アクティビティを宣言することもできます。例えば:
<activity Android:name="com.example.app.ChildActivity" Android:label="@string/title_child_activity" Android:parentActivityName="com.example.myfirstapp.MainActivity" > <!-- Parent activity meta-data to support API level 4+ --> <meta-data Android:name="Android.support.PARENT_ACTIVITY" Android:value="com.example.app.MainActivity" /> </activity>
onOptionsItemSelected()
で通常行うことを行います。これはAndroid Docsの一部であるため、副作用はありません。
ハッピーコーディング。 :)
Lollipopをターゲットにしている場合、このソリューションは機能しなくなります。 AppCompatを使用している場合、 this answerが探しているはずです。
getActionBar()
を使用してAndroid.app.Actionbar
を取得できました。最初はヌル値を返しました...その後、マニフェストに行き、テーマを次のように変更しました:
Android:theme="@style/Theme.AppCompat"
それから私は再びアクションバーを持つことができました。これは特定のビルドレベルでのみ機能すると想定しています。そのため、ビルド番号を確認するか、返される値がnullかどうかを確認することができます。
私が取り組んでいるアプリはICS/4.0
+用であるため、私にとっては問題ありません。
現在、この問題に対する公式の回答が公開されています。 v7/v14 Preference Support ライブラリです。
v7/v14プリファレンスサポートライブラリの使用方法 を参照して、使用方法の説明を確認してください。