// Copyright 2023 System76 // SPDX-License-Identifier: MPL-2.0 //! Application API example use cosmic::app::{Core, Settings, Task}; use cosmic::iced::widget::column; use cosmic::iced_core::Size; use cosmic::widget::nav_bar; use cosmic::{executor, iced, ApplicationExt, Element}; #[derive(Clone, Copy)] pub enum Page { Page1, Page2, Page3, Page4, } impl Page { const fn as_str(self) -> &'static str { match self { Page::Page1 => "Page 1", Page::Page2 => "Page 2", Page::Page3 => "Page 3", Page::Page4 => "Page 4", } } } /// Runs application with these settings #[rustfmt::skip] fn main() -> Result<(), Box> { tracing_subscriber::fmt::init(); let _ = tracing_log::LogTracer::init(); let input = vec![ (Page::Page1, "🖖 Hello from libcosmic.".into()), (Page::Page2, "🌟 This is an example application.".into()), (Page::Page3, "🚧 The libcosmic API is not stable yet.".into()), (Page::Page4, "🚀 Copy the source code and experiment today!".into()), ]; let settings = Settings::default() .size(Size::new(1024., 768.)); cosmic::app::run::(settings, input)?; Ok(()) } /// Messages that are used specifically by our [`App`]. #[derive(Clone, Debug)] pub enum Message { Input1(String), Input2(String), Ignore, ToggleHide, } /// The [`App`] stores application-specific state. pub struct App { core: Core, nav_model: nav_bar::Model, input_1: String, input_2: String, hidden: bool, } /// Implement [`cosmic::Application`] to integrate with COSMIC. impl cosmic::Application for App { /// Default async executor to use with the app. type Executor = executor::Default; /// Argument received [`cosmic::Application::new`]. type Flags = Vec<(Page, String)>; /// Message type specific to our [`App`]. type Message = Message; /// The unique application ID to supply to the window manager. const APP_ID: &'static str = "org.cosmic.AppDemo"; fn core(&self) -> &Core { &self.core } fn core_mut(&mut self) -> &mut Core { &mut self.core } /// Creates the application, and optionally emits command on initialize. fn init(core: Core, input: Self::Flags) -> (Self, Task) { let mut nav_model = nav_bar::Model::default(); for (title, content) in input { nav_model.insert().text(title.as_str()).data(content); } nav_model.activate_position(0); let mut app = App { core, nav_model, input_1: String::new(), input_2: String::new(), hidden: true, }; let command = app.update_title(); (app, command) } /// Allows COSMIC to integrate with your application's [`nav_bar::Model`]. fn nav_model(&self) -> Option<&nav_bar::Model> { Some(&self.nav_model) } /// Called when a navigation item is selected. fn on_nav_select(&mut self, id: nav_bar::Id) -> Task { self.nav_model.activate(id); self.update_title() } /// Handle application events here. fn update(&mut self, message: Self::Message) -> Task { match message { Message::Input1(v) => { self.input_1 = v; } Message::Input2(v) => { self.input_2 = v; } Message::Ignore => {} Message::ToggleHide => { self.hidden = !self.hidden; } } Task::none() } /// Creates a view after each update. fn view(&self) -> Element { let page_content = self .nav_model .active_data::() .map_or("No page selected", String::as_str); let text = cosmic::widget::text(page_content); let centered = cosmic::widget::container( column![ text, cosmic::widget::text_input::text_input("", &self.input_1) .on_input(Message::Input1) .on_clear(Message::Ignore), cosmic::widget::text_input::secure_input( "", &self.input_1, Some(Message::ToggleHide), self.hidden ) .on_input(Message::Input1), cosmic::widget::text_input::text_input("", &self.input_1).on_input(Message::Input1), cosmic::widget::text_input::search_input("", &self.input_2) .on_input(Message::Input2) .on_clear(Message::Ignore), ] .width(iced::Length::Fill) .height(iced::Length::Shrink) .align_x(iced::alignment::Horizontal::Center), ) .width(iced::Length::Fill) .height(iced::Length::Shrink) .align_x(iced::alignment::Horizontal::Center) .align_y(iced::alignment::Vertical::Center); Element::from(centered) } } impl App where Self: cosmic::Application, { fn active_page_title(&mut self) -> &str { self.nav_model .text(self.nav_model.active()) .unwrap_or("Unknown Page") } fn update_title(&mut self) -> Task { let header_title = self.active_page_title().to_owned(); let window_title = format!("{header_title} — COSMIC AppDemo"); self.set_header_title(header_title); if let Some(id) = self.core.main_window_id() { self.set_window_title(window_title, id) } else { Task::none() } } }