// SPDX-License-Identifier: GPL-3.0-only use cosmic::widget::menu::Item as MenuItem; use cosmic::widget::menu::key_bind::KeyBind; use cosmic::{ Element, app::Core, iced::{Background, Length, advanced::widget::text::Style as TextStyle, widget::column}, iced_core::Border, theme, widget::{ self, divider, menu::{ItemHeight, ItemWidth, menu_button}, responsive_menu_bar, segmented_button, space, }, }; use std::{collections::HashMap, path::PathBuf, sync::LazyLock}; use crate::{Action, Config, ConfigState, Message, fl}; static MENU_ID: LazyLock = LazyLock::new(|| cosmic::widget::Id::new("responsive-menu")); // Menu rows are fixed-height in libcosmic; wrapped labels get visually clipped. // Keep recent path labels short enough to stay on one line. const RECENT_MENU_LABEL_MAX_CHARS: usize = 40; fn char_count(value: &str) -> usize { value.chars().count() } fn take_prefix_chars(value: &str, count: usize) -> String { value.chars().take(count).collect() } fn take_suffix_chars(value: &str, count: usize) -> String { let total = char_count(value); if count >= total { value.to_string() } else { value.chars().skip(total - count).collect() } } fn truncate_middle(value: &str, max_chars: usize) -> String { const ELLIPSIS: &str = "..."; if char_count(value) <= max_chars { return value.to_string(); } if max_chars <= ELLIPSIS.len() + 2 { return take_prefix_chars(value, max_chars); } let prefix_len = (max_chars - ELLIPSIS.len()) / 2; let suffix_len = max_chars - ELLIPSIS.len() - prefix_len; format!( "{}{}{}", take_prefix_chars(value, prefix_len), ELLIPSIS, take_suffix_chars(value, suffix_len) ) } fn format_recent_menu_path(path: &PathBuf, home_dir_opt: Option<&PathBuf>) -> String { const ELLIPSIS: &str = "..."; let display = if let Some(home_dir) = home_dir_opt { if let Ok(part) = path.strip_prefix(home_dir) { format!("~/{}", part.display()) } else { path.display().to_string() } } else { path.display().to_string() }; if char_count(&display) <= RECENT_MENU_LABEL_MAX_CHARS { return display; } let file_name = path.file_name().and_then(|name| name.to_str()); let parent_name = path .parent() .and_then(|parent| parent.file_name()) .and_then(|name| name.to_str()); let mut tails = Vec::new(); if let Some(file_name) = file_name { tails.push(format!("/{}", file_name)); if let Some(parent_name) = parent_name { tails.push(format!("/{}/{}", parent_name, file_name)); } } for tail in tails.into_iter().rev() { let tail_len = char_count(&tail); if tail_len + ELLIPSIS.len() >= RECENT_MENU_LABEL_MAX_CHARS { continue; } let prefix_len = RECENT_MENU_LABEL_MAX_CHARS - ELLIPSIS.len() - tail_len; if prefix_len < 6 { continue; } return format!("{}{}{}", take_prefix_chars(&display, prefix_len), ELLIPSIS, tail); } truncate_middle(&display, RECENT_MENU_LABEL_MAX_CHARS) } pub fn context_menu<'a>( key_binds: &HashMap, entity: segmented_button::Entity, ) -> Element<'a, Message> { fn key_style(theme: &cosmic::Theme) -> TextStyle { let mut color = theme.cosmic().background.component.on; color.alpha *= 0.75; TextStyle { color: Some(color.into()), } } let menu_item = |menu_label, menu_action| { let mut key = String::new(); for (key_bind, key_action) in key_binds.iter() { if key_action == &menu_action { key = key_bind.to_string(); break; } } menu_button(vec![ widget::text(menu_label).into(), space::horizontal().into(), widget::text(key) .class(theme::Text::Custom(key_style)) .into(), ]) .on_press(Message::TabContextAction(entity, menu_action)) }; widget::container(column!( menu_item(fl!("undo"), Action::Undo), menu_item(fl!("redo"), Action::Redo), divider::horizontal::light(), menu_item(fl!("cut"), Action::Cut), menu_item(fl!("copy"), Action::Copy), menu_item(fl!("paste"), Action::Paste), menu_item(fl!("select-all"), Action::SelectAll), )) .padding(1) //TODO: move style to libcosmic .style(|theme| { let cosmic = theme.cosmic(); let component = &cosmic.background.component; widget::container::Style { icon_color: Some(component.on.into()), text_color: Some(component.on.into()), background: Some(Background::Color(component.base.into())), border: Border { radius: cosmic.radius_s().map(|x| x + 1.0).into(), width: 1.0, color: component.divider.into(), }, ..Default::default() } }) .width(Length::Fixed(240.0)) .into() } pub fn menu_bar<'a>( core: &Core, config: &Config, config_state: &ConfigState, key_binds: &HashMap, projects: &Vec<(String, PathBuf)>, ) -> Element<'a, Message> { //TODO: port to libcosmic let menu_tab_width = |tab_width: u16| { MenuItem::CheckBox( fl!("tab-width", tab_width = tab_width), None, config.tab_width == tab_width, Action::TabWidth(tab_width), ) }; let home_dir_opt = dirs::home_dir(); let mut recent_files = Vec::with_capacity(config_state.recent_files.len()); for (i, path) in config_state.recent_files.iter().enumerate() { recent_files.push(MenuItem::Button( format_recent_menu_path(path, home_dir_opt.as_ref()), None, Action::OpenRecentFile(i), )); } let mut recent_projects = Vec::with_capacity(config_state.recent_projects.len()); for (i, path) in config_state.recent_projects.iter().enumerate() { recent_projects.push(MenuItem::Button( format_recent_menu_path(path, home_dir_opt.as_ref()), None, Action::OpenRecentProject(i), )); } let mut close_projects = Vec::with_capacity(projects.len()); for (project_i, (name, _path)) in projects.iter().enumerate() { close_projects.push(MenuItem::Button( name.clone(), None, Action::CloseProject(project_i), )); } responsive_menu_bar() .item_height(ItemHeight::Dynamic(40)) .item_width(ItemWidth::Uniform(320)) .spacing(4.0) .into_element( core, key_binds, MENU_ID.clone(), Message::Surface, vec![ ( (fl!("file")), vec![ MenuItem::Button(fl!("new-file"), None, Action::NewFile), MenuItem::Button(fl!("new-window"), None, Action::NewWindow), MenuItem::Divider, MenuItem::Button(fl!("open-file"), None, Action::OpenFileDialog), MenuItem::Folder(fl!("open-recent-file"), recent_files), MenuItem::Button(fl!("close-file"), None, Action::CloseFile), MenuItem::Divider, MenuItem::Button(fl!("menu-open-project"), None, Action::OpenProjectDialog), MenuItem::Folder(fl!("open-recent-project"), recent_projects), MenuItem::Folder(fl!("close-project"), close_projects), MenuItem::Divider, MenuItem::Button(fl!("save"), None, Action::Save), MenuItem::Button(fl!("save-as"), None, Action::SaveAsDialog), MenuItem::Divider, MenuItem::Button(fl!("revert-all-changes"), None, Action::RevertAllChanges), MenuItem::Divider, MenuItem::Button( fl!("menu-document-statistics"), None, Action::ToggleDocumentStatistics, ), //TODO MenuItem::Button(fl!("document-type"), Action::Todo), //TODO MenuItem::Button(fl!("encoding"), Action::Todo), MenuItem::Button( fl!("menu-git-management"), None, Action::ToggleGitManagement, ), //TODO MenuItem::Button(fl!("print"), Action::Todo), MenuItem::Divider, MenuItem::Button(fl!("quit"), None, Action::Quit), ], ), ( (fl!("edit")), vec![ MenuItem::Button(fl!("undo"), None, Action::Undo), MenuItem::Button(fl!("redo"), None, Action::Redo), MenuItem::Divider, MenuItem::Button(fl!("cut"), None, Action::Cut), MenuItem::Button(fl!("copy"), None, Action::Copy), MenuItem::Button(fl!("paste"), None, Action::Paste), MenuItem::Button(fl!("select-all"), None, Action::SelectAll), MenuItem::Divider, MenuItem::Button(fl!("find"), None, Action::Find), MenuItem::Button(fl!("replace"), None, Action::FindAndReplace), MenuItem::Button(fl!("find-in-project"), None, Action::ToggleProjectSearch), /*TODO: implement spell-check MenuItem::Divider, MenuItem::Button(fl!("spell-check"), None, Action::Todo), */ ], ), ( (fl!("view")), vec![ MenuItem::Folder( fl!("indentation"), vec![ MenuItem::CheckBox( fl!("automatic-indentation"), None, config.auto_indent, Action::ToggleAutoIndent, ), MenuItem::Divider, menu_tab_width(1), menu_tab_width(2), menu_tab_width(3), menu_tab_width(4), menu_tab_width(5), menu_tab_width(6), menu_tab_width(7), menu_tab_width(8), //TODO MenuItem::Divider, //TODO MenuItem::Button(fl!("convert-indentation-to-spaces"), Action::Todo), //TODO MenuItem::Button(fl!("convert-indentation-to-tabs"), Action::Todo), ], ), MenuItem::Divider, MenuItem::Button(fl!("zoom-in"), None, Action::ZoomIn), MenuItem::Button(fl!("default-size"), None, Action::ZoomReset), MenuItem::Button(fl!("zoom-out"), None, Action::ZoomOut), MenuItem::Divider, MenuItem::CheckBox( fl!("word-wrap"), None, config.word_wrap, Action::ToggleWordWrap, ), MenuItem::CheckBox( fl!("show-line-numbers"), None, config.line_numbers, Action::ToggleLineNumbers, ), MenuItem::CheckBox( fl!("highlight-current-line"), None, config.highlight_current_line, Action::ToggleHighlightCurrentLine, ), //TODO: MenuItem::CheckBox(fl!("syntax-highlighting"), Action::Todo), MenuItem::Divider, MenuItem::Button(fl!("menu-settings"), None, Action::ToggleSettingsPage), //TODO MenuItem::Divider, //TODO MenuItem::Button(fl!("menu-keyboard-shortcuts"), Action::Todo), MenuItem::Divider, MenuItem::Button(fl!("menu-about"), None, Action::About), ], ), ], ) }