首次运行时显示叠加工具提示

时间:2013-08-22 18:02:39

标签: android user-interface

首次运行时(或经过一些重大更新后),某些Google应用或核心Android系统本身有时会显示透明覆盖图,并提供简短的工具提示如何使用新功能。一个例子可能是下面的图像。

Android框架中是否有一些用于创建这些API的API,还是全部是自定义的?在后一种情况下如何实现它?最后,但并非最不重要的是,这些工具提示是否具有某种官方/技术名称,通过它可以引用它们(在谷歌搜索有关该主题的某些信息时可能有用)?感谢。

修改

我已经获得了一个截图,更准确地说明了我的意思。除了“指针”图形外,此工具提示还在时钟应用程序图标周围设置了一个突出显示的圆圈。这不仅仅是一个半透明的叠加层,它在点击后消失:此刻可以点击时钟图标甚至长按,但在工具包消失之前无法访问其他图标(圆圈外)。所有这些行为是专门为此目的而定制的,还是存在一些内置设施?

enter image description here

3 个答案:

答案 0 :(得分:3)

这只是一个半透明的对话框。

在res文件夹

中的themes.xml中插入以下代码
 <style name="Theme.TranparentDialog" parent="@android:style/Theme.NoTitleBar">
        <item name="android:colorBackgroundCacheHint">@null</item>
        <item name="android:windowIsTranslucent">true</item>
        <item name="android:colorForeground">@color/transparent</item>
        <item name="android:windowIsFloating">false</item>
        <item name="android:backgroundDimEnabled">false</item>
        <item name="android:windowAnimationStyle">@android:style/Animation.Dialog</item>
        <item name="android:windowBackground">@color/transparent</item>
    </style>

在屏幕上创建对话框以获得半透明效果/

Dialog dialog= new Dialog(context,  R.style.Theme_TutorialDialog);

dialog.show();

答案 1 :(得分:1)

我认为这是使用ShowcaseView Library。见this related question

答案 2 :(得分:0)

如果在上面的对话框中用作背景,则此视图可用于在屏幕上绘制透明矩形。

public class WindowView extends View {
private static final int LOW_DPI_STATUS_BAR_HEIGHT = 19;

private static final int MEDIUM_DPI_STATUS_BAR_HEIGHT = 25;

private static final int HIGH_DPI_STATUS_BAR_HEIGHT = 38;
Context mContext;

int mHeight, mWidth, mTop, mRight, mLeft, mBottom;
Rect mLeftRect, mRightRect, mTopRect, mBottomRect;

Rect mRect;
Paint mPaint;
View mAnchorView;
Paint mTempPaint1;

public WindowView(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}

public WindowView(Context context) {
    this(context, null);
}

public WindowView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    mContext = context;
    initVariables();
}

boolean mReset = false;

GestureDetector mGestureDetector;
GestureDetector.SimpleOnGestureListener mSimpleOnGestureListener;

private View getView() {
    return this;
}

@Override
public boolean onTouchEvent(MotionEvent event) {
    if (mSimpleOnGestureListener == null) {
        mSimpleOnGestureListener = new SimpleOnGestureListener() {
            @Override
            public boolean onSingleTapConfirmed(MotionEvent e) {
                if (mListener != null && e.getX() >= mRect.left
                        && e.getX() <= mRect.right && e.getY() >= mRect.top
                        && e.getY() <= mRect.bottom)
                    mListener.onClick(getView());
                return true;
            }
        };
    }
    if (mGestureDetector == null) {
        mGestureDetector = new GestureDetector(mContext,
                mSimpleOnGestureListener);
    }
    mGestureDetector.onTouchEvent(event);
    return true;
}

private void initVariables() {
    mPaint = new Paint();
    mPaint.setStyle(Style.FILL);
    mPaint.setColor(mContext.getResources().getColor(
            R.color.tut_transcluscent_bg));
    mRect = new Rect();

    mTempPaint1 = new Paint(Paint.ANTI_ALIAS_FLAG);
    mTempPaint1.setAlpha(64);
    mTempPaint1.setColorFilter(new PorterDuffColorFilter(Color.WHITE,
            Mode.SRC_IN));
}

OnClickListener mListener;

public void setTransparentBackGroundColor(int color) {
    mPaint.setColor(color);
    invalidate();
}

@Override
public void setOnClickListener(OnClickListener l) {
    mListener = l;
}

public View getAnchorView() {
    return mAnchorView;
}

float[] mPaddings;
Rect mAnchorRect;;

public void setAnchorView(View mView, Rect rect, float[] paddings) {
    mAnchorRect = rect;
    if (mAnchorRect == null)
        mAnchorView = mView;
    else
        mAnchorView = null;
    mPaddings = paddings;
    requestLayout();
    invalidate();
}

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    if (mAnchorView != null || mAnchorRect != null)
        initView();
    mReset = true;
}

