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