Skip to content

Commit

Permalink
Dynamically adjust the column widths (LiveSplit#813)
Browse files Browse the repository at this point in the history
Instead of assuming each column has the width of the text `88:88:88`, we
now determine the largest string per column (based on unicode scalar
values) and adjust the column width accordingly. There is still a fixed
minimum width based on the string `88:88`, which ensures that the
columns don't switch sizes too often, especially if you start with empty
splits. We could've also chosen `8:88:88`, which would ensure there's no
column size change as you approach the hour mark. However, this would
increase the size of delta columns unnecessarily. So it's a little bit
of a trade-off.
  • Loading branch information
CryZe authored and AlexKnauth committed Jun 13, 2024
1 parent 707ee63 commit 9b95a58
Show file tree
Hide file tree
Showing 2 changed files with 121 additions and 39 deletions.
132 changes: 107 additions & 25 deletions src/rendering/component/splits.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
use core::iter;

use crate::{
component::splits::State,
layout::{LayoutDirection, LayoutState},
Expand All @@ -20,10 +18,45 @@ use crate::{
pub struct Cache<L> {
splits: Vec<SplitCache<L>>,
column_labels: Vec<CachedLabel<L>>,
column_width_label: CachedLabel<L>,
column_label_widths: Vec<f32>,
column_width_labels: Vec<(f32, CachedLabel<L>)>,
longest_column_values: Vec<ShortLivedStr>,
}

#[derive(Copy, Clone)]
struct ShortLivedStr {
str: *const str,
char_count: usize,
}

impl ShortLivedStr {
// We use this as the smallest possible "space" we use for each column. This
// prevents the columns from changing size too much. We could bump this up
// to include hours, but that's not ideal for delta based columns, which are
// usually smaller.
const MIN: Self = Self {
str: "88:88",
char_count: 5,
};

fn new(s: &str) -> Self {
Self {
str: s,
char_count: s.chars().count(),
}
}

/// # Safety
/// Only call this function for a string that's still valid.
const unsafe fn get(&self) -> &str {
&*self.str
}
}

// SAFETY: These strings are never actually kept across calls to render.
unsafe impl Send for ShortLivedStr {}
// SAFETY: These strings are never actually kept across calls to render.
unsafe impl Sync for ShortLivedStr {}

struct SplitCache<L> {
name: CachedLabel<L>,
columns: Vec<CachedLabel<L>>,
Expand All @@ -43,8 +76,8 @@ impl<L> Cache<L> {
Self {
splits: Vec::new(),
column_labels: Vec::new(),
column_width_label: CachedLabel::new(),
column_label_widths: Vec::new(),
column_width_labels: Vec::new(),
longest_column_values: Vec::new(),
}
}
}
Expand All @@ -56,11 +89,57 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
component: &State,
layout_state: &LayoutState,
) {
const COLUMN_PADDING: f32 = 0.2;
let max_column_width =
context.measure_numbers("88:88:88", &mut cache.column_width_label, DEFAULT_TEXT_SIZE);
// We measure the longest value in each column in terms of unicode scalar
// values. This is not perfect, but gives a decent enough approximation for
// now. Long term we probably want to shape all the texts first and then lay
// them out.

let text_color = solid(&layout_state.text_color);
cache.longest_column_values.clear();

for split in &component.splits {
if split.columns.len() > cache.longest_column_values.len() {
cache
.longest_column_values
.resize(split.columns.len(), ShortLivedStr::MIN);
}
for (column, longest_column_value) in
split.columns.iter().zip(&mut cache.longest_column_values)
{
let column_value = ShortLivedStr::new(column.value.as_str());
if column_value.char_count > longest_column_value.char_count {
*longest_column_value = column_value;
}
}
}

cache.column_width_labels.resize_with(
cache.longest_column_values.len().max(
component
.column_labels
.as_ref()
.map(|labels| labels.len())
.unwrap_or_default(),
),
|| (0.0, CachedLabel::new()),
);

for (longest_column_value, (column_width, column_width_label)) in cache
.longest_column_values
.iter()
.zip(&mut cache.column_width_labels)
{
// SAFETY: The longest_column_values vector is cleared on every render
// call. We only store references to the column values in the vector and
// a 'static default string. All of these are valid for the entire
// duration of the render call.
unsafe {
*column_width = context.measure_numbers(
longest_column_value.get(),
column_width_label,
DEFAULT_TEXT_SIZE,
);
}
}

let split_background = match component.background {
ListGradient::Same(gradient) => {
Expand Down Expand Up @@ -102,8 +181,7 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
};

let transform = context.transform;

cache.column_label_widths.clear();
let text_color = solid(&layout_state.text_color);

if let Some(column_labels) = &component.column_labels {
if layout_state.direction == LayoutDirection::Vertical {
Expand All @@ -112,8 +190,11 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
.resize_with(column_labels.len(), CachedLabel::new);

let mut right_x = width - PADDING;
for (label, column_cache) in column_labels.iter().zip(&mut cache.column_labels) {
// FIXME: The column width should depend on the column type too.
for ((label, column_cache), (max_width, _)) in column_labels
.iter()
.zip(&mut cache.column_labels)
.zip(&mut cache.column_width_labels)
{
let left_x = context.render_text_right_align(
label,
column_cache,
Expand All @@ -123,8 +204,10 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
text_color,
);
let label_width = right_x - left_x;
cache.column_label_widths.push(right_x - left_x);
right_x -= label_width.max(max_column_width) + COLUMN_PADDING;
if label_width > *max_width {
*max_width = label_width;
}
right_x -= *max_width + PADDING;
}

context.translate(0.0, DEFAULT_COMPONENT_HEIGHT);
Expand Down Expand Up @@ -178,14 +261,11 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
.columns
.resize_with(split.columns.len(), CachedLabel::new);

for ((column, column_cache), column_label_width) in
split.columns.iter().zip(&mut split_cache.columns).zip(
cache
.column_label_widths
.iter()
.cloned()
.chain(iter::repeat(max_column_width)),
)
for ((column, column_cache), (max_width, _)) in split
.columns
.iter()
.zip(&mut split_cache.columns)
.zip(&cache.column_width_labels)
{
if !column.value.is_empty() {
left_x = context.render_numbers(
Expand All @@ -197,7 +277,7 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
solid(&column.visual_color),
);
}
right_x -= max_column_width.max(column_label_width) + COLUMN_PADDING;
right_x -= max_width + PADDING;
}

if display_two_rows {
Expand All @@ -215,6 +295,7 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
}
context.translate(delta_x, delta_y);
}

if component.show_final_separator {
let (pos, end) = if layout_state.direction == LayoutDirection::Horizontal {
(
Expand All @@ -229,5 +310,6 @@ pub(in crate::rendering) fn render<A: ResourceAllocator>(
};
context.render_rectangle(pos, end, &Gradient::Plain(layout_state.separators_color));
}

context.transform = transform;
}
28 changes: 14 additions & 14 deletions tests/rendering.rs
Original file line number Diff line number Diff line change
Expand Up @@ -44,8 +44,8 @@ fn default() {
check(
&state,
&image_cache,
"a98ef51c25f115fe",
"08cbb11aa1719035",
"e32259a84233e364",
"0520a7a32958470a",
"default",
);
}
Expand Down Expand Up @@ -127,8 +127,8 @@ fn font_fallback() {
check(
&state,
&image_cache,
"e3c55d333d082bab",
"267615d875c8cf61",
"924286709a5b32f3",
"88f140654343de5f",
"font_fallback",
);
}
Expand Down Expand Up @@ -216,17 +216,17 @@ fn all_components() {
&state,
&image_cache,
[300, 800],
"9c3a68e4c9c6b73c",
"f94f2b05c06f8d16",
"7e7aa83a3b80e1da",
"39b5d1923053c5d9",
"all_components",
);

check_dims(
&state,
&image_cache,
[150, 800],
"37cc2602ef0402a8",
"1ceb8d7ff13d1741",
"97afa51bfd8a8597",
"82b26ae781d58b78",
"all_components_thin",
);
}
Expand Down Expand Up @@ -294,8 +294,8 @@ fn subsplits_layout() {
&layout.state(&mut image_cache, &timer.snapshot()),
&image_cache,
[300, 800],
"78250961341ef747",
"95ba6dcf34c60078",
"39ab965781d0ceee",
"405baac87e52acc5",
"subsplits_layout",
);
}
Expand All @@ -318,8 +318,8 @@ fn background_image() {
&layout.state(&mut image_cache, &timer.snapshot()),
&image_cache,
[300, 300],
"efc369e681d98dfe",
"f0ec188c38f0e26d",
"b5238ec57ba70c3a",
"e4df7276b1603cd5",
"background_image",
);
}
Expand Down Expand Up @@ -410,8 +410,8 @@ fn horizontal() {
&layout.state(&mut image_cache, &timer.snapshot()),
&image_cache,
[1500, 40],
"944a515a48627b31",
"65b9819a3e903307",
"4a9c12d00233f3c1",
"9353fac22b4cfde4",
"horizontal",
);
}
Expand Down

0 comments on commit 9b95a58

Please sign in to comment.