Rect.h revision 034a10bf216cdef251928edf72d93668d81515f8
1bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy/*
2bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * Copyright (C) 2010 The Android Open Source Project
3bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy *
4bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * Licensed under the Apache License, Version 2.0 (the "License");
5bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * you may not use this file except in compliance with the License.
6bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * You may obtain a copy of the License at
7bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy *
8bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy *      http://www.apache.org/licenses/LICENSE-2.0
9bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy *
10bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * Unless required by applicable law or agreed to in writing, software
11bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * distributed under the License is distributed on an "AS IS" BASIS,
12bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * See the License for the specific language governing permissions and
14bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy * limitations under the License.
15bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy */
16bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
1791eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#pragma once
185b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy
1991eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include "Vertex.h"
20bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
215cbbce535744b89df5ecea95de21ee3733298260Romain Guy#include <utils/Log.h>
225cbbce535744b89df5ecea95de21ee3733298260Romain Guy
2391eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include <algorithm>
2491eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include <cmath>
2591eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include <iomanip>
2691eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include <ostream>
2791eff22b5d7f8fe551bae01331948858ce932a96Chris Craik#include <SkRect.h>
2832f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik
29bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guynamespace android {
309d5316e3f56d138504565ff311145ac01621dff4Romain Guynamespace uirenderer {
31bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
3262d307c2402777d5e53b4590af5f32f8c55afd81Chris Craik#define RECT_STRING "%5.2f %5.2f %5.2f %5.2f"
3328ce94a4ffc7576f40776d212f1ada79fafaa061Chris Craik#define RECT_ARGS(r) \
3428ce94a4ffc7576f40776d212f1ada79fafaa061Chris Craik    (r).left, (r).top, (r).right, (r).bottom
353f085429fd47ebd32ac2463b3eae2a5a6c17be25Chris Craik#define SK_RECT_ARGS(r) \
363f085429fd47ebd32ac2463b3eae2a5a6c17be25Chris Craik    (r).left(), (r).top(), (r).right(), (r).bottom()
3728ce94a4ffc7576f40776d212f1ada79fafaa061Chris Craik
38bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy///////////////////////////////////////////////////////////////////////////////
39bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy// Structs
40bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy///////////////////////////////////////////////////////////////////////////////
41bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
4283b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopianclass Rect {
4383b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopianpublic:
447ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    float left;
457ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    float top;
467ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    float right;
477ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    float bottom;
487ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
495b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    // Used by Region
505b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    typedef float value_type;
515b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy
5283b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian    // we don't provide copy-ctor and operator= on purpose
5383b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian    // because we want the compiler generated versions
5483b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian
555b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline Rect():
567ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            left(0),
577ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            top(0),
587ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            right(0),
597ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            bottom(0) {
607ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
617ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
625b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline Rect(float left, float top, float right, float bottom):
637ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            left(left),
647ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            top(top),
657ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            right(right),
667ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy            bottom(bottom) {
677ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
687ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
695b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline Rect(float width, float height):
705b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy            left(0.0f),
715b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy            top(0.0f),
725b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy            right(width),
735b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy            bottom(height) {
745b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    }
755b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy
76af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui    inline Rect(const SkRect& rect):
77af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui            left(rect.fLeft),
78af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui            top(rect.fTop),
79af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui            right(rect.fRight),
80af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui            bottom(rect.fBottom) {
81af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui    }
82af6f7ed8dd4288a41d0a07a1f0f0be7d6d035b33ztenghui
837ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    friend int operator==(const Rect& a, const Rect& b) {
847ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        return !memcmp(&a, &b, sizeof(a));
857ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
867ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
877ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    friend int operator!=(const Rect& a, const Rect& b) {
887ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        return memcmp(&a, &b, sizeof(a));
897ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
907ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
915b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline void clear() {
925b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        left = top = right = bottom = 0.0f;
935b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    }
945b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy
955b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline bool isEmpty() const {
9683b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian        // this is written in such way this it'll handle NANs to return
9783b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian        // true (empty)
9883b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian        return !((left < right) && (top < bottom));
997ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1007ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1015b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline void setEmpty() {
1025b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        left = top = right = bottom = 0.0f;
1037ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1047ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1055b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline void set(float left, float top, float right, float bottom) {
1067ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        this->left = left;
1077ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        this->right = right;
1087ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        this->top = top;
1097ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        this->bottom = bottom;
1107ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1117ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1125b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    inline void set(const Rect& r) {
1137ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        set(r.left, r.top, r.right, r.bottom);
1147ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1157ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
116487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    inline void set(const SkIRect& r) {
117487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk        set(r.left(), r.top(), r.right(), r.bottom());
118487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    }
119487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk
1208aef54fa17f2a3753d9a8f2027629bc480088f69Romain Guy    inline float getWidth() const {
1217ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        return right - left;
1227ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1237ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1248aef54fa17f2a3753d9a8f2027629bc480088f69Romain Guy    inline float getHeight() const {
1257ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        return bottom - top;
1267ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1277ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
12883b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian    bool intersects(float l, float t, float r, float b) const {
129ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        float tempLeft = std::max(left, l);
130ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        float tempTop = std::max(top, t);
131ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        float tempRight = std::min(right, r);
132ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        float tempBottom = std::min(bottom, b);
133ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik
134ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        return ((tempLeft < tempRight) && (tempTop < tempBottom)); // !isEmpty
1357ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1367ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1377ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    bool intersects(const Rect& r) const {
1387ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy        return intersects(r.left, r.top, r.right, r.bottom);
1397ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1407ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
141ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik    /**
142ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik     * This method is named 'doIntersect' instead of 'intersect' so as not to be confused with
143ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik     * SkRect::intersect / android.graphics.Rect#intersect behavior, which do not modify the object
144ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik     * if the intersection of the rects would be empty.
145ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik     */
146ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik    void doIntersect(float l, float t, float r, float b) {
147ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        left = std::max(left, l);
148ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        top = std::max(top, t);
149ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        right = std::min(right, r);
150ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        bottom = std::min(bottom, b);
1517ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1527ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
153ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik    void doIntersect(const Rect& r) {
154ac02eb9035a13a3d09c2def9ed63d04225eb2509Chris Craik        doIntersect(r.left, r.top, r.right, r.bottom);
1557ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
1567ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy
1572db5e993b626794eb07a0ff354269f9a77da81b3Romain Guy    inline bool contains(float l, float t, float r, float b) const {
158ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy        return l >= left && t >= top && r <= right && b <= bottom;
159ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy    }
160ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy
1612db5e993b626794eb07a0ff354269f9a77da81b3Romain Guy    inline bool contains(const Rect& r) const {
162ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy        return contains(r.left, r.top, r.right, r.bottom);
163ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy    }
164ec31f83bd3af1f900d1ee9116b15f56904c66dcdRomain Guy
165079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy    bool unionWith(const Rect& r) {
166079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy        if (r.left < r.right && r.top < r.bottom) {
167079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy            if (left < right && top < bottom) {
168079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                if (left > r.left) left = r.left;
169079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                if (top > r.top) top = r.top;
170079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                if (right < r.right) right = r.right;
171079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                if (bottom < r.bottom) bottom = r.bottom;
172079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                return true;
173079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy            } else {
174079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                left = r.left;
175079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                top = r.top;
176079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                right = r.right;
177079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                bottom = r.bottom;
178079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy                return true;
179079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy            }
180079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy        }
181079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy        return false;
182079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy    }
183079ba2c85b15e882629b8d188f5fbdb42f7f8eeaRomain Guy
1845b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    void translate(float dx, float dy) {
1855b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        left += dx;
1865b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        right += dx;
1875b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        top += dy;
1885b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy        bottom += dy;
1895b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy    }
1905b3b35296e8b2c8d3f07d32bb645d5414db41a1dRomain Guy
191e4aa95e3627226bcb0d8cc3e42dca6e4df8f421cChris Craik    void inset(float delta) {
192e4aa95e3627226bcb0d8cc3e42dca6e4df8f421cChris Craik        outset(-delta);
193e4aa95e3627226bcb0d8cc3e42dca6e4df8f421cChris Craik    }
194e4aa95e3627226bcb0d8cc3e42dca6e4df8f421cChris Craik
195c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik    void outset(float delta) {
196c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik        left -= delta;
197c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik        top -= delta;
198c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik        right += delta;
199c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik        bottom += delta;
200c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik    }
201c3566d06421c8acc0aafb18f7e307e5725ce87e1Chris Craik
20205f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik    void outset(float xdelta, float ydelta) {
20305f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik        left -= xdelta;
20405f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik        top -= ydelta;
20505f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik        right += xdelta;
20605f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik        bottom += ydelta;
20705f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik    }
20805f3d6e5111fd08df5cd9aae2c3d28399dc0e7f5Chris Craik
2095e49b307eb99269db2db257760508b8efd7bb97dChris Craik    /**
21032f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik     * Similar to snapToPixelBoundaries, but estimates bounds conservatively to handle GL rounding
21132f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik     * errors.
2125e49b307eb99269db2db257760508b8efd7bb97dChris Craik     *
21332f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik     * This function should be used whenever estimating the damage rect of geometry already mapped
21432f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik     * into layer space.
2155e49b307eb99269db2db257760508b8efd7bb97dChris Craik     */
21632f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik    void snapGeometryToPixelBoundaries(bool snapOut) {
21732f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik        if (snapOut) {
21832f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik            /* For AA geometry with a ramp perimeter, don't snap by rounding - AA geometry will have
21932f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * a 0.5 pixel perimeter not accounted for in its bounds. Instead, snap by
22032f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * conservatively rounding out the bounds with floor/ceil.
22132f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             *
22232f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * In order to avoid changing integer bounds with floor/ceil due to rounding errors
22332f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * inset the bounds first by the fudge factor. Very small fraction-of-a-pixel errors
22432f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * from this inset will only incur similarly small errors in output, due to transparency
22532f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * in extreme outside of the geometry.
22632f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             */
227564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            left = floorf(left + Vertex::GeometryFudgeFactor());
228564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            top = floorf(top + Vertex::GeometryFudgeFactor());
229564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            right = ceilf(right - Vertex::GeometryFudgeFactor());
230564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            bottom = ceilf(bottom - Vertex::GeometryFudgeFactor());
23132f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik        } else {
23232f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik            /* For other geometry, we do the regular rounding in order to snap, but also outset the
23332f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * bounds by a fudge factor. This ensures that ambiguous geometry (e.g. a non-AA Rect
23432f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             * with top left at (0.5, 0.5)) will err on the side of a larger damage rect.
23532f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik             */
236564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            left = floorf(left + 0.5f - Vertex::GeometryFudgeFactor());
237564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            top = floorf(top + 0.5f - Vertex::GeometryFudgeFactor());
238564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            right = floorf(right + 0.5f + Vertex::GeometryFudgeFactor());
239564acf7c9bff822f608cda0d5df0a64a9f9aaefdChris Craik            bottom = floorf(bottom + 0.5f + Vertex::GeometryFudgeFactor());
24032f05e343c5ffb17f3235942bcda651bd3b9f1d6Chris Craik        }
2415e49b307eb99269db2db257760508b8efd7bb97dChris Craik    }
2425e49b307eb99269db2db257760508b8efd7bb97dChris Craik
243bf434114cbf55b216fdc76fc8d65a75e84c9dab5Romain Guy    void snapToPixelBoundaries() {
244ae88e5e8e9cb6c9539314c4360c5b20f8ec1fefcRomain Guy        left = floorf(left + 0.5f);
245ae88e5e8e9cb6c9539314c4360c5b20f8ec1fefcRomain Guy        top = floorf(top + 0.5f);
246ae88e5e8e9cb6c9539314c4360c5b20f8ec1fefcRomain Guy        right = floorf(right + 0.5f);
247ae88e5e8e9cb6c9539314c4360c5b20f8ec1fefcRomain Guy        bottom = floorf(bottom + 0.5f);
248bf434114cbf55b216fdc76fc8d65a75e84c9dab5Romain Guy    }
249bf434114cbf55b216fdc76fc8d65a75e84c9dab5Romain Guy
250f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik    void roundOut() {
251f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik        left = floorf(left);
252f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik        top = floorf(top);
253f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik        right = ceilf(right);
254f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik        bottom = ceilf(bottom);
255f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik    }
256f0a590781b2c3e34132b2011d3956135add73ae0Chris Craik
25715c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik    /*
25815c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik     * Similar to unionWith, except this makes the assumption that both rects are non-empty
25915c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik     * to avoid both emptiness checks.
26015c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik     */
26115c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik    void expandToCover(const Rect& other) {
26215c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik        left = std::min(left, other.left);
26315c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik        top = std::min(top, other.top);
26415c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik        right = std::max(right, other.right);
26515c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik        bottom = std::max(bottom, other.bottom);
26615c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik    }
26715c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik
26815c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik    void expandToCover(float x, float y) {
269df72b63928cc1492b72ba9a4e99d5e714f93ccc6Chris Craik        left = std::min(left, x);
270df72b63928cc1492b72ba9a4e99d5e714f93ccc6Chris Craik        top = std::min(top, y);
271df72b63928cc1492b72ba9a4e99d5e714f93ccc6Chris Craik        right = std::max(right, x);
272df72b63928cc1492b72ba9a4e99d5e714f93ccc6Chris Craik        bottom = std::max(bottom, y);
273c93e45cf045f41aea95f856173e4043d988a5a5cChris Craik    }
274c93e45cf045f41aea95f856173e4043d988a5a5cChris Craik
275487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    SkRect toSkRect() const {
276487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk        return SkRect::MakeLTRB(left, top, right, bottom);
277487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    }
278487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk
279487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    SkIRect toSkIRect() const {
280487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk        return SkIRect::MakeLTRB(left, top, right, bottom);
281487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk    }
282487a92caef2eb90a62e8f8d7a6fe6315f1c1d8d8Rob Tsuk
283e84a208317e0ed388fcdad1e6743c7849acb51b0Chris Craik    void dump(const char* label = nullptr) const {
2848ecf41c61a5185207a21d64681e8ebc2502b7b2aChris Craik        ALOGD("%s[l=%.2f t=%.2f r=%.2f b=%.2f]", label ? label : "Rect", left, top, right, bottom);
2857ae7ac48aa2b53453c9805075171ecd5bcafd7deRomain Guy    }
28691eff22b5d7f8fe551bae01331948858ce932a96Chris Craik
28791eff22b5d7f8fe551bae01331948858ce932a96Chris Craik    friend std::ostream& operator<<(std::ostream& os, const Rect& rect) {
28891eff22b5d7f8fe551bae01331948858ce932a96Chris Craik        if (rect.isEmpty()) {
289034a10bf216cdef251928edf72d93668d81515f8Chris Craik            // Print empty, but continue, since empty rects may still have useful coordinate info
290034a10bf216cdef251928edf72d93668d81515f8Chris Craik            os << "(empty)";
29191eff22b5d7f8fe551bae01331948858ce932a96Chris Craik        }
29291eff22b5d7f8fe551bae01331948858ce932a96Chris Craik
29391eff22b5d7f8fe551bae01331948858ce932a96Chris Craik        if (rect.left == 0 && rect.top == 0) {
29491eff22b5d7f8fe551bae01331948858ce932a96Chris Craik            return os << "[" << rect.right << " x " << rect.bottom << "]";
29591eff22b5d7f8fe551bae01331948858ce932a96Chris Craik        }
29691eff22b5d7f8fe551bae01331948858ce932a96Chris Craik
29791eff22b5d7f8fe551bae01331948858ce932a96Chris Craik        return os << "[" << rect.left
29891eff22b5d7f8fe551bae01331948858ce932a96Chris Craik                << " " << rect.top
29991eff22b5d7f8fe551bae01331948858ce932a96Chris Craik                << " " << rect.right
30091eff22b5d7f8fe551bae01331948858ce932a96Chris Craik                << " " << rect.bottom << "]";
30191eff22b5d7f8fe551bae01331948858ce932a96Chris Craik    }
30283b186a246e8ffd52b91a17c0019dd8c9c9d21b1Mathias Agopian}; // class Rect
303bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
3049d5316e3f56d138504565ff311145ac01621dff4Romain Guy}; // namespace uirenderer
305bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy}; // namespace android
306bb9524b6bdddc7ac77d8628daa8b366b8a7be4a4Romain Guy
307