使用错误的Canvas调用View.onDraw

时间:2012-11-21 16:50:04

标签: android graphics android-2.3-gingerbread

我的应用程序支持API级别10(Gingerbread)向上。其中一个活动绘制了一个完全适用于更高版本的图表,但是当运行10级模拟器时,我会使用错误的画布ID对View.onDraw进行额外调用,这会导致屏幕变为空白。 (这不仅仅是模拟器 - 有人在姜饼手机上运行时报告了这个问题。)

正常操作是为onDraw调用两次 - 第一次来自框架,我从中获取画布ID,第二次从我调用invalidate(),它传递相同的画布ID。这两个调用是在10级模拟器中发生的,但是后来有第三个调用具有不同的画布ID - 即不属于视图,这会使其空白。

该活动源自SherlockActivity以提供操作栏,我相信这是造成问题的原因。

我的活动类的相关代码:

public class Chart extends SherlockActivity implements OnGestureListener, OnDoubleTapListener, OnScaleGestureListener
{
    public static boolean   mDataSet = false;

    private ChartView              mView;
    private Menu                   mMenu;
    private GestureDetector        mDetector;
    private ScaleGestureDetector   mScaleDetector;
    private ActionBar              mActionBar;

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);

        mDataSet = false;

        mActionBar = getSupportActionBar();
        mActionBar.setHomeButtonEnabled(true);
        mActionBar.setDisplayHomeAsUpEnabled(true);
        mActionBar.setDisplayOptions(0, ActionBar.DISPLAY_SHOW_TITLE);

        mView = new ChartView(this, mCentreLat, mCentreLong, mRadius);
        setContentView(mView);
        setTitle("");

        Context context = getApplicationContext();

        mDetector   = new GestureDetector(context, this);
        mScaleDetector = new ScaleGestureDetector(context, this);
    }

    @Override
    public void onConfigurationChanged(Configuration config)
    {
        super.onConfigurationChanged(config);
        mDataSet = false;
    }

    // Menu handling
    @Override
    public boolean onCreateOptionsMenu(Menu menu)
    {
        mMenu = menu;
        MenuInflater inflater = getSupportMenuInflater();
        inflater.inflate(R.menu.chart_menu, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item)
    {
        boolean handled = true;

        switch (item.getItemId())
        {
        case android.R.id.home:
        finish();
        break;

        case R.id.viewOptions:
            mDataSet = false;
            Intent i = new Intent(getBaseContext(), ChartSettings.class);
            startActivity(i);
            break;

        // Other menu options here...

        default:
            handled = false;
        }
        return handled;
    }

我的View类中的相关代码:

public class ChartView extends View
{

    public ChartView(Context context, float centreLat, float centreLong, float radius)
    {
        super(context);
        mContext    = context;
        mCentreLat  = centreLat;
        mCentreLong = centreLong;
        mRadius = radius;
    }

    @Override
    protected void onDraw(Canvas canvas)
    {
        // First time
        // (We pick up the canvas id mCanvas from the parameter.)
        // (Nothing much else relevant here, except this is where stuff
        // gets initialized, then in setDataInfo(), mDataSet is set true
        // and invalidate() is called at the end.)
        if (!Chart.mDataSet)
        {
            setBackgroundColor(Color.WHITE);
            mCanvas = canvas;
            initPaint();
            setDataInfo();      // invalidate() called at end
        }
        else
        {
            // Second call to onDraw() with same canvas id comes here,
            // then an unwanted third call with a different canvas id, only
            // with the Level 10 (Gingerbread) emulator.

            // This is where the various lines and shapes are plotted
            // on the canvas (mCanvas)
            plotLinesAndShapes();
        }

任何人都可以解释为什么姜饼模拟器(或手机)会发生第三次通话?效果是屏幕被消隐(全白)。

当它到达这个功能时,为时已晚,并且不能忽略调用 - 当调用堆栈展开时,屏幕变为空白。

有一个workround - 如果用户从菜单中选择视图选项,然后立即返回到图表,它将被重绘并从此开始正常运行。

1 个答案:

答案 0 :(得分:4)

问题是您保留对作为参数接收的Canvas的引用。在当前帧完成后,无法保证此Canvas实例有效。例如,您可以在每个帧上接收不同的Canvas实例。当视图呈现为位图(例如View.setDrawingCacheEnabled())时,您通常会收到不同的Canvas。)

不要在mCanvas中保留引用,只需将您收到的画布传递给plotLinesAndShapes()。