blob: ab9ca1f19ef691bc492f7c4fbb04d8c3de19ab87 [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 Reck113e0822014-03-18 09:22:59 -070024#include "Caches.h"
Chris Craikc3566d02013-02-04 16:16:33 -080025#include "DeferredDisplayList.h"
Chet Haase9c1e23b2011-03-24 10:51:31 -070026#include "DisplayListLogBuffer.h"
Chris Craik2af46352012-11-26 18:30:17 -080027#include "DisplayListOp.h"
Romain Guy4aa90572010-09-26 18:40:37 -070028#include "DisplayListRenderer.h"
John Reck113e0822014-03-18 09:22:59 -070029#include "RenderNode.h"
Romain Guy13631f32012-01-30 17:41:55 -080030
Romain Guy4aa90572010-09-26 18:40:37 -070031namespace android {
32namespace uirenderer {
33
Chris Craikcce47eb2014-07-16 15:12:15 -070034DisplayListRenderer::DisplayListRenderer()
35 : mCaches(Caches::getInstance())
Chris Craik8afd0f22014-08-21 17:41:57 -070036 , mDisplayListData(NULL)
Chris Craikcce47eb2014-07-16 15:12:15 -070037 , mTranslateX(0.0f)
38 , mTranslateY(0.0f)
Chris Craik8afd0f22014-08-21 17:41:57 -070039 , mDeferredBarrierType(kBarrier_None)
Chris Craikcce47eb2014-07-16 15:12:15 -070040 , mHighContrastText(false)
41 , mRestoreSaveCount(-1) {
Romain Guy4aa90572010-09-26 18:40:37 -070042}
43
44DisplayListRenderer::~DisplayListRenderer() {
John Reck44fd8d22014-02-26 11:00:11 -080045 LOG_ALWAYS_FATAL_IF(mDisplayListData,
46 "Destroyed a DisplayListRenderer during a record!");
Romain Guy4aa90572010-09-26 18:40:37 -070047}
48
49///////////////////////////////////////////////////////////////////////////////
50// Operations
51///////////////////////////////////////////////////////////////////////////////
52
John Reck44fd8d22014-02-26 11:00:11 -080053DisplayListData* DisplayListRenderer::finishRecording() {
John Reck44fd8d22014-02-26 11:00:11 -080054 mPaintMap.clear();
55 mRegionMap.clear();
56 mPathMap.clear();
57 DisplayListData* data = mDisplayListData;
58 mDisplayListData = 0;
59 return data;
Chet Haase5977baa2011-01-05 18:01:22 -080060}
61
Tom Hudson107843d2014-09-08 11:26:26 -040062void DisplayListRenderer::prepareDirty(float left, float top,
Romain Guy7d7b5492011-01-24 16:33:45 -080063 float right, float bottom, bool opaque) {
John Reck44fd8d22014-02-26 11:00:11 -080064
65 LOG_ALWAYS_FATAL_IF(mDisplayListData,
66 "prepareDirty called a second time during a recording!");
67 mDisplayListData = new DisplayListData();
68
Chris Craik69e5adf2014-08-14 13:34:01 -070069 initializeSaveStack(0, 0, getWidth(), getHeight(), Vector3());
Romain Guy45e4c3d2012-09-11 17:17:07 -070070
Chris Craik8afd0f22014-08-21 17:41:57 -070071 mDeferredBarrierType = kBarrier_InOrder;
Romain Guy45e4c3d2012-09-11 17:17:07 -070072 mDirtyClip = opaque;
Romain Guy27454a42011-01-23 12:01:41 -080073 mRestoreSaveCount = -1;
74}
75
Tom Hudson107843d2014-09-08 11:26:26 -040076bool DisplayListRenderer::finish() {
Chris Craik8afd0f22014-08-21 17:41:57 -070077 flushRestoreToCount();
78 flushTranslate();
Tom Hudson107843d2014-09-08 11:26:26 -040079 return false;
Romain Guyb051e892010-09-28 19:09:36 -070080}
81
Chet Haasedaf98e92011-01-10 14:10:36 -080082void DisplayListRenderer::interrupt() {
Chet Haasedaf98e92011-01-10 14:10:36 -080083}
Romain Guy2b1847e2011-01-26 13:43:01 -080084
Chet Haasedaf98e92011-01-10 14:10:36 -080085void DisplayListRenderer::resume() {
Romain Guy4aa90572010-09-26 18:40:37 -070086}
87
Tom Hudson107843d2014-09-08 11:26:26 -040088void DisplayListRenderer::callDrawGLFunction(Functor *functor, Rect& dirty) {
Romain Guycabfcc12011-03-07 18:06:46 -080089 // Ignore dirty during recording, it matters only when we replay
Chris Craik2af46352012-11-26 18:30:17 -080090 addDrawOp(new (alloc()) DrawFunctorOp(functor));
John Reck09d5cdd2014-07-24 10:36:08 -070091 mDisplayListData->functors.add(functor);
Chet Haasedaf98e92011-01-10 14:10:36 -080092}
93
Romain Guy4aa90572010-09-26 18:40:37 -070094int DisplayListRenderer::save(int flags) {
Chris Craik2af46352012-11-26 18:30:17 -080095 addStateOp(new (alloc()) SaveOp(flags));
Chris Craik14e51302013-12-30 15:32:54 -080096 return StatefulBaseRenderer::save(flags);
Romain Guy4aa90572010-09-26 18:40:37 -070097}
98
99void DisplayListRenderer::restore() {
Romain Guy04c9d8c2011-08-25 14:01:48 -0700100 if (mRestoreSaveCount < 0) {
Romain Guy33f6beb2012-02-16 19:24:51 -0800101 restoreToCount(getSaveCount() - 1);
102 return;
Romain Guy04c9d8c2011-08-25 14:01:48 -0700103 }
Romain Guy33f6beb2012-02-16 19:24:51 -0800104
105 mRestoreSaveCount--;
Chris Craik8afd0f22014-08-21 17:41:57 -0700106 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800107 StatefulBaseRenderer::restore();
Romain Guy4aa90572010-09-26 18:40:37 -0700108}
109
110void DisplayListRenderer::restoreToCount(int saveCount) {
Romain Guy27454a42011-01-23 12:01:41 -0800111 mRestoreSaveCount = saveCount;
Chris Craik8afd0f22014-08-21 17:41:57 -0700112 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800113 StatefulBaseRenderer::restoreToCount(saveCount);
Romain Guy4aa90572010-09-26 18:40:37 -0700114}
115
116int DisplayListRenderer::saveLayer(float left, float top, float right, float bottom,
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500117 const SkPaint* paint, int flags) {
Chris Craik4ace7302014-09-14 15:49:54 -0700118 // force matrix/clip isolation for layer
119 flags |= SkCanvas::kClip_SaveFlag | SkCanvas::kMatrix_SaveFlag;
120
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500121 paint = refPaint(paint);
122 addStateOp(new (alloc()) SaveLayerOp(left, top, right, bottom, paint, flags));
Chris Craik14e51302013-12-30 15:32:54 -0800123 return StatefulBaseRenderer::save(flags);
Romain Guy5b3b3522010-10-27 18:57:51 -0700124}
125
Chris Craikb4589422013-12-26 15:13:13 -0800126void DisplayListRenderer::translate(float dx, float dy, float dz) {
127 // ignore dz, not used at defer time
Chris Craik8afd0f22014-08-21 17:41:57 -0700128 mHasDeferredTranslate = true;
Romain Guy33f6beb2012-02-16 19:24:51 -0800129 mTranslateX += dx;
130 mTranslateY += dy;
Chris Craik8afd0f22014-08-21 17:41:57 -0700131 flushRestoreToCount();
Chris Craik14e51302013-12-30 15:32:54 -0800132 StatefulBaseRenderer::translate(dx, dy, dz);
Romain Guy4aa90572010-09-26 18:40:37 -0700133}
134
135void DisplayListRenderer::rotate(float degrees) {
Chris Craik2af46352012-11-26 18:30:17 -0800136 addStateOp(new (alloc()) RotateOp(degrees));
Chris Craik14e51302013-12-30 15:32:54 -0800137 StatefulBaseRenderer::rotate(degrees);
Romain Guy4aa90572010-09-26 18:40:37 -0700138}
139
140void DisplayListRenderer::scale(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800141 addStateOp(new (alloc()) ScaleOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800142 StatefulBaseRenderer::scale(sx, sy);
Romain Guy4aa90572010-09-26 18:40:37 -0700143}
144
Romain Guy807daf72011-01-18 11:19:19 -0800145void DisplayListRenderer::skew(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800146 addStateOp(new (alloc()) SkewOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800147 StatefulBaseRenderer::skew(sx, sy);
Romain Guy807daf72011-01-18 11:19:19 -0800148}
149
Derek Sollenberger13908822013-12-10 12:28:58 -0500150void DisplayListRenderer::setMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800151 addStateOp(new (alloc()) SetMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800152 StatefulBaseRenderer::setMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700153}
154
Derek Sollenberger13908822013-12-10 12:28:58 -0500155void DisplayListRenderer::concatMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800156 addStateOp(new (alloc()) ConcatMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800157 StatefulBaseRenderer::concatMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700158}
159
160bool DisplayListRenderer::clipRect(float left, float top, float right, float bottom,
161 SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800162 addStateOp(new (alloc()) ClipRectOp(left, top, right, bottom, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800163 return StatefulBaseRenderer::clipRect(left, top, right, bottom, op);
Romain Guy4aa90572010-09-26 18:40:37 -0700164}
165
Chris Craikd218a922014-01-02 17:13:34 -0800166bool DisplayListRenderer::clipPath(const SkPath* path, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800167 path = refPath(path);
168 addStateOp(new (alloc()) ClipPathOp(path, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800169 return StatefulBaseRenderer::clipPath(path, op);
Romain Guy735738c2012-12-03 12:34:51 -0800170}
171
Chris Craikd218a922014-01-02 17:13:34 -0800172bool DisplayListRenderer::clipRegion(const SkRegion* region, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800173 region = refRegion(region);
174 addStateOp(new (alloc()) ClipRegionOp(region, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800175 return StatefulBaseRenderer::clipRegion(region, op);
Romain Guy735738c2012-12-03 12:34:51 -0800176}
177
Tom Hudson107843d2014-09-08 11:26:26 -0400178void DisplayListRenderer::drawRenderNode(RenderNode* renderNode, Rect& dirty, int32_t flags) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700179 LOG_ALWAYS_FATAL_IF(!renderNode, "missing rendernode");
180
Romain Guycabfcc12011-03-07 18:06:46 -0800181 // dirty is an out parameter and should not be recorded,
182 // it matters only when replaying the display list
Chris Craikb3cca872014-08-08 18:42:51 -0700183 DrawRenderNodeOp* op = new (alloc()) DrawRenderNodeOp(renderNode, flags, *currentTransform());
Chris Craik8afd0f22014-08-21 17:41:57 -0700184 addRenderNodeOp(op);
Romain Guy0fe478e2010-11-08 12:08:41 -0800185}
186
Tom Hudson107843d2014-09-08 11:26:26 -0400187void DisplayListRenderer::drawLayer(Layer* layer, float x, float y) {
John Reck0e89e2b2014-10-31 14:49:06 -0700188 mDisplayListData->ref(layer);
Chris Craika08f95c2013-03-15 17:24:33 -0700189 addDrawOp(new (alloc()) DrawLayerOp(layer, x, y));
Romain Guy6c319ca2011-01-11 14:29:25 -0800190}
191
Tom Hudson107843d2014-09-08 11:26:26 -0400192void DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800193 bitmap = refBitmap(bitmap);
194 paint = refPaint(paint);
195
Chris Craik79647502014-08-06 13:42:24 -0700196 addDrawOp(new (alloc()) DrawBitmapOp(bitmap, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700197}
198
Tom Hudson107843d2014-09-08 11:26:26 -0400199void DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, float srcLeft, float srcTop,
Romain Guy4aa90572010-09-26 18:40:37 -0700200 float srcRight, float srcBottom, float dstLeft, float dstTop,
Chris Craikd218a922014-01-02 17:13:34 -0800201 float dstRight, float dstBottom, const SkPaint* paint) {
Chris Craik79647502014-08-06 13:42:24 -0700202 if (srcLeft == 0 && srcTop == 0
203 && srcRight == bitmap->width() && srcBottom == bitmap->height()
204 && (srcBottom - srcTop == dstBottom - dstTop)
205 && (srcRight - srcLeft == dstRight - dstLeft)) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800206 // transform simple rect to rect drawing case into position bitmap ops, since they merge
Chris Craik79647502014-08-06 13:42:24 -0700207 save(SkCanvas::kMatrix_SaveFlag);
208 translate(dstLeft, dstTop);
209 drawBitmap(bitmap, paint);
210 restore();
211 } else {
212 bitmap = refBitmap(bitmap);
213 paint = refPaint(paint);
Chris Craik527a3aa2013-03-04 10:19:31 -0800214
Chris Craik79647502014-08-06 13:42:24 -0700215 addDrawOp(new (alloc()) DrawBitmapRectOp(bitmap,
216 srcLeft, srcTop, srcRight, srcBottom,
217 dstLeft, dstTop, dstRight, dstBottom, paint));
218 }
Romain Guy4aa90572010-09-26 18:40:37 -0700219}
220
Tom Hudson107843d2014-09-08 11:26:26 -0400221void DisplayListRenderer::drawBitmapData(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800222 bitmap = refBitmapData(bitmap);
223 paint = refPaint(paint);
224
Chris Craik79647502014-08-06 13:42:24 -0700225 addDrawOp(new (alloc()) DrawBitmapDataOp(bitmap, paint));
Romain Guye651cc62012-05-14 19:44:40 -0700226}
227
Tom Hudson107843d2014-09-08 11:26:26 -0400228void DisplayListRenderer::drawBitmapMesh(const SkBitmap* bitmap, int meshWidth, int meshHeight,
Chris Craikd218a922014-01-02 17:13:34 -0800229 const float* vertices, const int* colors, const SkPaint* paint) {
Chris Craik0664fef2014-04-11 13:40:05 -0700230 int vertexCount = (meshWidth + 1) * (meshHeight + 1);
Chris Craik2af46352012-11-26 18:30:17 -0800231 bitmap = refBitmap(bitmap);
Chris Craik0664fef2014-04-11 13:40:05 -0700232 vertices = refBuffer<float>(vertices, vertexCount * 2); // 2 floats per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800233 paint = refPaint(paint);
Chris Craik0664fef2014-04-11 13:40:05 -0700234 colors = refBuffer<int>(colors, vertexCount); // 1 color per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800235
236 addDrawOp(new (alloc()) DrawBitmapMeshOp(bitmap, meshWidth, meshHeight,
237 vertices, colors, paint));
Romain Guy5a7b4662011-01-20 19:09:30 -0800238}
239
Tom Hudson107843d2014-09-08 11:26:26 -0400240void DisplayListRenderer::drawPatch(const SkBitmap* bitmap, const Res_png_9patch* patch,
Chris Craikd218a922014-01-02 17:13:34 -0800241 float left, float top, float right, float bottom, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800242 bitmap = refBitmap(bitmap);
Romain Guye3b0a012013-06-26 15:45:41 -0700243 patch = refPatch(patch);
Romain Guy16ea8d32013-06-21 11:35:52 -0700244 paint = refPaint(paint);
Chris Craik2af46352012-11-26 18:30:17 -0800245
Romain Guy03c00b52013-06-20 18:30:28 -0700246 addDrawOp(new (alloc()) DrawPatchOp(bitmap, patch, left, top, right, bottom, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700247}
248
Tom Hudson107843d2014-09-08 11:26:26 -0400249void DisplayListRenderer::drawColor(int color, SkXfermode::Mode mode) {
Chris Craik2af46352012-11-26 18:30:17 -0800250 addDrawOp(new (alloc()) DrawColorOp(color, mode));
Romain Guy4aa90572010-09-26 18:40:37 -0700251}
252
Tom Hudson107843d2014-09-08 11:26:26 -0400253void DisplayListRenderer::drawRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800254 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800255 paint = refPaint(paint);
256 addDrawOp(new (alloc()) DrawRectOp(left, top, right, bottom, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700257}
258
Tom Hudson107843d2014-09-08 11:26:26 -0400259void DisplayListRenderer::drawRoundRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800260 float rx, float ry, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800261 paint = refPaint(paint);
262 addDrawOp(new (alloc()) DrawRoundRectOp(left, top, right, bottom, rx, ry, paint));
Romain Guy01d58e42011-01-19 21:54:02 -0800263}
264
Tom Hudson107843d2014-09-08 11:26:26 -0400265void DisplayListRenderer::drawRoundRect(
Jorim Jaggi072707d2014-09-15 17:20:08 +0200266 CanvasPropertyPrimitive* left, CanvasPropertyPrimitive* top,
267 CanvasPropertyPrimitive* right, CanvasPropertyPrimitive* bottom,
268 CanvasPropertyPrimitive* rx, CanvasPropertyPrimitive* ry,
269 CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700270 mDisplayListData->ref(left);
271 mDisplayListData->ref(top);
272 mDisplayListData->ref(right);
273 mDisplayListData->ref(bottom);
274 mDisplayListData->ref(rx);
275 mDisplayListData->ref(ry);
276 mDisplayListData->ref(paint);
Jorim Jaggi072707d2014-09-15 17:20:08 +0200277 addDrawOp(new (alloc()) DrawRoundRectPropsOp(&left->value, &top->value,
278 &right->value, &bottom->value, &rx->value, &ry->value, &paint->value));
Jorim Jaggi072707d2014-09-15 17:20:08 +0200279}
280
Tom Hudson107843d2014-09-08 11:26:26 -0400281void DisplayListRenderer::drawCircle(float x, float y, float radius, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800282 paint = refPaint(paint);
283 addDrawOp(new (alloc()) DrawCircleOp(x, y, radius, paint));
Romain Guy01d58e42011-01-19 21:54:02 -0800284}
285
Tom Hudson107843d2014-09-08 11:26:26 -0400286void DisplayListRenderer::drawCircle(CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y,
John Reck52244ff2014-05-01 21:27:37 -0700287 CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700288 mDisplayListData->ref(x);
289 mDisplayListData->ref(y);
290 mDisplayListData->ref(radius);
291 mDisplayListData->ref(paint);
John Reck52244ff2014-05-01 21:27:37 -0700292 addDrawOp(new (alloc()) DrawCirclePropsOp(&x->value, &y->value,
293 &radius->value, &paint->value));
John Reck52244ff2014-05-01 21:27:37 -0700294}
295
Tom Hudson107843d2014-09-08 11:26:26 -0400296void DisplayListRenderer::drawOval(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800297 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800298 paint = refPaint(paint);
299 addDrawOp(new (alloc()) DrawOvalOp(left, top, right, bottom, paint));
Romain Guyc1cd9ba32011-01-23 14:18:41 -0800300}
301
Tom Hudson107843d2014-09-08 11:26:26 -0400302void DisplayListRenderer::drawArc(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800303 float startAngle, float sweepAngle, bool useCenter, const SkPaint* paint) {
Chris Craik544e5242014-07-11 10:58:10 -0700304 if (fabs(sweepAngle) >= 360.0f) {
Tom Hudson107843d2014-09-08 11:26:26 -0400305 drawOval(left, top, right, bottom, paint);
306 } else {
307 paint = refPaint(paint);
308 addDrawOp(new (alloc()) DrawArcOp(left, top, right, bottom,
309 startAngle, sweepAngle, useCenter, paint));
Chris Craik6ac174b2014-06-17 13:47:05 -0700310 }
Romain Guy8b2f5262011-01-23 16:15:02 -0800311}
312
Tom Hudson107843d2014-09-08 11:26:26 -0400313void DisplayListRenderer::drawPath(const SkPath* path, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800314 path = refPath(path);
315 paint = refPaint(paint);
Romain Guy33f6beb2012-02-16 19:24:51 -0800316
Chris Craik2af46352012-11-26 18:30:17 -0800317 addDrawOp(new (alloc()) DrawPathOp(path, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700318}
319
Tom Hudson107843d2014-09-08 11:26:26 -0400320void DisplayListRenderer::drawLines(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800321 points = refBuffer<float>(points, count);
322 paint = refPaint(paint);
323
324 addDrawOp(new (alloc()) DrawLinesOp(points, count, paint));
Romain Guy4aa90572010-09-26 18:40:37 -0700325}
326
Tom Hudson107843d2014-09-08 11:26:26 -0400327void DisplayListRenderer::drawPoints(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800328 points = refBuffer<float>(points, count);
329 paint = refPaint(paint);
330
331 addDrawOp(new (alloc()) DrawPointsOp(points, count, paint));
Romain Guyed6fcb02011-03-21 13:11:28 -0700332}
333
Tom Hudson107843d2014-09-08 11:26:26 -0400334void DisplayListRenderer::drawTextOnPath(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800335 const SkPath* path, float hOffset, float vOffset, const SkPaint* paint) {
Tom Hudson107843d2014-09-08 11:26:26 -0400336 if (!text || count <= 0) return;
Chris Craik2af46352012-11-26 18:30:17 -0800337
Chris Craik2af46352012-11-26 18:30:17 -0800338 text = refText(text, bytesCount);
339 path = refPath(path);
340 paint = refPaint(paint);
341
342 DrawOp* op = new (alloc()) DrawTextOnPathOp(text, bytesCount, count, path,
343 hOffset, vOffset, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800344 addDrawOp(op);
Romain Guy325740f2012-02-24 16:48:34 -0800345}
346
Tom Hudson107843d2014-09-08 11:26:26 -0400347void DisplayListRenderer::drawPosText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800348 const float* positions, const SkPaint* paint) {
Tom Hudson107843d2014-09-08 11:26:26 -0400349 if (!text || count <= 0) return;
Chris Craik2af46352012-11-26 18:30:17 -0800350
Chris Craik2af46352012-11-26 18:30:17 -0800351 text = refText(text, bytesCount);
352 positions = refBuffer<float>(positions, count * 2);
353 paint = refPaint(paint);
354
355 DrawOp* op = new (alloc()) DrawPosTextOp(text, bytesCount, count, positions, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800356 addDrawOp(op);
Romain Guyeb9a5362012-01-17 17:39:26 -0800357}
358
Chris Craikcce47eb2014-07-16 15:12:15 -0700359static void simplifyPaint(int color, SkPaint* paint) {
360 paint->setColor(color);
361 paint->setShader(NULL);
362 paint->setColorFilter(NULL);
363 paint->setLooper(NULL);
364 paint->setStrokeWidth(4 + 0.04 * paint->getTextSize());
365 paint->setStrokeJoin(SkPaint::kRound_Join);
366 paint->setLooper(NULL);
367}
368
Tom Hudson107843d2014-09-08 11:26:26 -0400369void DisplayListRenderer::drawText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800370 float x, float y, const float* positions, const SkPaint* paint,
Chris Craik41541822013-05-03 16:35:54 -0700371 float totalAdvance, const Rect& bounds, DrawOpMode drawOpMode) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800372
Tom Hudson107843d2014-09-08 11:26:26 -0400373 if (!text || count <= 0 || paintWillNotDrawText(*paint)) return;
Raph Levien996e57c2012-07-23 15:22:52 -0700374
Chris Craik2af46352012-11-26 18:30:17 -0800375 text = refText(text, bytesCount);
376 positions = refBuffer<float>(positions, count * 2);
Raph Levien996e57c2012-07-23 15:22:52 -0700377
Chris Craikcce47eb2014-07-16 15:12:15 -0700378 if (CC_UNLIKELY(mHighContrastText)) {
379 // high contrast draw path
380 int color = paint->getColor();
381 int channelSum = SkColorGetR(color) + SkColorGetG(color) + SkColorGetB(color);
382 bool darken = channelSum < (128 * 3);
383
384 // outline
385 SkPaint* outlinePaint = copyPaint(paint);
386 simplifyPaint(darken ? SK_ColorWHITE : SK_ColorBLACK, outlinePaint);
387 outlinePaint->setStyle(SkPaint::kStrokeAndFill_Style);
388 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
389 x, y, positions, outlinePaint, totalAdvance, bounds)); // bounds?
390
391 // inner
392 SkPaint* innerPaint = copyPaint(paint);
393 simplifyPaint(darken ? SK_ColorBLACK : SK_ColorWHITE, innerPaint);
394 innerPaint->setStyle(SkPaint::kFill_Style);
395 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
396 x, y, positions, innerPaint, totalAdvance, bounds));
397 } else {
398 // standard draw path
399 paint = refPaint(paint);
400
401 DrawOp* op = new (alloc()) DrawTextOp(text, bytesCount, count,
402 x, y, positions, paint, totalAdvance, bounds);
403 addDrawOp(op);
404 }
Raph Levien996e57c2012-07-23 15:22:52 -0700405}
406
Tom Hudson107843d2014-09-08 11:26:26 -0400407void DisplayListRenderer::drawRects(const float* rects, int count, const SkPaint* paint) {
408 if (count <= 0) return;
Romain Guy672433d2013-01-04 19:05:13 -0800409
Chris Craik2af46352012-11-26 18:30:17 -0800410 rects = refBuffer<float>(rects, count);
411 paint = refPaint(paint);
412 addDrawOp(new (alloc()) DrawRectsOp(rects, count, paint));
Romain Guy672433d2013-01-04 19:05:13 -0800413}
414
Derek Sollenberger09c2d4f2014-10-15 09:21:10 -0400415void DisplayListRenderer::setDrawFilter(SkDrawFilter* filter) {
416 mDrawFilter.reset(filter);
Chris Craik2af46352012-11-26 18:30:17 -0800417}
418
Chris Craik8afd0f22014-08-21 17:41:57 -0700419void DisplayListRenderer::insertReorderBarrier(bool enableReorder) {
420 flushRestoreToCount();
421 flushTranslate();
422 mDeferredBarrierType = enableReorder ? kBarrier_OutOfOrder : kBarrier_InOrder;
423}
424
425void DisplayListRenderer::flushRestoreToCount() {
Chris Craik2af46352012-11-26 18:30:17 -0800426 if (mRestoreSaveCount >= 0) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700427 addOpAndUpdateChunk(new (alloc()) RestoreToCountOp(mRestoreSaveCount));
Chris Craik2af46352012-11-26 18:30:17 -0800428 mRestoreSaveCount = -1;
429 }
430}
431
Chris Craik8afd0f22014-08-21 17:41:57 -0700432void DisplayListRenderer::flushTranslate() {
433 if (mHasDeferredTranslate) {
Chris Craik2af46352012-11-26 18:30:17 -0800434 if (mTranslateX != 0.0f || mTranslateY != 0.0f) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700435 addOpAndUpdateChunk(new (alloc()) TranslateOp(mTranslateX, mTranslateY));
Chris Craik2af46352012-11-26 18:30:17 -0800436 mTranslateX = mTranslateY = 0.0f;
437 }
Chris Craik8afd0f22014-08-21 17:41:57 -0700438 mHasDeferredTranslate = false;
Chris Craik2af46352012-11-26 18:30:17 -0800439 }
440}
441
Chris Craik8afd0f22014-08-21 17:41:57 -0700442size_t DisplayListRenderer::addOpAndUpdateChunk(DisplayListOp* op) {
443 int insertIndex = mDisplayListData->displayListOps.add(op);
444 if (mDeferredBarrierType != kBarrier_None) {
445 // op is first in new chunk
446 mDisplayListData->chunks.push();
447 DisplayListData::Chunk& newChunk = mDisplayListData->chunks.editTop();
448 newChunk.beginOpIndex = insertIndex;
449 newChunk.endOpIndex = insertIndex + 1;
450 newChunk.reorderChildren = (mDeferredBarrierType == kBarrier_OutOfOrder);
451
452 int nextChildIndex = mDisplayListData->children().size();
453 newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex;
454 mDeferredBarrierType = kBarrier_None;
455 } else {
456 // standard case - append to existing chunk
457 mDisplayListData->chunks.editTop().endOpIndex = insertIndex + 1;
458 }
459 return insertIndex;
Chris Craik2af46352012-11-26 18:30:17 -0800460}
461
Chris Craik8afd0f22014-08-21 17:41:57 -0700462size_t DisplayListRenderer::flushAndAddOp(DisplayListOp* op) {
463 flushRestoreToCount();
464 flushTranslate();
465 return addOpAndUpdateChunk(op);
466}
467
468size_t DisplayListRenderer::addStateOp(StateOp* op) {
469 return flushAndAddOp(op);
470}
471
472size_t DisplayListRenderer::addDrawOp(DrawOp* op) {
Chris Craik2af46352012-11-26 18:30:17 -0800473 Rect localBounds;
John Reck3b202512014-06-23 13:13:08 -0700474 if (op->getLocalBounds(localBounds)) {
Chris Craikf0a59072013-11-19 18:00:46 -0800475 bool rejected = quickRejectConservative(localBounds.left, localBounds.top,
Chris Craik2af46352012-11-26 18:30:17 -0800476 localBounds.right, localBounds.bottom);
477 op->setQuickRejected(rejected);
478 }
Chris Craikc1c5f082013-09-11 16:23:37 -0700479
John Reck44fd8d22014-02-26 11:00:11 -0800480 mDisplayListData->hasDrawOps = true;
Chris Craik8afd0f22014-08-21 17:41:57 -0700481 return flushAndAddOp(op);
482}
483
484size_t DisplayListRenderer::addRenderNodeOp(DrawRenderNodeOp* op) {
485 int opIndex = addDrawOp(op);
486 int childIndex = mDisplayListData->addChild(op);
487
488 // update the chunk's child indices
489 DisplayListData::Chunk& chunk = mDisplayListData->chunks.editTop();
490 chunk.endChildIndex = childIndex + 1;
491
492 if (op->renderNode()->stagingProperties().isProjectionReceiver()) {
493 // use staging property, since recording on UI thread
494 mDisplayListData->projectionReceiveIndex = opIndex;
495 }
496 return opIndex;
Romain Guy5ff9df62012-01-23 17:09:05 -0800497}
498
Romain Guy4aa90572010-09-26 18:40:37 -0700499}; // namespace uirenderer
500}; // namespace android