Flutter Engine Uber Docs
Docs for the entire Flutter Engine repo.
 
Loading...
Searching...
No Matches
texture_contents.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
7#include <memory>
8#include <optional>
9#include <utility>
10
14#include "impeller/entity/texture_fill.frag.h"
15#include "impeller/entity/texture_fill.vert.h"
16#include "impeller/entity/texture_fill_strict_src.frag.h"
17#include "impeller/entity/tiled_texture_fill_external.frag.h"
21
22namespace impeller {
23
25
27
28std::shared_ptr<TextureContents> TextureContents::MakeRect(Rect destination) {
29 auto contents = std::make_shared<TextureContents>();
30 contents->destination_rect_ = destination;
31 return contents;
32}
33
34void TextureContents::SetLabel(std::string_view label) {
35 label_ = label;
36}
37
39 destination_rect_ = rect;
40}
41
42void TextureContents::SetTexture(std::shared_ptr<Texture> texture) {
43 texture_ = std::move(texture);
44}
45
46std::shared_ptr<Texture> TextureContents::GetTexture() const {
47 return texture_;
48}
49
51 opacity_ = opacity;
52}
53
55 stencil_enabled_ = enabled;
56}
57
59 inherited_opacity_ = opacity;
60}
61
63 return opacity_ * inherited_opacity_;
64}
65
66std::optional<Rect> TextureContents::GetCoverage(const Entity& entity) const {
67 if (GetOpacity() == 0) {
68 return std::nullopt;
69 }
70 return destination_rect_.TransformBounds(entity.GetTransform());
71};
72
73std::optional<Snapshot> TextureContents::RenderToSnapshot(
74 const ContentContext& renderer,
75 const Entity& entity,
76 const SnapshotOptions& options) const {
77 // Passthrough textures that have simple rectangle paths and complete source
78 // rects.
79 auto bounds = destination_rect_;
80 auto opacity = GetOpacity();
81 if (source_rect_ == Rect::MakeSize(texture_->GetSize()) &&
82 (opacity >= 1 - kEhCloseEnough || defer_applying_opacity_)) {
83 auto scale = Vector2(bounds.GetSize() / Size(texture_->GetSize()));
84 return Snapshot{.texture = texture_,
85 .transform = entity.GetTransform() *
86 Matrix::MakeTranslation(bounds.GetOrigin()) *
87 Matrix::MakeScale(scale),
88 .sampler_descriptor = options.sampler_descriptor.value_or(
89 sampler_descriptor_),
90 .opacity = opacity,
91 .needs_rasterization_for_runtime_effects =
92 snapshots_need_rasterization_for_runtime_effects_};
93 }
95 renderer, entity,
96 {.coverage_limit = std::nullopt,
97 .sampler_descriptor =
98 options.sampler_descriptor.value_or(sampler_descriptor_),
99 .msaa_enabled = true,
100 .mip_count = options.mip_count,
101 .label = options.label,
102 .coverage_expansion = options.coverage_expansion});
103}
104
106 const Entity& entity,
107 RenderPass& pass) const {
108 using VS = TextureFillVertexShader;
109 using FS = TextureFillFragmentShader;
110 using FSStrict = TextureFillStrictSrcFragmentShader;
111
112 if (destination_rect_.IsEmpty() || source_rect_.IsEmpty() ||
113 texture_ == nullptr || texture_->GetSize().IsEmpty()) {
114 return true; // Nothing to render.
115 }
116
117#ifdef IMPELLER_ENABLE_OPENGLES
118 using FSExternal = TiledTextureFillExternalFragmentShader;
119 bool is_external_texture =
120 texture_->GetTextureDescriptor().type == TextureType::kTextureExternalOES;
121#endif // IMPELLER_ENABLE_OPENGLES
122
123 auto texture_coords =
124 Rect::MakeSize(texture_->GetSize()).Project(source_rect_);
125 auto& data_host_buffer = renderer.GetTransientsDataBuffer();
126
127 std::array<VS::PerVertexData, 4> vertices = {
128 VS::PerVertexData{destination_rect_.GetLeftTop(),
129 texture_coords.GetLeftTop()},
130 VS::PerVertexData{destination_rect_.GetRightTop(),
131 texture_coords.GetRightTop()},
132 VS::PerVertexData{destination_rect_.GetLeftBottom(),
133 texture_coords.GetLeftBottom()},
134 VS::PerVertexData{destination_rect_.GetRightBottom(),
135 texture_coords.GetRightBottom()},
136 };
137 auto vertex_buffer = CreateVertexBuffer(vertices, data_host_buffer);
138
139 VS::FrameInfo frame_info;
140 frame_info.mvp = entity.GetShaderTransform(pass);
141 frame_info.texture_sampler_y_coord_scale = texture_->GetYCoordScale();
142
143#ifdef IMPELLER_DEBUG
144 if (label_.empty()) {
145 pass.SetCommandLabel("Texture Fill");
146 } else {
147 pass.SetCommandLabel("Texture Fill: " + label_);
148 }
149#endif // IMPELLER_DEBUG
150
151 auto pipeline_options = OptionsFromPassAndEntity(pass, entity);
152 if (!stencil_enabled_) {
153 pipeline_options.stencil_mode = ContentContextOptions::StencilMode::kIgnore;
154 }
155 pipeline_options.primitive_type = PrimitiveType::kTriangleStrip;
156
157 pipeline_options.depth_write_enabled =
158 stencil_enabled_ && pipeline_options.blend_mode == BlendMode::kSrc;
159
160#ifdef IMPELLER_ENABLE_OPENGLES
161 if (is_external_texture) {
162 pass.SetPipeline(
163 renderer.GetTiledTextureExternalPipeline(pipeline_options));
164 } else {
165 pass.SetPipeline(
166 strict_source_rect_enabled_
167 ? renderer.GetTextureStrictSrcPipeline(pipeline_options)
168 : renderer.GetTexturePipeline(pipeline_options));
169 }
170#else
171 pass.SetPipeline(strict_source_rect_enabled_
172 ? renderer.GetTextureStrictSrcPipeline(pipeline_options)
173 : renderer.GetTexturePipeline(pipeline_options));
174#endif // IMPELLER_ENABLE_OPENGLES
175
176 pass.SetVertexBuffer(vertex_buffer);
177 VS::BindFrameInfo(pass, data_host_buffer.EmplaceUniform(frame_info));
178
179 if (strict_source_rect_enabled_) {
180 // For a strict source rect, shrink the texture coordinate range by half a
181 // texel to ensure that linear filtering does not sample anything outside
182 // the source rect bounds.
183 auto strict_texture_coords =
184 Rect::MakeSize(texture_->GetSize()).Project(source_rect_.Expand(-0.5));
185
186 FSStrict::FragInfo frag_info;
187 frag_info.source_rect = Vector4(strict_texture_coords.GetLTRB());
188 frag_info.alpha = GetOpacity();
189 FSStrict::BindFragInfo(pass, data_host_buffer.EmplaceUniform((frag_info)));
190 FSStrict::BindTextureSampler(
191 pass, texture_,
192 renderer.GetContext()->GetSamplerLibrary()->GetSampler(
193 sampler_descriptor_));
194#ifdef IMPELLER_ENABLE_OPENGLES
195 } else if (is_external_texture) {
196 FSExternal::FragInfo frag_info;
197 frag_info.x_tile_mode =
198 static_cast<Scalar>(sampler_descriptor_.width_address_mode);
199 frag_info.y_tile_mode =
200 static_cast<Scalar>(sampler_descriptor_.height_address_mode);
201 frag_info.alpha = GetOpacity();
202 FSExternal::BindFragInfo(pass, data_host_buffer.EmplaceUniform(frag_info));
203
204 SamplerDescriptor sampler_desc;
205 // OES_EGL_image_external states that only CLAMP_TO_EDGE is valid, so
206 // we emulate all other tile modes here by remapping the texture
207 // coordinates.
210 sampler_desc.min_filter = sampler_descriptor_.min_filter;
211 sampler_desc.mag_filter = sampler_descriptor_.mag_filter;
212 sampler_desc.mip_filter = MipFilter::kBase;
213
214 FSExternal::BindSAMPLEREXTERNALOESTextureSampler(
215 pass, texture_,
216 renderer.GetContext()->GetSamplerLibrary()->GetSampler(sampler_desc));
217#endif // IMPELLER_ENABLE_OPENGLES
218 } else {
219 FS::FragInfo frag_info;
220 frag_info.alpha = GetOpacity();
221 FS::BindFragInfo(pass, data_host_buffer.EmplaceUniform((frag_info)));
222 FS::BindTextureSampler(
223 pass, texture_,
224 renderer.GetContext()->GetSamplerLibrary()->GetSampler(
225 sampler_descriptor_));
226 }
227 return pass.Draw().ok();
228}
229
230void TextureContents::SetSourceRect(const Rect& source_rect) {
231 source_rect_ = source_rect;
232}
233
235 return source_rect_;
236}
237
239 strict_source_rect_enabled_ = strict;
240}
241
243 return strict_source_rect_enabled_;
244}
245
247 sampler_descriptor_ = desc;
248}
249
251 return sampler_descriptor_;
252}
253
254void TextureContents::SetDeferApplyingOpacity(bool defer_applying_opacity) {
255 defer_applying_opacity_ = defer_applying_opacity;
256}
257
259 snapshots_need_rasterization_for_runtime_effects_ = value;
260}
261
262} // namespace impeller
bool ok() const
Definition status.h:71
HostBuffer & GetTransientsDataBuffer() const
Retrieve the current host buffer for transient storage of other non-index data.
PipelineRef GetTextureStrictSrcPipeline(ContentContextOptions opts) const
PipelineRef GetTexturePipeline(ContentContextOptions opts) const
std::shared_ptr< Context > GetContext() const
virtual std::optional< Snapshot > RenderToSnapshot(const ContentContext &renderer, const Entity &entity, const SnapshotOptions &options) const
Render this contents to a snapshot, respecting the entity's transform, path, clip depth,...
Definition contents.cc:56
Matrix GetShaderTransform(const RenderPass &pass) const
Definition entity.cc:50
const Matrix & GetTransform() const
Get the global transform matrix for this Entity.
Definition entity.cc:46
Render passes encode render commands directed as one specific render target into an underlying comman...
Definition render_pass.h:30
virtual bool SetVertexBuffer(VertexBuffer buffer)
Specify the vertex and index buffer to use for this command.
virtual void SetPipeline(PipelineRef pipeline)
The pipeline to use for this command.
virtual fml::Status Draw()
Record the currently pending command.
virtual void SetCommandLabel(std::string_view label)
The debugging label to use for the command.
std::shared_ptr< Texture > GetTexture() const
void SetSourceRect(const Rect &source_rect)
bool Render(const ContentContext &renderer, const Entity &entity, RenderPass &pass) const override
void SetSamplerDescriptor(const SamplerDescriptor &desc)
void SetStrictSourceRect(bool strict)
void SetDeferApplyingOpacity(bool defer_applying_opacity)
const SamplerDescriptor & GetSamplerDescriptor() const
void SetOpacity(Scalar opacity)
static std::shared_ptr< TextureContents > MakeRect(Rect destination)
const Rect & GetSourceRect() const
void SetInheritedOpacity(Scalar opacity) override
Inherit the provided opacity.
std::optional< Rect > GetCoverage(const Entity &entity) const override
Get the area of the render pass that will be affected when this contents is rendered.
void SetTexture(std::shared_ptr< Texture > texture)
void SetLabel(std::string_view label)
std::optional< Snapshot > RenderToSnapshot(const ContentContext &renderer, const Entity &entity, const SnapshotOptions &options) const override
Render this contents to a snapshot, respecting the entity's transform, path, clip depth,...
void SetStencilEnabled(bool enabled)
void SetDestinationRect(Rect rect)
void SetNeedsRasterizationForRuntimeEffects(bool value)
int32_t value
FlTexture * texture
Point Vector2
Definition point.h:331
float Scalar
Definition scalar.h:19
constexpr float kEhCloseEnough
Definition constants.h:57
LinePipeline::FragmentShader FS
VertexBuffer CreateVertexBuffer(std::array< VertexType, size > input, HostBuffer &data_host_buffer)
Create an index-less vertex buffer from a fixed size array.
@ kBase
The texture is sampled as if it only had a single mipmap level.
LinePipeline::VertexShader VS
ContentContextOptions OptionsFromPassAndEntity(const RenderPass &pass, const Entity &entity)
Definition contents.cc:34
const std::optional< SamplerDescriptor > & sampler_descriptor
Definition contents.h:87
static constexpr Matrix MakeTranslation(const Vector3 &t)
Definition matrix.h:95
static constexpr Matrix MakeScale(const Vector3 &s)
Definition matrix.h:104
SamplerAddressMode width_address_mode
SamplerAddressMode height_address_mode
Represents a texture and its intended draw transform/sampler configuration.
Definition snapshot.h:24
std::shared_ptr< Texture > texture
Definition snapshot.h:25
constexpr TRect TransformBounds(const Matrix &transform) const
Creates a new bounding box that contains this transformed rectangle.
Definition rect.h:472
constexpr TRect< T > Project(TRect< T > source) const
Returns a new rectangle that represents the projection of the source rectangle onto this rectangle....
Definition rect.h:669
constexpr bool IsEmpty() const
Returns true if either of the width or height are 0, negative, or NaN.
Definition rect.h:297
constexpr TPoint< T > GetLeftTop() const
Definition rect.h:359
constexpr TPoint< T > GetRightBottom() const
Definition rect.h:371
constexpr TPoint< T > GetLeftBottom() const
Definition rect.h:367
static constexpr TRect MakeSize(const TSize< U > &size)
Definition rect.h:150
constexpr TPoint< T > GetRightTop() const
Definition rect.h:363
constexpr TRect< T > Expand(T left, T top, T right, T bottom) const
Returns a rectangle with expanded edges. Negative expansion results in shrinking.
Definition rect.h:618