/* -*- Mode: c++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef MOZILLA_GFX_RECT_H_ #define MOZILLA_GFX_RECT_H_ #include "BaseRect.h" #include "BaseMargin.h" #include "NumericTools.h" #include "Point.h" #include "Tools.h" #include namespace mozilla { template struct IsPixel; namespace gfx { template struct IntMarginTyped: public BaseMargin >, public units { static_assert(IsPixel::value, "'units' must be a coordinate system tag"); typedef BaseMargin > Super; IntMarginTyped() : Super() {} IntMarginTyped(int32_t aTop, int32_t aRight, int32_t aBottom, int32_t aLeft) : Super(aTop, aRight, aBottom, aLeft) {} // XXX When all of the code is ported, the following functions to convert // to and from unknown types should be removed. static IntMarginTyped FromUnknownMargin(const IntMarginTyped& aMargin) { return IntMarginTyped(aMargin.top, aMargin.right, aMargin.bottom, aMargin.left); } IntMarginTyped ToUnknownMargin() const { return IntMarginTyped(this->top, this->right, this->bottom, this->left); } }; typedef IntMarginTyped IntMargin; template struct MarginTyped: public BaseMargin >, public units { static_assert(IsPixel::value, "'units' must be a coordinate system tag"); typedef BaseMargin > Super; MarginTyped() : Super() {} MarginTyped(F aTop, F aRight, F aBottom, F aLeft) : Super(aTop, aRight, aBottom, aLeft) {} explicit MarginTyped(const IntMarginTyped& aMargin) : Super(F(aMargin.top), F(aMargin.right), F(aMargin.bottom), F(aMargin.left)) {} }; typedef MarginTyped Margin; typedef MarginTyped MarginDouble; template IntMarginTyped RoundedToInt(const MarginTyped& aMargin) { return IntMarginTyped(int32_t(floorf(aMargin.top + 0.5f)), int32_t(floorf(aMargin.right + 0.5f)), int32_t(floorf(aMargin.bottom + 0.5f)), int32_t(floorf(aMargin.left + 0.5f))); } template struct IntRectTyped : public BaseRect, IntPointTyped, IntSizeTyped, IntMarginTyped >, public units { static_assert(IsPixel::value, "'units' must be a coordinate system tag"); typedef BaseRect, IntPointTyped, IntSizeTyped, IntMarginTyped > Super; IntRectTyped() : Super() {} IntRectTyped(const IntPointTyped& aPos, const IntSizeTyped& aSize) : Super(aPos, aSize) {} IntRectTyped(int32_t _x, int32_t _y, int32_t _width, int32_t _height) : Super(_x, _y, _width, _height) {} // Rounding isn't meaningful on an integer rectangle. void Round() {} void RoundIn() {} void RoundOut() {} // XXX When all of the code is ported, the following functions to convert // to and from unknown types should be removed. static IntRectTyped FromUnknownRect(const IntRectTyped& rect) { return IntRectTyped(rect.x, rect.y, rect.width, rect.height); } IntRectTyped ToUnknownRect() const { return IntRectTyped(this->x, this->y, this->width, this->height); } bool Overflows() const { CheckedInt xMost = this->x; xMost += this->width; CheckedInt yMost = this->y; yMost += this->height; return !xMost.isValid() || !yMost.isValid(); } // This is here only to keep IPDL-generated code happy. DO NOT USE. bool operator==(const IntRectTyped& aRect) const { return IntRectTyped::IsEqualEdges(aRect); } void InflateToMultiple(const IntSizeTyped& aTileSize) { int32_t yMost = this->YMost(); int32_t xMost = this->XMost(); this->x = mozilla::RoundDownToMultiple(this->x, aTileSize.width); this->y = mozilla::RoundDownToMultiple(this->y, aTileSize.height); xMost = mozilla::RoundUpToMultiple(xMost, aTileSize.width); yMost = mozilla::RoundUpToMultiple(yMost, aTileSize.height); this->width = xMost - this->x; this->height = yMost - this->y; } }; typedef IntRectTyped IntRect; template struct RectTyped : public BaseRect, PointTyped, SizeTyped, MarginTyped >, public units { static_assert(IsPixel::value, "'units' must be a coordinate system tag"); typedef BaseRect, PointTyped, SizeTyped, MarginTyped > Super; RectTyped() : Super() {} RectTyped(const PointTyped& aPos, const SizeTyped& aSize) : Super(aPos, aSize) {} RectTyped(F _x, F _y, F _width, F _height) : Super(_x, _y, _width, _height) {} explicit RectTyped(const IntRectTyped& rect) : Super(F(rect.x), F(rect.y), F(rect.width), F(rect.height)) {} // Returns the largest rectangle that can be represented with 32-bit // signed integers, centered around a point at 0,0. As BaseRect's represent // the dimensions as a top-left point with a width and height, the width // and height will be the largest positive 32-bit value. The top-left // position coordinate is divided by two to center the rectangle around a // point at 0,0. static RectTyped MaxIntRect() { return RectTyped( -std::numeric_limits::max() * 0.5, -std::numeric_limits::max() * 0.5, std::numeric_limits::max(), std::numeric_limits::max() ); }; void NudgeToIntegers() { NudgeToInteger(&(this->x)); NudgeToInteger(&(this->y)); NudgeToInteger(&(this->width)); NudgeToInteger(&(this->height)); } bool ToIntRect(IntRectTyped *aOut) const { *aOut = IntRectTyped(int32_t(this->X()), int32_t(this->Y()), int32_t(this->Width()), int32_t(this->Height())); return RectTyped(F(aOut->x), F(aOut->y), F(aOut->width), F(aOut->height)) .IsEqualEdges(*this); } // XXX When all of the code is ported, the following functions to convert to and from // unknown types should be removed. static RectTyped FromUnknownRect(const RectTyped& rect) { return RectTyped(rect.x, rect.y, rect.width, rect.height); } RectTyped ToUnknownRect() const { return RectTyped(this->x, this->y, this->width, this->height); } // This is here only to keep IPDL-generated code happy. DO NOT USE. bool operator==(const RectTyped& aRect) const { return RectTyped::IsEqualEdges(aRect); } }; typedef RectTyped Rect; typedef RectTyped RectDouble; template IntRectTyped RoundedToInt(const RectTyped& aRect) { RectTyped copy(aRect); copy.Round(); return IntRectTyped(int32_t(copy.x), int32_t(copy.y), int32_t(copy.width), int32_t(copy.height)); } template IntRectTyped RoundedIn(const RectTyped& aRect) { RectTyped copy(aRect); copy.RoundIn(); return IntRectTyped(int32_t(copy.x), int32_t(copy.y), int32_t(copy.width), int32_t(copy.height)); } template IntRectTyped RoundedOut(const RectTyped& aRect) { RectTyped copy(aRect); copy.RoundOut(); return IntRectTyped(int32_t(copy.x), int32_t(copy.y), int32_t(copy.width), int32_t(copy.height)); } template IntRectTyped TruncatedToInt(const RectTyped& aRect) { return IntRectTyped(int32_t(aRect.x), int32_t(aRect.y), int32_t(aRect.width), int32_t(aRect.height)); } template RectTyped IntRectToRect(const IntRectTyped& aRect) { return RectTyped(aRect.x, aRect.y, aRect.width, aRect.height); } } // namespace gfx } // namespace mozilla #endif /* MOZILLA_GFX_RECT_H_ */