2010-10-18 14:34:14 -04:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/* Simple box operations */
|
|
|
|
|
2014-05-02 09:34:02 -04:00
|
|
|
/*
|
2010-10-18 14:34:14 -04:00
|
|
|
* Copyright (C) 2005, 2006 Elijah Newren
|
2014-05-02 09:34:02 -04:00
|
|
|
*
|
2010-10-18 14:34:14 -04:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
2014-05-02 09:34:02 -04:00
|
|
|
*
|
2010-10-18 14:34:14 -04:00
|
|
|
* You should have received a copy of the GNU General Public License
|
2014-01-11 20:42:06 -05:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2010-10-18 14:34:14 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef META_BOXES_PRIVATE_H
|
|
|
|
#define META_BOXES_PRIVATE_H
|
|
|
|
|
|
|
|
#include <glib-object.h>
|
2018-07-10 04:36:24 -04:00
|
|
|
|
2018-12-21 11:12:49 -05:00
|
|
|
#include "backends/meta-backend-types.h"
|
2019-01-22 21:25:35 -05:00
|
|
|
#include "core/util-private.h"
|
2018-07-10 04:36:24 -04:00
|
|
|
#include "meta/boxes.h"
|
|
|
|
#include "meta/common.h"
|
2010-10-18 14:34:14 -04:00
|
|
|
|
|
|
|
#define BOX_LEFT(box) ((box).x) /* Leftmost pixel of rect */
|
|
|
|
#define BOX_RIGHT(box) ((box).x + (box).width) /* One pixel past right */
|
|
|
|
#define BOX_TOP(box) ((box).y) /* Topmost pixel of rect */
|
|
|
|
#define BOX_BOTTOM(box) ((box).y + (box).height) /* One pixel past bottom */
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
FIXED_DIRECTION_NONE = 0,
|
|
|
|
FIXED_DIRECTION_X = 1 << 0,
|
|
|
|
FIXED_DIRECTION_Y = 1 << 1,
|
|
|
|
} FixedDirections;
|
|
|
|
|
2018-12-20 10:58:03 -05:00
|
|
|
typedef enum _MetaRoundingStrategy
|
|
|
|
{
|
|
|
|
META_ROUNDING_STRATEGY_SHRINK,
|
|
|
|
META_ROUNDING_STRATEGY_GROW,
|
2020-01-15 13:27:54 -05:00
|
|
|
META_ROUNDING_STRATEGY_ROUND,
|
2018-12-20 10:58:03 -05:00
|
|
|
} MetaRoundingStrategy;
|
|
|
|
|
2010-10-18 14:34:14 -04:00
|
|
|
/* Output functions -- note that the output buffer had better be big enough:
|
|
|
|
* rect_to_string: RECT_LENGTH
|
|
|
|
* region_to_string: (RECT_LENGTH+strlen(separator_string)) *
|
|
|
|
* g_list_length (region)
|
|
|
|
* edge_to_string: EDGE_LENGTH
|
2014-05-02 09:34:02 -04:00
|
|
|
* edge_list_to_...: (EDGE_LENGTH+strlen(separator_string)) *
|
2010-10-18 14:34:14 -04:00
|
|
|
* g_list_length (edge_list)
|
|
|
|
*/
|
|
|
|
#define RECT_LENGTH 27
|
|
|
|
#define EDGE_LENGTH 37
|
|
|
|
char* meta_rectangle_to_string (const MetaRectangle *rect,
|
|
|
|
char *output);
|
|
|
|
char* meta_rectangle_region_to_string (GList *region,
|
|
|
|
const char *separator_string,
|
|
|
|
char *output);
|
|
|
|
char* meta_rectangle_edge_to_string (const MetaEdge *edge,
|
|
|
|
char *output);
|
|
|
|
char* meta_rectangle_edge_list_to_string (
|
|
|
|
GList *edge_list,
|
|
|
|
const char *separator_string,
|
|
|
|
char *output);
|
|
|
|
|
|
|
|
/* Resize old_rect to the given new_width and new_height, but store the
|
|
|
|
* result in rect. NOTE THAT THIS IS RESIZE ONLY SO IT CANNOT BE USED FOR
|
2020-08-26 05:49:50 -04:00
|
|
|
* A MOVERESIZE OPERATION (that simplifies the routine a little bit as it
|
2020-02-14 03:44:43 -05:00
|
|
|
* means there's no difference between META_GRAVITY_NORTH_WEST and
|
|
|
|
* META_GRAVITY_STATIC. Also, I lied a little bit--technically, you could use
|
|
|
|
* it in a MoveResize operation if you muck with old_rect just right).
|
2010-10-18 14:34:14 -04:00
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_resize_with_gravity (const MetaRectangle *old_rect,
|
|
|
|
MetaRectangle *rect,
|
2020-02-14 03:44:43 -05:00
|
|
|
MetaGravity gravity,
|
2010-10-18 14:34:14 -04:00
|
|
|
int new_width,
|
|
|
|
int new_height);
|
|
|
|
|
|
|
|
/* find a list of rectangles with the property that a window is contained
|
|
|
|
* in the given region if and only if it is contained in one of the
|
|
|
|
* rectangles in the list.
|
|
|
|
*
|
|
|
|
* In this case, the region is given by taking basic_rect, removing from
|
|
|
|
* it the intersections with all the rectangles in the all_struts list,
|
|
|
|
* then expanding all the rectangles in the resulting list by the given
|
|
|
|
* amounts on each side.
|
|
|
|
*
|
|
|
|
* See boxes.c for more details.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
GList* meta_rectangle_get_minimal_spanning_set_for_region (
|
|
|
|
const MetaRectangle *basic_rect,
|
|
|
|
const GSList *all_struts);
|
|
|
|
|
|
|
|
/* Expand all rectangles in region by the given amount on each side */
|
|
|
|
GList* meta_rectangle_expand_region (GList *region,
|
|
|
|
const int left_expand,
|
|
|
|
const int right_expand,
|
|
|
|
const int top_expand,
|
|
|
|
const int bottom_expand);
|
|
|
|
/* Same as for meta_rectangle_expand_region except that rectangles not at
|
|
|
|
* least min_x or min_y in size are not expanded in that direction
|
|
|
|
*/
|
|
|
|
GList* meta_rectangle_expand_region_conditionally (
|
|
|
|
GList *region,
|
|
|
|
const int left_expand,
|
|
|
|
const int right_expand,
|
|
|
|
const int top_expand,
|
|
|
|
const int bottom_expand,
|
|
|
|
const int min_x,
|
|
|
|
const int min_y);
|
|
|
|
|
|
|
|
/* Expand rect in direction to the size of expand_to, and then clip out any
|
2020-08-26 05:49:50 -04:00
|
|
|
* overlapping struts oriented orthogonal to the expansion direction. (Think
|
2010-10-18 14:34:14 -04:00
|
|
|
* horizontal or vertical maximization)
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_expand_to_avoiding_struts (
|
|
|
|
MetaRectangle *rect,
|
|
|
|
const MetaRectangle *expand_to,
|
|
|
|
const MetaDirection direction,
|
|
|
|
const GSList *all_struts);
|
|
|
|
|
|
|
|
/* Free the list created by
|
|
|
|
* meta_rectangle_get_minimal_spanning_set_for_region()
|
|
|
|
* or
|
|
|
|
* meta_rectangle_find_onscreen_edges ()
|
|
|
|
* or
|
|
|
|
* meta_rectangle_find_nonintersected_monitor_edges()
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_free_list_and_elements (GList *filled_list);
|
|
|
|
|
|
|
|
/* could_fit_in_region determines whether one of the spanning_rects is
|
|
|
|
* big enough to contain rect. contained_in_region checks whether one
|
|
|
|
* actually contains it.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
gboolean meta_rectangle_could_fit_in_region (
|
|
|
|
const GList *spanning_rects,
|
|
|
|
const MetaRectangle *rect);
|
2019-01-22 21:25:35 -05:00
|
|
|
|
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
gboolean meta_rectangle_contained_in_region (
|
|
|
|
const GList *spanning_rects,
|
|
|
|
const MetaRectangle *rect);
|
2019-01-22 21:25:35 -05:00
|
|
|
|
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
gboolean meta_rectangle_overlaps_with_region (
|
|
|
|
const GList *spanning_rects,
|
|
|
|
const MetaRectangle *rect);
|
|
|
|
|
2019-03-15 19:19:08 -04:00
|
|
|
gboolean meta_rectangle_is_adjacent_to_any_in_region (
|
|
|
|
const GList *spanning_rects,
|
|
|
|
MetaRectangle *rect);
|
|
|
|
|
2010-10-18 14:34:14 -04:00
|
|
|
/* Make the rectangle small enough to fit into one of the spanning_rects,
|
|
|
|
* but make it no smaller than min_size.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_clamp_to_fit_into_region (
|
|
|
|
const GList *spanning_rects,
|
|
|
|
FixedDirections fixed_directions,
|
|
|
|
MetaRectangle *rect,
|
|
|
|
const MetaRectangle *min_size);
|
|
|
|
|
|
|
|
/* Clip the rectangle so that it fits into one of the spanning_rects, assuming
|
|
|
|
* it overlaps with at least one of them
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_clip_to_region (const GList *spanning_rects,
|
|
|
|
FixedDirections fixed_directions,
|
|
|
|
MetaRectangle *rect);
|
|
|
|
|
|
|
|
/* Shove the rectangle into one of the spanning_rects, assuming it fits in
|
|
|
|
* one of them.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_shove_into_region(
|
|
|
|
const GList *spanning_rects,
|
|
|
|
FixedDirections fixed_directions,
|
|
|
|
MetaRectangle *rect);
|
|
|
|
|
|
|
|
/* Finds the point on the line connecting (x1,y1) to (x2,y2) which is closest
|
|
|
|
* to (px, py). Useful for finding an optimal rectangle size when given a
|
|
|
|
* range between two sizes that are all candidates.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
void meta_rectangle_find_linepoint_closest_to_point (double x1, double y1,
|
|
|
|
double x2, double y2,
|
|
|
|
double px, double py,
|
|
|
|
double *valx, double *valy);
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* Switching gears to code for edges instead of just rectangles */
|
|
|
|
/* */
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
/* Return whether an edge overlaps or is adjacent to the rectangle in the
|
|
|
|
* nonzero-width dimension of the edge.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2014-05-02 09:34:02 -04:00
|
|
|
gboolean meta_rectangle_edge_aligns (const MetaRectangle *rect,
|
2010-10-18 14:34:14 -04:00
|
|
|
const MetaEdge *edge);
|
|
|
|
|
|
|
|
/* Compare two edges, so that sorting functions can put a list of edges in
|
|
|
|
* canonical order.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
gint meta_rectangle_edge_cmp (gconstpointer a, gconstpointer b);
|
|
|
|
|
|
|
|
/* Compare two edges, so that sorting functions can put a list of edges in
|
|
|
|
* order. This function doesn't separate left edges first, then right edges,
|
|
|
|
* etc., but rather compares only upon location.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
gint meta_rectangle_edge_cmp_ignore_type (gconstpointer a, gconstpointer b);
|
|
|
|
|
|
|
|
/* Removes an parts of edges in the given list that intersect any box in the
|
|
|
|
* given rectangle list. Returns the result.
|
|
|
|
*/
|
|
|
|
GList* meta_rectangle_remove_intersections_with_boxes_from_edges (
|
|
|
|
GList *edges,
|
|
|
|
const GSList *rectangles);
|
|
|
|
|
|
|
|
/* Finds all the edges of an onscreen region, returning a GList* of
|
|
|
|
* MetaEdgeRect's.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
GList* meta_rectangle_find_onscreen_edges (const MetaRectangle *basic_rect,
|
|
|
|
const GSList *all_struts);
|
|
|
|
|
|
|
|
/* Finds edges between adjacent monitors which are not covered by the given
|
|
|
|
* struts.
|
|
|
|
*/
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2010-10-18 14:34:14 -04:00
|
|
|
GList* meta_rectangle_find_nonintersected_monitor_edges (
|
|
|
|
const GList *monitor_rects,
|
|
|
|
const GSList *all_struts);
|
|
|
|
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2019-03-15 11:39:08 -04:00
|
|
|
gboolean meta_rectangle_is_adjacent_to (MetaRectangle *rect,
|
2017-02-07 21:32:33 -05:00
|
|
|
MetaRectangle *other);
|
|
|
|
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2018-12-20 10:58:03 -05:00
|
|
|
void meta_rectangle_scale_double (const MetaRectangle *rect,
|
|
|
|
double scale,
|
|
|
|
MetaRoundingStrategy rounding_strategy,
|
|
|
|
MetaRectangle *dest);
|
|
|
|
|
2019-02-20 10:23:04 -05:00
|
|
|
static inline graphene_rect_t
|
|
|
|
meta_rectangle_to_graphene_rect (MetaRectangle *rect)
|
2017-06-08 10:13:16 -04:00
|
|
|
{
|
2019-02-20 10:23:04 -05:00
|
|
|
return (graphene_rect_t) {
|
2017-06-08 10:13:16 -04:00
|
|
|
.origin = {
|
|
|
|
.x = rect->x,
|
|
|
|
.y = rect->y
|
|
|
|
},
|
|
|
|
.size = {
|
|
|
|
.width = rect->width,
|
|
|
|
.height = rect->height
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-01-22 21:25:35 -05:00
|
|
|
META_EXPORT_TEST
|
2018-12-21 11:12:49 -05:00
|
|
|
void meta_rectangle_transform (const MetaRectangle *rect,
|
|
|
|
MetaMonitorTransform transform,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
MetaRectangle *dest);
|
|
|
|
|
2019-02-20 10:23:04 -05:00
|
|
|
void meta_rectangle_from_graphene_rect (const graphene_rect_t *rect,
|
|
|
|
MetaRoundingStrategy rounding_strategy,
|
|
|
|
MetaRectangle *dest);
|
2019-03-01 08:23:05 -05:00
|
|
|
|
2019-01-05 10:39:55 -05:00
|
|
|
void meta_rectangle_crop_and_scale (const MetaRectangle *rect,
|
2019-02-20 10:23:04 -05:00
|
|
|
graphene_rect_t *src_rect,
|
2019-01-05 10:39:55 -05:00
|
|
|
int dst_width,
|
|
|
|
int dst_height,
|
|
|
|
MetaRectangle *dest);
|
|
|
|
|
2010-10-18 14:34:14 -04:00
|
|
|
#endif /* META_BOXES_PRIVATE_H */
|