New layout for widget picker is still not up to final design, but this change prepares all the backend work Also, remove use of features filter from the picker, since this is being removed from the API Change-Id: I9b332e22cf34b98c20c720602c920d28aed01d6c
189 lines
7.3 KiB
Java
189 lines
7.3 KiB
Java
/*
|
|
* Copyright (C) 2009 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.android.settings;
|
|
|
|
import android.appwidget.AppWidgetManager;
|
|
import android.appwidget.AppWidgetProviderInfo;
|
|
import android.content.Context;
|
|
import android.content.DialogInterface;
|
|
import android.content.Intent;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.content.res.Resources;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.os.Bundle;
|
|
import android.util.DisplayMetrics;
|
|
import android.util.Log;
|
|
|
|
import com.android.settings.ActivityPicker.PickAdapter;
|
|
|
|
import java.util.List;
|
|
|
|
/**
|
|
* Displays a list of {@link AppWidgetProviderInfo} widgets, along with any
|
|
* injected special widgets specified through
|
|
* {@link AppWidgetManager#EXTRA_CUSTOM_INFO} and
|
|
* {@link AppWidgetManager#EXTRA_CUSTOM_EXTRAS}.
|
|
* <p>
|
|
* When an installed {@link AppWidgetProviderInfo} is selected, this activity
|
|
* will bind it to the given {@link AppWidgetManager#EXTRA_APPWIDGET_ID},
|
|
* otherwise it will return the requested extras.
|
|
*/
|
|
public class AppWidgetPickActivity extends ActivityPicker
|
|
implements AppWidgetLoader.ItemConstructor<PickAdapter.Item>{
|
|
private static final String TAG = "AppWidgetPickActivity";
|
|
static final boolean LOGD = false;
|
|
|
|
List<PickAdapter.Item> mItems;
|
|
|
|
/**
|
|
* The allocated {@link AppWidgetManager#EXTRA_APPWIDGET_ID} that this
|
|
* activity is binding.
|
|
*/
|
|
private int mAppWidgetId;
|
|
private AppWidgetLoader<PickAdapter.Item> mAppWidgetLoader;
|
|
private AppWidgetManager mAppWidgetManager;
|
|
private PackageManager mPackageManager;
|
|
|
|
@Override
|
|
public void onCreate(Bundle icicle) {
|
|
mPackageManager = getPackageManager();
|
|
mAppWidgetManager = AppWidgetManager.getInstance(this);
|
|
mAppWidgetLoader = new AppWidgetLoader<PickAdapter.Item>
|
|
(this, mAppWidgetManager, this);
|
|
|
|
super.onCreate(icicle);
|
|
|
|
// Set default return data
|
|
setResultData(RESULT_CANCELED, null);
|
|
|
|
// Read the appWidgetId passed our direction, otherwise bail if not found
|
|
final Intent intent = getIntent();
|
|
if (intent.hasExtra(AppWidgetManager.EXTRA_APPWIDGET_ID)) {
|
|
mAppWidgetId = intent.getIntExtra(AppWidgetManager.EXTRA_APPWIDGET_ID,
|
|
AppWidgetManager.INVALID_APPWIDGET_ID);
|
|
} else {
|
|
finish();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Build and return list of items to be shown in dialog. This will mix both
|
|
* installed {@link AppWidgetProviderInfo} and those provided through
|
|
* {@link AppWidgetManager#EXTRA_CUSTOM_INFO}, sorting them alphabetically.
|
|
*/
|
|
@Override
|
|
protected List<PickAdapter.Item> getItems() {
|
|
mItems = mAppWidgetLoader.getItems(getIntent());
|
|
return mItems;
|
|
}
|
|
|
|
@Override
|
|
public PickAdapter.Item createItem(Context context, AppWidgetProviderInfo info, Bundle extras) {
|
|
CharSequence label = info.label;
|
|
Drawable icon = null;
|
|
|
|
if (info.icon != 0) {
|
|
try {
|
|
final Resources res = context.getResources();
|
|
final int density = res.getDisplayMetrics().densityDpi;
|
|
int iconDensity;
|
|
switch (density) {
|
|
case DisplayMetrics.DENSITY_MEDIUM:
|
|
iconDensity = DisplayMetrics.DENSITY_LOW;
|
|
case DisplayMetrics.DENSITY_TV:
|
|
iconDensity = DisplayMetrics.DENSITY_MEDIUM;
|
|
case DisplayMetrics.DENSITY_HIGH:
|
|
iconDensity = DisplayMetrics.DENSITY_MEDIUM;
|
|
case DisplayMetrics.DENSITY_XHIGH:
|
|
iconDensity = DisplayMetrics.DENSITY_HIGH;
|
|
case DisplayMetrics.DENSITY_XXHIGH:
|
|
iconDensity = DisplayMetrics.DENSITY_XHIGH;
|
|
default:
|
|
// The density is some abnormal value. Return some other
|
|
// abnormal value that is a reasonable scaling of it.
|
|
iconDensity = (int)((density*0.75f)+.5f);
|
|
}
|
|
Resources packageResources = mPackageManager.
|
|
getResourcesForApplication(info.provider.getPackageName());
|
|
icon = packageResources.getDrawableForDensity(info.icon, iconDensity);
|
|
} catch (NameNotFoundException e) {
|
|
Log.w(TAG, "Can't load icon drawable 0x" + Integer.toHexString(info.icon)
|
|
+ " for provider: " + info.provider);
|
|
}
|
|
if (icon == null) {
|
|
Log.w(TAG, "Can't load icon drawable 0x" + Integer.toHexString(info.icon)
|
|
+ " for provider: " + info.provider);
|
|
}
|
|
}
|
|
|
|
PickAdapter.Item item = new PickAdapter.Item(context, label, icon);
|
|
item.packageName = info.provider.getPackageName();
|
|
item.className = info.provider.getClassName();
|
|
item.extras = extras;
|
|
return item;
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
@Override
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
Intent intent = getIntentForPosition(which);
|
|
PickAdapter.Item item = mItems.get(which);
|
|
|
|
int result;
|
|
if (item.extras != null) {
|
|
// If these extras are present it's because this entry is custom.
|
|
// Don't try to bind it, just pass it back to the app.
|
|
setResultData(RESULT_OK, intent);
|
|
} else {
|
|
try {
|
|
Bundle options = null;
|
|
if (intent.getExtras() != null) {
|
|
options = intent.getExtras().getBundle(
|
|
AppWidgetManager.EXTRA_APPWIDGET_OPTIONS);
|
|
}
|
|
mAppWidgetManager.bindAppWidgetId(mAppWidgetId, intent.getComponent(), options);
|
|
result = RESULT_OK;
|
|
} catch (IllegalArgumentException e) {
|
|
// This is thrown if they're already bound, or otherwise somehow
|
|
// bogus. Set the result to canceled, and exit. The app *should*
|
|
// clean up at this point. We could pass the error along, but
|
|
// it's not clear that that's useful -- the widget will simply not
|
|
// appear.
|
|
result = RESULT_CANCELED;
|
|
}
|
|
setResultData(result, null);
|
|
}
|
|
|
|
finish();
|
|
}
|
|
|
|
|
|
/**
|
|
* Convenience method for setting the result code and intent. This method
|
|
* correctly injects the {@link AppWidgetManager#EXTRA_APPWIDGET_ID} that
|
|
* most hosts expect returned.
|
|
*/
|
|
void setResultData(int code, Intent intent) {
|
|
Intent result = intent != null ? intent : new Intent();
|
|
result.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, mAppWidgetId);
|
|
setResult(code, result);
|
|
}
|
|
}
|