/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef CLIPAREA_H
#define CLIPAREA_H
#include "Matrix.h"
#include "Rect.h"
#include "utils/Pair.h"
#include <SkRegion.h>
namespace android {
namespace uirenderer {
class LinearAllocator;
Rect transformAndCalculateBounds(const Rect& r, const Matrix4& transform);
class TransformedRectangle {
public:
TransformedRectangle();
TransformedRectangle(const Rect& bounds, const Matrix4& transform);
bool canSimplyIntersectWith(const TransformedRectangle& other) const;
void intersectWith(const TransformedRectangle& other);
bool isEmpty() const;
const Rect& getBounds() const {
return mBounds;
}
Rect transformedBounds() const {
Rect transformedBounds(transformAndCalculateBounds(mBounds, mTransform));
return transformedBounds;
}
const Matrix4& getTransform() const {
return mTransform;
}
void transform(const Matrix4& transform) {
Matrix4 t;
t.loadMultiply(transform, mTransform);
mTransform = t;
}
private:
Rect mBounds;
Matrix4 mTransform;
};
class RectangleList {
public:
RectangleList();
bool isEmpty() const;
int getTransformedRectanglesCount() const;
const TransformedRectangle& getTransformedRectangle(int i) const;
void setEmpty();
void set(const Rect& bounds, const Matrix4& transform);
bool intersectWith(const Rect& bounds, const Matrix4& transform);
void transform(const Matrix4& transform);
SkRegion convertToRegion(const SkRegion& clip) const;
Rect calculateBounds() const;
enum {
kMaxTransformedRectangles = 5
};
private:
int mTransformedRectanglesCount;
TransformedRectangle mTransformedRectangles[kMaxTransformedRectangles];
};
enum class ClipMode {
Rectangle,
RectangleList,
// region and path - intersected. if either is empty, don't use
Region
};
struct ClipBase {
ClipBase(ClipMode mode)
: mode(mode) {}
ClipBase(const Rect& rect)
: mode(ClipMode::Rectangle)
, rect(rect) {}
const ClipMode mode;
bool intersectWithRoot = false;
// Bounds of the clipping area, used to define the scissor, and define which
// portion of the stencil is updated/used
Rect rect;
void dump() const;
};
struct ClipRect : ClipBase {
ClipRect(const Rect& rect)
: ClipBase(rect) {}
};
struct ClipRectList : ClipBase {
ClipRectList(const RectangleList& rectList)
: ClipBase(ClipMode::RectangleList)
, rectList(rectList) {}
RectangleList rectList;
};
struct ClipRegion : ClipBase {
ClipRegion(const SkRegion& region)
: ClipBase(ClipMode::Region)
, region(region) {}
ClipRegion()
: ClipBase(ClipMode::Region) {}
SkRegion region;
};
class ClipArea {
public:
ClipArea();
void setViewportDimensions(int width, int height);
bool isEmpty() const {
return mClipRect.isEmpty();
}
void setEmpty();
void setClip(float left, float top, float right, float bottom);
void clipRectWithTransform(const Rect& r, const mat4* transform,
SkRegion::Op op);
void clipRegion(const SkRegion& region, SkRegion::Op op);
void clipPathWithTransform(const SkPath& path, const mat4* transform,
SkRegion::Op op);
const Rect& getClipRect() const {
return mClipRect;
}
const SkRegion& getClipRegion() const {
return mClipRegion;
}
const RectangleList& getRectangleList() const {
return mRectangleList;
}
bool isRegion() const {
return ClipMode::Region == mMode;
}
bool isSimple() const {
return mMode == ClipMode::Rectangle;
}
bool isRectangleList() const {
return mMode == ClipMode::RectangleList;
}
WARN_UNUSED_RESULT const ClipBase* serializeClip(LinearAllocator& allocator);
WARN_UNUSED_RESULT const ClipBase* serializeIntersectedClip(LinearAllocator& allocator,
const ClipBase* recordedClip, const Matrix4& recordedClipTransform);
void applyClip(const ClipBase* recordedClip, const Matrix4& recordedClipTransform);
static void applyTransformToRegion(const Matrix4& transform, SkRegion* region);
private:
void enterRectangleMode();
void rectangleModeClipRectWithTransform(const Rect& r, const mat4* transform, SkRegion::Op op);
void enterRectangleListMode();
void rectangleListModeClipRectWithTransform(const Rect& r,
const mat4* transform, SkRegion::Op op);
void enterRegionModeFromRectangleMode();
void enterRegionModeFromRectangleListMode();
void enterRegionMode();
void regionModeClipRectWithTransform(const Rect& r, const mat4* transform,
SkRegion::Op op);
void ensureClipRegion();
void onClipRegionUpdated();
// Called by every state modifying public method.
void onClipUpdated() {
mPostViewportClipObserved = true;
mLastSerialization = nullptr;
mLastResolutionResult = nullptr;
}
SkRegion createViewportRegion() {
return SkRegion(mViewportBounds.toSkIRect());
}
void regionFromPath(const SkPath& path, SkRegion& pathAsRegion) {
// TODO: this should not mask every path to the viewport - this makes it impossible to use
// paths to clip to larger areas (which is valid e.g. with SkRegion::kReplace_Op)
pathAsRegion.setPath(path, createViewportRegion());
}
ClipMode mMode;
bool mPostViewportClipObserved = false;
bool mReplaceOpObserved = false;
/**
* If mLastSerialization is non-null, it represents an already serialized copy
* of the current clip state. If null, it has not been computed.
*/
const ClipBase* mLastSerialization = nullptr;
/**
* This pair of pointers is a single entry cache of most recently seen
*/
const ClipBase* mLastResolutionResult = nullptr;
const ClipBase* mLastResolutionClip = nullptr;
Matrix4 mLastResolutionTransform;
Rect mViewportBounds;
Rect mClipRect;
SkRegion mClipRegion;
RectangleList mRectangleList;
};
} /* namespace uirenderer */
} /* namespace android */
#endif /* CLIPAREA_H_ */