Ответ 1
Я придумал этот подход (Примечание: это моделируется после приложение Google I/O Session 2012 UIUtilis.java):
В AndroidManifest.xml
define Activity
включить <meta-data>
:
<!-- Note: specify the target device for Activities with target_device meta-data of "universal|phone|tablet"
see UIUtils.java (configureDeviceSpecificActivities) for more details. -->
<!-- Activities for both phones and tablets -->
<activity android:name=".ui.AccountActivity"
android:configChanges="orientation|keyboardHidden"
android:label="@string/app_name"
android:theme="@style/Theme.Accounts">
<meta-data android:name="target_device" android:value="universal"/>
</activity>
<!-- Activities for tablets -->
<activity android:name=".ui.CoolMultipaneActivity"
android:label="@string/app_name">
<meta-data android:name="target_device" android:value="tablet"/>
Тяжелая работа помещается в метод configureDeviceSpecificActivities(Context context)
/**
* Enables and disables {@linkplain android.app.Activity activities} based on their "target_device" meta-data and
* the current device. Add <meta-data name="target_device" value="tablet|phone|universal" /> to an activity to
* specify its target device.
* @param context the current context of the device
* @see #isHoneycombTablet(android.content.Context)
*/
public static void configureDeviceSpecificActivities(Context context) {
final PackageManager package_manager = context.getPackageManager();
final boolean is_honeycomb_tablet = isHoneycombTablet(context);
try {
final ActivityInfo[] activity_info = package_manager.getPackageInfo(context.getPackageName(),
PackageManager.GET_ACTIVITIES | PackageManager.GET_META_DATA).activities;
for (ActivityInfo info : activity_info) {
final String target_device = info.metaData.getString("target_device");
if (target_device == null) break;
target_device = target_device.toLowerCase(Locale.US);
final boolean is_for_tablet = target_device.equals("tablet");
final boolean is_for_phone = target_device.equals("phone");
final String class_name = info.name;
package_manager.setComponentEnabledSetting(new ComponentName(context, Class.forName(class_name)),
is_honeycomb_tablet && is_for_phone
? PackageManager.COMPONENT_ENABLED_STATE_DISABLED
: !is_honeycomb_tablet && is_for_tablet
? PackageManager.COMPONENT_ENABLED_STATE_DISABLED
: PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP);
}
} catch (PackageManager.NameNotFoundException error) {
Ln.w(error.getCause());
} catch (ClassNotFoundException error) {
Ln.w(error.getCause());
}
}
забавный факт: он не работает без флага GET_META_DATA
, так как metaData
всегда будет возвращать как null, если вы не включаете этот тег.
Последнее прикосновение - вызвать этот метод, вероятно, в onCreate
вашего начального Activity
@Override
protected void onCreate(Bundle icicle) {
super.onCreate(icicle);
// Anything else you want to do in the onCreate callback
// Set up to use the appropriate Activities for the given device
UIUtils.configureDeviceSpecificActivities(this);
}
Теперь вы можете иметь Activity
, специально предназначенные для телефонов и планшетов, в то время, когда просто изменить макет и, возможно, включить больше Fragment
, недостаточно.
ПРИМЕЧАНИЕ: final String class_name = info.packageName + info.name;
может быть final String class_name = info.name;
, если вы видите предупреждение.
ПРИМЕЧАНИЕ (2): final String target_device = info.metaData.getString("target_device", "").toLowerCase();
должен быть совместим с предыдущей версией API 12.
String target_device = info.metaData.getString("target_device");
if (target_device == null) break;
target_device = target_device.toLowerCase();
ПРИМЕЧАНИЕ (3): target_device.toLowerCase();
использует языковой стандарт по умолчанию неявно. Вместо этого используйте target_device.toLowerCase(Locale.US)
. И внесены все изменения в код выше.