| /* |
| * Copyright (C) 2024 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. |
| */ |
| |
| import {Point} from 'common/geometry_types'; |
| import {PropertyTreeNode} from 'trace/tree_node/property_tree_node'; |
| |
| export class Rect { |
| constructor(public x: number, public y: number, public w: number, public h: number) {} |
| |
| static from(node: PropertyTreeNode): Rect { |
| const left = node.getChildByName('left')?.getValue() ?? 0; |
| const top = node.getChildByName('top')?.getValue() ?? 0; |
| const right = node.getChildByName('right')?.getValue() ?? 0; |
| const bottom = node.getChildByName('bottom')?.getValue() ?? 0; |
| return new Rect(left, top, right - left, bottom - top); |
| } |
| |
| containsPoint(point: Point): boolean { |
| return ( |
| this.x <= point.x && |
| point.x <= this.x + this.w && |
| this.y <= point.y && |
| point.y <= this.y + this.h |
| ); |
| } |
| |
| cropRect(other: Rect): Rect { |
| const maxLeft = Math.max(this.x, other.x); |
| const minRight = Math.min(this.x + this.w, other.x + other.w); |
| const maxTop = Math.max(this.y, other.y); |
| const minBottom = Math.min(this.y + this.h, other.y + other.h); |
| return new Rect(maxLeft, maxTop, minRight - maxLeft, minBottom - maxTop); |
| } |
| |
| containsRect(other: Rect): boolean { |
| return ( |
| this.w > 0 && |
| this.h > 0 && |
| this.x <= other.x && |
| this.y <= other.y && |
| this.x + this.w >= other.x + other.w && |
| this.y + this.h >= other.y + other.h |
| ); |
| } |
| |
| intersectsRect(other: Rect): boolean { |
| if ( |
| this.x < other.x + other.w && |
| other.x < this.x + this.w && |
| this.y <= other.y + other.h && |
| other.y <= this.y + this.h |
| ) { |
| const intersectionRect = new Rect(this.x, this.y, this.w, this.h); |
| |
| if (this.x < other.x) { |
| intersectionRect.x = other.x; |
| } |
| if (this.y < other.y) { |
| intersectionRect.y = other.y; |
| } |
| if (this.x + this.w > other.x + other.w) { |
| intersectionRect.w = other.w; |
| } |
| if (this.y + this.h > other.y + other.h) { |
| intersectionRect.h = other.h; |
| } |
| |
| return !intersectionRect.isEmpty(); |
| } |
| |
| return false; |
| } |
| |
| isEmpty(): boolean { |
| const [x, y, w, h] = [this.x, this.y, this.w, this.h]; |
| const nullValuePresent = x === -1 || y === -1 || x + w === -1 || y + h === -1; |
| const nullHeightOrWidth = w <= 0 || h <= 0; |
| return nullValuePresent || nullHeightOrWidth; |
| } |
| } |