SkiaCanvasProxy.cpp revision b1476ae7d515d6c406b4367cfb4ada8ce2d116e1
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> 211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergernamespace android { 231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergernamespace uirenderer { 241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 25b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom HudsonSkiaCanvasProxy::SkiaCanvasProxy(Canvas* canvas, bool filterHwuiCalls) 261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger : INHERITED(canvas->width(), canvas->height()) 27b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson , mCanvas(canvas) 28b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson , mFilterHwuiCalls(filterHwuiCalls) {} 291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPaint(const SkPaint& paint) { 311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPaint(paint); 321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPoints(PointMode pointMode, size_t count, const SkPoint pts[], 351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 36b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (!pts || count == 0) { 37b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 38b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 39b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson 401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert the SkPoints into floats 411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const size_t floatCount = count << 1; 431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* floatArray = &pts[0].fX; 441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger switch (pointMode) { 461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kPoints_PointMode: { 471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPoints(floatArray, floatCount, paint); 481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kLines_PointMode: { 511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawLines(floatArray, floatCount, paint); 521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger case kPolygon_PointMode: { 551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPaint strokedPaint(paint); 561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger strokedPaint.setStyle(SkPaint::kStroke_Style); 571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (size_t i = 0; i < count - 1; i++) { 601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.moveTo(pts[i]); 611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.lineTo(pts[i+1]); 621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawPath(path, strokedPaint); 631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.rewind(); 641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger break; 661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger default: 681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger LOG_ALWAYS_FATAL("Unknown point type"); 691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawOval(const SkRect& rect, const SkPaint& paint) { 731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawOval(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint); 741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawRect(const SkRect& rect, const SkPaint& paint) { 771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint); 781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawRRect(const SkRRect& roundRect, const SkPaint& paint) { 811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (!roundRect.isComplex()) { 821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& rect = roundRect.rect(); 831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkVector radii = roundRect.getSimpleRadii(); 841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawRoundRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, 851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger radii.fX, radii.fY, paint); 861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(roundRect); 891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPath(path, paint); 901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPath(const SkPath& path, const SkPaint& paint) { 941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawPath(path, paint); 951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmap(const SkBitmap& bitmap, SkScalar left, SkScalar top, 981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint) { 991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawBitmap(bitmap, left, top, paint); 1001db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1011db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1021db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmapRect(const SkBitmap& bitmap, const SkRect* srcPtr, 1031db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& dst, const SkPaint* paint, DrawBitmapRectFlags) { 1041db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect src = (srcPtr) ? *srcPtr : SkRect::MakeWH(bitmap.width(), bitmap.height()); 1051db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawBitmap(bitmap, src.fLeft, src.fTop, src.fRight, src.fBottom, 1061db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger dst.fLeft, dst.fTop, dst.fRight, dst.fBottom, paint); 1071db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1081db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1091db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawBitmapNine(const SkBitmap& bitmap, const SkIRect& center, 1101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkRect& dst, const SkPaint*) { 1111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger //TODO make nine-patch drawing a method on Canvas.h 1121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onDrawBitmapNine is not yet supported"); 1131db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawSprite(const SkBitmap& bitmap, int left, int top, 1161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint) { 1171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->save(SkCanvas::kMatrixClip_SaveFlag); 1181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->setMatrix(SkMatrix::I()); 1191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawBitmap(bitmap, left, top, paint); 1201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->restore(); 1211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawVertices(VertexMode mode, int vertexCount, const SkPoint vertices[], 1241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint texs[], const SkColor colors[], SkXfermode*, const uint16_t indices[], 1251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int indexCount, const SkPaint& paint) { 126b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (mFilterHwuiCalls) { 127b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 128b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 1291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert the SkPoints into floats 1301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 1311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const int floatCount = vertexCount << 1; 1321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* vArray = &vertices[0].fX; 1331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const float* tArray = (texs) ? &texs[0].fX : NULL; 1341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const int* cArray = (colors) ? (int*)colors : NULL; 1351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawVertices(mode, floatCount, vArray, tArray, cArray, indices, indexCount, paint); 1361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1371db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek SollenbergerSkSurface* SkiaCanvasProxy::onNewSurface(const SkImageInfo&, const SkSurfaceProps&) { 1391db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onNewSurface is not supported"); 1401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger return NULL; 1411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::willSave() { 1441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->save(SkCanvas::kMatrixClip_SaveFlag); 1451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek SollenbergerSkCanvas::SaveLayerStrategy SkiaCanvasProxy::willSaveLayer(const SkRect* rectPtr, 1481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint* paint, SaveFlags flags) { 1491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect rect; 1501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (rectPtr) { 1511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger rect = *rectPtr; 1521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else if(!mCanvas->getClipBounds(&rect)) { 1531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger rect = SkRect::MakeEmpty(); 1541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 1551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->saveLayer(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, paint, flags); 1561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger return SkCanvas::kNoLayer_SaveLayerStrategy; 1571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::willRestore() { 1601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->restore(); 1611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::didConcat(const SkMatrix& matrix) { 1641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->concat(matrix); 1651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::didSetMatrix(const SkMatrix& matrix) { 1681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->setMatrix(matrix); 1691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawDRRect(const SkRRect& outer, const SkRRect& inner, 1721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 1731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 1741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(outer); 1751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(inner); 1761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.setFillType(SkPath::kEvenOdd_FillType); 1771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawPath(path, paint); 1781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 1791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger/** 1811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * Utility class that converts the incoming text & paint from the given encoding 1821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger * into glyphIDs. 1831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger */ 1841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerclass GlyphIDConverter { 1851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerpublic: 1861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter(const void* text, size_t byteLength, const SkPaint& origPaint) { 1871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint = origPaint; 1881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (paint.getTextEncoding() == SkPaint::kGlyphID_TextEncoding) { 1891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphIDs = (uint16_t*)text; 1901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger count = byteLength >> 1; 1911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 1921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger storage.reset(byteLength); // ensures space for one glyph per ID given UTF8 encoding. 1931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphIDs = storage.get(); 1941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger count = paint.textToGlyphs(text, byteLength, storage.get()); 1951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding); 1961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 1971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 1981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 1991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPaint paint; 2001db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger uint16_t* glyphIDs; 2011db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int count; 2021db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergerprivate: 2031db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, uint16_t> storage; 2041db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; 2051db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2061db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawText(const void* text, size_t byteLength, SkScalar x, SkScalar y, 2071db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& origPaint) { 2081db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 2091db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 2101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute the glyph positions 2121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> pointStorage(glyphs.count); 2131db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkScalar> glyphWidths(glyphs.count); 2141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.getTextWidths(glyphs.glyphIDs, glyphs.count << 1, glyphWidths.get()); 2151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute conservative bounds 2171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // NOTE: We could call the faster paint.getFontBounds for a less accurate, 2181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // but even more conservative bounds if this is too slow. 2191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect bounds; 2201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.measureText(glyphs.glyphIDs, glyphs.count << 1, &bounds); 2211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // adjust for non-left alignment 2231db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.getTextAlign() != SkPaint::kLeft_Align) { 2241db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkScalar stop = 0; 2251db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 0; i < glyphs.count; i++) { 2261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger stop += glyphWidths[i]; 2271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2281db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.getTextAlign() == SkPaint::kCenter_Align) { 2291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger stop = SkScalarHalf(stop); 2301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.isVerticalText()) { 2321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y -= stop; 2331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x -= stop; 2351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2371db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // setup the first glyph position and adjust bounds if needed 239806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson int xBaseline = 0; 240806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson int yBaseline = 0; 2411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (mCanvas->drawTextAbsolutePos()) { 2421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger bounds.offset(x,y); 243806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson xBaseline = x; 244806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson yBaseline = y; 2451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 246806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[0].set(xBaseline, yBaseline); 2471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // setup the remaining glyph positions 2491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (glyphs.paint.isVerticalText()) { 2501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 1; i < glyphs.count; i++) { 251806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[i].set(xBaseline, glyphWidths[i-1] + pointStorage[i-1].fY); 2521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 1; i < glyphs.count; i++) { 255806a6f07a37c16b37631d8707dd1f2b41276fafcTom Hudson pointStorage[i].set(glyphWidths[i-1] + pointStorage[i-1].fX, yBaseline); 2561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 2601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawText(glyphs.glyphIDs, &pointStorage[0].fX, glyphs.count, glyphs.paint, 2611db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x, y, bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom, 0); 2621db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 2631db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2641db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPosText(const void* text, size_t byteLength, const SkPoint pos[], 2651db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& origPaint) { 2661db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 2671db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 2681db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2691db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to relative positions if necessary 2701db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger int x, y; 2711db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint* posArray; 2721db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> pointStorage; 2731db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (mCanvas->drawTextAbsolutePos()) { 2741db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x = 0; 2751db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y = 0; 2761db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger posArray = pos; 2771db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } else { 2781db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger x = pos[0].fX; 2791db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger y = pos[0].fY; 2801db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger posArray = pointStorage.reset(glyphs.count); 2811db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (int i = 0; i < glyphs.count; i++) { 2821db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pointStorage[i].fX = pos[i].fX- x; 2831db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pointStorage[i].fY = pos[i].fY- y; 2841db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2851db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 2861db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2871db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // compute conservative bounds 2881db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // NOTE: We could call the faster paint.getFontBounds for a less accurate, 2891db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // but even more conservative bounds if this is too slow. 2901db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkRect bounds; 2911db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger glyphs.paint.measureText(glyphs.glyphIDs, glyphs.count << 1, &bounds); 2921db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2931db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SK_COMPILE_ASSERT(sizeof(SkPoint) == sizeof(float)*2, SkPoint_is_no_longer_2_floats); 2941db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawText(glyphs.glyphIDs, &posArray[0].fX, glyphs.count, glyphs.paint, x, y, 2951db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom, 0); 2961db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 2971db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 2981db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPosTextH(const void* text, size_t byteLength, const SkScalar xpos[], 2991db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkScalar constY, const SkPaint& paint) { 3001db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const size_t pointCount = byteLength >> 1; 3011db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkAutoSTMalloc<32, SkPoint> storage(pointCount); 3021db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPoint* pts = storage.get(); 3031db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger for (size_t i = 0; i < pointCount; i++) { 3041db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger pts[i].set(xpos[i], constY); 3051db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3061db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->onDrawPosText(text, byteLength, pts, paint); 3071db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3081db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3091db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawTextOnPath(const void* text, size_t byteLength, const SkPath& path, 3101db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkMatrix* matrix, const SkPaint& origPaint) { 3111db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // convert to glyphIDs if necessary 3121db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger GlyphIDConverter glyphs(text, byteLength, origPaint); 3131db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->drawTextOnPath(glyphs.glyphIDs, glyphs.count, path, 0, 0, glyphs.paint); 3141db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3151db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3161db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y, 3171db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPaint& paint) { 3181db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkDEBUGFAIL("SkiaCanvasProxy::onDrawTextBlob is not supported"); 3191db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3201db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3211db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onDrawPatch(const SkPoint cubics[12], const SkColor colors[4], 3221db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger const SkPoint texCoords[4], SkXfermode* xmode, const SkPaint& paint) { 323b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson if (mFilterHwuiCalls) { 324b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson return; 325b1476ae7d515d6c406b4367cfb4ada8ce2d116e1Tom Hudson } 3261db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPatchUtils::VertexData data; 3271db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3281db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkMatrix matrix; 3291db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->getMatrix(&matrix); 3301db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkISize lod = SkPatchUtils::GetLevelOfDetail(cubics, &matrix); 3311db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3321db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // It automatically adjusts lodX and lodY in case it exceeds the number of indices. 3331db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger // If it fails to generate the vertices, then we do not draw. 3341db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger if (SkPatchUtils::getVertexData(&data, cubics, colors, texCoords, lod.width(), lod.height())) { 3351db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger this->drawVertices(SkCanvas::kTriangles_VertexMode, data.fVertexCount, data.fPoints, 3361db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger data.fTexCoords, data.fColors, xmode, data.fIndices, data.fIndexCount, 3371db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger paint); 3381db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger } 3391db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3401db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3411db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRect(const SkRect& rect, SkRegion::Op op, ClipEdgeStyle) { 3421db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipRect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, op); 3431db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3441db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3451db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRRect(const SkRRect& roundRect, SkRegion::Op op, ClipEdgeStyle) { 3461db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger SkPath path; 3471db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger path.addRRect(roundRect); 3481db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipPath(&path, op); 3491db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3501db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3511db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipPath(const SkPath& path, SkRegion::Op op, ClipEdgeStyle) { 3521db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipPath(&path, op); 3531db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3541db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3551db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenbergervoid SkiaCanvasProxy::onClipRegion(const SkRegion& region, SkRegion::Op op) { 3561db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger mCanvas->clipRegion(®ion, op); 3571db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger} 3581db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger 3591db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; // namespace uirenderer 3601db141f93c4fe79a4669440c3d14f63bc87b2e34Derek Sollenberger}; // namespace android 361