mirror of
https://github.com/emilk/egui.git
synced 2026-06-27 15:13:12 -04:00
Properly handle row repositioning
This commit is contained in:
@@ -39,11 +39,11 @@ pub fn update_accesskit_for_text_widget(
|
||||
};
|
||||
|
||||
ctx.with_accessibility_parent(parent_id, || {
|
||||
for (row_index, (row, offset)) in galley.rows.iter().enumerate() {
|
||||
for (row_index, row) in galley.rows.iter().enumerate() {
|
||||
let row_id = parent_id.with(row_index);
|
||||
ctx.accesskit_node_builder(row_id, |builder| {
|
||||
builder.set_role(accesskit::Role::TextRun);
|
||||
let rect = row.rect.translate(offset.to_vec2() + galley_pos.to_vec2());
|
||||
let rect = row.rect().translate(galley_pos.to_vec2());
|
||||
builder.set_bounds(accesskit::Rect {
|
||||
x0: rect.min.x.into(),
|
||||
y0: rect.min.y.into(),
|
||||
@@ -74,14 +74,14 @@ pub fn update_accesskit_for_text_widget(
|
||||
let old_len = value.len();
|
||||
value.push(glyph.chr);
|
||||
character_lengths.push((value.len() - old_len) as _);
|
||||
character_positions.push(glyph.pos.x - row.rect.min.x);
|
||||
character_positions.push(glyph.pos.x - row.pos.x);
|
||||
character_widths.push(glyph.advance_width);
|
||||
}
|
||||
|
||||
if row.ends_with_newline {
|
||||
value.push('\n');
|
||||
character_lengths.push(1);
|
||||
character_positions.push(row.rect.max.x - row.rect.min.x);
|
||||
character_positions.push(row.size.x);
|
||||
character_widths.push(0.0);
|
||||
}
|
||||
word_lengths.push((character_lengths.len() - last_word_start) as _);
|
||||
|
||||
@@ -284,7 +284,7 @@ fn ccursor_from_accesskit_text_position(
|
||||
position: &accesskit::TextPosition,
|
||||
) -> Option<CCursor> {
|
||||
let mut total_length = 0usize;
|
||||
for (i, (row, _)) in galley.rows.iter().enumerate() {
|
||||
for (i, row) in galley.rows.iter().enumerate() {
|
||||
let row_id = id.with(i);
|
||||
if row_id.accesskit_id() == position.node {
|
||||
return Some(CCursor {
|
||||
|
||||
@@ -179,10 +179,10 @@ impl LabelSelectionState {
|
||||
if let epaint::Shape::Text(text_shape) = &mut shape.shape {
|
||||
let galley = Arc::make_mut(&mut text_shape.galley);
|
||||
for row_selection in row_selections {
|
||||
if let Some((row, _)) =
|
||||
if let Some(placed_row) =
|
||||
galley.rows.get_mut(row_selection.row)
|
||||
{
|
||||
let row = Arc::make_mut(row);
|
||||
let row = Arc::make_mut(&mut placed_row.row);
|
||||
for vertex_index in row_selection.vertex_indices {
|
||||
if let Some(vertex) = row
|
||||
.visuals
|
||||
@@ -662,8 +662,8 @@ fn selected_text(galley: &Galley, cursor_range: &CursorRange) -> String {
|
||||
}
|
||||
|
||||
fn estimate_row_height(galley: &Galley) -> f32 {
|
||||
if let Some((row, _)) = galley.rows.first() {
|
||||
row.rect.height()
|
||||
if let Some(placed_row) = galley.rows.first() {
|
||||
placed_row.height()
|
||||
} else {
|
||||
galley.size().y
|
||||
}
|
||||
|
||||
@@ -31,26 +31,27 @@ pub fn paint_text_selection(
|
||||
let max = max.rcursor;
|
||||
|
||||
for ri in min.row..=max.row {
|
||||
let (row, _) = &mut galley.rows[ri];
|
||||
let row = Arc::make_mut(row);
|
||||
let placed_row = &mut galley.rows[ri];
|
||||
|
||||
let left = if ri == min.row {
|
||||
row.x_offset(min.column)
|
||||
placed_row.x_offset(min.column)
|
||||
} else {
|
||||
row.rect.left()
|
||||
0.0
|
||||
};
|
||||
let right = if ri == max.row {
|
||||
row.x_offset(max.column)
|
||||
placed_row.x_offset(max.column)
|
||||
} else {
|
||||
let newline_size = if row.ends_with_newline {
|
||||
row.height() / 2.0 // visualize that we select the newline
|
||||
let newline_size = if placed_row.ends_with_newline {
|
||||
placed_row.height() / 2.0 // visualize that we select the newline
|
||||
} else {
|
||||
0.0
|
||||
};
|
||||
row.rect.right() + newline_size
|
||||
placed_row.size.x + newline_size
|
||||
};
|
||||
|
||||
let rect = Rect::from_min_max(pos2(left, row.min_y()), pos2(right, row.max_y()));
|
||||
let rect = Rect::from_min_max(pos2(left, 0.0), pos2(right, placed_row.size.y));
|
||||
|
||||
let row = Arc::make_mut(&mut placed_row.row);
|
||||
let mesh = &mut row.visuals.mesh;
|
||||
|
||||
// Time to insert the selection rectangle into the row mesh.
|
||||
|
||||
@@ -640,8 +640,8 @@ impl WidgetText {
|
||||
Self::RichText(text) => text.font_height(fonts, style),
|
||||
Self::LayoutJob(job) => job.font_height(fonts),
|
||||
Self::Galley(galley) => {
|
||||
if let Some((row, _)) = galley.rows.first() {
|
||||
row.height()
|
||||
if let Some(placed_row) = galley.rows.first() {
|
||||
placed_row.height()
|
||||
} else {
|
||||
galley.size().y
|
||||
}
|
||||
|
||||
@@ -194,13 +194,10 @@ impl Label {
|
||||
let pos = pos2(ui.max_rect().left(), ui.cursor().top());
|
||||
assert!(!galley.rows.is_empty(), "Galleys are never empty");
|
||||
// collect a response from many rows:
|
||||
let rect = galley.rows[0]
|
||||
.0
|
||||
.rect
|
||||
.translate(galley.rows[0].1.to_vec2() + pos.to_vec2());
|
||||
let rect = galley.rows[0].rect().translate(pos.to_vec2());
|
||||
let mut response = ui.allocate_rect(rect, sense);
|
||||
for (row, offset) in galley.rows.iter().skip(1) {
|
||||
let rect = row.rect.translate(offset.to_vec2() + pos.to_vec2());
|
||||
for placed_row in galley.rows.iter().skip(1) {
|
||||
let rect = placed_row.rect().translate(pos.to_vec2());
|
||||
response |= ui.allocate_rect(rect, sense);
|
||||
}
|
||||
(pos, galley, response)
|
||||
|
||||
@@ -433,9 +433,8 @@ impl Shape {
|
||||
|
||||
// Scale text:
|
||||
let galley = Arc::make_mut(&mut text_shape.galley);
|
||||
for (row, offset) in &mut galley.rows {
|
||||
let row = Arc::make_mut(row);
|
||||
*offset = *offset * transform.scaling;
|
||||
for placed_row in &mut galley.rows {
|
||||
let row = Arc::make_mut(&mut placed_row.row);
|
||||
row.visuals.mesh_bounds = transform.scaling * row.visuals.mesh_bounds;
|
||||
for v in &mut row.visuals.mesh.vertices {
|
||||
v.pos = Pos2::new(transform.scaling * v.pos.x, transform.scaling * v.pos.y);
|
||||
|
||||
@@ -88,8 +88,8 @@ pub fn adjust_colors(
|
||||
|
||||
if !galley.is_empty() {
|
||||
let galley = std::sync::Arc::make_mut(galley);
|
||||
for (row, _) in &mut galley.rows {
|
||||
let row = Arc::make_mut(row);
|
||||
for placed_row in &mut galley.rows {
|
||||
let row = Arc::make_mut(&mut placed_row.row);
|
||||
for vertex in &mut row.visuals.mesh.vertices {
|
||||
adjust_color(&mut vertex.color);
|
||||
}
|
||||
|
||||
@@ -88,14 +88,10 @@ impl AllocInfo {
|
||||
pub fn from_galley(galley: &Galley) -> Self {
|
||||
Self::from_slice(galley.text().as_bytes())
|
||||
+ Self::from_slice(&galley.rows)
|
||||
+ galley
|
||||
.rows
|
||||
.iter()
|
||||
.map(|(row, _)| Self::from_galley_row(row))
|
||||
.sum()
|
||||
+ galley.rows.iter().map(Self::from_galley_row).sum()
|
||||
}
|
||||
|
||||
fn from_galley_row(row: &crate::text::Row) -> Self {
|
||||
fn from_galley_row(row: &crate::text::PlacedRow) -> Self {
|
||||
Self::from_mesh(&row.visuals.mesh) + Self::from_slice(&row.glyphs)
|
||||
}
|
||||
|
||||
@@ -217,8 +213,8 @@ impl PaintStats {
|
||||
self.shape_text += AllocInfo::from_galley(&text_shape.galley);
|
||||
|
||||
for row in &text_shape.galley.rows {
|
||||
self.text_shape_indices += AllocInfo::from_slice(&row.0.visuals.mesh.indices);
|
||||
self.text_shape_vertices += AllocInfo::from_slice(&row.0.visuals.mesh.vertices);
|
||||
self.text_shape_indices += AllocInfo::from_slice(&row.visuals.mesh.indices);
|
||||
self.text_shape_vertices += AllocInfo::from_slice(&row.visuals.mesh.vertices);
|
||||
}
|
||||
}
|
||||
Shape::Mesh(mesh) => {
|
||||
|
||||
@@ -1778,12 +1778,12 @@ impl Tessellator {
|
||||
|
||||
let rotator = Rot2::from_angle(*angle);
|
||||
|
||||
for (row, row_pos) in &galley.rows {
|
||||
for row in &galley.rows {
|
||||
if row.visuals.mesh.is_empty() {
|
||||
continue;
|
||||
}
|
||||
|
||||
let final_pos = galley_pos + row_pos.to_vec2();
|
||||
let final_pos = galley_pos + row.pos.to_vec2();
|
||||
|
||||
let mut row_rect = row.visuals.mesh_bounds;
|
||||
if *angle != 0.0 {
|
||||
|
||||
@@ -805,12 +805,21 @@ impl GalleyCache {
|
||||
let current_offset = emath::vec2(0.0, merged_galley.rect.height());
|
||||
merged_galley
|
||||
.rows
|
||||
.extend(galley.rows.iter().map(|(row, prev_offset)| {
|
||||
merged_galley.mesh_bounds =
|
||||
merged_galley.mesh_bounds.union(row.visuals.mesh_bounds);
|
||||
.extend(galley.rows.iter().map(|placed_row| {
|
||||
let new_pos = placed_row.pos + current_offset;
|
||||
merged_galley.mesh_bounds = merged_galley
|
||||
.mesh_bounds
|
||||
.union(placed_row.visuals.mesh_bounds.translate(new_pos.to_vec2()));
|
||||
|
||||
(row.clone(), *prev_offset + current_offset)
|
||||
super::PlacedRow {
|
||||
row: placed_row.row.clone(),
|
||||
pos: new_pos,
|
||||
ends_with_newline: placed_row.ends_with_newline,
|
||||
}
|
||||
}));
|
||||
if let Some(last) = merged_galley.rows.last_mut() {
|
||||
last.ends_with_newline = true;
|
||||
}
|
||||
merged_galley.rect = merged_galley
|
||||
.rect
|
||||
.union(galley.rect.translate(current_offset));
|
||||
@@ -822,6 +831,10 @@ impl GalleyCache {
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(last) = merged_galley.rows.last_mut() {
|
||||
last.ends_with_newline = false;
|
||||
}
|
||||
|
||||
merged_galley
|
||||
}
|
||||
|
||||
|
||||
@@ -5,7 +5,7 @@ use emath::{pos2, vec2, Align, NumExt, Pos2, Rect, Vec2};
|
||||
|
||||
use crate::{stroke::PathStroke, text::font::Font, Color32, Mesh, Stroke, Vertex};
|
||||
|
||||
use super::{FontsImpl, Galley, Glyph, LayoutJob, LayoutSection, Row, RowVisuals};
|
||||
use super::{FontsImpl, Galley, Glyph, LayoutJob, LayoutSection, PlacedRow, Row, RowVisuals};
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
|
||||
@@ -96,11 +96,11 @@ pub fn layout(fonts: &mut FontsImpl, job: Arc<LayoutJob>) -> Galley {
|
||||
let mut elided = false;
|
||||
let mut rows = rows_from_paragraphs(paragraphs, &job, &mut elided);
|
||||
if elided {
|
||||
if let Some((last_row, _)) = rows.last_mut() {
|
||||
let last_row = Arc::get_mut(last_row).unwrap();
|
||||
if let Some(last_placed) = rows.last_mut() {
|
||||
let last_row = Arc::get_mut(&mut last_placed.row).unwrap();
|
||||
replace_last_glyph_with_overflow_character(fonts, &job, last_row);
|
||||
if let Some(last) = last_row.glyphs.last() {
|
||||
last_row.rect.max.x = last.max_x();
|
||||
last_row.size.x = last.max_x();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -109,12 +109,13 @@ pub fn layout(fonts: &mut FontsImpl, job: Arc<LayoutJob>) -> Galley {
|
||||
|
||||
if justify || job.halign != Align::LEFT {
|
||||
let num_rows = rows.len();
|
||||
for (i, (row, _)) in rows.iter_mut().enumerate() {
|
||||
for (i, placed_row) in rows.iter_mut().enumerate() {
|
||||
let is_last_row = i + 1 == num_rows;
|
||||
let justify_row = justify && !row.ends_with_newline && !is_last_row;
|
||||
let justify_row = justify && !placed_row.ends_with_newline && !is_last_row;
|
||||
halign_and_justify_row(
|
||||
point_scale,
|
||||
Arc::get_mut(row).unwrap(),
|
||||
Arc::get_mut(&mut placed_row.row).unwrap(),
|
||||
&mut placed_row.pos,
|
||||
job.halign,
|
||||
job.wrap.max_width,
|
||||
justify_row,
|
||||
@@ -199,7 +200,7 @@ fn rows_from_paragraphs(
|
||||
paragraphs: Vec<Paragraph>,
|
||||
job: &LayoutJob,
|
||||
elided: &mut bool,
|
||||
) -> Vec<(Arc<Row>, Pos2)> {
|
||||
) -> Vec<PlacedRow> {
|
||||
let num_paragraphs = paragraphs.len();
|
||||
|
||||
let mut rows = vec![];
|
||||
@@ -213,38 +214,35 @@ fn rows_from_paragraphs(
|
||||
let is_last_paragraph = (i + 1) == num_paragraphs;
|
||||
|
||||
if paragraph.glyphs.is_empty() {
|
||||
rows.push((
|
||||
Arc::new(Row {
|
||||
rows.push(PlacedRow {
|
||||
row: Arc::new(Row {
|
||||
section_index_at_start: paragraph.section_index_at_start,
|
||||
glyphs: vec![],
|
||||
visuals: Default::default(),
|
||||
rect: Rect::from_min_size(
|
||||
pos2(paragraph.cursor_x, 0.0),
|
||||
vec2(0.0, paragraph.empty_paragraph_height),
|
||||
),
|
||||
ends_with_newline: !is_last_paragraph,
|
||||
size: vec2(0.0, paragraph.empty_paragraph_height),
|
||||
}),
|
||||
Pos2::ZERO,
|
||||
));
|
||||
pos: pos2(paragraph.cursor_x, 0.0),
|
||||
ends_with_newline: !is_last_paragraph,
|
||||
});
|
||||
} else {
|
||||
let paragraph_max_x = paragraph.glyphs.last().unwrap().max_x();
|
||||
if paragraph_max_x <= job.effective_wrap_width() {
|
||||
// Early-out optimization: the whole paragraph fits on one row.
|
||||
let paragraph_min_x = paragraph.glyphs[0].pos.x;
|
||||
rows.push((
|
||||
Arc::new(Row {
|
||||
let rect = rect_from_x_range(paragraph_min_x..=paragraph_max_x);
|
||||
rows.push(PlacedRow {
|
||||
row: Arc::new(Row {
|
||||
section_index_at_start: paragraph.section_index_at_start,
|
||||
glyphs: paragraph.glyphs,
|
||||
visuals: Default::default(),
|
||||
rect: rect_from_x_range(paragraph_min_x..=paragraph_max_x),
|
||||
ends_with_newline: !is_last_paragraph,
|
||||
size: rect.size(),
|
||||
}),
|
||||
Pos2::ZERO,
|
||||
));
|
||||
pos: rect.min,
|
||||
ends_with_newline: !is_last_paragraph,
|
||||
});
|
||||
} else {
|
||||
line_break(¶graph, job, &mut rows, elided);
|
||||
let last_row = Arc::get_mut(&mut rows.last_mut().unwrap().0).unwrap();
|
||||
last_row.ends_with_newline = !is_last_paragraph;
|
||||
rows.last_mut().unwrap().ends_with_newline = !is_last_paragraph;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -255,7 +253,7 @@ fn rows_from_paragraphs(
|
||||
fn line_break(
|
||||
paragraph: &Paragraph,
|
||||
job: &LayoutJob,
|
||||
out_rows: &mut Vec<(Arc<Row>, Pos2)>,
|
||||
out_rows: &mut Vec<PlacedRow>,
|
||||
elided: &mut bool,
|
||||
) {
|
||||
let wrap_width = job.effective_wrap_width();
|
||||
@@ -283,16 +281,17 @@ fn line_break(
|
||||
{
|
||||
// Allow the first row to be completely empty, because we know there will be more space on the next row:
|
||||
// TODO(emilk): this records the height of this first row as zero, though that is probably fine since first_row_indentation usually comes with a first_row_min_height.
|
||||
out_rows.push((
|
||||
Arc::new(Row {
|
||||
let rect = rect_from_x_range(first_row_indentation..=first_row_indentation);
|
||||
out_rows.push(PlacedRow {
|
||||
row: Arc::new(Row {
|
||||
section_index_at_start: paragraph.section_index_at_start,
|
||||
glyphs: vec![],
|
||||
visuals: Default::default(),
|
||||
rect: rect_from_x_range(first_row_indentation..=first_row_indentation),
|
||||
ends_with_newline: false,
|
||||
size: rect.size(),
|
||||
}),
|
||||
Pos2::ZERO,
|
||||
));
|
||||
pos: rect.min,
|
||||
ends_with_newline: false,
|
||||
});
|
||||
row_start_x += first_row_indentation;
|
||||
first_row_indentation = 0.0;
|
||||
} else if let Some(last_kept_index) = row_break_candidates.get(job.wrap.break_anywhere)
|
||||
@@ -310,16 +309,17 @@ fn line_break(
|
||||
let paragraph_min_x = glyphs[0].pos.x;
|
||||
let paragraph_max_x = glyphs.last().unwrap().max_x();
|
||||
|
||||
out_rows.push((
|
||||
Arc::new(Row {
|
||||
let rect = rect_from_x_range(paragraph_min_x..=paragraph_max_x);
|
||||
out_rows.push(PlacedRow {
|
||||
row: Arc::new(Row {
|
||||
section_index_at_start,
|
||||
glyphs,
|
||||
visuals: Default::default(),
|
||||
rect: rect_from_x_range(paragraph_min_x..=paragraph_max_x),
|
||||
ends_with_newline: false,
|
||||
size: rect.size(),
|
||||
}),
|
||||
Pos2::ZERO,
|
||||
));
|
||||
pos: rect.min,
|
||||
ends_with_newline: false,
|
||||
});
|
||||
|
||||
// Start a new row:
|
||||
row_start_idx = last_kept_index + 1;
|
||||
@@ -352,16 +352,17 @@ fn line_break(
|
||||
let paragraph_min_x = glyphs[0].pos.x;
|
||||
let paragraph_max_x = glyphs.last().unwrap().max_x();
|
||||
|
||||
out_rows.push((
|
||||
Arc::new(Row {
|
||||
let rect = rect_from_x_range(paragraph_min_x..=paragraph_max_x);
|
||||
out_rows.push(PlacedRow {
|
||||
row: Arc::new(Row {
|
||||
section_index_at_start,
|
||||
glyphs,
|
||||
visuals: Default::default(),
|
||||
rect: rect_from_x_range(paragraph_min_x..=paragraph_max_x),
|
||||
ends_with_newline: false,
|
||||
size: rect.size(),
|
||||
}),
|
||||
Pos2::ZERO,
|
||||
));
|
||||
pos: rect.min,
|
||||
ends_with_newline: false,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -523,6 +524,7 @@ fn replace_last_glyph_with_overflow_character(
|
||||
fn halign_and_justify_row(
|
||||
point_scale: PointScale,
|
||||
row: &mut Row,
|
||||
pos: &mut Pos2,
|
||||
halign: Align,
|
||||
wrap_width: f32,
|
||||
justify: bool,
|
||||
@@ -606,24 +608,25 @@ fn halign_and_justify_row(
|
||||
}
|
||||
|
||||
// Note we ignore the leading/trailing whitespace here!
|
||||
row.rect.min.x = target_min_x;
|
||||
row.rect.max.x = target_max_x;
|
||||
pos.x = target_min_x;
|
||||
row.size.x = target_max_x - target_min_x;
|
||||
}
|
||||
|
||||
/// Calculate the Y positions and tessellate the text.
|
||||
fn galley_from_rows(
|
||||
point_scale: PointScale,
|
||||
job: Arc<LayoutJob>,
|
||||
mut rows: Vec<(Arc<Row>, Pos2)>,
|
||||
mut rows: Vec<PlacedRow>,
|
||||
elided: bool,
|
||||
) -> Galley {
|
||||
let mut first_row_min_height = job.first_row_min_height;
|
||||
let mut cursor_y = 0.0;
|
||||
let mut min_x: f32 = 0.0;
|
||||
let mut max_x: f32 = 0.0;
|
||||
for (row, _) in &mut rows {
|
||||
let row = Arc::get_mut(row).unwrap();
|
||||
let mut max_row_height = first_row_min_height.max(row.rect.height());
|
||||
for placed_row in &mut rows {
|
||||
let mut max_row_height = first_row_min_height.max(placed_row.rect().height());
|
||||
let row = Arc::get_mut(&mut placed_row.row).unwrap();
|
||||
|
||||
first_row_min_height = 0.0;
|
||||
for glyph in &row.glyphs {
|
||||
max_row_height = max_row_height.max(glyph.line_height);
|
||||
@@ -634,8 +637,7 @@ fn galley_from_rows(
|
||||
for glyph in &mut row.glyphs {
|
||||
let format = &job.sections[glyph.section_index as usize].format;
|
||||
|
||||
glyph.pos.y = cursor_y
|
||||
+ glyph.font_impl_ascent
|
||||
glyph.pos.y = glyph.font_impl_ascent
|
||||
|
||||
// Apply valign to the different in height of the entire row, and the height of this `Font`:
|
||||
+ format.valign.to_factor() * (max_row_height - glyph.line_height)
|
||||
@@ -644,14 +646,16 @@ fn galley_from_rows(
|
||||
// we always center the difference:
|
||||
+ 0.5 * (glyph.font_height - glyph.font_impl_height);
|
||||
|
||||
glyph.pos.y = point_scale.round_to_pixel(glyph.pos.y);
|
||||
// FIXME(afishhh): HACK! change the proper code above instead!!
|
||||
// this should probably not be merged like this!
|
||||
glyph.pos.x -= placed_row.pos.x;
|
||||
}
|
||||
|
||||
row.rect.min.y = cursor_y;
|
||||
row.rect.max.y = cursor_y + max_row_height;
|
||||
placed_row.pos.y = cursor_y;
|
||||
row.size.y = max_row_height;
|
||||
|
||||
min_x = min_x.min(row.rect.min.x);
|
||||
max_x = max_x.max(row.rect.max.x);
|
||||
min_x = min_x.min(placed_row.rect().min.x);
|
||||
max_x = max_x.max(placed_row.rect().max.x);
|
||||
cursor_y += max_row_height;
|
||||
cursor_y = point_scale.round_to_pixel(cursor_y);
|
||||
}
|
||||
@@ -662,8 +666,8 @@ fn galley_from_rows(
|
||||
let mut num_vertices = 0;
|
||||
let mut num_indices = 0;
|
||||
|
||||
for (row, _) in &mut rows {
|
||||
let row = Arc::get_mut(row).unwrap();
|
||||
for placed_row in &mut rows {
|
||||
let row = Arc::get_mut(&mut placed_row.row).unwrap();
|
||||
row.visuals = tessellate_row(point_scale, &job, &format_summary, row);
|
||||
mesh_bounds = mesh_bounds.union(row.visuals.mesh_bounds);
|
||||
num_vertices += row.visuals.mesh.vertices.len();
|
||||
@@ -1096,7 +1100,7 @@ mod tests {
|
||||
|
||||
assert!(galley.elided);
|
||||
assert_eq!(galley.rows.len(), 1);
|
||||
let row_text = galley.rows[0].0.text();
|
||||
let row_text = galley.rows[0].text();
|
||||
assert!(
|
||||
row_text.ends_with('…'),
|
||||
"Expected row to end with `…`, got {row_text:?} when line-breaking the text {text:?} with max_width {max_width} and break_anywhere {break_anywhere}.",
|
||||
@@ -1115,7 +1119,7 @@ mod tests {
|
||||
|
||||
assert!(galley.elided);
|
||||
assert_eq!(galley.rows.len(), 1);
|
||||
let row_text = galley.rows[0].0.text();
|
||||
let row_text = galley.rows[0].text();
|
||||
assert_eq!(row_text, "Hello…");
|
||||
}
|
||||
}
|
||||
@@ -1130,11 +1134,7 @@ mod tests {
|
||||
layout_job.wrap.max_width = 90.0;
|
||||
let galley = layout(&mut fonts, layout_job.into());
|
||||
assert_eq!(
|
||||
galley
|
||||
.rows
|
||||
.iter()
|
||||
.map(|row| row.0.text())
|
||||
.collect::<Vec<_>>(),
|
||||
galley.rows.iter().map(|row| row.text()).collect::<Vec<_>>(),
|
||||
vec!["日本語と", "Englishの混在", "した文章"]
|
||||
);
|
||||
}
|
||||
@@ -1149,11 +1149,7 @@ mod tests {
|
||||
layout_job.wrap.max_width = 110.0;
|
||||
let galley = layout(&mut fonts, layout_job.into());
|
||||
assert_eq!(
|
||||
galley
|
||||
.rows
|
||||
.iter()
|
||||
.map(|row| row.0.text())
|
||||
.collect::<Vec<_>>(),
|
||||
galley.rows.iter().map(|row| row.text()).collect::<Vec<_>>(),
|
||||
vec!["日本語とEnglish", "の混在した文章"]
|
||||
);
|
||||
}
|
||||
@@ -1168,14 +1164,11 @@ mod tests {
|
||||
let galley = layout(&mut fonts, layout_job.into());
|
||||
assert!(galley.elided);
|
||||
assert_eq!(
|
||||
galley
|
||||
.rows
|
||||
.iter()
|
||||
.map(|row| row.0.text())
|
||||
.collect::<Vec<_>>(),
|
||||
galley.rows.iter().map(|row| row.text()).collect::<Vec<_>>(),
|
||||
vec!["# DNA…"]
|
||||
);
|
||||
let row = &galley.rows[0];
|
||||
assert_eq!(row.0.rect.max.x, row.0.glyphs.last().unwrap().max_x());
|
||||
assert_eq!(row.pos, Pos2::ZERO);
|
||||
assert_eq!(row.rect().max.x, row.glyphs.last().unwrap().max_x());
|
||||
}
|
||||
}
|
||||
|
||||
@@ -506,7 +506,7 @@ pub struct Galley {
|
||||
///
|
||||
/// Note that a paragraph (a piece of text separated with `\n`)
|
||||
/// can be split up into multiple rows.
|
||||
pub rows: Vec<(Arc<Row>, Pos2)>,
|
||||
pub rows: Vec<PlacedRow>,
|
||||
|
||||
/// Set to true the text was truncated due to [`TextWrapping::max_rows`].
|
||||
pub elided: bool,
|
||||
@@ -538,6 +538,39 @@ pub struct Galley {
|
||||
pub pixels_per_point: f32,
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
|
||||
pub struct PlacedRow {
|
||||
/// The underlying row unpositioned [`Row`].
|
||||
pub row: Arc<Row>,
|
||||
|
||||
/// The position of this [`Row`] relative to the galley.
|
||||
pub pos: Pos2,
|
||||
|
||||
/// If true, this [`PlacedRow`] came from a paragraph ending with a `\n`.
|
||||
/// The `\n` itself is omitted from [`Row::glyphs`].
|
||||
/// A `\n` in the input text always creates a new [`PlacedRow`] below it,
|
||||
/// so that text that ends with `\n` has an empty [`PlacedRow`] last.
|
||||
/// This also implies that the last [`PlacedRow`] in a [`Galley`] always has `ends_with_newline == false`.
|
||||
pub ends_with_newline: bool,
|
||||
}
|
||||
|
||||
impl PlacedRow {
|
||||
/// Logical bounding rectangle on font heights etc.
|
||||
/// Use this when drawing a selection or similar!
|
||||
pub fn rect(&self) -> Rect {
|
||||
Rect::from_min_size(self.pos, self.row.size)
|
||||
}
|
||||
}
|
||||
|
||||
impl std::ops::Deref for PlacedRow {
|
||||
type Target = Row;
|
||||
|
||||
fn deref(&self) -> &Self::Target {
|
||||
&self.row
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
|
||||
pub struct Row {
|
||||
@@ -547,20 +580,12 @@ pub struct Row {
|
||||
/// One for each `char`.
|
||||
pub glyphs: Vec<Glyph>,
|
||||
|
||||
/// Logical bounding rectangle based on font heights etc.
|
||||
/// Use this when drawing a selection or similar!
|
||||
/// Logical size based on font heights etc.
|
||||
/// Includes leading and trailing whitespace.
|
||||
pub rect: Rect,
|
||||
pub size: Vec2,
|
||||
|
||||
/// The mesh, ready to be rendered.
|
||||
pub visuals: RowVisuals,
|
||||
|
||||
/// If true, this [`Row`] came from a paragraph ending with a `\n`.
|
||||
/// The `\n` itself is omitted from [`Self::glyphs`].
|
||||
/// A `\n` in the input text always creates a new [`Row`] below it,
|
||||
/// so that text that ends with `\n` has an empty [`Row`] last.
|
||||
/// This also implies that the last [`Row`] in a [`Galley`] always has `ends_with_newline == false`.
|
||||
pub ends_with_newline: bool,
|
||||
}
|
||||
|
||||
/// The tessellated output of a row.
|
||||
@@ -668,28 +693,7 @@ impl Row {
|
||||
self.glyphs.len()
|
||||
}
|
||||
|
||||
/// Includes the implicit `\n` after the [`Row`], if any.
|
||||
#[inline]
|
||||
pub fn char_count_including_newline(&self) -> usize {
|
||||
self.glyphs.len() + (self.ends_with_newline as usize)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn min_y(&self) -> f32 {
|
||||
self.rect.top()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn max_y(&self) -> f32 {
|
||||
self.rect.bottom()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn height(&self) -> f32 {
|
||||
self.rect.height()
|
||||
}
|
||||
|
||||
/// Closest char at the desired x coordinate.
|
||||
/// Closest char at the desired x coordinate in row-relative coordinates.
|
||||
/// Returns something in the range `[0, char_count_excluding_newline()]`.
|
||||
pub fn char_at(&self, desired_x: f32) -> usize {
|
||||
for (i, glyph) in self.glyphs.iter().enumerate() {
|
||||
@@ -704,11 +708,34 @@ impl Row {
|
||||
if let Some(glyph) = self.glyphs.get(column) {
|
||||
glyph.pos.x
|
||||
} else {
|
||||
self.rect.right()
|
||||
self.size.x
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl PlacedRow {
|
||||
/// Includes the implicit `\n` after the [`Row`], if any.
|
||||
#[inline]
|
||||
pub fn char_count_including_newline(&self) -> usize {
|
||||
self.glyphs.len() + (self.ends_with_newline as usize)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn min_y(&self) -> f32 {
|
||||
self.rect().top()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn max_y(&self) -> f32 {
|
||||
self.rect().bottom()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn height(&self) -> f32 {
|
||||
self.row.size.y
|
||||
}
|
||||
}
|
||||
|
||||
impl Galley {
|
||||
#[inline]
|
||||
pub fn is_empty(&self) -> bool {
|
||||
@@ -755,8 +782,8 @@ impl std::ops::Deref for Galley {
|
||||
impl Galley {
|
||||
/// Zero-width rect past the last character.
|
||||
fn end_pos(&self) -> Rect {
|
||||
if let Some((row, _)) = self.rows.last() {
|
||||
let x = row.rect.right();
|
||||
if let Some(row) = self.rows.last() {
|
||||
let x = row.rect().right();
|
||||
Rect::from_min_max(pos2(x, row.min_y()), pos2(x, row.max_y()))
|
||||
} else {
|
||||
// Empty galley
|
||||
@@ -773,7 +800,7 @@ impl Galley {
|
||||
pub fn pos_from_pcursor(&self, pcursor: PCursor) -> Rect {
|
||||
let mut it = PCursor::default();
|
||||
|
||||
for (row, offset) in &self.rows {
|
||||
for row in &self.rows {
|
||||
if it.paragraph == pcursor.paragraph {
|
||||
// Right paragraph, but is it the right row in the paragraph?
|
||||
|
||||
@@ -787,11 +814,8 @@ impl Galley {
|
||||
&& !row.ends_with_newline
|
||||
&& column >= row.char_count_excluding_newline();
|
||||
if !select_next_row_instead {
|
||||
let x = row.x_offset(column) + offset.x;
|
||||
return Rect::from_min_max(
|
||||
pos2(x, row.min_y() + offset.y),
|
||||
pos2(x, row.max_y() + offset.y),
|
||||
);
|
||||
let x = row.x_offset(column);
|
||||
return Rect::from_min_max(pos2(x, row.min_y()), pos2(x, row.max_y()));
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -825,13 +849,13 @@ impl Galley {
|
||||
/// same as a cursor at the end.
|
||||
/// This allows implementing text-selection by dragging above/below the galley.
|
||||
pub fn cursor_from_pos(&self, pos: Vec2) -> Cursor {
|
||||
if let Some((first_row, offset)) = self.rows.first() {
|
||||
if pos.y < first_row.min_y() + offset.y {
|
||||
if let Some(first_row) = self.rows.first() {
|
||||
if pos.y < first_row.min_y() {
|
||||
return self.begin();
|
||||
}
|
||||
}
|
||||
if let Some((last_row, offset)) = self.rows.last() {
|
||||
if last_row.max_y() + offset.y < pos.y {
|
||||
if let Some(last_row) = self.rows.last() {
|
||||
if last_row.max_y() < pos.y {
|
||||
return self.end();
|
||||
}
|
||||
}
|
||||
@@ -842,15 +866,16 @@ impl Galley {
|
||||
let mut ccursor_index = 0;
|
||||
let mut pcursor_it = PCursor::default();
|
||||
|
||||
for (row_nr, (row, offset)) in self.rows.iter().enumerate() {
|
||||
let min_y = row.min_y() + offset.y;
|
||||
let max_y = row.max_y() + offset.y;
|
||||
for (row_nr, row) in self.rows.iter().enumerate() {
|
||||
let min_y = row.min_y();
|
||||
let max_y = row.max_y();
|
||||
|
||||
let is_pos_within_row = min_y <= pos.y && pos.y <= max_y;
|
||||
let y_dist = (min_y - pos.y).abs().min((max_y - pos.y).abs());
|
||||
if is_pos_within_row || y_dist < best_y_dist {
|
||||
best_y_dist = y_dist;
|
||||
let column = row.char_at(pos.x);
|
||||
// char_at is `Row` not `PlacedRow` relative which means we have to subtract the pos.
|
||||
let column = row.char_at(pos.x - row.pos.x);
|
||||
let prefer_next_row = column < row.char_count_excluding_newline();
|
||||
cursor = Cursor {
|
||||
ccursor: CCursor {
|
||||
@@ -910,7 +935,7 @@ impl Galley {
|
||||
offset: 0,
|
||||
prefer_next_row: true,
|
||||
};
|
||||
for (row, _) in &self.rows {
|
||||
for row in &self.rows {
|
||||
let row_char_count = row.char_count_including_newline();
|
||||
ccursor.index += row_char_count;
|
||||
if row.ends_with_newline {
|
||||
@@ -928,7 +953,7 @@ impl Galley {
|
||||
}
|
||||
|
||||
pub fn end_rcursor(&self) -> RCursor {
|
||||
if let Some((last_row, _)) = self.rows.last() {
|
||||
if let Some(last_row) = self.rows.last() {
|
||||
RCursor {
|
||||
row: self.rows.len() - 1,
|
||||
column: last_row.char_count_including_newline(),
|
||||
@@ -954,7 +979,7 @@ impl Galley {
|
||||
prefer_next_row,
|
||||
};
|
||||
|
||||
for (row_nr, (row, _)) in self.rows.iter().enumerate() {
|
||||
for (row_nr, row) in self.rows.iter().enumerate() {
|
||||
let row_char_count = row.char_count_excluding_newline();
|
||||
|
||||
if ccursor_it.index <= ccursor.index
|
||||
@@ -999,7 +1024,7 @@ impl Galley {
|
||||
}
|
||||
|
||||
let prefer_next_row =
|
||||
rcursor.column < self.rows[rcursor.row].0.char_count_excluding_newline();
|
||||
rcursor.column < self.rows[rcursor.row].char_count_excluding_newline();
|
||||
let mut ccursor_it = CCursor {
|
||||
index: 0,
|
||||
prefer_next_row,
|
||||
@@ -1010,7 +1035,7 @@ impl Galley {
|
||||
prefer_next_row,
|
||||
};
|
||||
|
||||
for (row_nr, (row, _)) in self.rows.iter().enumerate() {
|
||||
for (row_nr, row) in self.rows.iter().enumerate() {
|
||||
if row_nr == rcursor.row {
|
||||
ccursor_it.index += rcursor.column.at_most(row.char_count_excluding_newline());
|
||||
|
||||
@@ -1054,7 +1079,7 @@ impl Galley {
|
||||
prefer_next_row,
|
||||
};
|
||||
|
||||
for (row_nr, (row, _)) in self.rows.iter().enumerate() {
|
||||
for (row_nr, row) in self.rows.iter().enumerate() {
|
||||
if pcursor_it.paragraph == pcursor.paragraph {
|
||||
// Right paragraph, but is it the right row in the paragraph?
|
||||
|
||||
@@ -1128,9 +1153,7 @@ impl Galley {
|
||||
let new_row = cursor.rcursor.row - 1;
|
||||
|
||||
let cursor_is_beyond_end_of_current_row = cursor.rcursor.column
|
||||
>= self.rows[cursor.rcursor.row]
|
||||
.0
|
||||
.char_count_excluding_newline();
|
||||
>= self.rows[cursor.rcursor.row].char_count_excluding_newline();
|
||||
|
||||
let new_rcursor = if cursor_is_beyond_end_of_current_row {
|
||||
// keep same column
|
||||
@@ -1141,8 +1164,8 @@ impl Galley {
|
||||
} else {
|
||||
// keep same X coord
|
||||
let x = self.pos_from_cursor(cursor).center().x;
|
||||
let (row, offset) = &self.rows[new_row];
|
||||
let column = if x > row.rect.right() + offset.x {
|
||||
let row = &self.rows[new_row];
|
||||
let column = if x > row.rect().right() {
|
||||
// beyond the end of this row - keep same column
|
||||
cursor.rcursor.column
|
||||
} else {
|
||||
@@ -1162,9 +1185,7 @@ impl Galley {
|
||||
let new_row = cursor.rcursor.row + 1;
|
||||
|
||||
let cursor_is_beyond_end_of_current_row = cursor.rcursor.column
|
||||
>= self.rows[cursor.rcursor.row]
|
||||
.0
|
||||
.char_count_excluding_newline();
|
||||
>= self.rows[cursor.rcursor.row].char_count_excluding_newline();
|
||||
|
||||
let new_rcursor = if cursor_is_beyond_end_of_current_row {
|
||||
// keep same column
|
||||
@@ -1175,8 +1196,8 @@ impl Galley {
|
||||
} else {
|
||||
// keep same X coord
|
||||
let x = self.pos_from_cursor(cursor).center().x;
|
||||
let (row, offset) = &self.rows[new_row];
|
||||
let column = if x > row.rect.right() + offset.x {
|
||||
let row = &self.rows[new_row];
|
||||
let column = if x > row.rect().right() {
|
||||
// beyond the end of the next row - keep same column
|
||||
cursor.rcursor.column
|
||||
} else {
|
||||
@@ -1204,9 +1225,7 @@ impl Galley {
|
||||
pub fn cursor_end_of_row(&self, cursor: &Cursor) -> Cursor {
|
||||
self.from_rcursor(RCursor {
|
||||
row: cursor.rcursor.row,
|
||||
column: self.rows[cursor.rcursor.row]
|
||||
.0
|
||||
.char_count_excluding_newline(),
|
||||
column: self.rows[cursor.rcursor.row].char_count_excluding_newline(),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user