blob: 8dcaa7de5151d5a7cf7b5f005a500673879edb91 [file] [log] [blame] [edit]
// Copyright 2013 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef FLUTTER_TESTING_DISPLAY_LIST_TESTING_H_
#define FLUTTER_TESTING_DISPLAY_LIST_TESTING_H_
#include <ostream>
#include "flutter/display_list/display_list.h"
#include "flutter/display_list/dl_op_receiver.h"
namespace flutter {
namespace testing {
[[nodiscard]] bool DisplayListsEQ_Verbose(const DisplayList* a,
const DisplayList* b);
[[nodiscard]] bool inline DisplayListsEQ_Verbose(const DisplayList& a,
const DisplayList& b) {
return DisplayListsEQ_Verbose(&a, &b);
}
[[nodiscard]] bool inline DisplayListsEQ_Verbose(
const sk_sp<const DisplayList>& a,
const sk_sp<const DisplayList>& b) {
return DisplayListsEQ_Verbose(a.get(), b.get());
}
[[nodiscard]] bool DisplayListsNE_Verbose(const DisplayList* a,
const DisplayList* b);
[[nodiscard]] bool inline DisplayListsNE_Verbose(const DisplayList& a,
const DisplayList& b) {
return DisplayListsNE_Verbose(&a, &b);
}
[[nodiscard]] bool inline DisplayListsNE_Verbose(
const sk_sp<const DisplayList>& a,
const sk_sp<const DisplayList>& b) {
return DisplayListsNE_Verbose(a.get(), b.get());
}
} // namespace testing
} // namespace flutter
namespace std {
extern std::ostream& operator<<(std::ostream& os,
const flutter::DisplayList& display_list);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlPaint& paint);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlBlendMode& mode);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlCanvas::ClipOp& op);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlCanvas::PointMode& op);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlCanvas::SrcRectConstraint& op);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlStrokeCap& cap);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlStrokeJoin& join);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlDrawStyle& style);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlBlurStyle& style);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlFilterMode& mode);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlColor& color);
extern std::ostream& operator<<(std::ostream& os,
flutter::DlImageSampling sampling);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlVertexMode& mode);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlTileMode& mode);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DlImage* image);
extern std::ostream& operator<<(std::ostream& os,
const flutter::SaveLayerOptions& image);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DisplayListOpType& type);
extern std::ostream& operator<<(std::ostream& os,
const flutter::DisplayListOpCategory& category);
extern std::ostream& operator<<(std::ostream& os, const flutter::DlPath& path);
} // namespace std
namespace flutter {
namespace testing {
class DisplayListStreamDispatcher final : public DlOpReceiver {
public:
explicit DisplayListStreamDispatcher(std::ostream& os,
int cur_indent = 2,
int indent = 2)
: os_(os), cur_indent_(cur_indent), indent_(indent) {}
void setAntiAlias(bool aa) override;
void setDrawStyle(DlDrawStyle style) override;
void setColor(DlColor color) override;
void setStrokeWidth(DlScalar width) override;
void setStrokeMiter(DlScalar limit) override;
void setStrokeCap(DlStrokeCap cap) override;
void setStrokeJoin(DlStrokeJoin join) override;
void setColorSource(const DlColorSource* source) override;
void setColorFilter(const DlColorFilter* filter) override;
void setInvertColors(bool invert) override;
void setBlendMode(DlBlendMode mode) override;
void setMaskFilter(const DlMaskFilter* filter) override;
void setImageFilter(const DlImageFilter* filter) override;
void save() override;
void saveLayer(const DlRect& bounds,
const SaveLayerOptions options,
const DlImageFilter* backdrop) override;
void restore() override;
void translate(DlScalar tx, DlScalar ty) override;
void scale(DlScalar sx, DlScalar sy) override;
void rotate(DlScalar degrees) override;
void skew(DlScalar sx, DlScalar sy) override;
// clang-format off
void transform2DAffine(DlScalar mxx, DlScalar mxy, DlScalar mxt,
DlScalar myx, DlScalar myy, DlScalar myt) override;
void transformFullPerspective(
DlScalar mxx, DlScalar mxy, DlScalar mxz, DlScalar mxt,
DlScalar myx, DlScalar myy, DlScalar myz, DlScalar myt,
DlScalar mzx, DlScalar mzy, DlScalar mzz, DlScalar mzt,
DlScalar mwx, DlScalar mwy, DlScalar mwz, DlScalar mwt) override;
// clang-format on
void transformReset() override;
void clipRect(const DlRect& rect, ClipOp clip_op, bool is_aa) override;
void clipOval(const DlRect& bounds, ClipOp clip_op, bool is_aa) override;
void clipRRect(const SkRRect& rrect, ClipOp clip_op, bool is_aa) override;
void clipPath(const DlPath& path, ClipOp clip_op, bool is_aa) override;
void drawColor(DlColor color, DlBlendMode mode) override;
void drawPaint() override;
void drawLine(const DlPoint& p0, const DlPoint& p1) override;
void drawDashedLine(const DlPoint& p0,
const DlPoint& p1,
DlScalar on_length,
DlScalar off_length) override;
void drawRect(const DlRect& rect) override;
void drawOval(const DlRect& bounds) override;
void drawCircle(const DlPoint& center, DlScalar radius) override;
void drawRRect(const SkRRect& rrect) override;
void drawDRRect(const SkRRect& outer, const SkRRect& inner) override;
void drawPath(const DlPath& path) override;
void drawArc(const DlRect& oval_bounds,
DlScalar start_degrees,
DlScalar sweep_degrees,
bool use_center) override;
void drawPoints(PointMode mode,
uint32_t count,
const DlPoint points[]) override;
void drawVertices(const std::shared_ptr<DlVertices>& vertices,
DlBlendMode mode) override;
void drawImage(const sk_sp<DlImage> image,
const DlPoint& point,
DlImageSampling sampling,
bool render_with_attributes) override;
void drawImageRect(const sk_sp<DlImage> image,
const DlRect& src,
const DlRect& dst,
DlImageSampling sampling,
bool render_with_attributes,
SrcRectConstraint constraint) override;
void drawImageNine(const sk_sp<DlImage> image,
const DlIRect& center,
const DlRect& dst,
DlFilterMode filter,
bool render_with_attributes) override;
void drawAtlas(const sk_sp<DlImage> atlas,
const SkRSXform xform[],
const DlRect tex[],
const DlColor colors[],
int count,
DlBlendMode mode,
DlImageSampling sampling,
const DlRect* cull_rect,
bool render_with_attributes) override;
void drawDisplayList(const sk_sp<DisplayList> display_list,
DlScalar opacity) override;
void drawTextBlob(const sk_sp<SkTextBlob> blob,
DlScalar x,
DlScalar y) override;
void drawTextFrame(const std::shared_ptr<impeller::TextFrame>& text_frame,
DlScalar x,
DlScalar y) override;
void drawShadow(const DlPath& path,
const DlColor color,
const DlScalar elevation,
bool transparent_occluder,
DlScalar dpr) override;
private:
std::ostream& os_;
int cur_indent_;
int indent_;
void indent() { indent(indent_); }
void outdent() { outdent(indent_); }
void indent(int spaces) { cur_indent_ += spaces; }
void outdent(int spaces) { cur_indent_ -= spaces; }
template <class T>
std::ostream& out_array(std::string name, int count, const T array[]);
std::ostream& startl();
void out(const DlColorFilter& filter);
void out(const DlColorFilter* filter);
void out(const DlImageFilter& filter);
void out(const DlImageFilter* filter);
};
class DisplayListGeneralReceiver : public DlOpReceiver {
public:
DisplayListGeneralReceiver() {
type_counts_.fill(0u);
category_counts_.fill(0u);
}
void setAntiAlias(bool aa) override {
RecordByType(DisplayListOpType::kSetAntiAlias);
}
void setInvertColors(bool invert) override {
RecordByType(DisplayListOpType::kSetInvertColors);
}
void setStrokeCap(DlStrokeCap cap) override {
RecordByType(DisplayListOpType::kSetStrokeCap);
}
void setStrokeJoin(DlStrokeJoin join) override {
RecordByType(DisplayListOpType::kSetStrokeJoin);
}
void setDrawStyle(DlDrawStyle style) override {
RecordByType(DisplayListOpType::kSetStyle);
}
void setStrokeWidth(float width) override {
RecordByType(DisplayListOpType::kSetStrokeWidth);
}
void setStrokeMiter(float limit) override {
RecordByType(DisplayListOpType::kSetStrokeMiter);
}
void setColor(DlColor color) override {
RecordByType(DisplayListOpType::kSetColor);
}
void setBlendMode(DlBlendMode mode) override {
RecordByType(DisplayListOpType::kSetBlendMode);
}
void setColorSource(const DlColorSource* source) override {
if (source) {
switch (source->type()) {
case DlColorSourceType::kImage:
RecordByType(DisplayListOpType::kSetImageColorSource);
break;
case DlColorSourceType::kRuntimeEffect:
RecordByType(DisplayListOpType::kSetRuntimeEffectColorSource);
break;
case DlColorSourceType::kColor:
case DlColorSourceType::kLinearGradient:
case DlColorSourceType::kRadialGradient:
case DlColorSourceType::kConicalGradient:
case DlColorSourceType::kSweepGradient:
RecordByType(DisplayListOpType::kSetPodColorSource);
break;
}
} else {
RecordByType(DisplayListOpType::kClearColorSource);
}
}
void setImageFilter(const DlImageFilter* filter) override {
if (filter) {
switch (filter->type()) {
case DlImageFilterType::kBlur:
case DlImageFilterType::kDilate:
case DlImageFilterType::kErode:
case DlImageFilterType::kMatrix:
RecordByType(DisplayListOpType::kSetPodImageFilter);
break;
case DlImageFilterType::kCompose:
case DlImageFilterType::kLocalMatrix:
case DlImageFilterType::kColorFilter:
RecordByType(DisplayListOpType::kSetSharedImageFilter);
break;
}
} else {
RecordByType(DisplayListOpType::kClearImageFilter);
}
}
void setColorFilter(const DlColorFilter* filter) override {
if (filter) {
switch (filter->type()) {
case DlColorFilterType::kBlend:
case DlColorFilterType::kMatrix:
case DlColorFilterType::kLinearToSrgbGamma:
case DlColorFilterType::kSrgbToLinearGamma:
RecordByType(DisplayListOpType::kSetPodColorFilter);
break;
}
} else {
RecordByType(DisplayListOpType::kClearColorFilter);
}
}
void setMaskFilter(const DlMaskFilter* filter) override {
if (filter) {
switch (filter->type()) {
case DlMaskFilterType::kBlur:
RecordByType(DisplayListOpType::kSetPodMaskFilter);
break;
}
} else {
RecordByType(DisplayListOpType::kClearMaskFilter);
}
}
void translate(DlScalar tx, DlScalar ty) override {
RecordByType(DisplayListOpType::kTranslate);
}
void scale(DlScalar sx, DlScalar sy) override {
RecordByType(DisplayListOpType::kScale);
}
void rotate(DlScalar degrees) override {
RecordByType(DisplayListOpType::kRotate);
}
void skew(DlScalar sx, DlScalar sy) override {
RecordByType(DisplayListOpType::kSkew);
}
// clang-format off
// 2x3 2D affine subset of a 4x4 transform in row major order
void transform2DAffine(DlScalar mxx, DlScalar mxy, DlScalar mxt,
DlScalar myx, DlScalar myy, DlScalar myt) override {
RecordByType(DisplayListOpType::kTransform2DAffine);
}
// full 4x4 transform in row major order
void transformFullPerspective(
DlScalar mxx, DlScalar mxy, DlScalar mxz, DlScalar mxt,
DlScalar myx, DlScalar myy, DlScalar myz, DlScalar myt,
DlScalar mzx, DlScalar mzy, DlScalar mzz, DlScalar mzt,
DlScalar mwx, DlScalar mwy, DlScalar mwz, DlScalar mwt) override {
RecordByType(DisplayListOpType::kTransformFullPerspective);
}
// clang-format on
void transformReset() override {
RecordByType(DisplayListOpType::kTransformReset);
}
void clipRect(const DlRect& rect,
DlCanvas::ClipOp clip_op,
bool is_aa) override {
switch (clip_op) {
case DlCanvas::ClipOp::kIntersect:
RecordByType(DisplayListOpType::kClipIntersectRect);
break;
case DlCanvas::ClipOp::kDifference:
RecordByType(DisplayListOpType::kClipDifferenceRect);
break;
}
}
void clipOval(const DlRect& bounds,
DlCanvas::ClipOp clip_op,
bool is_aa) override {
switch (clip_op) {
case DlCanvas::ClipOp::kIntersect:
RecordByType(DisplayListOpType::kClipIntersectOval);
break;
case DlCanvas::ClipOp::kDifference:
RecordByType(DisplayListOpType::kClipDifferenceOval);
break;
}
}
void clipRRect(const SkRRect& rrect,
DlCanvas::ClipOp clip_op,
bool is_aa) override {
switch (clip_op) {
case DlCanvas::ClipOp::kIntersect:
RecordByType(DisplayListOpType::kClipIntersectRRect);
break;
case DlCanvas::ClipOp::kDifference:
RecordByType(DisplayListOpType::kClipDifferenceRRect);
break;
}
}
void clipPath(const DlPath& path,
DlCanvas::ClipOp clip_op,
bool is_aa) override {
switch (clip_op) {
case DlCanvas::ClipOp::kIntersect:
RecordByType(DisplayListOpType::kClipIntersectPath);
break;
case DlCanvas::ClipOp::kDifference:
RecordByType(DisplayListOpType::kClipDifferencePath);
break;
}
}
void save() override { RecordByType(DisplayListOpType::kSave); }
void saveLayer(const DlRect& bounds,
const SaveLayerOptions options,
const DlImageFilter* backdrop) override {
if (backdrop) {
RecordByType(DisplayListOpType::kSaveLayerBackdrop);
} else {
RecordByType(DisplayListOpType::kSaveLayer);
}
}
void restore() override { RecordByType(DisplayListOpType::kRestore); }
void drawColor(DlColor color, DlBlendMode mode) override {
RecordByType(DisplayListOpType::kDrawColor);
}
void drawPaint() override { RecordByType(DisplayListOpType::kDrawPaint); }
void drawLine(const DlPoint& p0, const DlPoint& p1) override {
RecordByType(DisplayListOpType::kDrawLine);
}
void drawDashedLine(const DlPoint& p0,
const DlPoint& p1,
DlScalar on_length,
DlScalar off_length) override {
RecordByType(DisplayListOpType::kDrawDashedLine);
}
void drawRect(const DlRect& rect) override {
RecordByType(DisplayListOpType::kDrawRect);
}
void drawOval(const DlRect& bounds) override {
RecordByType(DisplayListOpType::kDrawOval);
}
void drawCircle(const DlPoint& center, DlScalar radius) override {
RecordByType(DisplayListOpType::kDrawCircle);
}
void drawRRect(const SkRRect& rrect) override {
RecordByType(DisplayListOpType::kDrawRRect);
}
void drawDRRect(const SkRRect& outer, const SkRRect& inner) override {
RecordByType(DisplayListOpType::kDrawDRRect);
}
void drawPath(const DlPath& path) override {
RecordByType(DisplayListOpType::kDrawPath);
}
void drawArc(const DlRect& oval_bounds,
DlScalar start_degrees,
DlScalar sweep_degrees,
bool use_center) override {
RecordByType(DisplayListOpType::kDrawArc);
}
void drawPoints(DlCanvas::PointMode mode,
uint32_t count,
const DlPoint points[]) override {
switch (mode) {
case DlCanvas::PointMode::kPoints:
RecordByType(DisplayListOpType::kDrawPoints);
break;
case DlCanvas::PointMode::kLines:
RecordByType(DisplayListOpType::kDrawLines);
break;
case DlCanvas::PointMode::kPolygon:
RecordByType(DisplayListOpType::kDrawPolygon);
break;
}
}
void drawVertices(const std::shared_ptr<DlVertices>& vertices,
DlBlendMode mode) override {
RecordByType(DisplayListOpType::kDrawVertices);
}
void drawImage(const sk_sp<DlImage> image,
const DlPoint& point,
DlImageSampling sampling,
bool render_with_attributes) override {
if (render_with_attributes) {
RecordByType(DisplayListOpType::kDrawImageWithAttr);
} else {
RecordByType(DisplayListOpType::kDrawImage);
}
}
void drawImageRect(const sk_sp<DlImage> image,
const DlRect& src,
const DlRect& dst,
DlImageSampling sampling,
bool render_with_attributes,
SrcRectConstraint constraint) override {
RecordByType(DisplayListOpType::kDrawImageRect);
}
void drawImageNine(const sk_sp<DlImage> image,
const DlIRect& center,
const DlRect& dst,
DlFilterMode filter,
bool render_with_attributes) override {
if (render_with_attributes) {
RecordByType(DisplayListOpType::kDrawImageNineWithAttr);
} else {
RecordByType(DisplayListOpType::kDrawImageNine);
}
}
void drawAtlas(const sk_sp<DlImage> atlas,
const SkRSXform xform[],
const DlRect tex[],
const DlColor colors[],
int count,
DlBlendMode mode,
DlImageSampling sampling,
const DlRect* cull_rect,
bool render_with_attributes) override {
if (cull_rect) {
RecordByType(DisplayListOpType::kDrawAtlasCulled);
} else {
RecordByType(DisplayListOpType::kDrawAtlas);
}
}
void drawDisplayList(const sk_sp<DisplayList> display_list,
DlScalar opacity) override {
RecordByType(DisplayListOpType::kDrawDisplayList);
}
void drawTextBlob(const sk_sp<SkTextBlob> blob,
DlScalar x,
DlScalar y) override {
RecordByType(DisplayListOpType::kDrawTextBlob);
}
void drawTextFrame(const std::shared_ptr<impeller::TextFrame>& text_frame,
DlScalar x,
DlScalar y) override {
RecordByType(DisplayListOpType::kDrawTextFrame);
}
void drawShadow(const DlPath& path,
const DlColor color,
const DlScalar elevation,
bool transparent_occluder,
DlScalar dpr) override {
if (transparent_occluder) {
RecordByType(DisplayListOpType::kDrawShadowTransparentOccluder);
} else {
RecordByType(DisplayListOpType::kDrawShadow);
}
}
uint32_t GetOpsReceived() { return op_count_; }
uint32_t GetOpsReceived(DisplayListOpCategory category) {
return category_counts_[static_cast<int>(category)];
}
uint32_t GetOpsReceived(DisplayListOpType type) {
return type_counts_[static_cast<int>(type)];
}
protected:
virtual void RecordByType(DisplayListOpType type) {
type_counts_[static_cast<int>(type)]++;
RecordByCategory(DisplayList::GetOpCategory(type));
}
virtual void RecordByCategory(DisplayListOpCategory category) {
category_counts_[static_cast<int>(category)]++;
switch (category) {
case DisplayListOpCategory::kAttribute:
RecordAttribute();
break;
case DisplayListOpCategory::kTransform:
RecordTransform();
break;
case DisplayListOpCategory::kClip:
RecordClip();
break;
case DisplayListOpCategory::kSave:
RecordSave();
break;
case DisplayListOpCategory::kSaveLayer:
RecordSaveLayer();
break;
case DisplayListOpCategory::kRestore:
RecordRestore();
break;
case DisplayListOpCategory::kRendering:
RecordRendering();
break;
case DisplayListOpCategory::kSubDisplayList:
RecordSubDisplayList();
break;
case DisplayListOpCategory::kInvalidCategory:
RecordInvalid();
break;
}
}
virtual void RecordAttribute() { RecordOp(); }
virtual void RecordTransform() { RecordOp(); }
virtual void RecordClip() { RecordOp(); }
virtual void RecordSave() { RecordOp(); }
virtual void RecordSaveLayer() { RecordOp(); }
virtual void RecordRestore() { RecordOp(); }
virtual void RecordRendering() { RecordOp(); }
virtual void RecordSubDisplayList() { RecordOp(); }
virtual void RecordInvalid() { RecordOp(); }
virtual void RecordOp() { op_count_++; }
static constexpr size_t kTypeCount =
static_cast<size_t>(DisplayListOpType::kMaxOp) + 1;
static constexpr size_t kCategoryCount =
static_cast<size_t>(DisplayListOpCategory::kMaxCategory) + 1;
std::array<uint32_t, kTypeCount> type_counts_;
std::array<uint32_t, kCategoryCount> category_counts_;
uint32_t op_count_ = 0u;
};
} // namespace testing
} // namespace flutter
#endif // FLUTTER_TESTING_DISPLAY_LIST_TESTING_H_