mirror of
https://github.com/LawnchairLauncher/lawnchair.git
synced 2026-03-03 17:36:49 +00:00
-> Pulling out the parts of device profile which can (and need to be) initialized and accessed without access to an Activity context, ie. the invariant bits. -> The invariant bits are stored in InvariantDeviceProfile which is initialized statically from LauncherAppState. -> The DeviceProfile contains the Activity context-dependent bits, and we will create one of these for each Activity instance, and this instance is accessed through the Launcher activity. -> It's possible that we can continue to refactor this such that all appropriate dimensions can be computed without an Activity context (by only specifying orientation). This would be an extension of this CL and allow us to know exactly how launcher will look in both orientations from any context. Sets the stage for some improvements around b/19514688 Change-Id: Ia7daccf14d8ca2b9cb340b8780b684769e9f1892
257 lines
9.8 KiB
Java
257 lines
9.8 KiB
Java
/*
|
|
* Copyright (C) 2014 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.launcher3;
|
|
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.res.Resources.Theme;
|
|
import android.graphics.Bitmap;
|
|
import android.graphics.Canvas;
|
|
import android.graphics.Rect;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.os.Bundle;
|
|
import android.text.Layout;
|
|
import android.text.StaticLayout;
|
|
import android.text.TextPaint;
|
|
import android.util.TypedValue;
|
|
import android.view.View;
|
|
import android.view.View.OnClickListener;
|
|
|
|
public class PendingAppWidgetHostView extends LauncherAppWidgetHostView implements OnClickListener {
|
|
|
|
private static Theme sPreloaderTheme;
|
|
|
|
private final Rect mRect = new Rect();
|
|
private View mDefaultView;
|
|
private OnClickListener mClickListener;
|
|
private final LauncherAppWidgetInfo mInfo;
|
|
private final int mStartState;
|
|
private final Intent mIconLookupIntent;
|
|
private final boolean mDisabledForSafeMode;
|
|
private Launcher mLauncher;
|
|
|
|
private Bitmap mIcon;
|
|
|
|
private Drawable mCenterDrawable;
|
|
private Drawable mTopCornerDrawable;
|
|
|
|
private boolean mDrawableSizeChanged;
|
|
|
|
private final TextPaint mPaint;
|
|
private Layout mSetupTextLayout;
|
|
|
|
public PendingAppWidgetHostView(Context context, LauncherAppWidgetInfo info,
|
|
boolean disabledForSafeMode) {
|
|
super(context);
|
|
|
|
mLauncher = (Launcher) context;
|
|
mInfo = info;
|
|
mStartState = info.restoreStatus;
|
|
mIconLookupIntent = new Intent().setComponent(info.providerName);
|
|
mDisabledForSafeMode = disabledForSafeMode;
|
|
|
|
mPaint = new TextPaint();
|
|
mPaint.setColor(0xFFFFFFFF);
|
|
mPaint.setTextSize(TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_PX,
|
|
mLauncher.getDeviceProfile().iconTextSizePx, getResources().getDisplayMetrics()));
|
|
setBackgroundResource(R.drawable.quantum_panel_dark);
|
|
setWillNotDraw(false);
|
|
}
|
|
|
|
@Override
|
|
public void updateAppWidgetSize(Bundle newOptions, int minWidth, int minHeight, int maxWidth,
|
|
int maxHeight) {
|
|
// No-op
|
|
}
|
|
|
|
@Override
|
|
protected View getDefaultView() {
|
|
if (mDefaultView == null) {
|
|
mDefaultView = mInflater.inflate(R.layout.appwidget_not_ready, this, false);
|
|
mDefaultView.setOnClickListener(this);
|
|
applyState();
|
|
}
|
|
return mDefaultView;
|
|
}
|
|
|
|
@Override
|
|
public void setOnClickListener(OnClickListener l) {
|
|
mClickListener = l;
|
|
}
|
|
|
|
@Override
|
|
public boolean isReinflateRequired() {
|
|
// Re inflate is required any time the widget restore status changes
|
|
return mStartState != mInfo.restoreStatus;
|
|
}
|
|
|
|
@Override
|
|
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
|
|
super.onSizeChanged(w, h, oldw, oldh);
|
|
mDrawableSizeChanged = true;
|
|
}
|
|
|
|
public void updateIcon(IconCache cache) {
|
|
Bitmap icon = cache.getIcon(mIconLookupIntent, mInfo.user);
|
|
if (mIcon == icon) {
|
|
return;
|
|
}
|
|
mIcon = icon;
|
|
if (mCenterDrawable != null) {
|
|
mCenterDrawable.setCallback(null);
|
|
mCenterDrawable = null;
|
|
}
|
|
if (mIcon != null) {
|
|
// The view displays three modes,
|
|
// 1) App icon in the center
|
|
// 2) Preload icon in the center
|
|
// 3) Setup icon in the center and app icon in the top right corner.
|
|
if (mDisabledForSafeMode) {
|
|
FastBitmapDrawable disabledIcon = Utilities.createIconDrawable(mIcon);
|
|
disabledIcon.setGhostModeEnabled(true);
|
|
mCenterDrawable = disabledIcon;
|
|
mTopCornerDrawable = null;
|
|
} else if (isReadyForClickSetup()) {
|
|
mCenterDrawable = getResources().getDrawable(R.drawable.ic_setting);
|
|
mTopCornerDrawable = new FastBitmapDrawable(mIcon);
|
|
} else {
|
|
if (sPreloaderTheme == null) {
|
|
sPreloaderTheme = getResources().newTheme();
|
|
sPreloaderTheme.applyStyle(R.style.PreloadIcon, true);
|
|
}
|
|
|
|
FastBitmapDrawable drawable = Utilities.createIconDrawable(mIcon);
|
|
mCenterDrawable = new PreloadIconDrawable(drawable, sPreloaderTheme);
|
|
mCenterDrawable.setCallback(this);
|
|
mTopCornerDrawable = null;
|
|
applyState();
|
|
}
|
|
mDrawableSizeChanged = true;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected boolean verifyDrawable(Drawable who) {
|
|
return (who == mCenterDrawable) || super.verifyDrawable(who);
|
|
}
|
|
|
|
public void applyState() {
|
|
if (mCenterDrawable != null) {
|
|
mCenterDrawable.setLevel(Math.max(mInfo.installProgress, 0));
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onClick(View v) {
|
|
// AppWidgetHostView blocks all click events on the root view. Instead handle click events
|
|
// on the content and pass it along.
|
|
if (mClickListener != null) {
|
|
mClickListener.onClick(this);
|
|
}
|
|
}
|
|
|
|
public boolean isReadyForClickSetup() {
|
|
return (mInfo.restoreStatus & LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY) == 0
|
|
&& (mInfo.restoreStatus & LauncherAppWidgetInfo.FLAG_UI_NOT_READY) != 0;
|
|
}
|
|
|
|
@Override
|
|
protected void onDraw(Canvas canvas) {
|
|
if (mCenterDrawable == null) {
|
|
// Nothing to draw
|
|
return;
|
|
}
|
|
|
|
DeviceProfile grid = mLauncher.getDeviceProfile();
|
|
if (mTopCornerDrawable == null) {
|
|
if (mDrawableSizeChanged) {
|
|
int outset = (mCenterDrawable instanceof PreloadIconDrawable) ?
|
|
((PreloadIconDrawable) mCenterDrawable).getOutset() : 0;
|
|
int maxSize = grid.iconSizePx + 2 * outset;
|
|
int size = Math.min(maxSize, Math.min(
|
|
getWidth() - getPaddingLeft() - getPaddingRight(),
|
|
getHeight() - getPaddingTop() - getPaddingBottom()));
|
|
|
|
mRect.set(0, 0, size, size);
|
|
mRect.inset(outset, outset);
|
|
mRect.offsetTo((getWidth() - mRect.width()) / 2, (getHeight() - mRect.height()) / 2);
|
|
mCenterDrawable.setBounds(mRect);
|
|
mDrawableSizeChanged = false;
|
|
}
|
|
mCenterDrawable.draw(canvas);
|
|
} else {
|
|
// Draw the top corner icon and "Setup" text is possible
|
|
if (mDrawableSizeChanged) {
|
|
int iconSize = grid.iconSizePx;
|
|
int paddingTop = getPaddingTop();
|
|
int paddingBottom = getPaddingBottom();
|
|
int paddingLeft = getPaddingLeft();
|
|
int paddingRight = getPaddingRight();
|
|
|
|
int availableWidth = getWidth() - paddingLeft - paddingRight;
|
|
int availableHeight = getHeight() - paddingTop - paddingBottom;
|
|
|
|
// Recreate the setup text.
|
|
mSetupTextLayout = new StaticLayout(
|
|
getResources().getText(R.string.gadget_setup_text), mPaint, availableWidth,
|
|
Layout.Alignment.ALIGN_CENTER, 1, 0, true);
|
|
if (mSetupTextLayout.getLineCount() == 1) {
|
|
// The text fits in a single line. No need to draw the setup icon.
|
|
int size = Math.min(iconSize, Math.min(availableWidth,
|
|
availableHeight - mSetupTextLayout.getHeight()));
|
|
mRect.set(0, 0, size, size);
|
|
mRect.offsetTo((getWidth() - mRect.width()) / 2,
|
|
(getHeight() - mRect.height() - mSetupTextLayout.getHeight()
|
|
- grid.iconDrawablePaddingPx) / 2);
|
|
|
|
mTopCornerDrawable.setBounds(mRect);
|
|
|
|
// Update left and top to indicate the position where the text will be drawn.
|
|
mRect.left = paddingLeft;
|
|
mRect.top = mRect.bottom + grid.iconDrawablePaddingPx;
|
|
} else {
|
|
// The text can't be drawn in a single line. Draw a setup icon instead.
|
|
mSetupTextLayout = null;
|
|
int size = Math.min(iconSize, Math.min(
|
|
getWidth() - paddingLeft - paddingRight,
|
|
getHeight() - paddingTop - paddingBottom));
|
|
mRect.set(0, 0, size, size);
|
|
mRect.offsetTo((getWidth() - mRect.width()) / 2, (getHeight() - mRect.height()) / 2);
|
|
mCenterDrawable.setBounds(mRect);
|
|
|
|
size = Math.min(size / 2,
|
|
Math.max(mRect.top - paddingTop, mRect.left - paddingLeft));
|
|
mTopCornerDrawable.setBounds(paddingLeft, paddingTop,
|
|
paddingLeft + size, paddingTop + size);
|
|
}
|
|
mDrawableSizeChanged = false;
|
|
}
|
|
|
|
if (mSetupTextLayout == null) {
|
|
mCenterDrawable.draw(canvas);
|
|
mTopCornerDrawable.draw(canvas);
|
|
} else {
|
|
canvas.save();
|
|
canvas.translate(mRect.left, mRect.top);
|
|
mSetupTextLayout.draw(canvas);
|
|
canvas.restore();
|
|
mTopCornerDrawable.draw(canvas);
|
|
}
|
|
}
|
|
}
|
|
}
|