Flutter Engine Uber Docs
Docs for the entire Flutter Engine repo.
 
Loading...
Searching...
No Matches
dl_sk_conversions.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
12#include "third_party/skia/include/core/SkColorFilter.h"
13#include "third_party/skia/include/effects/SkGradient.h"
14#include "third_party/skia/include/effects/SkImageFilters.h"
15#include "third_party/skia/include/effects/SkRuntimeEffect.h"
16
17namespace flutter {
18
19// clang-format off
20constexpr float kInvertColorMatrix[20] = {
21 -1.0, 0, 0, 1.0, 0,
22 0, -1.0, 0, 1.0, 0,
23 0, 0, -1.0, 1.0, 0,
24 1.0, 1.0, 1.0, 1.0, 0
25};
26// clang-format on
27
28SkPaint ToSk(const DlPaint& paint) {
29 SkPaint sk_paint;
30
31 sk_paint.setAntiAlias(paint.isAntiAlias());
32 sk_paint.setColor(ToSkColor4f(paint.getColor()));
33 sk_paint.setBlendMode(ToSk(paint.getBlendMode()));
34 sk_paint.setStyle(ToSk(paint.getDrawStyle()));
35 sk_paint.setStrokeWidth(paint.getStrokeWidth());
36 sk_paint.setStrokeMiter(paint.getStrokeMiter());
37 sk_paint.setStrokeCap(ToSk(paint.getStrokeCap()));
38 sk_paint.setStrokeJoin(ToSk(paint.getStrokeJoin()));
39 sk_paint.setImageFilter(ToSk(paint.getImageFilterPtr()));
40 auto color_filter = ToSk(paint.getColorFilterPtr());
41 if (paint.isInvertColors()) {
42 auto invert_filter = SkColorFilters::Matrix(kInvertColorMatrix);
43 if (color_filter) {
44 invert_filter = invert_filter->makeComposed(color_filter);
45 }
46 color_filter = invert_filter;
47 }
48 sk_paint.setColorFilter(color_filter);
49
50 auto color_source = paint.getColorSourcePtr();
51 if (color_source) {
52 // Unconditionally set dither to true for gradient shaders.
53 sk_paint.setDither(color_source->isGradient());
54 sk_paint.setShader(ToSk(color_source));
55 }
56
57 sk_paint.setMaskFilter(ToSk(paint.getMaskFilterPtr()));
58
59 return sk_paint;
60}
61
62SkPaint ToStrokedSk(const DlPaint& paint) {
63 DlPaint stroked_paint = paint;
65 return ToSk(stroked_paint);
66}
67
68SkPaint ToNonShaderSk(const DlPaint& paint) {
69 DlPaint non_shader_paint = paint;
70 non_shader_paint.setColorSource(nullptr);
71 return ToSk(non_shader_paint);
72}
73
74sk_sp<SkShader> ToSk(const DlColorSource* source) {
75 if (!source) {
76 return nullptr;
77 }
78 SkMatrix scratch_matrix;
79 std::vector<SkColor4f> scratch_colors;
80 static auto ToSkGradient =
81 [](const DlGradientColorSourceBase* gradient,
82 std::vector<SkColor4f>& scratch_colors) -> SkGradient {
83 scratch_colors.clear();
84 scratch_colors.reserve(gradient->stop_count());
85 for (int i = 0; i < gradient->stop_count(); ++i) {
86 DlColor color = gradient->colors()[i];
87 // TODO(180904): avoid clamp and truncation
88 scratch_colors.push_back(SkColor4f::FromColor(color.argb()));
89 // DlColor esrgb = color.withColorSpace(DlColorSpace::kExtendedSRGB);
90 // scratch_colors.emplace_back(esrgb.getRedF(), esrgb.getGreenF(),
91 // esrgb.getBlueF(), esrgb.getAlphaF());
92 }
93 return SkGradient(
94 SkGradient::Colors(scratch_colors,
95 {gradient->stops(), gradient->stop_count()},
96 ToSk(gradient->tile_mode())),
97 SkGradient::Interpolation());
98 };
99 switch (source->type()) {
101 const DlImageColorSource* image_source = source->asImage();
102 FML_DCHECK(image_source != nullptr);
103 auto image = image_source->image();
104 if (!image || !image->skia_image()) {
105 return nullptr;
106 }
107 return image->skia_image()->makeShader(
108 ToSk(image_source->horizontal_tile_mode()),
109 ToSk(image_source->vertical_tile_mode()),
110 ToSk(image_source->sampling()),
111 ToSk(image_source->matrix_ptr(), scratch_matrix));
112 }
114 const DlLinearGradientColorSource* linear_source =
115 source->asLinearGradient();
116 FML_DCHECK(linear_source != nullptr);
117 SkPoint pts[] = {ToSkPoint(linear_source->start_point()),
118 ToSkPoint(linear_source->end_point())};
119 return SkShaders::LinearGradient(
120 pts, ToSkGradient(linear_source, scratch_colors),
121 ToSk(linear_source->matrix_ptr(), scratch_matrix));
122 }
124 const DlRadialGradientColorSource* radial_source =
125 source->asRadialGradient();
126 FML_DCHECK(radial_source != nullptr);
127 return SkShaders::RadialGradient(
128 ToSkPoint(radial_source->center()), radial_source->radius(),
129 ToSkGradient(radial_source, scratch_colors),
130 ToSk(radial_source->matrix_ptr(), scratch_matrix));
131 }
133 const DlConicalGradientColorSource* conical_source =
134 source->asConicalGradient();
135 FML_DCHECK(conical_source != nullptr);
136 return SkShaders::TwoPointConicalGradient(
137 ToSkPoint(conical_source->start_center()),
138 conical_source->start_radius(),
139 ToSkPoint(conical_source->end_center()), conical_source->end_radius(),
140 ToSkGradient(conical_source, scratch_colors),
141 ToSk(conical_source->matrix_ptr(), scratch_matrix));
142 }
144 const DlSweepGradientColorSource* sweep_source =
145 source->asSweepGradient();
146 FML_DCHECK(sweep_source != nullptr);
147 return SkShaders::SweepGradient(
148 SkPoint(sweep_source->center().x, sweep_source->center().y),
149 sweep_source->start(), sweep_source->end(),
150 ToSkGradient(sweep_source, scratch_colors),
151 ToSk(sweep_source->matrix_ptr(), scratch_matrix));
152 }
154 const DlRuntimeEffectColorSource* runtime_source =
155 source->asRuntimeEffect();
156 FML_DCHECK(runtime_source != nullptr);
157 auto runtime_effect = runtime_source->runtime_effect();
158 if (!runtime_effect || !runtime_effect->skia_runtime_effect()) {
159 return nullptr;
160 }
161
162 auto samplers = runtime_source->samplers();
163 std::vector<sk_sp<SkShader>> sk_samplers(samplers.size());
164 for (size_t i = 0; i < samplers.size(); i++) {
165 const auto& sampler = samplers[i];
166 if (sampler == nullptr) {
167 return nullptr;
168 }
169 sk_samplers[i] = ToSk(sampler);
170 }
171
172 auto uniform_data = runtime_source->uniform_data();
173 auto ref = new std::shared_ptr<std::vector<uint8_t>>(uniform_data);
174 auto sk_uniform_data = SkData::MakeWithProc(
175 uniform_data->data(), uniform_data->size(),
176 [](const void* ptr, void* context) {
177 delete reinterpret_cast<std::shared_ptr<std::vector<uint8_t>>*>(
178 context);
179 },
180 ref);
181
182 return runtime_effect->skia_runtime_effect()->makeShader(
183 sk_uniform_data, sk_samplers.data(), sk_samplers.size());
184 }
185 }
186}
187
188sk_sp<SkImageFilter> ToSk(const DlImageFilter* filter) {
189 if (!filter) {
190 return nullptr;
191 }
192 switch (filter->type()) {
194 const DlBlurImageFilter* blur_filter = filter->asBlur();
195 FML_DCHECK(blur_filter != nullptr);
196 return SkImageFilters::Blur(blur_filter->sigma_x(),
197 blur_filter->sigma_y(),
198 ToSk(blur_filter->tile_mode()), nullptr);
199 }
201 const DlDilateImageFilter* dilate_filter = filter->asDilate();
202 FML_DCHECK(dilate_filter != nullptr);
203 return SkImageFilters::Dilate(dilate_filter->radius_x(),
204 dilate_filter->radius_y(), nullptr);
205 }
207 const DlErodeImageFilter* erode_filter = filter->asErode();
208 FML_DCHECK(erode_filter != nullptr);
209 return SkImageFilters::Erode(erode_filter->radius_x(),
210 erode_filter->radius_y(), nullptr);
211 }
213 const DlMatrixImageFilter* matrix_filter = filter->asMatrix();
214 FML_DCHECK(matrix_filter != nullptr);
215 return SkImageFilters::MatrixTransform(
216 ToSkMatrix(matrix_filter->matrix()), ToSk(matrix_filter->sampling()),
217 nullptr);
218 }
220 const DlComposeImageFilter* compose_filter = filter->asCompose();
221 FML_DCHECK(compose_filter != nullptr);
222 return SkImageFilters::Compose(ToSk(compose_filter->outer()),
223 ToSk(compose_filter->inner()));
224 }
226 const DlColorFilterImageFilter* cf_filter = filter->asColorFilter();
227 FML_DCHECK(cf_filter != nullptr);
228 return SkImageFilters::ColorFilter(ToSk(cf_filter->color_filter()),
229 nullptr);
230 }
232 const DlLocalMatrixImageFilter* lm_filter = filter->asLocalMatrix();
233 FML_DCHECK(lm_filter != nullptr);
234 sk_sp<SkImageFilter> skia_filter = ToSk(lm_filter->image_filter());
235 // The image_filter property itself might have been null, or the
236 // construction of the SkImageFilter might be optimized to null
237 // for any number of reasons. In any case, if the filter is null
238 // or optimizaed away, let's then optimize away this local matrix
239 // case by returning null.
240 if (!skia_filter) {
241 return nullptr;
242 }
243 return skia_filter->makeWithLocalMatrix(ToSkMatrix(lm_filter->matrix()));
244 }
246 // UNSUPPORTED.
247 return nullptr;
248 }
249}
250
251sk_sp<SkColorFilter> ToSk(const DlColorFilter* filter) {
252 if (!filter) {
253 return nullptr;
254 }
255 switch (filter->type()) {
257 const DlBlendColorFilter* blend_filter = filter->asBlend();
258 FML_DCHECK(blend_filter != nullptr);
259 return SkColorFilters::Blend(ToSkColor4f(blend_filter->color()), nullptr,
260 ToSk(blend_filter->mode()));
261 }
263 const DlMatrixColorFilter* matrix_filter = filter->asMatrix();
264 FML_DCHECK(matrix_filter != nullptr);
265 float matrix[20];
266 matrix_filter->get_matrix(matrix);
267 return SkColorFilters::Matrix(matrix);
268 }
270 return SkColorFilters::SRGBToLinearGamma();
271 }
273 return SkColorFilters::LinearToSRGBGamma();
274 }
275 }
276}
277
278sk_sp<SkMaskFilter> ToSk(const DlMaskFilter* filter) {
279 if (!filter) {
280 return nullptr;
281 }
282 switch (filter->type()) {
284 const DlBlurMaskFilter* blur_filter = filter->asBlur();
285 FML_DCHECK(blur_filter != nullptr);
286 return SkMaskFilter::MakeBlur(ToSk(blur_filter->style()),
287 blur_filter->sigma(),
288 blur_filter->respectCTM());
289 }
290 }
291}
292
293sk_sp<SkVertices> ToSk(const std::shared_ptr<DlVertices>& vertices) {
294 std::vector<SkColor> sk_colors;
295 const SkColor* sk_colors_ptr = nullptr;
296 if (vertices->colors()) {
297 sk_colors.reserve(vertices->vertex_count());
298 for (int i = 0; i < vertices->vertex_count(); ++i) {
299 sk_colors.push_back(vertices->colors()[i].argb());
300 }
301 sk_colors_ptr = sk_colors.data();
302 }
303 return SkVertices::MakeCopy(ToSk(vertices->mode()), vertices->vertex_count(),
304 ToSkPoints(vertices->vertex_data()),
305 ToSkPoints(vertices->texture_coordinate_data()),
306 sk_colors_ptr, vertices->index_count(),
307 vertices->indices());
308}
309
310} // namespace flutter
virtual T type() const =0
DlBlurStyle style() const
virtual const DlBlendColorFilter * asBlend() const
virtual const DlMatrixColorFilter * asMatrix() const
const std::shared_ptr< const DlColorFilter > color_filter() const
virtual const DlRuntimeEffectColorSource * asRuntimeEffect() const
virtual const DlRadialGradientColorSource * asRadialGradient() const
virtual const DlLinearGradientColorSource * asLinearGradient() const
virtual const DlImageColorSource * asImage() const
virtual const DlSweepGradientColorSource * asSweepGradient() const
virtual const DlConicalGradientColorSource * asConicalGradient() const
std::shared_ptr< DlImageFilter > inner() const
std::shared_ptr< DlImageFilter > outer() const
DlImageSampling sampling() const
DlTileMode horizontal_tile_mode() const
sk_sp< const DlImage > image() const
virtual const DlLocalMatrixImageFilter * asLocalMatrix() const
virtual const DlColorFilterImageFilter * asColorFilter() const
virtual const DlMatrixImageFilter * asMatrix() const
virtual const DlComposeImageFilter * asCompose() const
virtual const DlBlurImageFilter * asBlur() const
virtual const DlDilateImageFilter * asDilate() const
virtual const DlErodeImageFilter * asErode() const
const std::shared_ptr< DlImageFilter > image_filter() const
virtual const DlBlurMaskFilter * asBlur() const
void get_matrix(float matrix[20]) const
const DlMatrix & matrix() const
bool isAntiAlias() const
Definition dl_paint.h:57
DlStrokeCap getStrokeCap() const
Definition dl_paint.h:98
DlColor getColor() const
Definition dl_paint.h:69
DlBlendMode getBlendMode() const
Definition dl_paint.h:82
float getStrokeMiter() const
Definition dl_paint.h:120
DlStrokeJoin getStrokeJoin() const
Definition dl_paint.h:106
const DlColorSource * getColorSourcePtr() const
Definition dl_paint.h:129
const DlMaskFilter * getMaskFilterPtr() const
Definition dl_paint.h:183
const DlColorFilter * getColorFilterPtr() const
Definition dl_paint.h:147
const DlImageFilter * getImageFilterPtr() const
Definition dl_paint.h:165
DlDrawStyle getDrawStyle() const
Definition dl_paint.h:90
float getStrokeWidth() const
Definition dl_paint.h:114
DlPaint & setDrawStyle(DlDrawStyle style)
Definition dl_paint.h:93
DlPaint & setColorSource(std::nullptr_t source)
Definition dl_paint.h:131
bool isInvertColors() const
Definition dl_paint.h:63
const std::shared_ptr< std::vector< uint8_t > > uniform_data() const
const sk_sp< DlRuntimeEffect > runtime_effect() const
const std::vector< std::shared_ptr< DlColorSource > > samplers() const
FlutterVulkanImage * image
#define FML_DCHECK(condition)
Definition logging.h:122
SkPaint ToSk(const DlPaint &paint)
const SkPoint & ToSkPoint(const DlPoint &point)
SkPaint ToStrokedSk(const DlPaint &paint)
SkColor4f ToSkColor4f(DlColor color)
SkPaint ToNonShaderSk(const DlPaint &paint)
@ kStroke
strokes boundary of shapes
constexpr float kInvertColorMatrix[20]
SkMatrix ToSkMatrix(const DlMatrix &matrix)
const SkPoint * ToSkPoints(const DlPoint *points)
uint32_t argb() const
Definition dl_color.h:158