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