boolean mFlagInvalid = false;
Handler mAnrHandler;

public void setHandler(Handler handler) {
    mAnrHandler = handler;
}

private int getStatusBarHeight() {
    DisplayMetrics displayMetrics = new DisplayMetrics();
    ((WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE))
            .getDefaultDisplay().getMetrics(displayMetrics);

    int statusBarHeight;

    switch (displayMetrics.densityDpi) {
    case DisplayMetrics.DENSITY_HIGH:
        statusBarHeight = HIGH_DPI_STATUS_BAR_HEIGHT;
        break;
    case DisplayMetrics.DENSITY_MEDIUM:
        statusBarHeight = MEDIUM_DPI_STATUS_BAR_HEIGHT;
        break;
    case DisplayMetrics.DENSITY_LOW:
        statusBarHeight = LOW_DPI_STATUS_BAR_HEIGHT;
        break;
    default:
        statusBarHeight = MEDIUM_DPI_STATUS_BAR_HEIGHT;
    }
    return statusBarHeight;
}

public void initView() {

    mFlagInvalid = false;
    int top = 0, left = 0;
    if (mAnchorView != null) {
        top = mAnchorView.getTop();
        left = mAnchorView.getLeft();
        View temp = mAnchorView;
        int cnt = 0;
        try {
            while (((View) temp.getParent()).getId() != android.R.id.content) {
                temp = (View) temp.getParent();
                int scrolledHeight = 0;
                if (temp instanceof ScrollView) {
                    scrolledHeight = ((ScrollView) temp).getScrollY();
                }
                top = top + temp.getTop() - scrolledHeight;
                left = left + temp.getLeft();
                cnt++;
                if (cnt > 100) {
                    if (mAnrHandler != null)
                        mAnrHandler.sendEmptyMessage(9);
                    mFlagInvalid = true;
                    break;
                }
            }
        } catch (Exception e) {
            mFlagInvalid = true;
            if (mAnrHandler != null)
                mAnrHandler.sendEmptyMessage(8);
            e.printStackTrace();
        }
        TypedValue tv = new TypedValue();
        if (getContext()
                .getTheme()
                .resolveAttribute(
                        VERSION.SDK_INT >= VERSION_CODES.HONEYCOMB ? android.R.attr.actionBarSize
                                : R.attr.actionBarSize, tv, true)) {
            int actionBarHeight = TypedValue.complexToDimensionPixelSize(
                    tv.data, getResources().getDisplayMetrics());
            top = top + actionBarHeight;
        }

    } else if (mAnchorRect != null) {
    }

    if (mFlagInvalid)
        init(20, 20, 50, 50);
    else {
        if (mAnchorRect != null) {
            init(mAnchorRect.bottom, mAnchorRect.right, mAnchorRect.left,
                    mAnchorRect.top);
        } else
            init(mAnchorView.getHeight(), mAnchorView.getWidth(), left, top);
    }
}

public void init(int height, int width, int left, int top) {
    mWidth = width;
    mHeight = height;
    mTop = top;
    mBottom = top + height;
    mLeft = left;
    mRight = left + width;

    mLeft = (int) (mLeft + mPaddings[0] * mWidth);
    mTop = (int) (mTop + mPaddings[1] * mHeight);
    mRight = (int) (mRight - mPaddings[2] * mWidth);
    mBottom = (int) (mBottom - mPaddings[3] * mHeight);
    mRect = new Rect(mLeft, mTop, mRight, mBottom);

    invalidate();
}

public Rect getAnchorRect() {
    return mRect;
}

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    initOnceOnDraw();
    canvas.drawRect(mLeftRect, mPaint);
    canvas.drawRect(mTopRect, mPaint);
    canvas.drawRect(mRightRect, mPaint);
    canvas.drawRect(mBottomRect, mPaint);
    canvas.drawRect(mRect, mTempPaint1);
}

private void initOnceOnDraw() {
    if (mReset) {
        mReset = false;
        mTopRect = new Rect(0, 0, getWidth(), mRect.top);
        mLeftRect = new Rect(0, mRect.top, mRect.left, mRect.bottom);
        mRightRect = new Rect(mRect.right, mRect.top, getWidth(),
                mRect.bottom);
        mBottomRect = new Rect(0, mRect.bottom, getWidth(), getHeight());
    }
}}

方法setAnchorView()给出了必须绘制叠加层的视图..

    mWindowView = (WindowView) mTutorialView
                            .findViewById(R.id.windowview_tutorial);
                        mWindowView.setTransparentBackGroundColor(backgroundColor);

                    mWindowView.setAnchorView(view, null, padding);

希望这有帮助