Динамическая настройка содержимого и макета виджета до размера, определенного пользователем при изменении размера. Android
В руководстве по шаблонам дизайна Android говорится, что содержание и макет виджета могут быть динамически отрегулированы до размера, определенного пользователем через операцию изменения размера: Дизайн руководство по виджетам
Пример, приведенный в руководстве по проектированию:
![Example image provided in the design guide.]()
Но я ничего не вижу в документах относительно того, как это сделать. Как изменить макет в соответствии с операцией изменения размера? Любые идеи относительно подхода будут оценены.
Ответы
Ответ 1
Благодаря A-C это возможно для Jellybean и выше устройств и прост в реализации.
Ниже приведен пример кода с использованием метода onAppWidgetOptionsChanged
@TargetApi(Build.VERSION_CODES.JELLY_BEAN)
@Override
public void onAppWidgetOptionsChanged(Context context,
AppWidgetManager appWidgetManager, int appWidgetId, Bundle newOptions) {
Log.d(DEBUG_TAG, "Changed dimensions");
// See the dimensions and
Bundle options = appWidgetManager.getAppWidgetOptions(appWidgetId);
// Get min width and height.
int minWidth = options.getInt(AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH);
int minHeight = options
.getInt(AppWidgetManager.OPTION_APPWIDGET_MIN_HEIGHT);
// Obtain appropriate widget and update it.
appWidgetManager.updateAppWidget(appWidgetId,
getRemoteViews(context, minWidth, minHeight));
super.onAppWidgetOptionsChanged(context, appWidgetManager, appWidgetId,
newOptions);
}
/**
* Determine appropriate view based on width provided.
*
* @param minWidth
* @param minHeight
* @return
*/
private RemoteViews getRemoteViews(Context context, int minWidth,
int minHeight) {
// First find out rows and columns based on width provided.
int rows = getCellsForSize(minHeight);
int columns = getCellsForSize(minWidth);
if (columns == 4) {
// Get 4 column widget remote view and return
} else {
// Get appropriate remote view.
return new RemoteViews(context.getPackageName(),
R.layout.quick_add_widget_3_1);
}
}
/**
* Returns number of cells needed for given size of the widget.
*
* @param size Widget size in dp.
* @return Size in number of cells.
*/
private static int getCellsForSize(int size) {
int n = 2;
while (70 * n - 30 < size) {
++n;
}
return n - 1;
}
Ответ 2
@Choletski
@azendh
После изменения макетов некоторые события click больше не вызываются
Я решил эту проблему, создав функцию setOnClickPendingIntent
на вид, а затем верните его.
Например, код похож на
private RemoteViews getConfiguredView (RemoteViews remoteViews, Context context){
Intent refreshIntent = new Intent(context, EarningsWidget.class);
refreshIntent.setAction(REFRESH_ACTION);
PendingIntent toastPendingIntent = PendingIntent.getBroadcast(context, 3, refreshIntent, PendingIntent.FLAG_UPDATE_CURRENT);
remoteViews.setOnClickPendingIntent(R.id.refreshButton, toastPendingIntent);
return remoteViews;
}
И чем функция вызывается, где вы делаете "Получить соответствующий удаленный просмотр".
return getConfiguredView(new RemoteViews(context.getPackageName(), R.layout.activity_widget), context);
Ответ 3
Основываясь на ответе Gogu и в соответствии с этим ответом о том, как получить размер виджета, я создал этот WidgetClass
в Kotlin
:
class WidgetClass: AppWidgetProvider() {
override fun onUpdate(context: Context?, appWidgetManager: AppWidgetManager?, appWidgetIds: IntArray?) {
for (id in appWidgetIds!!) {
//get widget options for later get widget dimensions
val options = appWidgetManager?.getAppWidgetOptions(id)
//get widget view based on widget size
val view = getView(context, options)
//update widget
appWidgetManager!!.updateAppWidget(id, view)
}
}
//listen for widget changes
override fun onAppWidgetOptionsChanged(context: Context?, appWidgetManager: AppWidgetManager?,
appWidgetId: Int, newOptions: Bundle?) {
//update widget view based on new options
appWidgetManager?.updateAppWidget(appWidgetId, getView(context, newOptions))
super.onAppWidgetOptionsChanged(context, appWidgetManager, appWidgetId, newOptions)
}
private fun getView(context: Context?, options: Bundle?): RemoteViews {
val minWidth: Int
val minHeight: Int
if (context!!.resources.configuration.orientation == ActivityInfo.SCREEN_ORIENTATION_PORTRAIT
|| context.resources.configuration.orientation == ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT) {
minWidth = options?.getInt(AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH) ?: 0
minHeight = options?.getInt(AppWidgetManager.OPTION_APPWIDGET_MAX_HEIGHT) ?: 0
} else {
minWidth = options?.getInt(AppWidgetManager.OPTION_APPWIDGET_MAX_WIDTH) ?: 0
minHeight = options?.getInt(AppWidgetManager.OPTION_APPWIDGET_MIN_HEIGHT) ?: 0
}
//get widget view accordin widget size
return if (minWidth >= 240)
RemoteViews(context.packageName, R.layout.widget_large)
else
RemoteViews(context.packageName, R.layout.widget_small)
}
}