#include "config.h"
#include "pango-lines-private.h"
#include "pango-line-private.h"
#include "pango-item-private.h"
#include "pango-run-private.h"
#include "pango-line-iter-private.h"
/**
* Pango2Lines:
*
* A `Pango2Lines` object represents the result of formatting an
* entire paragraph (or more) of text.
*
* A `Pango2Lines` object contains a list of `Pango2Line` objects,
* together with information about where to position each line
* in layout coordinates.
*
* `Pango2Lines` has APIs to query collective information about
* its lines (such as ellipsization or unknown glyphs), to translate
* between logical character positions within the text and the physical
* position of the resulting glyphs, and to determine cursor positions
* for editing the text.
*
* One way to obtain a `Pango2Lines` object is to use a [class@Pango2.Layout].
* But it is also possible to populate a `Pango2Lines` manually with lines
* produced by a [class@Pango2.LineBreaker] object.
*
* Note that the lines that make up a `Pango2Lines` object don't have to
* share the same underlying text. Therefore, using byte indexes to refer
* to positions within the `Pango2Lines` is, in general, ambiguous. All the
* `Pango2Lines` APIs that take a byte index as argument or return one have
* a `Pango2Line*` companion argument to handle this situation. When all
* the lines in the `Pango2Lines` share the same text (such as when they
* originate from the same `Pango2Layout`), it is safe to always pass `NULL`
* for the `Pango2Lines*`.
*
* The most convenient way to access the visual extents and components
* of a `Pango2Lines` is via a [struct@Pango2.LineIter] iterator.
*/
/* {{{ Pango2Lines implementation */
struct _Pango2LinesClass
{
GObjectClass parent_class;
};
G_DEFINE_TYPE (Pango2Lines, pango2_lines, G_TYPE_OBJECT)
enum {
PROP_UNKNOWN_GLYPHS_COUNT = 1,
PROP_WRAPPED,
PROP_ELLIPSIZED,
PROP_HYPHENATED,
N_PROPERTIES
};
static GParamSpec *properties[N_PROPERTIES] = { NULL, };
static void
pango2_lines_init (Pango2Lines *lines)
{
lines->serial = 1;
lines->lines = g_ptr_array_new_with_free_func ((GDestroyNotify) pango2_line_free);
lines->positions = g_array_new (FALSE, FALSE, sizeof (Position));
}
static void
pango2_lines_finalize (GObject *object)
{
Pango2Lines *lines = PANGO2_LINES (object);
g_ptr_array_unref (lines->lines);
g_array_unref (lines->positions);
G_OBJECT_CLASS (pango2_lines_parent_class)->finalize (object);
}
static void
pango2_lines_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
Pango2Lines *self = PANGO2_LINES (object);
switch (prop_id)
{
case PROP_UNKNOWN_GLYPHS_COUNT:
g_value_set_int (value, pango2_lines_get_unknown_glyphs_count (self));
break;
case PROP_WRAPPED:
g_value_set_boolean (value, pango2_lines_is_wrapped (self));
break;
case PROP_ELLIPSIZED:
g_value_set_boolean (value, pango2_lines_is_ellipsized (self));
break;
case PROP_HYPHENATED:
g_value_set_boolean (value, pango2_lines_is_hyphenated (self));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
pango2_lines_class_init (Pango2LinesClass *class)
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
object_class->finalize = pango2_lines_finalize;
object_class->get_property = pango2_lines_get_property;
/**
* Pango2Lines:unknown-glyphs-count: (attributes org.gtk.Property.get=pango2_lines_get_unknown_glyphs_count)
*
* The number of unknown glyphs in the `Pango2Lines`.
*/
properties[PROP_UNKNOWN_GLYPHS_COUNT] =
g_param_spec_uint ("unknown-glyphs-count", NULL, NULL, 0, G_MAXUINT, 0,
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
/**
* Pango2Lines:wrapped: (attributes org.gtk.Property.get=pango2_lines_is_wrapped)
*
* `TRUE` if the `Pango2Lines` contains any wrapped lines.
*/
properties[PROP_WRAPPED] =
g_param_spec_boolean ("wrapped", NULL, NULL, FALSE,
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
/**
* Pango2Lines:ellipsized: (attributes org.gtk.Property.get=pango2_lines_is_ellipsized)
*
* `TRUE` if the `Pango2Lines` contains any ellipsized lines.
*/
properties[PROP_ELLIPSIZED] =
g_param_spec_boolean ("ellipsized", NULL, NULL, FALSE,
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
/**
* Pango2Lines:hyphenated: (attributes org.gtk.Property.get=pango2_lines_is_hyphenated)
*
* `TRUE` if the `Pango2Lines` contains any hyphenated lines.
*/
properties[PROP_HYPHENATED] =
g_param_spec_boolean ("hyphenated", NULL, NULL, FALSE,
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
g_object_class_install_properties (object_class, N_PROPERTIES, properties);
}
/* }}} */
/* {{{ Utilities*/
typedef struct {
int x;
int pos;
} CursorPos;
static int
compare_cursor (gconstpointer v1,
gconstpointer v2)
{
const CursorPos *c1 = v1;
const CursorPos *c2 = v2;
return c1->x - c2->x;
}
static void
pango2_line_get_cursors (Pango2Lines *lines,
Pango2Line *line,
gboolean strong,
GArray *cursors)
{
const char *start, *end;
int start_offset;
int j;
const char *p;
Pango2Rectangle pos;
Position offset;
g_assert (g_array_get_element_size (cursors) == sizeof (CursorPos));
g_assert (cursors->len == 0);
for (int i = 0; i < lines->lines->len; i++)
{
if (line == g_ptr_array_index (lines->lines, i))
{
offset = g_array_index (lines->positions, Position, i);
break;
}
}
start = line->data->text + line->start_index;
end = start + line->length;
start_offset = line->start_offset;
if (line->ends_paragraph)
end++;
for (j = start_offset, p = start; p < end; j++, p = g_utf8_next_char (p))
{
int idx = p - line->data->text;
if (line->data->log_attrs[j].is_cursor_position)
{
CursorPos cursor;
pango2_line_get_cursor_pos (line, idx,
strong ? &pos : NULL,
strong ? NULL : &pos);
cursor.x = pos.x + offset.x;
cursor.pos = idx;
g_array_append_val (cursors, cursor);
}
}
g_array_sort (cursors, compare_cursor);
}
/* }}} */
/* {{{ Public API */
/**
* pango2_lines_new:
*
* Creates an empty `Pango2Lines` object.
*
* Returns: a newly allocated `Pango2Lines`
*/
Pango2Lines *
pango2_lines_new (void)
{
return g_object_new (PANGO2_TYPE_LINES, NULL);
}
/**
* pango2_lines_get_serial:
* @lines: a `Pango2Lines`
*
* Returns the current serial number of @lines.
*
* The serial number is initialized to an small number larger than zero
* when a new layout is created and is increased whenever the @lines
* object is changed (i.e. more lines are added).
*
* The serial may wrap, but will never have the value 0. Since it can
* wrap, never compare it with "less than", always use "not equals".
*
* This can be used to automatically detect changes to a `Pango2Lines`,
* and is useful for example to decide whether a layout needs redrawing.
*
* Return value: The current serial number of @lines
*/
guint
pango2_lines_get_serial (Pango2Lines *lines)
{
return lines->serial;
}
/**
* pango2_lines_add_line:
* @lines: a `Pango2Lines`
* @line: (transfer full): the `Pango2Line` to add
* @line_x: X coordinate of the position
* @line_y: Y coordinate of the position
*
* Adds a line to the `Pango2Lines`.
*
* The coordinates are the position
* at which @line is placed.
*
* Note that this function takes ownership of the line.
*/
void
pango2_lines_add_line (Pango2Lines *lines,
Pango2Line *line,
int x_line,
int y_line)
{
Position pos = { x_line, y_line };
g_ptr_array_add (lines->lines, line);
g_array_append_val (lines->positions, pos);
lines->serial++;
if (lines->serial == 0)
lines->serial++;
g_object_notify_by_pspec (G_OBJECT (lines), properties[PROP_UNKNOWN_GLYPHS_COUNT]);
g_object_notify_by_pspec (G_OBJECT (lines), properties[PROP_WRAPPED]);
g_object_notify_by_pspec (G_OBJECT (lines), properties[PROP_ELLIPSIZED]);
g_object_notify_by_pspec (G_OBJECT (lines), properties[PROP_HYPHENATED]);
}
/**
* pango2_lines_get_iter:
* @lines: a `Pango2Lines`
*
* Returns an iterator to iterate over the visual extents of the lines.
*
* The returned iterator will be invaliated when more
* lines are added to @lines, and can't be used anymore
* after that point.
*
* Note that the iter holds a reference to @lines.
*
* Return value: the new `Pango2LineIter`
*/
Pango2LineIter *
pango2_lines_get_iter (Pango2Lines *lines)
{
return pango2_line_iter_new (lines);
}
/**
* pango2_lines_get_line_count:
* @lines: a `Pango2Lines`
*
* Gets the number of lines in @lines.
*
* Returns: the number of lines'
*/
int
pango2_lines_get_line_count (Pango2Lines *lines)
{
return lines->lines->len;
}
/**
* pango2_lines_get_lines:
* @lines: a `Pango2Lines`
*
* Gets the lines.
*
* The length of the returned array can be obtained with
* [method@Pango2.Lines.get_line_count].
*
* Returns: (transfer none): the array of lines
*/
Pango2Line **
pango2_lines_get_lines (Pango2Lines *lines)
{
g_return_val_if_fail (PANGO2_IS_LINES (lines), NULL);
return (Pango2Line **)lines->lines->pdata;
}
/**
* pango2_lines_get_line_position:
* @lines: a `Pango2Lines`
* @num: the index of the line to get the position for
* @line_x: (out) (optional): return location for the X coordinate
* @line_y: (out) (optional): return location for the Y coordinate
*
* Gets the position of a line.
*/
void
pango2_lines_get_line_position (Pango2Lines *lines,
int num,
int *line_x,
int *line_y)
{
Position *pos;
g_return_if_fail (PANGO2_IS_LINES (lines));
if (num >= lines->lines->len)
return;
pos = &g_array_index (lines->positions, Position, num);
if (line_x)
*line_x = pos->x;
if (line_y)
*line_y = pos->y;
}
/* {{{ Miscellaneous */
/**
* pango2_lines_get_unknown_glyphs_count:
* @lines: a `Pango2Lines` object
*
* Counts the number of unknown glyphs in the `Pango2Lines`
*
* This function can be used to determine if there are any fonts
* available to render all characters in a certain string, or when
* used in combination with `PANGO2_ATTR_FALLBACK`, to check if a
* certain font supports all the characters in the string.
*
* Return value: The number of unknown glyphs in @lines
*/
int
pango2_lines_get_unknown_glyphs_count (Pango2Lines *lines)
{
int count = 0;
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
for (GSList *l = line->runs; l; l = l->next)
{
Pango2GlyphItem *run = l->data;
for (int j = 0; j < run->glyphs->num_glyphs; j++)
{
if (run->glyphs->glyphs[j].glyph & PANGO2_GLYPH_UNKNOWN_FLAG)
count++;
}
}
}
return count;
}
/**
* pango2_lines_is_wrapped:
* @lines: a `Pango2Lines` object
*
* Returns whether any of the lines is wrapped.
*
* Returns: `TRUE` if @lines is wrapped
*/
gboolean
pango2_lines_is_wrapped (Pango2Lines *lines)
{
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
if (pango2_line_is_wrapped (line))
return TRUE;
}
return FALSE;
}
/**
* pango2_lines_is_ellipsized:
* @lines: a `Pango2Lines` object
*
* Returns whether any of the lines is ellipsized.
*
* Returns: `TRUE` if @lines is ellipsized
*/
gboolean
pango2_lines_is_ellipsized (Pango2Lines *lines)
{
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
if (pango2_line_is_ellipsized (line))
return TRUE;
}
return FALSE;
}
/**
* pango2_lines_is_hyphenated:
* @lines: a `Pango2Lines` object
*
* Returns whether any of the lines is hyphenated.
*
* Returns: `TRUE` if @lines is hyphenated
*/
gboolean
pango2_lines_is_hyphenated (Pango2Lines *lines)
{
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
if (pango2_line_is_hyphenated (line))
return TRUE;
}
return FALSE;
}
/* }}} */
/* {{{ Extents */
/**
* pango2_lines_get_trimmd_extents:
* @lines: a `Pango2Lines` object
* @trim: `Pango2LeadingTrim` flags
* @ink_rect: (out) (optional): return location for the ink extents
* @logical_rect: (out) (optional): return location for the logical extents
*
* Computes the extents of the lines, trimmed according to `trim`.
*
* Logical extents are usually what you want for positioning things. Note
* that the extents may have non-zero x and y. You may want to use those
* to offset where you render the layout. Not doing that is a very typical
* bug that shows up as right-to-left layouts not being correctly positioned
* in a layout with a set width.
*
* The extents are given in layout coordinates and in Pango units; layout
* coordinates begin at the top left corner.
*/
void
pango2_lines_get_trimmed_extents (Pango2Lines *lines,
Pango2LeadingTrim trim,
Pango2Rectangle *ink_rect,
Pango2Rectangle *logical_rect)
{
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
Position *pos = &g_array_index (lines->positions, Position, i);
Pango2Rectangle line_ink;
Pango2Rectangle line_logical;
Pango2LeadingTrim line_trim = PANGO2_LEADING_TRIM_NONE;
if (line->starts_paragraph)
line_trim |= PANGO2_LEADING_TRIM_START;
if (line->ends_paragraph)
line_trim |= PANGO2_LEADING_TRIM_END;
pango2_line_get_extents (line, &line_ink, NULL);
pango2_line_get_trimmed_extents (line, trim & line_trim, &line_logical);
line_ink.x += pos->x;
line_ink.y += pos->y;
line_logical.x += pos->x;
line_logical.y += pos->y;
if (i == 0)
{
if (ink_rect)
*ink_rect = line_ink;
if (logical_rect)
*logical_rect = line_logical;
}
else
{
int new_pos;
if (ink_rect)
{
new_pos = MIN (ink_rect->x, line_ink.x);
ink_rect->width = MAX (ink_rect->x + ink_rect->width, line_ink.x + line_ink.width) - new_pos;
ink_rect->x = new_pos;
new_pos = MIN (ink_rect->y, line_ink.y);
ink_rect->height = MAX (ink_rect->y + ink_rect->height, line_ink.y + line_ink.height) - new_pos;
ink_rect->y = new_pos;
}
if (logical_rect)
{
new_pos = MIN (logical_rect->x, line_logical.x);
logical_rect->width = MAX (logical_rect->x + logical_rect->width, line_logical.x + line_logical.width) - new_pos;
logical_rect->x = new_pos;
new_pos = MIN (logical_rect->y, line_logical.y);
logical_rect->height = MAX (logical_rect->y + logical_rect->height, line_logical.y + line_logical.height) - new_pos;
logical_rect->y = new_pos;
}
}
}
}
/**
* pango2_lines_get_extents:
* @lines: a `Pango2Lines` object
* @ink_rect: (out) (optional): return location for the ink extents
* @logical_rect: (out) (optional): return location for the logical extents
*
* Computes the extents of the lines.
*
* Logical extents are usually what you want for positioning things. Note
* that the extents may have non-zero x and y. You may want to use those
* to offset where you render the layout. Not doing that is a very typical
* bug that shows up as right-to-left layouts not being correctly positioned
* in a layout with a set width.
*
* The extents are given in layout coordinates and in Pango units; layout
* coordinates begin at the top left corner.
*/
void
pango2_lines_get_extents (Pango2Lines *lines,
Pango2Rectangle *ink_rect,
Pango2Rectangle *logical_rect)
{
pango2_lines_get_trimmed_extents (lines, PANGO2_LEADING_TRIM_NONE, ink_rect, logical_rect);
}
/**
* pango2_lines_get_size:
* @lines: a `Pango2Lines`
* @width: (out) (optional): location to store the logical width
* @height: (out) (optional): location to store the logical height
*
* Determines the logical width and height of a `Pango2Lines`
* in Pango units.
*
* This is simply a convenience function around
* [method@Pango2.Lines.get_extents].
*/
void
pango2_lines_get_size (Pango2Lines *lines,
int *width,
int *height)
{
Pango2Rectangle ext;
pango2_lines_get_extents (lines, NULL, &ext);
if (width)
*width = ext.width;
if (height)
*height = ext.height;
}
/**
* pango2_lines_get_baseline:
* @lines: a `Pango2Lines` object
*
* Gets the Y position of baseline of the first line.
*
* Return value: baseline of first line
*/
int
pango2_lines_get_baseline (Pango2Lines *lines)
{
Position *pos;
g_return_val_if_fail (PANGO2_IS_LINES (lines), 0);
if (lines->lines->len == 0)
return 0;
pos = &g_array_index (lines->positions, Position, 0);
return pos->y;
}
/**
* pango2_lines_get_x_ranges:
* @lines: a `Pango2Lines` object
* @line: the `Pango2Line` in @lines whose x ranges will be reported
* @start_line: (nullable): `Pango2Line` wrt to which @start_index is
* interpreted or `NULL` for the first matching line
* @start_index: Start byte index of the logical range. If this value
* is less than the start index for the line, then the first range
* will extend all the way to the leading edge of the layout. Otherwise,
* it will start at the leading edge of the first character.
* @end_line: (nullable): `Pango2Line` wrt to which @end_index is
* interpreted or `NULL` for the first matching line
* @end_index: Ending byte index of the logical range. If this value is
* greater than the end index for the line, then the last range will
* extend all the way to the trailing edge of the layout. Otherwise,
* it will end at the trailing edge of the last character.
* @ranges: (out) (array length=n_ranges) (transfer full): location to
* store a pointer to an array of ranges. The array will be of length
* `2*n_ranges`, with each range starting at `(*ranges)[2*n]` and of
* width `(*ranges)[2*n + 1] - (*ranges)[2*n]`. This array must be freed
* with [GLib.free]. The coordinates are relative to the layout and are in
* Pango units.
* @n_ranges: The number of ranges stored in @ranges
*
* Gets a list of visual ranges corresponding to a given logical range.
*
* This list is not necessarily minimal - there may be consecutive
* ranges which are adjacent. The ranges will be sorted from left to
* right. The ranges are with respect to the left edge of the entire
* layout, not with respect to the line.
*/
void
pango2_lines_get_x_ranges (Pango2Lines *lines,
Pango2Line *line,
Pango2Line *start_line,
int start_index,
Pango2Line *end_line,
int end_index,
int **ranges,
int *n_ranges)
{
int x_offset;
int line_no, start_line_no, end_line_no;
Pango2Direction dir;
int range_count;
int width;
Pango2Rectangle ext;
int accumulated_width;
g_return_if_fail (PANGO2_IS_LINES (lines));
g_return_if_fail (ranges != NULL);
g_return_if_fail (n_ranges != NULL);
pango2_lines_index_to_line (lines, line->start_index, &line, &line_no, &x_offset, NULL);
g_return_if_fail (line != NULL);
pango2_lines_index_to_line (lines, start_index, &start_line, &start_line_no, NULL, NULL);
g_return_if_fail (start_line != NULL);
g_return_if_fail (start_line_no <= line_no);
pango2_lines_index_to_line (lines, end_index, &end_line, &end_line_no, NULL, NULL);
g_return_if_fail (end_line != NULL);
g_return_if_fail (end_line_no >= line_no);
/* clamp start_index and end_index to be inside line, we'll use
* the line numbers to determine overshoot.
*/
if (start_line_no < line_no || line_no < end_line_no)
{
if (start_line_no < line_no)
start_index = line->start_index;
if (end_line_no > line_no)
end_index = line->start_index + line->length;
}
*ranges = g_new (int, 2 * (2 + g_slist_length (line->runs)));
range_count = 0;
dir = pango2_line_get_resolved_direction (line);
if (x_offset > 0 &&
((dir == PANGO2_DIRECTION_LTR && start_line_no < line_no) ||
(dir == PANGO2_DIRECTION_RTL && end_line_no > line_no)))
{
/* add range from left edge of layout to line start */
(*ranges)[2*range_count] = 0;
(*ranges)[2*range_count + 1] = x_offset;
range_count++;
}
accumulated_width = 0;
for (int i = 0; i < pango2_line_get_run_count (line); i++)
{
Pango2GlyphItem *run = pango2_run_get_glyph_item (pango2_line_get_runs (line)[i]);
if ((start_index < run->item->offset + run->item->length &&
end_index > run->item->offset))
{
int run_start_index = MAX (start_index, run->item->offset);
int run_end_index = MIN (end_index, run->item->offset + run->item->length);
int run_start_x, run_end_x;
int attr_offset;
g_assert (run_end_index > 0);
/* Back the end_index off one since we want to find the trailing edge of the
* preceding character
*/
run_end_index = g_utf8_prev_char (line->data->text + run_end_index) - line->data->text;
attr_offset = run->item->char_offset;
pango2_glyph_string_index_to_x_full (run->glyphs,
line->data->text + run->item->offset,
run->item->length,
&run->item->analysis,
line->data->log_attrs + attr_offset,
run_start_index - run->item->offset, FALSE,
&run_start_x);
pango2_glyph_string_index_to_x_full (run->glyphs,
line->data->text + run->item->offset,
run->item->length,
&run->item->analysis,
line->data->log_attrs + attr_offset,
run_end_index - run->item->offset, TRUE,
&run_end_x);
(*ranges)[2*range_count] = x_offset + accumulated_width + MIN (run_start_x, run_end_x);
(*ranges)[2*range_count + 1] = x_offset + accumulated_width + MAX (run_start_x, run_end_x);
}
range_count++;
if (i + 1 < pango2_line_get_run_count (line))
accumulated_width += pango2_glyph_string_get_width (run->glyphs);
}
pango2_line_get_extents (line, NULL, &ext);
pango2_lines_get_size (lines, &width, NULL);
if (x_offset + ext.width < width &&
((dir == PANGO2_DIRECTION_LTR && start_line_no < line_no) ||
(dir == PANGO2_DIRECTION_RTL && end_line_no > line_no)))
{
/* add range from line end to rigth edge of layout */
(*ranges)[2*range_count] = x_offset + ext.width;
(*ranges)[2*range_count + 1] = width;
range_count ++;
}
*n_ranges = range_count;
}
/* }}} */
/* { {{ Editing API */
/**
* pango2_lines_index_to_line:
* @lines: a `Pango2Lines`
* @idx: index in @lines
* @line: (inout): if *@line is `NULL`, the found line will be returned here
* @line_no: (out) (optional): return location for line number
* @x_offset: (out) (optional): return location for X offset of line
* @y_offset: (out) (optional): return location for Y offset of line
*
* Given an index (and possibly line), determine the line number,
* and offset for the line.
*
* @idx may refer to any byte position inside @lines, as well as
* the position before the first or after the last character (i.e.
* line->start_index + line->length, for the last line).
*
* If @lines contains lines with different backing data (i.e.
* more than one line with line->start_index of zero), then
* *@line must be specified to disambiguate. If all lines
* are backed by the same data, it is save to pass `NULL`
* as *@line and use this function to find the line at @idx.
*/
void
pango2_lines_index_to_line (Pango2Lines *lines,
int idx,
Pango2Line **line,
int *line_no,
int *x_offset,
int *y_offset)
{
Pango2Line *found = NULL;
int num;
int i;
Position pos;
g_return_if_fail (PANGO2_IS_LINES (lines));
for (i = 0; i < lines->lines->len; i++)
{
Pango2Line *l = g_ptr_array_index (lines->lines, i);
if (l->start_index > idx && found)
break;
found = l;
pos = g_array_index (lines->positions, Position, i);
num = i;
if (*line && *line == found)
break;
if (l->start_index + l->length > idx)
break;
}
if (found)
{
*line = found;
if (line_no)
*line_no = num;
if (x_offset)
*x_offset = pos.x;
if (y_offset)
*y_offset = pos.y;
return;
}
*line = NULL;
}
/**
* pango2_lines_pos_to_line:
* @lines: a `Pango2Lines` object
* @x: the X position (in Pango units)
* @y: the Y position (in Pango units)
* @line_x: (out) (optional): return location for the X offset of the line
* @line_y: (out) (optional): return location for the Y offset of the line
*
* Finds the line at the given position.
*
* If either the X or Y positions were not inside the layout, then the
* function returns `NULL`; on an exact hit, it returns the found line.
* Returns: (transfer none) (nullable): the line that was found
*/
Pango2Line *
pango2_lines_pos_to_line (Pango2Lines *lines,
int x,
int y,
int *line_x,
int *line_y)
{
g_return_val_if_fail (PANGO2_IS_LINES (lines), FALSE);
for (int i = 0; i < lines->lines->len; i++)
{
Pango2Line *line = g_ptr_array_index (lines->lines, i);
Position pos = g_array_index (lines->positions, Position, i);
Pango2Rectangle ext;
pango2_line_get_extents (line, NULL, &ext);
ext.x += pos.x;
ext.y += pos.y;
if (ext.x <= x && x <= ext.x + ext.width &&
ext.y <= y && y <= ext.y + ext.height)
{
if (line_x)
*line_x = pos.x;
if (line_y)
*line_y = pos.y;
return line;
}
}
if (line_x)
*line_x = 0;
if (line_y)
*line_y = 0;
return NULL;
}
/**
* pango2_lines_index_to_pos:
* @lines: a `Pango2Lines` object
* @line: (nullable): `Pango2Line` wrt to which @idx is interpreted
* or `NULL` for the first matching line
* @idx: byte index within @line
* @pos: (out): rectangle in which to store the position of the grapheme
*
* Converts from an index within a `Pango2Line` to the
* position corresponding to the grapheme at that index.
*
* The return value is represented as rectangle. Note that `pos->x`
* is always the leading edge of the grapheme and `pos->x + pos->width`
* the trailing edge of the grapheme. If the directionality of the
* grapheme is right-to-left, then `pos->width` will be negative.
*
* Note that @idx is allowed to be @line->start_index + @line->length
* for the position off the end of the last line.
*/
void
pango2_lines_index_to_pos (Pango2Lines *lines,
Pango2Line *line,
int idx,
Pango2Rectangle *pos)
{
int x_offset, y_offset;
g_return_if_fail (PANGO2_IS_LINES (lines));
g_return_if_fail (idx >= 0);
g_return_if_fail (pos != NULL);
pango2_lines_index_to_line (lines, idx, &line, NULL, &x_offset, &y_offset);
g_return_if_fail (line != NULL);
pango2_line_index_to_pos (line, idx, pos);
pos->x += x_offset;
pos->y += y_offset;
}
/**
* pango2_lines_pos_to_index:
* @lines: a `Pango2Lines` object
* @x: the X offset (in Pango units) from the left edge of the layout
* @y: the Y offset (in Pango units) from the top edge of the layout
* @idx: (out): location to store calculated byte index
* @trailing: (out): location to store a integer indicating where
* in the grapheme the user clicked. It will either be zero, or the
* number of characters in the grapheme. 0 represents the leading edge
* of the grapheme.
*
* Converts from X and Y position within lines to the byte index of the
* character at that position.
*
* Returns: (transfer none) (nullable): the line that was found
*/
Pango2Line *
pango2_lines_pos_to_index (Pango2Lines *lines,
int x,
int y,
int *idx,
int *trailing)
{
Pango2Line *line;
int x_offset;
g_return_val_if_fail (PANGO2_IS_LINES (lines), FALSE);
g_return_val_if_fail (idx != NULL, FALSE);
g_return_val_if_fail (trailing != NULL, FALSE);
line = pango2_lines_pos_to_line (lines, x, y, &x_offset, NULL);
if (line)
pango2_line_x_to_index (line, x - x_offset, idx, trailing);
return line;
}
/* }}} */
/* {{{ Cursor positioning */
/**
* pango2_lines_get_cursor_pos:
* @lines: a `Pango2Lines` object
* @line: (nullable): `Pango2Line` wrt to which @idx is interpreted
* or `NULL` for the first matching line
* @idx: the byte index of the cursor
* @strong_pos: (out) (optional): location to store the strong cursor position
* @weak_pos: (out) (optional): location to store the weak cursor position
*
* Given an index within lines, determines the positions that of the
* strong and weak cursors if the insertion point is at that index.
*
* Note that @idx is allowed to be @line->start_index + @line->length
* for the position off the end of the last line.
*
* The position of each cursor is stored as a zero-width rectangle
* with the height of the run extents.
*
*
*
* The strong cursor location is the location where characters of the
* directionality equal to the base direction of the layout are inserted.
* The weak cursor location is the location where characters of the
* directionality opposite to the base direction of the layout are inserted.
*
* The following example shows text with both a strong and a weak cursor.
*
*
*
* The strong cursor has a little arrow pointing to the right, the weak
* cursor to the left. Typing a 'c' in this situation will insert the
* character after the 'b', and typing another Hebrew character, like 'ג',
* will insert it at the end.
*/
void
pango2_lines_get_cursor_pos (Pango2Lines *lines,
Pango2Line *line,
int idx,
Pango2Rectangle *strong_pos,
Pango2Rectangle *weak_pos)
{
int x_offset, y_offset;
Pango2Line *l;
g_return_if_fail (PANGO2_IS_LINES (lines));
l = line;
pango2_lines_index_to_line (lines, idx, &l, NULL, &x_offset, &y_offset);
g_return_if_fail (l != NULL);
g_return_if_fail (line == NULL || l == line);
line = l;
pango2_line_get_cursor_pos (line, idx, strong_pos, weak_pos);
if (strong_pos)
{
strong_pos->x += x_offset;
strong_pos->y += y_offset;
}
if (weak_pos)
{
weak_pos->x += x_offset;
weak_pos->y += y_offset;
}
}
/**
* pango2_lines_get_caret_pos:
* @lines: a `Pango2Lines` object
* @line: (nullable): `Pango2Line` wrt to which @idx is interpreted
* or `NULL` for the first matching line
* @idx: the byte index of the cursor
* @strong_pos: (out) (optional): location to store the strong cursor position
* @weak_pos: (out) (optional): location to store the weak cursor position
*
* Given an index within a layout, determines the positions of the
* strong and weak cursors if the insertion point is at that index.
*
* Note that @idx is allowed to be @line->start_index + @line->length
* for the position off the end of the last line.
*
* This is a variant of [method@Pango2.Lines.get_cursor_pos] that applies
* font metric information about caret slope and offset to the positions
* it returns.
*
*
*/
void
pango2_lines_get_caret_pos (Pango2Lines *lines,
Pango2Line *line,
int idx,
Pango2Rectangle *strong_pos,
Pango2Rectangle *weak_pos)
{
int x_offset, y_offset;
g_return_if_fail (PANGO2_IS_LINES (lines));
pango2_lines_index_to_line (lines, idx, &line, NULL, &x_offset, &y_offset);
g_return_if_fail (line != NULL);
pango2_line_get_caret_pos (line, idx, strong_pos, weak_pos);
if (strong_pos)
{
strong_pos->x += x_offset;
strong_pos->y += y_offset;
}
if (weak_pos)
{
weak_pos->x += x_offset;
weak_pos->y += y_offset;
}
}
/**
* pango2_lines_move_cursor:
* @lines: a `Pango2Lines` object
* @strong: whether the moving cursor is the strong cursor or the
* weak cursor. The strong cursor is the cursor corresponding
* to text insertion in the base direction for the layout.
* @line: (nullable): `Pango2Line` wrt to which @idx is interpreted
* or `NULL` for the first matching line
* @idx: the byte index of the current cursor position
* @trailing: if 0, the cursor was at the leading edge of the
* grapheme indicated by @old_index, if > 0, the cursor
* was at the trailing edge.
* @direction: direction to move cursor. A negative
* value indicates motion to the left
* @new_line: (nullable): `Pango2Line` wrt to which @new_idx is interpreted
* @new_idx: (out): location to store the new cursor byte index
* A value of -1 indicates that the cursor has been moved off the
* beginning of the layout. A value of %G_MAXINT indicates that
* the cursor has been moved off the end of the layout.
* @new_trailing: (out): number of characters to move forward from
* the location returned for @new_idx to get the position where
* the cursor should be displayed. This allows distinguishing the
* position at the beginning of one line from the position at the
* end of the preceding line. @new_idx is always on the line where
* the cursor should be displayed.
*
* Computes a new cursor position from an old position and a direction.
*
* If @direction is positive, then the new position will cause the strong
* or weak cursor to be displayed one position to right of where it was
* with the old cursor position. If @direction is negative, it will be
* moved to the left.
*
* In the presence of bidirectional text, the correspondence between
* logical and visual order will depend on the direction of the current
* run, and there may be jumps when the cursor is moved off of the end
* of a run.
*
* Motion here is in cursor positions, not in characters, so a single
* call to this function may move the cursor over multiple characters
* when multiple characters combine to form a single grapheme.
*/
void
pango2_lines_move_cursor (Pango2Lines *lines,
gboolean strong,
Pango2Line *line,
int idx,
int trailing,
int direction,
Pango2Line **new_line,
int *new_idx,
int *new_trailing)
{
int line_no;
GArray *cursors;
int n_vis;
int vis_pos;
int start_offset;
gboolean off_start = FALSE;
gboolean off_end = FALSE;
Pango2Rectangle pos;
int j;
g_return_if_fail (PANGO2_IS_LINES (lines));
g_return_if_fail (idx >= 0);
g_return_if_fail (trailing >= 0);
g_return_if_fail (new_idx != NULL);
g_return_if_fail (new_trailing != NULL);
direction = (direction >= 0 ? 1 : -1);
pango2_lines_index_to_line (lines, idx, &line, &line_no, NULL, NULL);
g_return_if_fail (line != NULL);
while (trailing--)
idx = g_utf8_next_char (line->data->text + idx) - line->data->text;
/* Clamp old_index to fit on the line */
if (idx > (line->start_index + line->length))
idx = line->start_index + line->length;
cursors = g_array_new (FALSE, FALSE, sizeof (CursorPos));
pango2_line_get_cursors (lines, line, strong, cursors);
pango2_lines_get_cursor_pos (lines, line, idx, strong ? &pos : NULL, strong ? NULL : &pos);
vis_pos = -1;
for (j = 0; j < cursors->len; j++)
{
CursorPos *cursor = &g_array_index (cursors, CursorPos, j);
if (cursor->x == pos.x)
{
vis_pos = j;
/* If moving left, we pick the leftmost match, otherwise
* the rightmost one. Without this, we can get stuck
*/
if (direction < 0)
break;
}
}
if (vis_pos == -1 &&
idx == line->start_index + line->length)
{
if (line->direction == PANGO2_DIRECTION_LTR)
vis_pos = cursors->len;
else
vis_pos = 0;
}
/* Handling movement between lines */
if (line->direction == PANGO2_DIRECTION_LTR)
{
if (idx == line->start_index && direction < 0)
off_start = TRUE;
if (idx == line->start_index + line->length && direction > 0)
off_end = TRUE;
}
else
{
if (idx == line->start_index + line->length && direction < 0)
off_start = TRUE;
if (idx == line->start_index && direction > 0)
off_end = TRUE;
}
if (off_start || off_end)
{
/* If we move over a paragraph boundary, count that as
* an extra position in the motion
*/
gboolean paragraph_boundary;
if (off_start)
{
if (line_no == 0)
{
if (new_line)
*new_line = NULL;
*new_idx = -1;
*new_trailing = 0;
g_array_unref (cursors);
return;
}
line_no--;
line = g_ptr_array_index (lines->lines, line_no);
paragraph_boundary = (line->start_index + line->length != idx);
}
else
{
if (line_no == lines->lines->len - 1)
{
if (new_line)
*new_line = NULL;
*new_idx = G_MAXINT;
*new_trailing = 0;
g_array_unref (cursors);
return;
}
line_no++;
line = g_ptr_array_index (lines->lines, line_no);
paragraph_boundary = (line->start_index != idx);
}
g_array_set_size (cursors, 0);
pango2_line_get_cursors (lines, line, strong, cursors);
n_vis = cursors->len;
if (off_start && direction < 0)
{
vis_pos = n_vis;
if (paragraph_boundary)
vis_pos++;
}
else if (off_end && direction > 0)
{
vis_pos = 0;
if (paragraph_boundary)
vis_pos--;
}
}
if (direction < 0)
vis_pos--;
else
vis_pos++;
if (0 <= vis_pos && vis_pos < cursors->len)
*new_idx = g_array_index (cursors, CursorPos, vis_pos).pos;
else if (vis_pos >= cursors->len - 1)
*new_idx = line->start_index + line->length;
*new_trailing = 0;
if (*new_idx == line->start_index + line->length && line->length > 0)
{
int log_pos;
start_offset = g_utf8_pointer_to_offset (line->data->text, line->data->text + line->start_index);
log_pos = start_offset + g_utf8_strlen (line->data->text + line->start_index, line->length);
do
{
log_pos--;
*new_idx = g_utf8_prev_char (line->data->text + *new_idx) - line->data->text;
(*new_trailing)++;
}
while (log_pos > start_offset && !line->data->log_attrs[log_pos].is_cursor_position);
}
if (new_line)
*new_line = line;
g_array_unref (cursors);
}
/* }}} */
/* }}} */
/* vim:set foldmethod=marker expandtab: */