Usage: zellij options --copy-command "xclip -sel clip" Co-authored-by: Christophe Verbinnen <christophev@knowbe4.com>
333 lines
13 KiB
Rust
333 lines
13 KiB
Rust
use ansi_term::{
|
|
ANSIStrings,
|
|
Color::{Fixed, RGB},
|
|
Style,
|
|
};
|
|
use zellij_tile::prelude::*;
|
|
|
|
use crate::{
|
|
tip::{data::TIPS, TipFn},
|
|
LinePart, MORE_MSG,
|
|
};
|
|
|
|
fn full_length_shortcut(
|
|
is_first_shortcut: bool,
|
|
letter: &str,
|
|
description: &str,
|
|
palette: Palette,
|
|
) -> LinePart {
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let green_color = match palette.green {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let separator = if is_first_shortcut { " " } else { " / " };
|
|
let separator = Style::new().fg(white_color).paint(separator);
|
|
let shortcut_len = letter.chars().count() + 3; // 2 for <>'s around shortcut, 1 for the space
|
|
let shortcut_left_separator = Style::new().fg(white_color).paint("<");
|
|
let shortcut = Style::new().fg(green_color).bold().paint(letter);
|
|
let shortcut_right_separator = Style::new().fg(white_color).paint("> ");
|
|
let description_len = description.chars().count();
|
|
let description = Style::new().fg(white_color).bold().paint(description);
|
|
let len = shortcut_len + description_len + separator.chars().count();
|
|
LinePart {
|
|
part: ANSIStrings(&[
|
|
separator,
|
|
shortcut_left_separator,
|
|
shortcut,
|
|
shortcut_right_separator,
|
|
description,
|
|
])
|
|
.to_string(),
|
|
len,
|
|
}
|
|
}
|
|
|
|
fn first_word_shortcut(
|
|
is_first_shortcut: bool,
|
|
letter: &str,
|
|
description: &str,
|
|
palette: Palette,
|
|
) -> LinePart {
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let green_color = match palette.green {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let separator = if is_first_shortcut { " " } else { " / " };
|
|
let separator = Style::new().fg(white_color).paint(separator);
|
|
let shortcut_len = letter.chars().count() + 3; // 2 for <>'s around shortcut, 1 for the space
|
|
let shortcut_left_separator = Style::new().fg(white_color).paint("<");
|
|
let shortcut = Style::new().fg(green_color).bold().paint(letter);
|
|
let shortcut_right_separator = Style::new().fg(white_color).paint("> ");
|
|
let description_first_word = description.split(' ').next().unwrap_or("");
|
|
let description_first_word_length = description_first_word.chars().count();
|
|
let description_first_word = Style::new()
|
|
.fg(white_color)
|
|
.bold()
|
|
.paint(description_first_word);
|
|
let len = shortcut_len + description_first_word_length + separator.chars().count();
|
|
LinePart {
|
|
part: ANSIStrings(&[
|
|
separator,
|
|
shortcut_left_separator,
|
|
shortcut,
|
|
shortcut_right_separator,
|
|
description_first_word,
|
|
])
|
|
.to_string(),
|
|
len,
|
|
}
|
|
}
|
|
|
|
fn locked_interface_indication(palette: Palette) -> LinePart {
|
|
let locked_text = " -- INTERFACE LOCKED -- ";
|
|
let locked_text_len = locked_text.chars().count();
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let locked_styled_text = Style::new().fg(white_color).bold().paint(locked_text);
|
|
LinePart {
|
|
part: locked_styled_text.to_string(),
|
|
len: locked_text_len,
|
|
}
|
|
}
|
|
|
|
fn select_pane_shortcut(is_first_shortcut: bool, palette: Palette) -> LinePart {
|
|
let shortcut = "ENTER";
|
|
let description = "Select pane";
|
|
let separator = if is_first_shortcut { " " } else { " / " };
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let orange_color = match palette.orange {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let separator = Style::new().fg(white_color).paint(separator);
|
|
let shortcut_len = shortcut.chars().count() + 3; // 2 for <>'s around shortcut, 1 for the space
|
|
let shortcut_left_separator = Style::new().fg(white_color).paint("<");
|
|
let shortcut = Style::new().fg(orange_color).bold().paint(shortcut);
|
|
let shortcut_right_separator = Style::new().fg(white_color).paint("> ");
|
|
let description_len = description.chars().count();
|
|
let description = Style::new().fg(white_color).bold().paint(description);
|
|
let len = shortcut_len + description_len + separator.chars().count();
|
|
LinePart {
|
|
part: ANSIStrings(&[
|
|
separator,
|
|
shortcut_left_separator,
|
|
shortcut,
|
|
shortcut_right_separator,
|
|
description,
|
|
])
|
|
.to_string(),
|
|
len,
|
|
}
|
|
}
|
|
|
|
fn full_shortcut_list(help: &ModeInfo, tip: TipFn) -> LinePart {
|
|
match help.mode {
|
|
InputMode::Normal => tip(help.palette),
|
|
InputMode::Locked => locked_interface_indication(help.palette),
|
|
_ => {
|
|
let mut line_part = LinePart::default();
|
|
for (i, (letter, description)) in help.keybinds.iter().enumerate() {
|
|
let shortcut = full_length_shortcut(i == 0, letter, description, help.palette);
|
|
line_part.len += shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, shortcut,);
|
|
}
|
|
let select_pane_shortcut = select_pane_shortcut(help.keybinds.is_empty(), help.palette);
|
|
line_part.len += select_pane_shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, select_pane_shortcut,);
|
|
line_part
|
|
}
|
|
}
|
|
}
|
|
|
|
fn shortened_shortcut_list(help: &ModeInfo, tip: TipFn) -> LinePart {
|
|
match help.mode {
|
|
InputMode::Normal => tip(help.palette),
|
|
InputMode::Locked => locked_interface_indication(help.palette),
|
|
_ => {
|
|
let mut line_part = LinePart::default();
|
|
for (i, (letter, description)) in help.keybinds.iter().enumerate() {
|
|
let shortcut = first_word_shortcut(i == 0, letter, description, help.palette);
|
|
line_part.len += shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, shortcut,);
|
|
}
|
|
let select_pane_shortcut = select_pane_shortcut(help.keybinds.is_empty(), help.palette);
|
|
line_part.len += select_pane_shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, select_pane_shortcut,);
|
|
line_part
|
|
}
|
|
}
|
|
}
|
|
|
|
fn best_effort_shortcut_list(help: &ModeInfo, tip: TipFn, max_len: usize) -> LinePart {
|
|
match help.mode {
|
|
InputMode::Normal => {
|
|
let line_part = tip(help.palette);
|
|
if line_part.len <= max_len {
|
|
line_part
|
|
} else {
|
|
LinePart::default()
|
|
}
|
|
}
|
|
InputMode::Locked => {
|
|
let line_part = locked_interface_indication(help.palette);
|
|
if line_part.len <= max_len {
|
|
line_part
|
|
} else {
|
|
LinePart::default()
|
|
}
|
|
}
|
|
_ => {
|
|
let mut line_part = LinePart::default();
|
|
for (i, (letter, description)) in help.keybinds.iter().enumerate() {
|
|
let shortcut = first_word_shortcut(i == 0, letter, description, help.palette);
|
|
if line_part.len + shortcut.len + MORE_MSG.chars().count() > max_len {
|
|
// TODO: better
|
|
line_part.part = format!("{}{}", line_part.part, MORE_MSG);
|
|
line_part.len += MORE_MSG.chars().count();
|
|
break;
|
|
}
|
|
line_part.len += shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, shortcut);
|
|
}
|
|
let select_pane_shortcut = select_pane_shortcut(help.keybinds.is_empty(), help.palette);
|
|
if line_part.len + select_pane_shortcut.len <= max_len {
|
|
line_part.len += select_pane_shortcut.len;
|
|
line_part.part = format!("{}{}", line_part.part, select_pane_shortcut,);
|
|
}
|
|
line_part
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn keybinds(help: &ModeInfo, tip_name: &str, max_width: usize) -> LinePart {
|
|
// It is assumed that there is at least one TIP data in the TIPS HasMap.
|
|
let tip_body = TIPS
|
|
.get(tip_name)
|
|
.unwrap_or_else(|| TIPS.get("quicknav").unwrap());
|
|
|
|
let full_shortcut_list = full_shortcut_list(help, tip_body.full);
|
|
if full_shortcut_list.len <= max_width {
|
|
return full_shortcut_list;
|
|
}
|
|
let shortened_shortcut_list = shortened_shortcut_list(help, tip_body.medium);
|
|
if shortened_shortcut_list.len <= max_width {
|
|
return shortened_shortcut_list;
|
|
}
|
|
best_effort_shortcut_list(help, tip_body.short, max_width)
|
|
}
|
|
|
|
pub fn text_copied_hint(palette: &Palette) -> LinePart {
|
|
let hint = " Text copied to clipboard";
|
|
let green_color = match palette.green {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
LinePart {
|
|
part: Style::new().fg(green_color).bold().paint(hint).to_string(),
|
|
len: hint.len(),
|
|
}
|
|
}
|
|
|
|
pub fn system_clipboard_error(palette: &Palette) -> LinePart {
|
|
let hint = " Error using the system clipboard.";
|
|
let red_color = match palette.red {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
LinePart {
|
|
part: Style::new().fg(red_color).bold().paint(hint).to_string(),
|
|
len: hint.len(),
|
|
}
|
|
}
|
|
|
|
pub fn fullscreen_panes_to_hide(palette: &Palette, panes_to_hide: usize) -> LinePart {
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let green_color = match palette.green {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let orange_color = match palette.orange {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let shortcut_left_separator = Style::new().fg(white_color).bold().paint(" (");
|
|
let shortcut_right_separator = Style::new().fg(white_color).bold().paint("): ");
|
|
let fullscreen = "FULLSCREEN";
|
|
let puls = "+ ";
|
|
let panes = panes_to_hide.to_string();
|
|
let hide = " hidden panes";
|
|
let len = fullscreen.chars().count()
|
|
+ puls.chars().count()
|
|
+ panes.chars().count()
|
|
+ hide.chars().count()
|
|
+ 5; // 3 for ():'s around shortcut, 2 for the space
|
|
LinePart {
|
|
part: format!(
|
|
"{}{}{}{}{}{}",
|
|
shortcut_left_separator,
|
|
Style::new().fg(orange_color).bold().paint(fullscreen),
|
|
shortcut_right_separator,
|
|
Style::new().fg(white_color).bold().paint(puls),
|
|
Style::new().fg(green_color).bold().paint(panes),
|
|
Style::new().fg(white_color).bold().paint(hide)
|
|
),
|
|
len,
|
|
}
|
|
}
|
|
|
|
pub fn locked_fullscreen_panes_to_hide(palette: &Palette, panes_to_hide: usize) -> LinePart {
|
|
let white_color = match palette.white {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let green_color = match palette.green {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let orange_color = match palette.orange {
|
|
PaletteColor::Rgb((r, g, b)) => RGB(r, g, b),
|
|
PaletteColor::EightBit(color) => Fixed(color),
|
|
};
|
|
let locked_text = " -- INTERFACE LOCKED -- ";
|
|
let shortcut_left_separator = Style::new().fg(white_color).bold().paint(" (");
|
|
let shortcut_right_separator = Style::new().fg(white_color).bold().paint("): ");
|
|
let fullscreen = "FULLSCREEN";
|
|
let puls = "+ ";
|
|
let panes = panes_to_hide.to_string();
|
|
let hide = " hidden panes";
|
|
let len = locked_text.chars().count()
|
|
+ fullscreen.chars().count()
|
|
+ puls.chars().count()
|
|
+ panes.chars().count()
|
|
+ hide.chars().count()
|
|
+ 5; // 3 for ():'s around shortcut, 2 for the space
|
|
LinePart {
|
|
part: format!(
|
|
"{}{}{}{}{}{}{}",
|
|
Style::new().fg(white_color).bold().paint(locked_text),
|
|
shortcut_left_separator,
|
|
Style::new().fg(orange_color).bold().paint(fullscreen),
|
|
shortcut_right_separator,
|
|
Style::new().fg(white_color).bold().paint(puls),
|
|
Style::new().fg(green_color).bold().paint(panes),
|
|
Style::new().fg(white_color).bold().paint(hide)
|
|
),
|
|
len,
|
|
}
|
|
}
|