blob: 5a132936757615d46383e1c5aaff938f1701a097 [file] [log] [blame]
Romain Guy4aa90572010-09-26 18:40:37 -07001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "OpenGLRenderer"
18
Romain Guyd5a85fb2012-03-13 11:18:20 -070019#include <SkCamera.h>
Romain Guyc46d07a2013-03-15 19:06:39 -070020#include <SkCanvas.h>
Chet Haase9c1e23b2011-03-24 10:51:31 -070021
Romain Guy65549432012-03-26 16:45:05 -070022#include <private/hwui/DrawGlInfo.h>
23
John Recka35778c72014-11-06 09:45:10 -080024#include "ResourceCache.h"
Chris Craikc3566d02013-02-04 16:16:33 -080025#include "DeferredDisplayList.h"
John Reck12f5e342014-11-07 07:53:43 -080026#include "DeferredLayerUpdater.h"
Chet Haase9c1e23b2011-03-24 10:51:31 -070027#include "DisplayListLogBuffer.h"
Chris Craik2af46352012-11-26 18:30:17 -080028#include "DisplayListOp.h"
Romain Guy4aa90572010-09-26 18:40:37 -070029#include "DisplayListRenderer.h"
John Reck113e0822014-03-18 09:22:59 -070030#include "RenderNode.h"
Tom Hudson8dfaa492014-12-09 15:03:44 -050031#include "utils/PaintUtils.h"
Romain Guy13631f32012-01-30 17:41:55 -080032
Romain Guy4aa90572010-09-26 18:40:37 -070033namespace android {
34namespace uirenderer {
35
Chris Craikcce47eb2014-07-16 15:12:15 -070036DisplayListRenderer::DisplayListRenderer()
Tom Hudson984162f2014-10-10 13:38:16 -040037 : mState(*this)
John Reck088c5142014-11-06 13:04:49 -080038 , mResourceCache(ResourceCache::getInstance())
Chris Craik8afd0f22014-08-21 17:41:57 -070039 , mDisplayListData(NULL)
Chris Craikcce47eb2014-07-16 15:12:15 -070040 , mTranslateX(0.0f)
41 , mTranslateY(0.0f)
Rob Tsuka5f2e072014-11-17 12:57:32 -080042 , mHasDeferredTranslate(false)
Chris Craik8afd0f22014-08-21 17:41:57 -070043 , mDeferredBarrierType(kBarrier_None)
Chris Craikcce47eb2014-07-16 15:12:15 -070044 , mHighContrastText(false)
45 , mRestoreSaveCount(-1) {
Romain Guy4aa90572010-09-26 18:40:37 -070046}
47
48DisplayListRenderer::~DisplayListRenderer() {
John Reck44fd8d22014-02-26 11:00:11 -080049 LOG_ALWAYS_FATAL_IF(mDisplayListData,
50 "Destroyed a DisplayListRenderer during a record!");
Romain Guy4aa90572010-09-26 18:40:37 -070051}
52
53///////////////////////////////////////////////////////////////////////////////
54// Operations
55///////////////////////////////////////////////////////////////////////////////
56
John Reck44fd8d22014-02-26 11:00:11 -080057DisplayListData* DisplayListRenderer::finishRecording() {
John Reck44fd8d22014-02-26 11:00:11 -080058 mPaintMap.clear();
59 mRegionMap.clear();
60 mPathMap.clear();
61 DisplayListData* data = mDisplayListData;
62 mDisplayListData = 0;
63 return data;
Chet Haase5977baa2011-01-05 18:01:22 -080064}
65
Andreas Gampe64bb4132014-11-22 00:35:09 +000066void DisplayListRenderer::prepareDirty(float left, float top,
Tom Hudson8dfaa492014-12-09 15:03:44 -050067 float right, float bottom) {
John Reck44fd8d22014-02-26 11:00:11 -080068
69 LOG_ALWAYS_FATAL_IF(mDisplayListData,
70 "prepareDirty called a second time during a recording!");
71 mDisplayListData = new DisplayListData();
72
Tom Hudson984162f2014-10-10 13:38:16 -040073 mState.initializeSaveStack(0, 0, mState.getWidth(), mState.getHeight(), Vector3());
Romain Guy45e4c3d2012-09-11 17:17:07 -070074
Chris Craik8afd0f22014-08-21 17:41:57 -070075 mDeferredBarrierType = kBarrier_InOrder;
Tom Hudson8dfaa492014-12-09 15:03:44 -050076 mState.setDirtyClip(false);
Romain Guy27454a42011-01-23 12:01:41 -080077 mRestoreSaveCount = -1;
78}
79
Tom Hudson107843d2014-09-08 11:26:26 -040080bool DisplayListRenderer::finish() {
Chris Craik8afd0f22014-08-21 17:41:57 -070081 flushRestoreToCount();
82 flushTranslate();
Tom Hudson107843d2014-09-08 11:26:26 -040083 return false;
Romain Guyb051e892010-09-28 19:09:36 -070084}
85
Chet Haasedaf98e92011-01-10 14:10:36 -080086void DisplayListRenderer::interrupt() {
Chet Haasedaf98e92011-01-10 14:10:36 -080087}
Romain Guy2b1847e2011-01-26 13:43:01 -080088
Chet Haasedaf98e92011-01-10 14:10:36 -080089void DisplayListRenderer::resume() {
Romain Guy4aa90572010-09-26 18:40:37 -070090}
91
Andreas Gampe64bb4132014-11-22 00:35:09 +000092void DisplayListRenderer::callDrawGLFunction(Functor *functor, Rect& dirty) {
Romain Guycabfcc12011-03-07 18:06:46 -080093 // Ignore dirty during recording, it matters only when we replay
Chris Craik2af46352012-11-26 18:30:17 -080094 addDrawOp(new (alloc()) DrawFunctorOp(functor));
John Reck09d5cdd2014-07-24 10:36:08 -070095 mDisplayListData->functors.add(functor);
Chet Haasedaf98e92011-01-10 14:10:36 -080096}
97
Tom Hudson8dfaa492014-12-09 15:03:44 -050098int DisplayListRenderer::save(SkCanvas::SaveFlags flags) {
99 addStateOp(new (alloc()) SaveOp((int) flags));
100 return mState.save((int) flags);
Romain Guy4aa90572010-09-26 18:40:37 -0700101}
102
103void DisplayListRenderer::restore() {
Romain Guy04c9d8c2011-08-25 14:01:48 -0700104 if (mRestoreSaveCount < 0) {
Romain Guy33f6beb2012-02-16 19:24:51 -0800105 restoreToCount(getSaveCount() - 1);
106 return;
Romain Guy04c9d8c2011-08-25 14:01:48 -0700107 }
Romain Guy33f6beb2012-02-16 19:24:51 -0800108
109 mRestoreSaveCount--;
Chris Craik8afd0f22014-08-21 17:41:57 -0700110 flushTranslate();
Tom Hudson984162f2014-10-10 13:38:16 -0400111 mState.restore();
Romain Guy4aa90572010-09-26 18:40:37 -0700112}
113
114void DisplayListRenderer::restoreToCount(int saveCount) {
Romain Guy27454a42011-01-23 12:01:41 -0800115 mRestoreSaveCount = saveCount;
Chris Craik8afd0f22014-08-21 17:41:57 -0700116 flushTranslate();
Tom Hudson984162f2014-10-10 13:38:16 -0400117 mState.restoreToCount(saveCount);
Romain Guy4aa90572010-09-26 18:40:37 -0700118}
119
120int DisplayListRenderer::saveLayer(float left, float top, float right, float bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500121 const SkPaint* paint, SkCanvas::SaveFlags flags) {
Chris Craik4ace7302014-09-14 15:49:54 -0700122 // force matrix/clip isolation for layer
123 flags |= SkCanvas::kClip_SaveFlag | SkCanvas::kMatrix_SaveFlag;
124
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500125 paint = refPaint(paint);
Tom Hudson8dfaa492014-12-09 15:03:44 -0500126 addStateOp(new (alloc()) SaveLayerOp(left, top, right, bottom, paint, (int) flags));
127 return mState.save((int) flags);
Romain Guy5b3b3522010-10-27 18:57:51 -0700128}
129
Tom Hudson8dfaa492014-12-09 15:03:44 -0500130void DisplayListRenderer::translate(float dx, float dy) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700131 mHasDeferredTranslate = true;
Romain Guy33f6beb2012-02-16 19:24:51 -0800132 mTranslateX += dx;
133 mTranslateY += dy;
Chris Craik8afd0f22014-08-21 17:41:57 -0700134 flushRestoreToCount();
Tom Hudson8dfaa492014-12-09 15:03:44 -0500135 mState.translate(dx, dy, 0.0f);
Romain Guy4aa90572010-09-26 18:40:37 -0700136}
137
138void DisplayListRenderer::rotate(float degrees) {
Chris Craik2af46352012-11-26 18:30:17 -0800139 addStateOp(new (alloc()) RotateOp(degrees));
Tom Hudson984162f2014-10-10 13:38:16 -0400140 mState.rotate(degrees);
Romain Guy4aa90572010-09-26 18:40:37 -0700141}
142
143void DisplayListRenderer::scale(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800144 addStateOp(new (alloc()) ScaleOp(sx, sy));
Tom Hudson984162f2014-10-10 13:38:16 -0400145 mState.scale(sx, sy);
Romain Guy4aa90572010-09-26 18:40:37 -0700146}
147
Romain Guy807daf72011-01-18 11:19:19 -0800148void DisplayListRenderer::skew(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800149 addStateOp(new (alloc()) SkewOp(sx, sy));
Tom Hudson984162f2014-10-10 13:38:16 -0400150 mState.skew(sx, sy);
Romain Guy807daf72011-01-18 11:19:19 -0800151}
152
Derek Sollenberger13908822013-12-10 12:28:58 -0500153void DisplayListRenderer::setMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800154 addStateOp(new (alloc()) SetMatrixOp(matrix));
Tom Hudson984162f2014-10-10 13:38:16 -0400155 mState.setMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700156}
157
Tom Hudson8dfaa492014-12-09 15:03:44 -0500158void DisplayListRenderer::concat(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800159 addStateOp(new (alloc()) ConcatMatrixOp(matrix));
Tom Hudson984162f2014-10-10 13:38:16 -0400160 mState.concatMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700161}
162
Tom Hudson8dfaa492014-12-09 15:03:44 -0500163bool DisplayListRenderer::getClipBounds(SkRect* outRect) const {
164 Rect bounds = mState.getLocalClipBounds();
165 *outRect = SkRect::MakeLTRB(bounds.left, bounds.top, bounds.right, bounds.bottom);
166 return !(outRect->isEmpty());
167}
168
169bool DisplayListRenderer::quickRejectRect(float left, float top, float right, float bottom) const {
170 return mState.quickRejectConservative(left, top, right, bottom);
171}
172
173bool DisplayListRenderer::quickRejectPath(const SkPath& path) const {
174 SkRect bounds = path.getBounds();
175 return mState.quickRejectConservative(bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
176}
177
178
Romain Guy4aa90572010-09-26 18:40:37 -0700179bool DisplayListRenderer::clipRect(float left, float top, float right, float bottom,
180 SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800181 addStateOp(new (alloc()) ClipRectOp(left, top, right, bottom, op));
Tom Hudson984162f2014-10-10 13:38:16 -0400182 return mState.clipRect(left, top, right, bottom, op);
Romain Guy4aa90572010-09-26 18:40:37 -0700183}
184
Chris Craikd218a922014-01-02 17:13:34 -0800185bool DisplayListRenderer::clipPath(const SkPath* path, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800186 path = refPath(path);
187 addStateOp(new (alloc()) ClipPathOp(path, op));
Tom Hudson984162f2014-10-10 13:38:16 -0400188 return mState.clipPath(path, op);
Romain Guy735738c2012-12-03 12:34:51 -0800189}
190
Chris Craikd218a922014-01-02 17:13:34 -0800191bool DisplayListRenderer::clipRegion(const SkRegion* region, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800192 region = refRegion(region);
193 addStateOp(new (alloc()) ClipRegionOp(region, op));
Tom Hudson984162f2014-10-10 13:38:16 -0400194 return mState.clipRegion(region, op);
Romain Guy735738c2012-12-03 12:34:51 -0800195}
196
Andreas Gampe64bb4132014-11-22 00:35:09 +0000197void DisplayListRenderer::drawRenderNode(RenderNode* renderNode, Rect& dirty, int32_t flags) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700198 LOG_ALWAYS_FATAL_IF(!renderNode, "missing rendernode");
199
Romain Guycabfcc12011-03-07 18:06:46 -0800200 // dirty is an out parameter and should not be recorded,
201 // it matters only when replaying the display list
Tom Hudson984162f2014-10-10 13:38:16 -0400202 DrawRenderNodeOp* op = new (alloc()) DrawRenderNodeOp(renderNode, flags, *mState.currentTransform());
Chris Craik8afd0f22014-08-21 17:41:57 -0700203 addRenderNodeOp(op);
Romain Guy0fe478e2010-11-08 12:08:41 -0800204}
205
John Reck65f2def2014-11-07 11:29:08 -0800206void DisplayListRenderer::drawLayer(DeferredLayerUpdater* layerHandle, float x, float y) {
John Reck12f5e342014-11-07 07:53:43 -0800207 // We ref the DeferredLayerUpdater due to its thread-safe ref-counting
208 // semantics.
209 mDisplayListData->ref(layerHandle);
210 addDrawOp(new (alloc()) DrawLayerOp(layerHandle->backingLayer(), x, y));
Romain Guy6c319ca2011-01-11 14:29:25 -0800211}
212
Tom Hudson107843d2014-09-08 11:26:26 -0400213void DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800214 bitmap = refBitmap(bitmap);
215 paint = refPaint(paint);
216
Chris Craik79647502014-08-06 13:42:24 -0700217 addDrawOp(new (alloc()) DrawBitmapOp(bitmap, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700218}
219
Tom Hudson8dfaa492014-12-09 15:03:44 -0500220void DisplayListRenderer::drawBitmap(const SkBitmap& bitmap, float left, float top,
221 const SkPaint* paint) {
222 save(SkCanvas::kMatrix_SaveFlag);
223 translate(left, top);
224 drawBitmap(&bitmap, paint);
225 restore();
226}
227
228void DisplayListRenderer::drawBitmap(const SkBitmap& bitmap, const SkMatrix& matrix,
229 const SkPaint* paint) {
230 if (matrix.isIdentity()) {
231 drawBitmap(&bitmap, paint);
232 } else if (!(matrix.getType() & ~(SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask))) {
233 // SkMatrix::isScaleTranslate() not available in L
234 SkRect src;
235 SkRect dst;
236 bitmap.getBounds(&src);
237 matrix.mapRect(&dst, src);
238 drawBitmap(bitmap, src.fLeft, src.fTop, src.fRight, src.fBottom,
239 dst.fLeft, dst.fTop, dst.fRight, dst.fBottom, paint);
240 } else {
241 save(SkCanvas::kMatrix_SaveFlag);
242 concat(matrix);
243 drawBitmap(&bitmap, paint);
244 restore();
245 }
246}
247
248void DisplayListRenderer::drawBitmap(const SkBitmap& bitmap, float srcLeft, float srcTop,
Romain Guy4aa90572010-09-26 18:40:37 -0700249 float srcRight, float srcBottom, float dstLeft, float dstTop,
Chris Craikd218a922014-01-02 17:13:34 -0800250 float dstRight, float dstBottom, const SkPaint* paint) {
Chris Craik79647502014-08-06 13:42:24 -0700251 if (srcLeft == 0 && srcTop == 0
Tom Hudson8dfaa492014-12-09 15:03:44 -0500252 && srcRight == bitmap.width() && srcBottom == bitmap.height()
Chris Craik79647502014-08-06 13:42:24 -0700253 && (srcBottom - srcTop == dstBottom - dstTop)
254 && (srcRight - srcLeft == dstRight - dstLeft)) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800255 // transform simple rect to rect drawing case into position bitmap ops, since they merge
Chris Craik79647502014-08-06 13:42:24 -0700256 save(SkCanvas::kMatrix_SaveFlag);
257 translate(dstLeft, dstTop);
Tom Hudson8dfaa492014-12-09 15:03:44 -0500258 drawBitmap(&bitmap, paint);
Chris Craik79647502014-08-06 13:42:24 -0700259 restore();
260 } else {
Chris Craik79647502014-08-06 13:42:24 -0700261 paint = refPaint(paint);
Chris Craik527a3aa2013-03-04 10:19:31 -0800262
Tom Hudson8dfaa492014-12-09 15:03:44 -0500263 addDrawOp(new (alloc()) DrawBitmapRectOp(refBitmap(&bitmap),
Chris Craik79647502014-08-06 13:42:24 -0700264 srcLeft, srcTop, srcRight, srcBottom,
265 dstLeft, dstTop, dstRight, dstBottom, paint));
266 }
Romain Guy4aa90572010-09-26 18:40:37 -0700267}
268
Tom Hudson107843d2014-09-08 11:26:26 -0400269void DisplayListRenderer::drawBitmapData(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800270 bitmap = refBitmapData(bitmap);
271 paint = refPaint(paint);
272
Chris Craik79647502014-08-06 13:42:24 -0700273 addDrawOp(new (alloc()) DrawBitmapDataOp(bitmap, paint));
Romain Guye651cc62012-05-14 19:44:40 -0700274}
275
Tom Hudson8dfaa492014-12-09 15:03:44 -0500276void DisplayListRenderer::drawBitmapMesh(const SkBitmap& bitmap, int meshWidth, int meshHeight,
Chris Craikd218a922014-01-02 17:13:34 -0800277 const float* vertices, const int* colors, const SkPaint* paint) {
Chris Craik0664fef2014-04-11 13:40:05 -0700278 int vertexCount = (meshWidth + 1) * (meshHeight + 1);
Chris Craik0664fef2014-04-11 13:40:05 -0700279 vertices = refBuffer<float>(vertices, vertexCount * 2); // 2 floats per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800280 paint = refPaint(paint);
Chris Craik0664fef2014-04-11 13:40:05 -0700281 colors = refBuffer<int>(colors, vertexCount); // 1 color per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800282
Tom Hudson8dfaa492014-12-09 15:03:44 -0500283 addDrawOp(new (alloc()) DrawBitmapMeshOp(refBitmap(&bitmap), meshWidth, meshHeight,
284 vertices, colors, paint));
Romain Guy5a7b4662011-01-20 19:09:30 -0800285}
286
Tom Hudson107843d2014-09-08 11:26:26 -0400287void DisplayListRenderer::drawPatch(const SkBitmap* bitmap, const Res_png_9patch* patch,
Chris Craikd218a922014-01-02 17:13:34 -0800288 float left, float top, float right, float bottom, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800289 bitmap = refBitmap(bitmap);
Romain Guye3b0a012013-06-26 15:45:41 -0700290 patch = refPatch(patch);
Romain Guy16ea8d32013-06-21 11:35:52 -0700291 paint = refPaint(paint);
Chris Craik2af46352012-11-26 18:30:17 -0800292
Romain Guy03c00b52013-06-20 18:30:28 -0700293 addDrawOp(new (alloc()) DrawPatchOp(bitmap, patch, left, top, right, bottom, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700294}
295
Tom Hudson107843d2014-09-08 11:26:26 -0400296void DisplayListRenderer::drawColor(int color, SkXfermode::Mode mode) {
Chris Craik2af46352012-11-26 18:30:17 -0800297 addDrawOp(new (alloc()) DrawColorOp(color, mode));
Romain Guy4aa90572010-09-26 18:40:37 -0700298}
299
Tom Hudson8dfaa492014-12-09 15:03:44 -0500300void DisplayListRenderer::drawPaint(const SkPaint& paint) {
301 SkRect bounds;
302 if (getClipBounds(&bounds)) {
303 drawRect(bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom, paint);
304 }
305}
306
307
Tom Hudson107843d2014-09-08 11:26:26 -0400308void DisplayListRenderer::drawRect(float left, float top, float right, float bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500309 const SkPaint& paint) {
310 addDrawOp(new (alloc()) DrawRectOp(left, top, right, bottom, refPaint(&paint)));
Romain Guy4aa90572010-09-26 18:40:37 -0700311}
312
Tom Hudson107843d2014-09-08 11:26:26 -0400313void DisplayListRenderer::drawRoundRect(float left, float top, float right, float bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500314 float rx, float ry, const SkPaint& paint) {
315 addDrawOp(new (alloc()) DrawRoundRectOp(left, top, right, bottom, rx, ry, refPaint(&paint)));
Romain Guy01d58e42011-01-19 21:54:02 -0800316}
317
Tom Hudson107843d2014-09-08 11:26:26 -0400318void DisplayListRenderer::drawRoundRect(
Jorim Jaggi072707d2014-09-15 17:20:08 +0200319 CanvasPropertyPrimitive* left, CanvasPropertyPrimitive* top,
320 CanvasPropertyPrimitive* right, CanvasPropertyPrimitive* bottom,
321 CanvasPropertyPrimitive* rx, CanvasPropertyPrimitive* ry,
322 CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700323 mDisplayListData->ref(left);
324 mDisplayListData->ref(top);
325 mDisplayListData->ref(right);
326 mDisplayListData->ref(bottom);
327 mDisplayListData->ref(rx);
328 mDisplayListData->ref(ry);
329 mDisplayListData->ref(paint);
Jorim Jaggi072707d2014-09-15 17:20:08 +0200330 addDrawOp(new (alloc()) DrawRoundRectPropsOp(&left->value, &top->value,
331 &right->value, &bottom->value, &rx->value, &ry->value, &paint->value));
Jorim Jaggi072707d2014-09-15 17:20:08 +0200332}
333
Tom Hudson8dfaa492014-12-09 15:03:44 -0500334void DisplayListRenderer::drawCircle(float x, float y, float radius, const SkPaint& paint) {
335 addDrawOp(new (alloc()) DrawCircleOp(x, y, radius, refPaint(&paint)));
Romain Guy01d58e42011-01-19 21:54:02 -0800336}
337
Tom Hudson107843d2014-09-08 11:26:26 -0400338void DisplayListRenderer::drawCircle(CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y,
John Reck52244ff2014-05-01 21:27:37 -0700339 CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700340 mDisplayListData->ref(x);
341 mDisplayListData->ref(y);
342 mDisplayListData->ref(radius);
343 mDisplayListData->ref(paint);
John Reck52244ff2014-05-01 21:27:37 -0700344 addDrawOp(new (alloc()) DrawCirclePropsOp(&x->value, &y->value,
345 &radius->value, &paint->value));
John Reck52244ff2014-05-01 21:27:37 -0700346}
347
Tom Hudson107843d2014-09-08 11:26:26 -0400348void DisplayListRenderer::drawOval(float left, float top, float right, float bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500349 const SkPaint& paint) {
350 addDrawOp(new (alloc()) DrawOvalOp(left, top, right, bottom, refPaint(&paint)));
Romain Guyc1cd9ba32011-01-23 14:18:41 -0800351}
352
Tom Hudson107843d2014-09-08 11:26:26 -0400353void DisplayListRenderer::drawArc(float left, float top, float right, float bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500354 float startAngle, float sweepAngle, bool useCenter, const SkPaint& paint) {
Chris Craik544e5242014-07-11 10:58:10 -0700355 if (fabs(sweepAngle) >= 360.0f) {
Tom Hudson107843d2014-09-08 11:26:26 -0400356 drawOval(left, top, right, bottom, paint);
357 } else {
Tom Hudson107843d2014-09-08 11:26:26 -0400358 addDrawOp(new (alloc()) DrawArcOp(left, top, right, bottom,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500359 startAngle, sweepAngle, useCenter, refPaint(&paint)));
Chris Craik6ac174b2014-06-17 13:47:05 -0700360 }
Romain Guy8b2f5262011-01-23 16:15:02 -0800361}
362
Tom Hudson8dfaa492014-12-09 15:03:44 -0500363void DisplayListRenderer::drawPath(const SkPath& path, const SkPaint& paint) {
364 addDrawOp(new (alloc()) DrawPathOp(refPath(&path), refPaint(&paint)));
Romain Guy4aa90572010-09-26 18:40:37 -0700365}
366
Tom Hudson8dfaa492014-12-09 15:03:44 -0500367void DisplayListRenderer::drawLines(const float* points, int count, const SkPaint& paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800368 points = refBuffer<float>(points, count);
Chris Craik2af46352012-11-26 18:30:17 -0800369
Tom Hudson8dfaa492014-12-09 15:03:44 -0500370 addDrawOp(new (alloc()) DrawLinesOp(points, count, refPaint(&paint)));
Romain Guy4aa90572010-09-26 18:40:37 -0700371}
372
Tom Hudson8dfaa492014-12-09 15:03:44 -0500373void DisplayListRenderer::drawPoints(const float* points, int count, const SkPaint& paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800374 points = refBuffer<float>(points, count);
Chris Craik2af46352012-11-26 18:30:17 -0800375
Tom Hudson8dfaa492014-12-09 15:03:44 -0500376 addDrawOp(new (alloc()) DrawPointsOp(points, count, refPaint(&paint)));
Romain Guyed6fcb02011-03-21 13:11:28 -0700377}
378
Tom Hudson8dfaa492014-12-09 15:03:44 -0500379void DisplayListRenderer::drawTextOnPath(const uint16_t* glyphs, int count,
380 const SkPath& path, float hOffset, float vOffset, const SkPaint& paint) {
381 if (!glyphs || count <= 0) return;
Chris Craik2af46352012-11-26 18:30:17 -0800382
Tom Hudson8dfaa492014-12-09 15:03:44 -0500383 int bytesCount = 2 * count;
384 DrawOp* op = new (alloc()) DrawTextOnPathOp(refText((const char*) glyphs, bytesCount),
385 bytesCount, count, refPath(&path),
386 hOffset, vOffset, refPaint(&paint));
Romain Guy0f667532013-03-01 14:31:04 -0800387 addDrawOp(op);
Romain Guy325740f2012-02-24 16:48:34 -0800388}
389
Tom Hudson8dfaa492014-12-09 15:03:44 -0500390void DisplayListRenderer::drawPosText(const uint16_t* text, const float* positions,
391 int count, int posCount, const SkPaint& paint) {
Tom Hudson107843d2014-09-08 11:26:26 -0400392 if (!text || count <= 0) return;
Chris Craik2af46352012-11-26 18:30:17 -0800393
Tom Hudson8dfaa492014-12-09 15:03:44 -0500394 int bytesCount = 2 * count;
Chris Craik2af46352012-11-26 18:30:17 -0800395 positions = refBuffer<float>(positions, count * 2);
Chris Craik2af46352012-11-26 18:30:17 -0800396
Tom Hudson8dfaa492014-12-09 15:03:44 -0500397 DrawOp* op = new (alloc()) DrawPosTextOp(refText((const char*) text, bytesCount),
398 bytesCount, count, positions, refPaint(&paint));
Romain Guy0f667532013-03-01 14:31:04 -0800399 addDrawOp(op);
Romain Guyeb9a5362012-01-17 17:39:26 -0800400}
401
Chris Craikcce47eb2014-07-16 15:12:15 -0700402static void simplifyPaint(int color, SkPaint* paint) {
403 paint->setColor(color);
404 paint->setShader(NULL);
405 paint->setColorFilter(NULL);
406 paint->setLooper(NULL);
407 paint->setStrokeWidth(4 + 0.04 * paint->getTextSize());
408 paint->setStrokeJoin(SkPaint::kRound_Join);
409 paint->setLooper(NULL);
410}
411
Tom Hudson8dfaa492014-12-09 15:03:44 -0500412void DisplayListRenderer::drawText(const uint16_t* glyphs, const float* positions,
413 int count, const SkPaint& paint, float x, float y,
414 float boundsLeft, float boundsTop, float boundsRight, float boundsBottom,
415 float totalAdvance) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800416
Tom Hudson8dfaa492014-12-09 15:03:44 -0500417 if (!glyphs || count <= 0 || PaintUtils::paintWillNotDrawText(paint)) return;
Raph Levien996e57c2012-07-23 15:22:52 -0700418
Tom Hudson8dfaa492014-12-09 15:03:44 -0500419 int bytesCount = count * 2;
420 const char* text = refText((const char*) glyphs, bytesCount);
Chris Craik2af46352012-11-26 18:30:17 -0800421 positions = refBuffer<float>(positions, count * 2);
Tom Hudson8dfaa492014-12-09 15:03:44 -0500422 Rect bounds(boundsLeft, boundsTop, boundsRight, boundsBottom);
Raph Levien996e57c2012-07-23 15:22:52 -0700423
Chris Craikcce47eb2014-07-16 15:12:15 -0700424 if (CC_UNLIKELY(mHighContrastText)) {
425 // high contrast draw path
Tom Hudson8dfaa492014-12-09 15:03:44 -0500426 int color = paint.getColor();
Chris Craikcce47eb2014-07-16 15:12:15 -0700427 int channelSum = SkColorGetR(color) + SkColorGetG(color) + SkColorGetB(color);
428 bool darken = channelSum < (128 * 3);
429
430 // outline
Tom Hudson8dfaa492014-12-09 15:03:44 -0500431 SkPaint* outlinePaint = copyPaint(&paint);
Chris Craikcce47eb2014-07-16 15:12:15 -0700432 simplifyPaint(darken ? SK_ColorWHITE : SK_ColorBLACK, outlinePaint);
433 outlinePaint->setStyle(SkPaint::kStrokeAndFill_Style);
434 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
435 x, y, positions, outlinePaint, totalAdvance, bounds)); // bounds?
436
437 // inner
Tom Hudson8dfaa492014-12-09 15:03:44 -0500438 SkPaint* innerPaint = copyPaint(&paint);
Chris Craikcce47eb2014-07-16 15:12:15 -0700439 simplifyPaint(darken ? SK_ColorBLACK : SK_ColorWHITE, innerPaint);
440 innerPaint->setStyle(SkPaint::kFill_Style);
441 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
442 x, y, positions, innerPaint, totalAdvance, bounds));
443 } else {
444 // standard draw path
Chris Craikcce47eb2014-07-16 15:12:15 -0700445 DrawOp* op = new (alloc()) DrawTextOp(text, bytesCount, count,
Tom Hudson8dfaa492014-12-09 15:03:44 -0500446 x, y, positions, refPaint(&paint), totalAdvance, bounds);
Chris Craikcce47eb2014-07-16 15:12:15 -0700447 addDrawOp(op);
448 }
Raph Levien996e57c2012-07-23 15:22:52 -0700449}
450
Tom Hudson107843d2014-09-08 11:26:26 -0400451void DisplayListRenderer::drawRects(const float* rects, int count, const SkPaint* paint) {
452 if (count <= 0) return;
Romain Guy672433d2013-01-04 19:05:13 -0800453
Chris Craik2af46352012-11-26 18:30:17 -0800454 rects = refBuffer<float>(rects, count);
455 paint = refPaint(paint);
456 addDrawOp(new (alloc()) DrawRectsOp(rects, count, paint));
Romain Guy672433d2013-01-04 19:05:13 -0800457}
458
Derek Sollenberger09c2d4f2014-10-15 09:21:10 -0400459void DisplayListRenderer::setDrawFilter(SkDrawFilter* filter) {
460 mDrawFilter.reset(filter);
Chris Craik2af46352012-11-26 18:30:17 -0800461}
462
Chris Craik8afd0f22014-08-21 17:41:57 -0700463void DisplayListRenderer::insertReorderBarrier(bool enableReorder) {
464 flushRestoreToCount();
465 flushTranslate();
466 mDeferredBarrierType = enableReorder ? kBarrier_OutOfOrder : kBarrier_InOrder;
467}
468
469void DisplayListRenderer::flushRestoreToCount() {
Chris Craik2af46352012-11-26 18:30:17 -0800470 if (mRestoreSaveCount >= 0) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700471 addOpAndUpdateChunk(new (alloc()) RestoreToCountOp(mRestoreSaveCount));
Chris Craik2af46352012-11-26 18:30:17 -0800472 mRestoreSaveCount = -1;
473 }
474}
475
Chris Craik8afd0f22014-08-21 17:41:57 -0700476void DisplayListRenderer::flushTranslate() {
477 if (mHasDeferredTranslate) {
Chris Craik2af46352012-11-26 18:30:17 -0800478 if (mTranslateX != 0.0f || mTranslateY != 0.0f) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700479 addOpAndUpdateChunk(new (alloc()) TranslateOp(mTranslateX, mTranslateY));
Chris Craik2af46352012-11-26 18:30:17 -0800480 mTranslateX = mTranslateY = 0.0f;
481 }
Chris Craik8afd0f22014-08-21 17:41:57 -0700482 mHasDeferredTranslate = false;
Chris Craik2af46352012-11-26 18:30:17 -0800483 }
484}
485
Chris Craik8afd0f22014-08-21 17:41:57 -0700486size_t DisplayListRenderer::addOpAndUpdateChunk(DisplayListOp* op) {
487 int insertIndex = mDisplayListData->displayListOps.add(op);
488 if (mDeferredBarrierType != kBarrier_None) {
489 // op is first in new chunk
490 mDisplayListData->chunks.push();
491 DisplayListData::Chunk& newChunk = mDisplayListData->chunks.editTop();
492 newChunk.beginOpIndex = insertIndex;
493 newChunk.endOpIndex = insertIndex + 1;
494 newChunk.reorderChildren = (mDeferredBarrierType == kBarrier_OutOfOrder);
495
496 int nextChildIndex = mDisplayListData->children().size();
497 newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex;
498 mDeferredBarrierType = kBarrier_None;
499 } else {
500 // standard case - append to existing chunk
501 mDisplayListData->chunks.editTop().endOpIndex = insertIndex + 1;
502 }
503 return insertIndex;
Chris Craik2af46352012-11-26 18:30:17 -0800504}
505
Chris Craik8afd0f22014-08-21 17:41:57 -0700506size_t DisplayListRenderer::flushAndAddOp(DisplayListOp* op) {
507 flushRestoreToCount();
508 flushTranslate();
509 return addOpAndUpdateChunk(op);
510}
511
512size_t DisplayListRenderer::addStateOp(StateOp* op) {
513 return flushAndAddOp(op);
514}
515
516size_t DisplayListRenderer::addDrawOp(DrawOp* op) {
Chris Craik2af46352012-11-26 18:30:17 -0800517 Rect localBounds;
John Reck3b202512014-06-23 13:13:08 -0700518 if (op->getLocalBounds(localBounds)) {
Tom Hudson8dfaa492014-12-09 15:03:44 -0500519 bool rejected = quickRejectRect(localBounds.left, localBounds.top,
Chris Craik2af46352012-11-26 18:30:17 -0800520 localBounds.right, localBounds.bottom);
521 op->setQuickRejected(rejected);
522 }
Chris Craikc1c5f082013-09-11 16:23:37 -0700523
John Reck44fd8d22014-02-26 11:00:11 -0800524 mDisplayListData->hasDrawOps = true;
Chris Craik8afd0f22014-08-21 17:41:57 -0700525 return flushAndAddOp(op);
526}
527
528size_t DisplayListRenderer::addRenderNodeOp(DrawRenderNodeOp* op) {
529 int opIndex = addDrawOp(op);
530 int childIndex = mDisplayListData->addChild(op);
531
532 // update the chunk's child indices
533 DisplayListData::Chunk& chunk = mDisplayListData->chunks.editTop();
534 chunk.endChildIndex = childIndex + 1;
535
536 if (op->renderNode()->stagingProperties().isProjectionReceiver()) {
537 // use staging property, since recording on UI thread
538 mDisplayListData->projectionReceiveIndex = opIndex;
539 }
540 return opIndex;
Romain Guy5ff9df62012-01-23 17:09:05 -0800541}
542
Romain Guy4aa90572010-09-26 18:40:37 -0700543}; // namespace uirenderer
544}; // namespace android