Flutter Impeller
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 
11 #include "impeller/core/formats.h"
13 #include "impeller/entity/entity.h"
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 
22 namespace impeller {
23 
25 
27 
28 std::shared_ptr<TextureContents> TextureContents::MakeRect(Rect destination) {
29  auto contents = std::make_shared<TextureContents>();
30  contents->destination_rect_ = destination;
31  return contents;
32 }
33 
34 void TextureContents::SetLabel(std::string_view label) {
35  label_ = label;
36 }
37 
39  destination_rect_ = rect;
40 }
41 
42 void TextureContents::SetTexture(std::shared_ptr<Texture> texture) {
43  texture_ = std::move(texture);
44 }
45 
46 std::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 
66 std::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 
73 std::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 #if defined(IMPELLER_ENABLE_OPENGLES) && !defined(FML_OS_EMSCRIPTEN)
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 #if defined(IMPELLER_ENABLE_OPENGLES) && !defined(FML_OS_EMSCRIPTEN)
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 #if defined(IMPELLER_ENABLE_OPENGLES) && !defined(FML_OS_EMSCRIPTEN)
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 
230 void 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 
254 void 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
PipelineRef GetTextureStrictSrcPipeline(ContentContextOptions opts) const
PipelineRef GetTexturePipeline(ContentContextOptions opts) const
HostBuffer & GetTransientsDataBuffer() const
Retrieve the current host buffer for transient storage of other non-index data.
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.
Definition: render_pass.cc:127
virtual void SetPipeline(PipelineRef pipeline)
The pipeline to use for this command.
Definition: render_pass.cc:86
virtual fml::Status Draw()
Record the currently pending command.
Definition: render_pass.cc:208
virtual void SetCommandLabel(std::string_view label)
The debugging label to use for the command.
Definition: render_pass.cc:97
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
Point Vector2
Definition: point.h:429
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
@ kIgnore
Turn the stencil test off. Used when drawing without stencil-then-cover.
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< 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
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 > GetLeftBottom() const
Definition: rect.h:367
constexpr TPoint< T > GetRightTop() const
Definition: rect.h:363
constexpr static TRect MakeSize(const TSize< U > &size)
Definition: rect.h:150
constexpr TPoint< T > GetRightBottom() const
Definition: rect.h:371
constexpr TPoint< T > GetLeftTop() const
Definition: rect.h:359