RecordingCanvas.cpp revision d2dfd8f128b632ed99418ab2b32949c939a9a369
1b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik/* 2b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Copyright (C) 2015 The Android Open Source Project 3b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * 4b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Licensed under the Apache License, Version 2.0 (the "License"); 5b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * you may not use this file except in compliance with the License. 6b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * You may obtain a copy of the License at 7b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * 8b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * http://www.apache.org/licenses/LICENSE-2.0 9b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * 10b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Unless required by applicable law or agreed to in writing, software 11b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * distributed under the License is distributed on an "AS IS" BASIS, 12b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * See the License for the specific language governing permissions and 14b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * limitations under the License. 15b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik */ 16b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 17b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include "RecordingCanvas.h" 18b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 19d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik#include "DeferredLayerUpdater.h" 20b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include "RecordedOp.h" 21b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include "RenderNode.h" 22b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 23b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace android { 24b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace uirenderer { 25b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 26b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris CraikRecordingCanvas::RecordingCanvas(size_t width, size_t height) 27b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik : mState(*this) 28b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik , mResourceCache(ResourceCache::getInstance()) { 29b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik reset(width, height); 30b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 31b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 32b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris CraikRecordingCanvas::~RecordingCanvas() { 33003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik LOG_ALWAYS_FATAL_IF(mDisplayList, 34b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik "Destroyed a RecordingCanvas during a record!"); 35b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 36b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 37b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::reset(int width, int height) { 38003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik LOG_ALWAYS_FATAL_IF(mDisplayList, 39b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik "prepareDirty called a second time during a recording!"); 40003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList = new DisplayList(); 41b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 42b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.initializeSaveStack(width, height, 0, 0, width, height, Vector3()); 43b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 44161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik mDeferredBarrierType = DeferredBarrierType::InOrder; 45b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.setDirtyClip(false); 46b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mRestoreSaveCount = -1; 47b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 48b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 49003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris CraikDisplayList* RecordingCanvas::finishRecording() { 50b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mPaintMap.clear(); 51b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mRegionMap.clear(); 52b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mPathMap.clear(); 53003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik DisplayList* displayList = mDisplayList; 54003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList = nullptr; 55b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mSkiaCanvasProxy.reset(nullptr); 56003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik return displayList; 57b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 58b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 59b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris CraikSkCanvas* RecordingCanvas::asSkCanvas() { 60003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik LOG_ALWAYS_FATAL_IF(!mDisplayList, 61b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik "attempting to get an SkCanvas when we are not recording!"); 62b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (!mSkiaCanvasProxy) { 63b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mSkiaCanvasProxy.reset(new SkiaCanvasProxy(this)); 64b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 65b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 66b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // SkCanvas instances default to identity transform, but should inherit 67b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // the state of this Canvas; if this code was in the SkiaCanvasProxy 68b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // constructor, we couldn't cache mSkiaCanvasProxy. 69b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkMatrix parentTransform; 70b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik getMatrix(&parentTransform); 71b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mSkiaCanvasProxy.get()->setMatrix(parentTransform); 72b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 73b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mSkiaCanvasProxy.get(); 74b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 75b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 76b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// ---------------------------------------------------------------------------- 776fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik// CanvasStateClient implementation 786fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik// ---------------------------------------------------------------------------- 796fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 806fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craikvoid RecordingCanvas::onViewportInitialized() { 816fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik} 826fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 836fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craikvoid RecordingCanvas::onSnapshotRestored(const Snapshot& removed, const Snapshot& restored) { 846fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik if (removed.flags & Snapshot::kFlagIsFboLayer) { 856fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik addOp(new (alloc()) EndLayerOp()); 866fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik } 876fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik} 886fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 896fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik// ---------------------------------------------------------------------------- 90b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// android/graphics/Canvas state operations 91b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// ---------------------------------------------------------------------------- 92b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Save (layer) 93b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikint RecordingCanvas::save(SkCanvas::SaveFlags flags) { 94b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.save((int) flags); 95b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 96b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 97b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::RecordingCanvas::restore() { 98b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (mRestoreSaveCount < 0) { 99b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik restoreToCount(getSaveCount() - 1); 100b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return; 101b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 102b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 103b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mRestoreSaveCount--; 104b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.restore(); 105b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 106b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 107b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::restoreToCount(int saveCount) { 108b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mRestoreSaveCount = saveCount; 109b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.restoreToCount(saveCount); 110b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 111b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 112b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikint RecordingCanvas::saveLayer(float left, float top, float right, float bottom, const SkPaint* paint, 113b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkCanvas::SaveFlags flags) { 1146fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik if (!(flags & SkCanvas::kClipToLayer_SaveFlag)) { 1156fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik LOG_ALWAYS_FATAL("unclipped layers not supported"); 1166fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik } 1176fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // force matrix/clip isolation for layer 1186fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik flags |= SkCanvas::kClip_SaveFlag | SkCanvas::kMatrix_SaveFlag; 1196fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1206fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1216fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik const Snapshot& previous = *mState.currentSnapshot(); 1226fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1236fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // initialize the snapshot as though it almost represents an FBO layer so deferred draw 1246fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // operations will be able to store and restore the current clip and transform info, and 1256fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // quick rejection will be correct (for display lists) 1266fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1276fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik const Rect untransformedBounds(left, top, right, bottom); 1286fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1296fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // determine clipped bounds relative to previous viewport. 1306fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Rect visibleBounds = untransformedBounds; 1316fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik previous.transform->mapRect(visibleBounds); 1326fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1336fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1346fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik visibleBounds.doIntersect(previous.getRenderTargetClip()); 1356fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik visibleBounds.snapToPixelBoundaries(); 1366fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1376fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Rect previousViewport(0, 0, previous.getViewportWidth(), previous.getViewportHeight()); 1386fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik visibleBounds.doIntersect(previousViewport); 1396fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1406fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // Map visible bounds back to layer space, and intersect with parameter bounds 1416fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Rect layerBounds = visibleBounds; 1426fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Matrix4 inverse; 1436fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik inverse.loadInverse(*previous.transform); 1446fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik inverse.mapRect(layerBounds); 1456fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik layerBounds.doIntersect(untransformedBounds); 1466fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1476fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik int saveValue = mState.save((int) flags); 1486fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Snapshot& snapshot = *mState.writableSnapshot(); 1496fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1506fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // layerBounds is now original bounds, but with clipped to clip 1516fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // and viewport to ensure it's minimal size. 1526fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik if (layerBounds.isEmpty() || untransformedBounds.isEmpty()) { 1536fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik // Don't bother recording layer, since it's been rejected 1546fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik snapshot.resetClip(0, 0, 0, 0); 1556fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik return saveValue; 1566fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik } 1576fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1586fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik snapshot.flags |= Snapshot::kFlagFboTarget | Snapshot::kFlagIsFboLayer; 1596fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik snapshot.initializeViewport(untransformedBounds.getWidth(), untransformedBounds.getHeight()); 16098787e6c9b2c10b1ab7820bdac168686025b924aChris Craik snapshot.transform->loadTranslate(-untransformedBounds.left, -untransformedBounds.top, 0.0f); 1616fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1626fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Rect clip = layerBounds; 1636fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik clip.translate(-untransformedBounds.left, -untransformedBounds.top); 1646fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik snapshot.resetClip(clip.left, clip.top, clip.right, clip.bottom); 1656fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik snapshot.roundRectClipState = nullptr; 1666fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1676fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik addOp(new (alloc()) BeginLayerOp( 1686fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik Rect(left, top, right, bottom), 1696fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik *previous.transform, // transform to *draw* with 1706fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik previous.getRenderTargetClip(), // clip to *draw* with 1716fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik refPaint(paint))); 1726fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik 1736fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik return saveValue; 174b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 175b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 176b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Matrix 177b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::rotate(float degrees) { 178b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (degrees == 0) return; 179b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 180b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.rotate(degrees); 181b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 182b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 183b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::scale(float sx, float sy) { 184b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (sx == 1 && sy == 1) return; 185b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 186b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.scale(sx, sy); 187b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 188b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 189b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::skew(float sx, float sy) { 190b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.skew(sx, sy); 191b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 192b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 193b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::translate(float dx, float dy) { 194b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (dx == 0 && dy == 0) return; 195b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 196b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.translate(dx, dy, 0); 197b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 198b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 199b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Clip 200b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::getClipBounds(SkRect* outRect) const { 201e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik *outRect = mState.getLocalClipBounds().toSkRect(); 202b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return !(outRect->isEmpty()); 203b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 204b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::quickRejectRect(float left, float top, float right, float bottom) const { 205b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.quickRejectConservative(left, top, right, bottom); 206b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 207b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::quickRejectPath(const SkPath& path) const { 208b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkRect bounds = path.getBounds(); 209b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.quickRejectConservative(bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom); 210b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 211b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::clipRect(float left, float top, float right, float bottom, SkRegion::Op op) { 212b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.clipRect(left, top, right, bottom, op); 213b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 214b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::clipPath(const SkPath* path, SkRegion::Op op) { 215b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.clipPath(path, op); 216b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 217b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikbool RecordingCanvas::clipRegion(const SkRegion* region, SkRegion::Op op) { 218b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return mState.clipRegion(region, op); 219b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 220b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 221b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// ---------------------------------------------------------------------------- 222b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// android/graphics/Canvas draw operations 223b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// ---------------------------------------------------------------------------- 224b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawColor(int color, SkXfermode::Mode mode) { 225b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkPaint paint; 226b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik paint.setColor(color); 227b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik paint.setXfermodeMode(mode); 228b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawPaint(paint); 229b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 230b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 231b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawPaint(const SkPaint& paint) { 232b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // TODO: more efficient recording? 233b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik addOp(new (alloc()) RectOp( 234b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 235a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik Matrix4::identity(), 236b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 237b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refPaint(&paint))); 238b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 239b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 240386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craikstatic Rect calcBoundsOfPoints(const float* points, int floatCount) { 241386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik Rect unmappedBounds(points[0], points[1], points[0], points[1]); 242386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik for (int i = 2; i < floatCount; i += 2) { 243386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik unmappedBounds.expandToCover(points[i], points[i + 1]); 244386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik } 245386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik return unmappedBounds; 246386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik} 247386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 248b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Geometry 249386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craikvoid RecordingCanvas::drawPoints(const float* points, int floatCount, const SkPaint& paint) { 250386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik if (floatCount < 2) return; 251386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik floatCount &= ~0x1; // round down to nearest two 252386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 253386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik addOp(new (alloc()) PointsOp( 254386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik calcBoundsOfPoints(points, floatCount), 255386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik *mState.currentSnapshot()->transform, 256386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik mState.getRenderTargetClipBounds(), 257386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik refPaint(&paint), refBuffer<float>(points, floatCount), floatCount)); 258b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 259a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik 260a1717271caac5e8ea3808c331d4141ac01a42134Chris Craikvoid RecordingCanvas::drawLines(const float* points, int floatCount, const SkPaint& paint) { 261a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik if (floatCount < 4) return; 262a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik floatCount &= ~0x3; // round down to nearest four 263a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik 264a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik addOp(new (alloc()) LinesOp( 265386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik calcBoundsOfPoints(points, floatCount), 266a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik *mState.currentSnapshot()->transform, 267a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik mState.getRenderTargetClipBounds(), 268a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik refPaint(&paint), refBuffer<float>(points, floatCount), floatCount)); 269b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 270a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik 271b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawRect(float left, float top, float right, float bottom, const SkPaint& paint) { 272b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik addOp(new (alloc()) RectOp( 273b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Rect(left, top, right, bottom), 274b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *(mState.currentSnapshot()->transform), 275b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 276b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refPaint(&paint))); 277b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 278b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 279b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawSimpleRects(const float* rects, int vertexCount, const SkPaint* paint) { 280b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (rects == nullptr) return; 281b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 282003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik Vertex* rectData = (Vertex*) mDisplayList->allocator.alloc(vertexCount * sizeof(Vertex)); 283b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vertex* vertex = rectData; 284b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 285b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float left = FLT_MAX; 286b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float top = FLT_MAX; 287b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float right = FLT_MIN; 288b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float bottom = FLT_MIN; 289b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik for (int index = 0; index < vertexCount; index += 4) { 290b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float l = rects[index + 0]; 291b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float t = rects[index + 1]; 292b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float r = rects[index + 2]; 293b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float b = rects[index + 3]; 294b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 295b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vertex::set(vertex++, l, t); 296b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vertex::set(vertex++, r, t); 297b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vertex::set(vertex++, l, b); 298b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vertex::set(vertex++, r, b); 299b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 300b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik left = std::min(left, l); 301b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik top = std::min(top, t); 302b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik right = std::max(right, r); 303b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik bottom = std::max(bottom, b); 304b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 305b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik addOp(new (alloc()) SimpleRectsOp( 306b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Rect(left, top, right, bottom), 307b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *(mState.currentSnapshot()->transform), 308b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 309b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refPaint(paint), rectData, vertexCount)); 310b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 311b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 312b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawRegion(const SkRegion& region, const SkPaint& paint) { 313b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (paint.getStyle() == SkPaint::kFill_Style 314b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && (!paint.isAntiAlias() || mState.currentTransform()->isSimple())) { 315b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik int count = 0; 316b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik Vector<float> rects; 317b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkRegion::Iterator it(region); 318b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik while (!it.done()) { 319b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const SkIRect& r = it.rect(); 320b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik rects.push(r.fLeft); 321b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik rects.push(r.fTop); 322b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik rects.push(r.fRight); 323b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik rects.push(r.fBottom); 324b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik count += 4; 325b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik it.next(); 326b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 327b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawSimpleRects(rects.array(), count, &paint); 328b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } else { 329b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkRegion::Iterator it(region); 330b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik while (!it.done()) { 331b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const SkIRect& r = it.rect(); 332b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawRect(r.fLeft, r.fTop, r.fRight, r.fBottom, paint); 333b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik it.next(); 334b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 335b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 336b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 337b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawRoundRect(float left, float top, float right, float bottom, 338b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float rx, float ry, const SkPaint& paint) { 339386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik addOp(new (alloc()) RoundRectOp( 340386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik Rect(left, top, right, bottom), 341386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik *(mState.currentSnapshot()->transform), 342386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik mState.getRenderTargetClipBounds(), 343386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik refPaint(&paint), rx, ry)); 344b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 345386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 346268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craikvoid RecordingCanvas::drawRoundRect( 347268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPrimitive* left, CanvasPropertyPrimitive* top, 348268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPrimitive* right, CanvasPropertyPrimitive* bottom, 349268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPrimitive* rx, CanvasPropertyPrimitive* ry, 350268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPaint* paint) { 351268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(left); 352268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(top); 353268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(right); 354268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(bottom); 355268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(rx); 356268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(ry); 357268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(paint); 358268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik refBitmapsInShader(paint->value.getShader()); 359268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik addOp(new (alloc()) RoundRectPropsOp( 360268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik *(mState.currentSnapshot()->transform), 361268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mState.getRenderTargetClipBounds(), 362268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik &paint->value, 363268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik &left->value, &top->value, &right->value, &bottom->value, 364268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik &rx->value, &ry->value)); 365268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik} 366268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik 367b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawCircle(float x, float y, float radius, const SkPaint& paint) { 368268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik // TODO: move to Canvas.h 369386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik if (radius <= 0) return; 370386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik drawOval(x - radius, y - radius, x + radius, y + radius, paint); 371b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 372386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 373268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craikvoid RecordingCanvas::drawCircle( 374268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y, 375268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) { 376268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(x); 377268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(y); 378268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(radius); 379268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mDisplayList->ref(paint); 380268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik refBitmapsInShader(paint->value.getShader()); 381268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik addOp(new (alloc()) CirclePropsOp( 382268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik *(mState.currentSnapshot()->transform), 383268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik mState.getRenderTargetClipBounds(), 384268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik &paint->value, 385268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik &x->value, &y->value, &radius->value)); 386268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik} 387268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik 388268a9c0f29c16a64d5819c7dbe8b0633baedab83Chris Craik 389b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawOval(float left, float top, float right, float bottom, const SkPaint& paint) { 390386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik addOp(new (alloc()) OvalOp( 391386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik Rect(left, top, right, bottom), 392386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik *(mState.currentSnapshot()->transform), 393386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik mState.getRenderTargetClipBounds(), 394386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik refPaint(&paint))); 395b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 396386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 397b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawArc(float left, float top, float right, float bottom, 398386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik float startAngle, float sweepAngle, bool useCenter, const SkPaint& paint) { 399386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik addOp(new (alloc()) ArcOp( 400386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik Rect(left, top, right, bottom), 401386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik *(mState.currentSnapshot()->transform), 402386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik mState.getRenderTargetClipBounds(), 403386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik refPaint(&paint), 404386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik startAngle, sweepAngle, useCenter)); 405b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 406386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 407b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawPath(const SkPath& path, const SkPaint& paint) { 408386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik addOp(new (alloc()) PathOp( 409386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik Rect(path.getBounds()), 410386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik *(mState.currentSnapshot()->transform), 411386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik mState.getRenderTargetClipBounds(), 412386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik refPaint(&paint), refPath(&path))); 413b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 414b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 415b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Bitmap-based 416b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawBitmap(const SkBitmap& bitmap, float left, float top, const SkPaint* paint) { 417b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik save(SkCanvas::kMatrix_SaveFlag); 418b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik translate(left, top); 419b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawBitmap(&bitmap, paint); 420b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik restore(); 421b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 422b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 423b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawBitmap(const SkBitmap& bitmap, const SkMatrix& matrix, 424b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const SkPaint* paint) { 425b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (matrix.isIdentity()) { 426b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawBitmap(&bitmap, paint); 427b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } else if (!(matrix.getType() & ~(SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask)) 428b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && MathUtils::isPositive(matrix.getScaleX()) 429b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && MathUtils::isPositive(matrix.getScaleY())) { 430b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // SkMatrix::isScaleTranslate() not available in L 431b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkRect src; 432b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkRect dst; 433b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik bitmap.getBounds(&src); 434b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik matrix.mapRect(&dst, src); 435b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawBitmap(bitmap, src.fLeft, src.fTop, src.fRight, src.fBottom, 436b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik dst.fLeft, dst.fTop, dst.fRight, dst.fBottom, paint); 437b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } else { 438b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik save(SkCanvas::kMatrix_SaveFlag); 439b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik concat(matrix); 440b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawBitmap(&bitmap, paint); 441b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik restore(); 442b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 443b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 444386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 445b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawBitmap(const SkBitmap& bitmap, float srcLeft, float srcTop, 446b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float srcRight, float srcBottom, float dstLeft, float dstTop, 447b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float dstRight, float dstBottom, const SkPaint* paint) { 448b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (srcLeft == 0 && srcTop == 0 449b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && srcRight == bitmap.width() 450b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && srcBottom == bitmap.height() 451b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && (srcBottom - srcTop == dstBottom - dstTop) 452b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik && (srcRight - srcLeft == dstRight - dstLeft)) { 453b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // transform simple rect to rect drawing case into position bitmap ops, since they merge 454b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik save(SkCanvas::kMatrix_SaveFlag); 455b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik translate(dstLeft, dstTop); 456b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik drawBitmap(&bitmap, paint); 457b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik restore(); 458b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } else { 459f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik addOp(new (alloc()) BitmapRectOp( 460f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik Rect(dstLeft, dstTop, dstRight, dstBottom), 461f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik *(mState.currentSnapshot()->transform), 462f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik mState.getRenderTargetClipBounds(), 463f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik refPaint(paint), refBitmap(bitmap), 464f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik Rect(srcLeft, srcTop, srcRight, srcBottom))); 465b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 466b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 467386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 468b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawBitmapMesh(const SkBitmap& bitmap, int meshWidth, int meshHeight, 469b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const float* vertices, const int* colors, const SkPaint* paint) { 470f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik int vertexCount = (meshWidth + 1) * (meshHeight + 1); 471f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik addOp(new (alloc()) BitmapMeshOp( 472f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik calcBoundsOfPoints(vertices, vertexCount * 2), 473f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik *(mState.currentSnapshot()->transform), 474f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik mState.getRenderTargetClipBounds(), 475f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik refPaint(paint), refBitmap(bitmap), meshWidth, meshHeight, 476f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik refBuffer<float>(vertices, vertexCount * 2), // 2 floats per vertex 477f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik refBuffer<int>(colors, vertexCount))); // 1 color per vertex 478b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 479386aa031793bb037ec43b6cdbd8908c343cc86cbChris Craik 480f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craikvoid RecordingCanvas::drawNinePatch(const SkBitmap& bitmap, const android::Res_png_9patch& patch, 481b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float dstLeft, float dstTop, float dstRight, float dstBottom, 482b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const SkPaint* paint) { 483f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik addOp(new (alloc()) PatchOp( 484f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik Rect(dstLeft, dstTop, dstRight, dstBottom), 485f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik *(mState.currentSnapshot()->transform), 486f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik mState.getRenderTargetClipBounds(), 487f09ff5aa57bff01cb17595fb7ca8e48d238a6acdChris Craik refPaint(paint), refBitmap(bitmap), refPatch(&patch))); 488b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 489b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 490b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik// Text 491a1717271caac5e8ea3808c331d4141ac01a42134Chris Craikvoid RecordingCanvas::drawText(const uint16_t* glyphs, const float* positions, int glyphCount, 492b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik const SkPaint& paint, float x, float y, float boundsLeft, float boundsTop, 493b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float boundsRight, float boundsBottom, float totalAdvance) { 494a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik if (!glyphs || !positions || glyphCount <= 0 || PaintUtils::paintWillNotDrawText(paint)) return; 495a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik glyphs = refBuffer<glyph_t>(glyphs, glyphCount); 496a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik positions = refBuffer<float>(positions, glyphCount * 2); 497a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik 49815c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik // TODO: either must account for text shadow in bounds, or record separate ops for text shadows 499a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik addOp(new (alloc()) TextOp( 500a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik Rect(boundsLeft, boundsTop, boundsRight, boundsBottom), 501a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik *(mState.currentSnapshot()->transform), 502a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik mState.getRenderTargetClipBounds(), 503a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik refPaint(&paint), glyphs, positions, glyphCount, x, y)); 504a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik drawTextDecorations(x, y, totalAdvance, paint); 505b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 506a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik 507d7448e65e243754f31890baef29dff187dc2e5e5Chris Craikvoid RecordingCanvas::drawTextOnPath(const uint16_t* glyphs, int glyphCount, const SkPath& path, 508b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik float hOffset, float vOffset, const SkPaint& paint) { 509d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik if (!glyphs || glyphCount <= 0 || PaintUtils::paintWillNotDrawText(paint)) return; 510d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik glyphs = refBuffer<glyph_t>(glyphs, glyphCount); 511d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik addOp(new (alloc()) TextOnPathOp( 512d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik mState.getRenderTargetClipBounds(), // TODO: explicitly define bounds 513d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik *(mState.currentSnapshot()->transform), 514d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik mState.getRenderTargetClipBounds(), 515d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik refPaint(&paint), glyphs, glyphCount, refPath(&path), hOffset, vOffset)); 516b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 517b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 518b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) { 519b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik addOp(new (alloc()) BitmapOp( 5205430ab220b231a96b71c3e030d0303d9ce008b05Chris Craik Rect(bitmap->width(), bitmap->height()), 521b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *(mState.currentSnapshot()->transform), 522b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 523b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refPaint(paint), refBitmap(*bitmap))); 524b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 525e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik 526b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::drawRenderNode(RenderNode* renderNode) { 52754fa17f667c285a5c9225e238c8132dfe830ef36Chris Craik auto&& stagingProps = renderNode->stagingProperties(); 528b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik RenderNodeOp* op = new (alloc()) RenderNodeOp( 52954fa17f667c285a5c9225e238c8132dfe830ef36Chris Craik Rect(stagingProps.getWidth(), stagingProps.getHeight()), 530b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *(mState.currentSnapshot()->transform), 531b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik mState.getRenderTargetClipBounds(), 532b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik renderNode); 533b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik int opIndex = addOp(op); 534003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik int childIndex = mDisplayList->addChild(op); 535b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 536b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // update the chunk's child indices 537003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik DisplayList::Chunk& chunk = mDisplayList->chunks.back(); 538b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik chunk.endChildIndex = childIndex + 1; 539b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 540b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (renderNode->stagingProperties().isProjectionReceiver()) { 541b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // use staging property, since recording on UI thread 542003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList->projectionReceiveIndex = opIndex; 543b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 544b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 545b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 546d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craikvoid RecordingCanvas::drawLayer(DeferredLayerUpdater* layerHandle) { 547d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik // We ref the DeferredLayerUpdater due to its thread-safe ref-counting semantics. 548d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik mDisplayList->ref(layerHandle); 549d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik 550d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik Layer* layer = layerHandle->backingLayer(); 551d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik Matrix4 totalTransform(*(mState.currentSnapshot()->transform)); 552d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik totalTransform.multiply(layer->getTransform()); 553d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik 554d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik addOp(new (alloc()) TextureLayerOp( 555d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik Rect(layer->getWidth(), layer->getHeight()), 556d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik totalTransform, 557d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik mState.getRenderTargetClipBounds(), 558d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik layer)); 559d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik} 560d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik 561e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craikvoid RecordingCanvas::callDrawGLFunction(Functor* functor) { 562e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik mDisplayList->functors.push_back(functor); 563e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik addOp(new (alloc()) FunctorOp( 564e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik mState.getRenderTargetClipBounds(), // TODO: explicitly define bounds 565e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik *(mState.currentSnapshot()->transform), 566e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik mState.getRenderTargetClipBounds(), 567e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik functor)); 568e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik} 569e29ce6f51d681af7649c0a7cddee97c471e43eb5Chris Craik 570b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiksize_t RecordingCanvas::addOp(RecordedOp* op) { 571b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // TODO: validate if "addDrawOp" quickrejection logic is useful before adding 572003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik int insertIndex = mDisplayList->ops.size(); 573003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList->ops.push_back(op); 574161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik if (mDeferredBarrierType != DeferredBarrierType::None) { 575b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // op is first in new chunk 576003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList->chunks.emplace_back(); 577003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik DisplayList::Chunk& newChunk = mDisplayList->chunks.back(); 578b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik newChunk.beginOpIndex = insertIndex; 579b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik newChunk.endOpIndex = insertIndex + 1; 580161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik newChunk.reorderChildren = (mDeferredBarrierType == DeferredBarrierType::OutOfOrder); 581b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 582b36af87f8275f4b982906f88193ec27600f2746aChris Craik int nextChildIndex = mDisplayList->children.size(); 583b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex; 584161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik mDeferredBarrierType = DeferredBarrierType::None; 585b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } else { 586b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // standard case - append to existing chunk 587003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik mDisplayList->chunks.back().endOpIndex = insertIndex + 1; 588b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 589b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return insertIndex; 590b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 591b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 592b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikvoid RecordingCanvas::refBitmapsInShader(const SkShader* shader) { 593b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (!shader) return; 594b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 595b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // If this paint has an SkShader that has an SkBitmap add 596b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik // it to the bitmap pile 597b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkBitmap bitmap; 598b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkShader::TileMode xy[2]; 599f35b989d26bb98900f6c5fa2e586326b30b6e161Leon Scroggins III if (shader->isABitmap(&bitmap, nullptr, xy)) { 600b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refBitmap(bitmap); 601b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return; 602b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 603b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik SkShader::ComposeRec rec; 604b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik if (shader->asACompose(&rec)) { 605b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refBitmapsInShader(rec.fShaderA); 606b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik refBitmapsInShader(rec.fShaderB); 607b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik return; 608b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik } 609b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} 610b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik 611b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik}; // namespace uirenderer 612b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik}; // namespace android 613