mirror of
https://codeberg.org/dnkl/foot.git
synced 2026-02-04 04:06:06 -05:00
At first, an OSC-8 URI range was added when we received the closing OSC-8 escape (i.e. with an empty URI). But, this meant that cursor movements while the OSC-8 escape was in effect wasn’t handled correctly, since we’d add a range that spanned the cursor movements. Attempts were made to handle this in the cursor movement functions, by closing and re-opening the URI. However, there are too many corner cases to make this a viable approach. Scrolling is one such example, line-wrapping another. This patch takes a different approach; emit, or update the URI range when we print to the grid. This models the intended behavior much more closely, where an active OSC-8 URI act like any other SGR attribute - it is applied to all cells printed to, but otherwise have no effect. To avoid killing performance, this is only done in the “generic” printer. This means OSC-8 open/close calls must now “switch” the ASCII printer. Note that the “fast” printer still needs to *erase* pre-existing OSC-8 URIs. Closes #816
101 lines
2.5 KiB
C
101 lines
2.5 KiB
C
#pragma once
|
|
|
|
#include <stddef.h>
|
|
#include "debug.h"
|
|
#include "terminal.h"
|
|
|
|
struct grid *grid_snapshot(const struct grid *grid);
|
|
void grid_free(struct grid *grid);
|
|
|
|
void grid_swap_row(struct grid *grid, int row_a, int row_b);
|
|
struct row *grid_row_alloc(int cols, bool initialize);
|
|
void grid_row_free(struct row *row);
|
|
|
|
void grid_resize_without_reflow(
|
|
struct grid *grid, int new_rows, int new_cols,
|
|
int old_screen_rows, int new_screen_rows);
|
|
|
|
void grid_resize_and_reflow(
|
|
struct grid *grid, int new_rows, int new_cols,
|
|
int old_screen_rows, int new_screen_rows,
|
|
size_t tracking_points_count,
|
|
struct coord *const _tracking_points[static tracking_points_count]);
|
|
|
|
static inline int
|
|
grid_row_absolute(const struct grid *grid, int row_no)
|
|
{
|
|
return (grid->offset + row_no) & (grid->num_rows - 1);
|
|
}
|
|
|
|
static inline int
|
|
grid_row_absolute_in_view(const struct grid *grid, int row_no)
|
|
{
|
|
return (grid->view + row_no) & (grid->num_rows - 1);
|
|
}
|
|
|
|
static inline struct row *
|
|
_grid_row_maybe_alloc(struct grid *grid, int row_no, bool alloc_if_null)
|
|
{
|
|
xassert(grid->offset >= 0);
|
|
|
|
int real_row = grid_row_absolute(grid, row_no);
|
|
struct row *row = grid->rows[real_row];
|
|
|
|
if (row == NULL && alloc_if_null) {
|
|
row = grid_row_alloc(grid->num_cols, false);
|
|
grid->rows[real_row] = row;
|
|
}
|
|
|
|
xassert(row != NULL);
|
|
return row;
|
|
}
|
|
|
|
static inline struct row *
|
|
grid_row(struct grid *grid, int row_no)
|
|
{
|
|
return _grid_row_maybe_alloc(grid, row_no, false);
|
|
}
|
|
|
|
static inline struct row *
|
|
grid_row_and_alloc(struct grid *grid, int row_no)
|
|
{
|
|
return _grid_row_maybe_alloc(grid, row_no, true);
|
|
}
|
|
|
|
static inline struct row *
|
|
grid_row_in_view(struct grid *grid, int row_no)
|
|
{
|
|
xassert(grid->view >= 0);
|
|
|
|
int real_row = grid_row_absolute_in_view(grid, row_no);
|
|
struct row *row = grid->rows[real_row];
|
|
|
|
xassert(row != NULL);
|
|
return row;
|
|
}
|
|
|
|
void grid_row_uri_range_put(
|
|
struct row *row, int col, const char *uri, uint64_t id);
|
|
void grid_row_uri_range_add(struct row *row, struct row_uri_range range);
|
|
void grid_row_uri_range_erase(struct row *row, int start, int end);
|
|
|
|
static inline void
|
|
grid_row_uri_range_destroy(struct row_uri_range *range)
|
|
{
|
|
free(range->uri);
|
|
}
|
|
|
|
static inline void
|
|
grid_row_reset_extra(struct row *row)
|
|
{
|
|
if (likely(row->extra == NULL))
|
|
return;
|
|
|
|
tll_foreach(row->extra->uri_ranges, it) {
|
|
grid_row_uri_range_destroy(&it->item);
|
|
tll_remove(row->extra->uri_ranges, it);
|
|
}
|
|
|
|
free(row->extra);
|
|
row->extra = NULL;
|
|
}
|