blob: 685a2a37f72c1c01543c02dc60a77862c3724780 [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef GrStyledShape_DEFINED
#define GrStyledShape_DEFINED
#include "include/core/SkPath.h"
#include "include/core/SkPathTypes.h"
#include "include/core/SkRRect.h"
#include "include/core/SkRect.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkScalar.h"
#include "include/private/base/SkAssert.h"
#include "include/private/base/SkTemplates.h"
#include "src/base/SkTLazy.h"
#include "src/core/SkPathEnums.h"
#include "src/core/SkPathPriv.h"
#include "src/gpu/ganesh/GrStyle.h"
#include "src/gpu/ganesh/geometry/GrShape.h"
#include <cstdint>
class SkIDChangeListener;
class SkPaint;
struct SkPoint;
/**
* Represents a geometric shape (rrect or path) and the GrStyle that it should be rendered with.
* It is possible to apply the style to the GrStyledShape to produce a new GrStyledShape where the
* geometry reflects the styling information (e.g. is stroked). It is also possible to apply just
* the path effect from the style. In this case the resulting shape will include any remaining
* stroking information that is to be applied after the path effect.
*
* Shapes can produce keys that represent only the geometry information, not the style. Note that
* when styling information is applied to produce a new shape then the style has been converted
* to geometric information and is included in the new shape's key. When the same style is applied
* to two shapes that reflect the same underlying geometry the computed keys of the stylized shapes
* will be the same.
*
* Currently this can only be constructed from a path, rect, or rrect though it can become a path
* applying style to the geometry. The idea is to expand this to cover most or all of the geometries
* that have fast paths in the GPU backend.
*/
class GrStyledShape {
public:
// Keys for paths may be extracted from the path data for small paths. Clients aren't supposed
// to have to worry about this. This value is exposed for unit tests.
inline static constexpr int kMaxKeyFromDataVerbCnt = 10;
GrStyledShape() {}
enum class DoSimplify : bool { kNo = false, kYes };
explicit GrStyledShape(const SkPath& path, DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(path, GrStyle::SimpleFill(), doSimplify) {}
explicit GrStyledShape(const SkRRect& rrect, DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(rrect, GrStyle::SimpleFill(), doSimplify) {}
explicit GrStyledShape(const SkRect& rect, DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(rect, GrStyle::SimpleFill(), doSimplify) {}
GrStyledShape(const SkPath& path, const SkPaint& paint,
DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(path, GrStyle(paint), doSimplify) {}
GrStyledShape(const SkRRect& rrect, const SkPaint& paint,
DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(rrect, GrStyle(paint), doSimplify) {}
GrStyledShape(const SkRect& rect, const SkPaint& paint,
DoSimplify doSimplify = DoSimplify::kYes)
: GrStyledShape(rect, GrStyle(paint), doSimplify) {}
GrStyledShape(const SkPath& path, const GrStyle& style,
DoSimplify doSimplify = DoSimplify::kYes)
: fShape(path), fStyle(style) {
if (doSimplify == DoSimplify::kYes) {
this->simplify();
}
}
GrStyledShape(const SkRRect& rrect, const GrStyle& style,
DoSimplify doSimplify = DoSimplify::kYes)
// Preserve legacy indices (6 for CW), see SkPathBuilder::addRRect().
: GrStyledShape(rrect, SkPathDirection::kCW, 6, false, style, doSimplify) {}
GrStyledShape(const SkRRect& rrect, SkPathDirection dir, unsigned start, bool inverted,
const GrStyle& style, DoSimplify doSimplify = DoSimplify::kYes)
: fShape(rrect)
, fStyle(style) {
fShape.setPathWindingParams(dir, start);
fShape.setInverted(inverted);
if (doSimplify == DoSimplify::kYes) {
this->simplify();
}
}
GrStyledShape(const SkRect& rect, const GrStyle& style,
DoSimplify doSimplify = DoSimplify::kYes)
: fShape(rect), fStyle(style) {
if (doSimplify == DoSimplify::kYes) {
this->simplify();
}
}
GrStyledShape(const GrStyledShape&);
static GrStyledShape MakeArc(const SkRect& oval, SkScalar startAngleDegrees,
SkScalar sweepAngleDegrees, bool useCenter, const GrStyle& style,
DoSimplify = DoSimplify::kYes);
GrStyledShape& operator=(const GrStyledShape& that);
/**
* Informs MakeFilled on how to modify that shape's fill rule when making a simple filled
* version of the shape.
*/
enum class FillInversion {
kPreserve,
kFlip,
kForceNoninverted,
kForceInverted
};
/**
* Makes a filled shape from the pre-styled original shape and optionally modifies whether
* the fill is inverted or not. It's important to note that the original shape's geometry
* may already have been modified if doing so was neutral with respect to its style
* (e.g. filled paths are always closed when stored in a shape and dashed paths are always
* made non-inverted since dashing ignores inverseness).
*/
static GrStyledShape MakeFilled(const GrStyledShape& original,
FillInversion = FillInversion::kPreserve);
const GrStyle& style() const { return fStyle; }
// True if the shape and/or style were modified into a simpler, equivalent pairing
bool simplified() const { return fSimplified; }
/**
* Returns a shape that has either applied the path effect or path effect and stroking
* information from this shape's style to its geometry. Scale is used when approximating the
* output geometry and typically is computed from the view matrix
*/
GrStyledShape applyStyle(GrStyle::Apply apply, SkScalar scale) const {
return GrStyledShape(*this, apply, scale);
}
bool isRect() const {
// Should have simplified a rrect to a rect if possible already.
SkASSERT(!fShape.isRRect() || !fShape.rrect().isRect());
return fShape.isRect();
}
/** Returns the unstyled geometry as a rrect if possible. */
bool asRRect(SkRRect* rrect, SkPathDirection* dir, unsigned* start, bool* inverted) const;
/**
* If the unstyled shape is a straight line segment, returns true and sets pts to the endpoints.
* An inverse filled line path is still considered a line.
*/
bool asLine(SkPoint pts[2], bool* inverted) const;
// Can this shape be drawn as a pair of filled nested rectangles?
bool asNestedRects(SkRect rects[2]) const;
/** Returns the unstyled geometry as a path. */
void asPath(SkPath* out) const {
fShape.asPath(out, fStyle.isSimpleFill());
}
/**
* Returns whether the geometry is empty. Note that applying the style could produce a
* non-empty shape. It also may have an inverse fill.
*/
bool isEmpty() const { return fShape.isEmpty(); }
/**
* Gets the bounds of the geometry without reflecting the shape's styling. This ignores
* the inverse fill nature of the geometry.
*/
SkRect bounds() const { return fShape.bounds(); }
/**
* Gets the bounds of the geometry reflecting the shape's styling (ignoring inverse fill
* status).
*/
SkRect styledBounds() const;
/**
* Is this shape known to be convex, before styling is applied. An unclosed but otherwise
* convex path is considered to be closed if they styling reflects a fill and not otherwise.
* This is because filling closes all contours in the path.
*/
bool knownToBeConvex() const {
return fShape.convex(fStyle.isSimpleFill());
}
/**
* Does the shape have a known winding direction. Some degenerate convex shapes may not have
* a computable direction, but this is not always a requirement for path renderers so it is
* kept separate from knownToBeConvex().
*/
bool knownDirection() const {
// Assuming this is called after knownToBeConvex(), this should just be relying on
// cached convexity and direction and will be cheap.
return !fShape.isPath() ||
SkPathPriv::ComputeFirstDirection(fShape.path()) != SkPathFirstDirection::kUnknown;
}
/** Is the pre-styled geometry inverse filled? */
bool inverseFilled() const {
// Since the path tracks inverted-fillness itself, it should match what was recorded.
SkASSERT(!fShape.isPath() || fShape.inverted() == fShape.path().isInverseFillType());
// Dashing ignores inverseness. We should have caught this earlier. skbug.com/5421
SkASSERT(!(fShape.inverted() && this->style().isDashed()));
return fShape.inverted();
}
/**
* Might applying the styling to the geometry produce an inverse fill. The "may" part comes in
* because an arbitrary path effect could produce an inverse filled path. In other cases this
* can be thought of as "inverseFilledAfterStyling()".
*/
bool mayBeInverseFilledAfterStyling() const {
// An arbitrary path effect can produce an arbitrary output path, which may be inverse
// filled.
if (this->style().hasNonDashPathEffect()) {
return true;
}
return this->inverseFilled();
}
/**
* Is it known that the unstyled geometry has no unclosed contours. This means that it will
* not have any caps if stroked (modulo the effect of any path effect).
*/
bool knownToBeClosed() const {
// This refers to the base shape and does not depend on invertedness.
return fShape.closed();
}
uint32_t segmentMask() const {
// This refers to the base shape and does not depend on invertedness.
return fShape.segmentMask();
}
/**
* Gets the size of the key for the shape represented by this GrStyledShape (ignoring its
* styling). A negative value is returned if the shape has no key (shouldn't be cached).
*/
int unstyledKeySize() const;
bool hasUnstyledKey() const { return this->unstyledKeySize() >= 0; }
/**
* Writes unstyledKeySize() bytes into the provided pointer. Assumes that there is enough
* space allocated for the key and that unstyledKeySize() does not return a negative value
* for this shape.
*/
void writeUnstyledKey(uint32_t* key) const;
/**
* Adds a listener to the *original* path. Typically used to invalidate cached resources when
* a path is no longer in-use. If the shape started out as something other than a path, this
* does nothing.
*/
void addGenIDChangeListener(sk_sp<SkIDChangeListener>) const;
/**
* Helpers that are only exposed for unit tests, to determine if the shape is a path, and get
* the generation ID of the *original* path. This is the path that will receive
* GenIDChangeListeners added to this shape.
*/
uint32_t testingOnly_getOriginalGenerationID() const;
bool testingOnly_isPath() const;
bool testingOnly_isNonVolatilePath() const;
/**
* Similar to GrShape::simplify but also takes into account style and stroking, possibly
* applying the style explicitly to produce a new analytic shape with a simpler style.
* Unless "doSimplify" is kNo, this method gets called automatically during construction.
*/
void simplify();
private:
/** Constructor used by the applyStyle() function */
GrStyledShape(const GrStyledShape& parentShape, GrStyle::Apply, SkScalar scale);
/**
* Determines the key we should inherit from the input shape's geometry and style when
* we are applying the style to create a new shape.
*/
void setInheritedKey(const GrStyledShape& parentShape, GrStyle::Apply, SkScalar scale);
/**
* As part of the simplification process, some shapes can have stroking trivially evaluated
* and form a new geometry with just a fill.
*/
void simplifyStroke();
/** Gets the path that gen id listeners should be added to. */
const SkPath* originalPathForListeners() const;
GrShape fShape;
GrStyle fStyle;
// Gen ID of the original path (path may be modified or simplified away).
int32_t fGenID = 0;
bool fClosed = false;
bool fSimplified = false;
SkTLazy<SkPath> fInheritedPathForListeners;
skia_private::AutoSTArray<8, uint32_t> fInheritedKey;
};
#endif