5

下一个类是红色三角形的视图:

public class FreeStyleViewII extends View {

private final Paint paint = new Paint();
private final int[] colors = new int[] {
        Color.RED,
        Color.RED,
        Color.RED,
        0xFF000000, 0xFF000000, 0xFF000000
};
private final float[] verts = new float[] {
    1f/2f * 200f, 1f/4f * 200f,
    1f/4f * 200f, 3f/4f * 200f,
    3f/4f * 200f, 3f/4f * 200f
};
private final Path path = new Path();
{
    path.moveTo(1/2 * 200, 1/4 * 200);
    path.lineTo(1/4 * 200, 3/4 * 200);
    path.lineTo(3/4 * 200, 3/4 * 200);
    path.lineTo(1/2 * 200, 1/4 * 200);
}

private final RectF bounds = new RectF();

    public FreeStyleViewII(Context context) {
        super(context);
    }

    public FreeStyleViewII(Context context, AttributeSet attrs) {
    super(context, attrs);
}

public FreeStyleViewII(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
}

    @Override
    protected void onDraw(Canvas canvas) {
        canvas.clipRect(bounds);
        canvas.drawRGB(0, 0, 0);

        paint.setStyle(Style.FILL);
        paint.setColor(Color.RED);

        // HERE. WHY DRAWVERTICES DOESN'T WORK BUT DRAWPATH DOES?... 
        canvas.drawVertices(Canvas.VertexMode.TRIANGLES, verts.length, verts, 0, null, 0, colors, 0, null, 0, 0, paint);
        // canvas.drawPath(path, paint);

        paint.setStyle(Style.STROKE);
        paint.setColor(Color.LTGRAY);
        canvas.drawLine(0, bounds.bottom / 2, bounds.right, bounds.bottom / 2, paint);
        canvas.drawLine(bounds.right / 2, 0, bounds.right / 2, bounds.bottom, paint);

        // Delay
    try {  
        Thread.sleep(30);  
    } catch (InterruptedException e) { }

        invalidate();

    }

@Override
public void onSizeChanged(int w, int h, int oldW, int oldH) {
    bounds.set(1, 1, w - 1, h - 1);
    System.out.println(bounds.left + " " + bounds.top + " " + bounds.right + " " + bounds.bottom);

    verts[0] = bounds.left + ((bounds.right - bounds.left) * (1f / 2f));
    verts[1] = bounds.top + ((bounds.bottom - bounds.top)  * (1f / 4f));
    System.out.println(" Point: X ----> " + verts[0] + " | Y ----> " + verts[1]);

    verts[2] = bounds.left + ((bounds.right - bounds.left) * (1f / 4f));
    verts[3] = bounds.top + ((bounds.bottom - bounds.top)  * (3f / 4f));
    System.out.println(" Point: X ----> " + verts[2] + " | Y ----> " + verts[3]);

    verts[4] = bounds.left + ((bounds.right - bounds.left) * (3f / 4f));
    verts[5] = bounds.top + ((bounds.bottom - bounds.top)  * (3f / 4f));
    System.out.println(" Point: X ----> " + verts[4] + " | Y ----> " + verts[5]);

    path.reset();

    path.moveTo(verts[0], verts[1]);
    path.lineTo(verts[2], verts[3]);
    path.lineTo(verts[4], verts[5]);
    path.lineTo(verts[0], verts[1]);

}
}

当我使用 Canvas.drawPath 方法时,它工作正常。但是,如果我更改为 Canvas.drawVertices,它不会绘制任何内容。我检查了Canvas.drawVertices 中的 Bug 中所说的内容?(使用 repro 代码和 logcat)Method drawVertices() 没有在 Android Canvas 上绘制任何东西,但在我的情况下结果是一样的。

我在VirtualBox(v 4.1.22)中使用AndroVM(v 4.1.1)进行测试。会不会是模拟器?

任何的想法?

4

1 回答 1

10

简短的回答:

禁用硬件加速,你会没事的!

长答案:

我有同样的问题。在我的自定义视图代码中,我调用了 Canvas.drawVertices(),如下所示:

canvas.drawVertices(Canvas.VertexMode.TRIANGLES, mTriVerts.length, mTriVerts, 0, null, 0, mTriangleColors, 0, null, 0, 0, mTriPaint);

这与许多设备上的预期完全一样,但是在我的 4.x 测试设备上,我试图绘制的三角形只是消失了,并且 logcat 中没有任何痕迹可以说明原因。

在 SO 没有找到解决方案后,我开始四处寻找,偶然发现了以下令人震惊的解决方案:如果启用了硬件加速,包括Canvas.drawVertices()对支持 Android 4.x 及更高版本的设备。

官方 android 文档中概述了该问题和一个好的解决方案。在这里总结一下,只需将其放在不受支持的呼叫之前的行上:

setLayerType(View.LAYER_TYPE_SOFTWARE, null);

这是在View水平上,不幸的是,目前无法在已设置为软件模式的视图上重新启用硬件加速,但阅读文档感觉这可能是未来的一种选择。如果您觉得您的代码需要硬件加速,只需避免使用调用,或在视图中单独收集您不兼容的代码,并对其进行分析以查看您实际上通过这种方式获得了一些东西。

重要编辑:

setLayerType()调用仅在 Android API 级别 11 及更高版本上受支持,这意味着您必须将对该函数的调用包装在版本检查中,以使其在早期版本上工作,如下所示:

if (android.os.Build.VERSION.SDK_INT >= 11) {
    setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}

现在你应该准备好了!

PS:希望这个答案能及时传达给你!

于 2013-04-27T21:37:26.850 回答