SkiaCanvasProxy.cpp revision 17c5adfa63872fbb6a903a5941e3c6455995b92d
11db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger/* 21db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * Copyright (C) 2015 The Android Open Source Project 31db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * 41db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * Licensed under the Apache License, Version 2.0 (the "License"); 51db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * you may not use this file except in compliance with the License. 61db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * You may obtain a copy of the License at 71db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * 81db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * http://www.apache.org/licenses/LICENSE-2.0 91db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * 101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * Unless required by applicable law or agreed to in writing, software 111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * distributed under the License is distributed on an "AS IS" BASIS, 121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 131db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * See the License for the specific language governing permissions and 141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * limitations under the License. 151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger */ 161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger#include "SkiaCanvasProxy.h" 181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger#include <cutils/log.h> 201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger#include <SkPatchUtils.h> 2117c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson#include <SkPixelRef.h> 221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergernamespace android { 241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergernamespace uirenderer { 251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 26b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom HudsonSkiaCanvasProxy::SkiaCanvasProxy(Canvas* canvas, bool filterHwuiCalls) 271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger : INHERITED(canvas->width(), canvas->height()) 28b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson , mCanvas(canvas) 29b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson , mFilterHwuiCalls(filterHwuiCalls) {} 301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPaint(const SkPaint& paint) { 321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPaint(paint); 331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPoints(PointMode pointMode, size_t count, const SkPoint pts[], 361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 37b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (!pts || count == 0) { 38b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 39b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 40b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson 411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert the SkPoints into floats 421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const size_t floatCount = count << 1; 441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* floatArray = &pts[0].fX; 451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger switch (pointMode) { 471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kPoints_PointMode: { 481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPoints(floatArray, floatCount, paint); 491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kLines_PointMode: { 521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawLines(floatArray, floatCount, paint); 531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kPolygon_PointMode: { 561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPaint strokedPaint(paint); 571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger strokedPaint.setStyle(SkPaint::kStroke_Style); 581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (size_t i = 0; i < count - 1; i++) { 611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.moveTo(pts[i]); 621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.lineTo(pts[i+1]); 631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawPath(path, strokedPaint); 641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.rewind(); 651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger default: 691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger LOG_ALWAYS_FATAL("Unknown point type"); 701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawOval(const SkRect& rect, const SkPaint& paint) { 741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawOval(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint); 751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawRect(const SkRect& rect, const SkPaint& paint) { 781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint); 791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawRRect(const SkRRect& roundRect, const SkPaint& paint) { 821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (!roundRect.isComplex()) { 831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& rect = roundRect.rect(); 841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkVector radii = roundRect.getSimpleRadii(); 851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawRoundRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, 861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger radii.fX, radii.fY, paint); 871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(roundRect); 901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPath(path, paint); 911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPath(const SkPath& path, const SkPaint& paint) { 951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPath(path, paint); 961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmap(const SkBitmap& bitmap, SkScalar left, SkScalar top, 991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint) { 10017c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson SkPixelRef* pxRef = bitmap.pixelRef(); 10117c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson 10217c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson // HWUI doesn't support extractSubset(), so convert any subsetted bitmap into 10317c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson // a drawBitmapRect(); pass through an un-subsetted bitmap. 10417c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson if (pxRef && bitmap.dimensions() != pxRef->info().dimensions()) { 10517c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson SkBitmap fullBitmap; 10617c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson fullBitmap.setInfo(pxRef->info()); 10717c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson fullBitmap.setPixelRef(pxRef, 0, 0); 10817c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson SkIPoint origin = bitmap.pixelRefOrigin(); 10917c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson mCanvas->drawBitmap(fullBitmap, origin.fX, origin.fY, 11017c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson origin.fX + bitmap.dimensions().width(), 11117c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson origin.fY + bitmap.dimensions().height(), 11217c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson left, top, 11317c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson left + bitmap.dimensions().width(), 11417c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson top + bitmap.dimensions().height(), 11517c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson paint); 11617c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson } else { 11717c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson mCanvas->drawBitmap(bitmap, left, top, paint); 11817c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson } 1191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmapRect(const SkBitmap& bitmap, const SkRect* srcPtr, 1221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& dst, const SkPaint* paint, DrawBitmapRectFlags) { 1231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect src = (srcPtr) ? *srcPtr : SkRect::MakeWH(bitmap.width(), bitmap.height()); 12417c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson // TODO: if bitmap is a subset, do we need to add pixelRefOrigin to src? 1251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawBitmap(bitmap, src.fLeft, src.fTop, src.fRight, src.fBottom, 1261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger dst.fLeft, dst.fTop, dst.fRight, dst.fBottom, paint); 1271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1281db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmapNine(const SkBitmap& bitmap, const SkIRect& center, 1301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& dst, const SkPaint*) { 1311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger //TODO make nine-patch drawing a method on Canvas.h 1321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onDrawBitmapNine is not yet supported"); 1331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawSprite(const SkBitmap& bitmap, int left, int top, 1361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint) { 13717c5adfa63872fbb6a903a5941e3c6455995b92dTom Hudson // TODO: if bitmap is a subset, do we need to add pixelRefOrigin to src? 1381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->save(SkCanvas::kMatrixClip_SaveFlag); 1391db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->setMatrix(SkMatrix::I()); 1401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawBitmap(bitmap, left, top, paint); 1411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->restore(); 1421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawVertices(VertexMode mode, int vertexCount, const SkPoint vertices[], 1451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint texs[], const SkColor colors[], SkXfermode*, const uint16_t indices[], 1461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int indexCount, const SkPaint& paint) { 147b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (mFilterHwuiCalls) { 148b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 149b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 1501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert the SkPoints into floats 1511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 1521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const int floatCount = vertexCount << 1; 1531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* vArray = &vertices[0].fX; 1541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* tArray = (texs) ? &texs[0].fX : NULL; 1551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const int* cArray = (colors) ? (int*)colors : NULL; 1561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawVertices(mode, floatCount, vArray, tArray, cArray, indices, indexCount, paint); 1571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek SollenbergerSkSurface* SkiaCanvasProxy::onNewSurface(const SkImageInfo&, const SkSurfaceProps&) { 1601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onNewSurface is not supported"); 1611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger return NULL; 1621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::willSave() { 1651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->save(SkCanvas::kMatrixClip_SaveFlag); 1661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek SollenbergerSkCanvas::SaveLayerStrategy SkiaCanvasProxy::willSaveLayer(const SkRect* rectPtr, 1691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint, SaveFlags flags) { 1701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect rect; 1711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (rectPtr) { 1721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger rect = *rectPtr; 1731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else if(!mCanvas->getClipBounds(&rect)) { 1741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger rect = SkRect::MakeEmpty(); 1751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 1761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->saveLayer(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint, flags); 1771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger return SkCanvas::kNoLayer_SaveLayerStrategy; 1781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::willRestore() { 1811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->restore(); 1821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::didConcat(const SkMatrix& matrix) { 1851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->concat(matrix); 1861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::didSetMatrix(const SkMatrix& matrix) { 1891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->setMatrix(matrix); 1901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawDRRect(const SkRRect& outer, const SkRRect& inner, 1931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 1941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 1951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(outer); 1961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(inner); 1971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.setFillType(SkPath::kEvenOdd_FillType); 1981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawPath(path, paint); 1991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 2001db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2011db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger/** 2021db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * Utility class that converts the incoming text & paint from the given encoding 2031db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * into glyphIDs. 2041db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger */ 2051db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerclass GlyphIDConverter { 2061db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerpublic: 2071db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter(const void* text, size_t byteLength, const SkPaint& origPaint) { 2081db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint = origPaint; 2091db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (paint.getTextEncoding() == SkPaint::kGlyphID_TextEncoding) { 2101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphIDs = (uint16_t*)text; 2111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger count = byteLength >> 1; 2121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2131db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger storage.reset(byteLength); // ensures space for one glyph per ID given UTF8 encoding. 2141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphIDs = storage.get(); 2151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger count = paint.textToGlyphs(text, byteLength, storage.get()); 2161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding); 2171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPaint paint; 2211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger uint16_t* glyphIDs; 2221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int count; 2231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerprivate: 2241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, uint16_t> storage; 2251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; 2261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawText(const void* text, size_t byteLength, SkScalar x, SkScalar y, 2281db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& origPaint) { 2291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 2301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 2311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute the glyph positions 2331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> pointStorage(glyphs.count); 2341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkScalar> glyphWidths(glyphs.count); 2351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.getTextWidths(glyphs.glyphIDs, glyphs.count << 1, glyphWidths.get()); 2361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2371db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute conservative bounds 2381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // NOTE: We could call the faster paint.getFontBounds for a less accurate, 2391db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // but even more conservative bounds if this is too slow. 2401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect bounds; 2411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.measureText(glyphs.glyphIDs, glyphs.count << 1, &bounds); 2421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // adjust for non-left alignment 2441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.getTextAlign() != SkPaint::kLeft_Align) { 2451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkScalar stop = 0; 2461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 0; i < glyphs.count; i++) { 2471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger stop += glyphWidths[i]; 2481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.getTextAlign() == SkPaint::kCenter_Align) { 2501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger stop = SkScalarHalf(stop); 2511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.isVerticalText()) { 2531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y -= stop; 2541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x -= stop; 2561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // setup the first glyph position and adjust bounds if needed 260806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson int xBaseline = 0; 261806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson int yBaseline = 0; 2621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (mCanvas->drawTextAbsolutePos()) { 2631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger bounds.offset(x,y); 264806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson xBaseline = x; 265806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson yBaseline = y; 2661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 267806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[0].set(xBaseline, yBaseline); 2681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // setup the remaining glyph positions 2701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.isVerticalText()) { 2711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 1; i < glyphs.count; i++) { 272806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[i].set(xBaseline, glyphWidths[i-1] + pointStorage[i-1].fY); 2731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 1; i < glyphs.count; i++) { 276806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[i].set(glyphWidths[i-1] + pointStorage[i-1].fX, yBaseline); 2771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 2811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawText(glyphs.glyphIDs, &pointStorage[0].fX, glyphs.count, glyphs.paint, 2821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x, y, bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom, 0); 2831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 2841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPosText(const void* text, size_t byteLength, const SkPoint pos[], 2861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& origPaint) { 2871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 2881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 2891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to relative positions if necessary 2911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int x, y; 2921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint* posArray; 2931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> pointStorage; 2941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (mCanvas->drawTextAbsolutePos()) { 2951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x = 0; 2961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y = 0; 2971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger posArray = pos; 2981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x = pos[0].fX; 3001db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y = pos[0].fY; 3011db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger posArray = pointStorage.reset(glyphs.count); 3021db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 0; i < glyphs.count; i++) { 3031db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pointStorage[i].fX = pos[i].fX- x; 3041db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pointStorage[i].fY = pos[i].fY- y; 3051db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3061db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3071db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3081db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute conservative bounds 3091db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // NOTE: We could call the faster paint.getFontBounds for a less accurate, 3101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // but even more conservative bounds if this is too slow. 3111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect bounds; 3121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.measureText(glyphs.glyphIDs, glyphs.count << 1, &bounds); 31320c2b3ef74ac96fee5ef1daa7959c434d6caeb97Tom Hudson bounds.offset(x, y); 3141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 3161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawText(glyphs.glyphIDs, &posArray[0].fX, glyphs.count, glyphs.paint, x, y, 3171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom, 0); 3181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPosTextH(const void* text, size_t byteLength, const SkScalar xpos[], 3211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkScalar constY, const SkPaint& paint) { 3221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const size_t pointCount = byteLength >> 1; 3231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> storage(pointCount); 3241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPoint* pts = storage.get(); 3251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (size_t i = 0; i < pointCount; i++) { 3261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pts[i].set(xpos[i], constY); 3271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3281db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->onDrawPosText(text, byteLength, pts, paint); 3291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawTextOnPath(const void* text, size_t byteLength, const SkPath& path, 3321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkMatrix* matrix, const SkPaint& origPaint) { 3331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 3341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 3351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawTextOnPath(glyphs.glyphIDs, glyphs.count, path, 0, 0, glyphs.paint); 3361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3371db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y, 3391db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 3401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onDrawTextBlob is not supported"); 3411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPatch(const SkPoint cubics[12], const SkColor colors[4], 3441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint texCoords[4], SkXfermode* xmode, const SkPaint& paint) { 345b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (mFilterHwuiCalls) { 346b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 347b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 3481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPatchUtils::VertexData data; 3491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkMatrix matrix; 3511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->getMatrix(&matrix); 3521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkISize lod = SkPatchUtils::GetLevelOfDetail(cubics, &matrix); 3531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // It automatically adjusts lodX and lodY in case it exceeds the number of indices. 3551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // If it fails to generate the vertices, then we do not draw. 3561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (SkPatchUtils::getVertexData(&data, cubics, colors, texCoords, lod.width(), lod.height())) { 3571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawVertices(SkCanvas::kTriangles_VertexMode, data.fVertexCount, data.fPoints, 3581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger data.fTexCoords, data.fColors, xmode, data.fIndices, data.fIndexCount, 3591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint); 3601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRect(const SkRect& rect, SkRegion::Op op, ClipEdgeStyle) { 3641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, op); 3651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRRect(const SkRRect& roundRect, SkRegion::Op op, ClipEdgeStyle) { 3681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 3691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(roundRect); 3701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipPath(&path, op); 3711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipPath(const SkPath& path, SkRegion::Op op, ClipEdgeStyle) { 3741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipPath(&path, op); 3751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRegion(const SkRegion& region, SkRegion::Op op) { 3781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipRegion(®ion, op); 3791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; // namespace uirenderer 3821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; // namespace android 383