update notifications applet with clear button, and times
This commit is contained in:
parent
68dec69d4c
commit
1767bc2a9d
6 changed files with 411 additions and 243 deletions
501
Cargo.lock
generated
501
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
|
@ -19,3 +19,8 @@ sendfd = { version = "0.4", features = [ "tokio" ] }
|
||||||
bytemuck = "1"
|
bytemuck = "1"
|
||||||
tracing-subscriber = "0.3"
|
tracing-subscriber = "0.3"
|
||||||
zbus = "3.14"
|
zbus = "3.14"
|
||||||
|
# Application i18n
|
||||||
|
i18n-embed = { version = "0.13.4", features = ["fluent-system", "desktop-requester"] }
|
||||||
|
i18n-embed-fl = "0.6.4"
|
||||||
|
rust-embed = "6.3.0"
|
||||||
|
rust-embed-utils = "7.5.0"
|
||||||
|
|
|
||||||
4
cosmic-applet-notifications/i18n.toml
Normal file
4
cosmic-applet-notifications/i18n.toml
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
fallback_language = "en"
|
||||||
|
|
||||||
|
[fluent]
|
||||||
|
assets_dir = "i18n"
|
||||||
|
|
@ -0,0 +1,3 @@
|
||||||
|
hours-ago = {$duration} Hours Ago
|
||||||
|
minutes-ago = {$duration} Minutes Ago
|
||||||
|
clear-all = Clear All Notifications
|
||||||
47
cosmic-applet-notifications/src/localize.rs
Normal file
47
cosmic-applet-notifications/src/localize.rs
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
// SPDX-License-Identifier: MPL-2.0-only
|
||||||
|
|
||||||
|
use cosmic_time::once_cell::sync::Lazy;
|
||||||
|
use i18n_embed::{
|
||||||
|
fluent::{fluent_language_loader, FluentLanguageLoader},
|
||||||
|
DefaultLocalizer, LanguageLoader, Localizer,
|
||||||
|
};
|
||||||
|
use rust_embed::RustEmbed;
|
||||||
|
|
||||||
|
#[derive(RustEmbed)]
|
||||||
|
#[folder = "i18n/"]
|
||||||
|
struct Localizations;
|
||||||
|
|
||||||
|
pub static LANGUAGE_LOADER: Lazy<FluentLanguageLoader> = Lazy::new(|| {
|
||||||
|
let loader: FluentLanguageLoader = fluent_language_loader!();
|
||||||
|
|
||||||
|
loader
|
||||||
|
.load_fallback_language(&Localizations)
|
||||||
|
.expect("Error while loading fallback language");
|
||||||
|
|
||||||
|
loader
|
||||||
|
});
|
||||||
|
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! fl {
|
||||||
|
($message_id:literal) => {{
|
||||||
|
i18n_embed_fl::fl!($crate::localize::LANGUAGE_LOADER, $message_id)
|
||||||
|
}};
|
||||||
|
|
||||||
|
($message_id:literal, $($args:expr),*) => {{
|
||||||
|
i18n_embed_fl::fl!($crate::localize::LANGUAGE_LOADER, $message_id, $($args), *)
|
||||||
|
}};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the `Localizer` to be used for localizing this library.
|
||||||
|
pub fn localizer() -> Box<dyn Localizer> {
|
||||||
|
Box::from(DefaultLocalizer::new(&*LANGUAGE_LOADER, &Localizations))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn localize() {
|
||||||
|
let localizer = localizer();
|
||||||
|
let requested_languages = i18n_embed::DesktopLanguageRequester::requested_languages();
|
||||||
|
|
||||||
|
if let Err(error) = localizer.select(&requested_languages) {
|
||||||
|
eprintln!("Error while loading language for App List {}", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,3 +1,4 @@
|
||||||
|
mod localize;
|
||||||
mod subscriptions;
|
mod subscriptions;
|
||||||
|
|
||||||
use cosmic::cosmic_config::{config_subscription, Config, CosmicConfigEntry};
|
use cosmic::cosmic_config::{config_subscription, Config, CosmicConfigEntry};
|
||||||
|
|
@ -6,7 +7,8 @@ use cosmic::iced::{
|
||||||
widget::{button, column, row, text, Row, Space},
|
widget::{button, column, row, text, Row, Space},
|
||||||
window, Alignment, Application, Color, Command, Length, Subscription,
|
window, Alignment, Application, Color, Command, Length, Subscription,
|
||||||
};
|
};
|
||||||
use cosmic::iced_core::image;
|
use cosmic::iced_core::alignment::Horizontal;
|
||||||
|
use cosmic::iced_core::{image, Widget};
|
||||||
use cosmic::iced_futures::subscription;
|
use cosmic::iced_futures::subscription;
|
||||||
use cosmic::iced_widget::button::StyleSheet;
|
use cosmic::iced_widget::button::StyleSheet;
|
||||||
use cosmic_applet::{applet_button_theme, CosmicAppletHelper};
|
use cosmic_applet::{applet_button_theme, CosmicAppletHelper};
|
||||||
|
|
@ -15,13 +17,14 @@ use cosmic::iced_style::application::{self, Appearance};
|
||||||
|
|
||||||
use cosmic::iced_widget::{horizontal_space, scrollable, Column};
|
use cosmic::iced_widget::{horizontal_space, scrollable, Column};
|
||||||
use cosmic::theme::{Button, Svg};
|
use cosmic::theme::{Button, Svg};
|
||||||
use cosmic::widget::{divider, icon};
|
use cosmic::widget::{container, divider, icon};
|
||||||
use cosmic::Renderer;
|
use cosmic::Renderer;
|
||||||
use cosmic::{Element, Theme};
|
use cosmic::{Element, Theme};
|
||||||
use cosmic_notifications_config::NotificationsConfig;
|
use cosmic_notifications_config::NotificationsConfig;
|
||||||
use cosmic_notifications_util::{AppletEvent, Notification};
|
use cosmic_notifications_util::{AppletEvent, Notification};
|
||||||
use cosmic_time::{anim, chain, id, once_cell::sync::Lazy, Instant, Timeline};
|
use cosmic_time::{anim, chain, id, once_cell::sync::Lazy, Instant, Timeline};
|
||||||
use std::borrow::Cow;
|
use std::borrow::Cow;
|
||||||
|
use std::collections::HashMap;
|
||||||
use std::process;
|
use std::process;
|
||||||
use tokio::sync::mpsc::Sender;
|
use tokio::sync::mpsc::Sender;
|
||||||
use tracing::info;
|
use tracing::info;
|
||||||
|
|
@ -29,6 +32,8 @@ use tracing::info;
|
||||||
#[tokio::main(flavor = "current_thread")]
|
#[tokio::main(flavor = "current_thread")]
|
||||||
pub async fn main() -> cosmic::iced::Result {
|
pub async fn main() -> cosmic::iced::Result {
|
||||||
tracing_subscriber::fmt::init();
|
tracing_subscriber::fmt::init();
|
||||||
|
// Prepare i18n
|
||||||
|
localize::localize();
|
||||||
|
|
||||||
info!("Notifications applet");
|
info!("Notifications applet");
|
||||||
|
|
||||||
|
|
@ -37,7 +42,6 @@ pub async fn main() -> cosmic::iced::Result {
|
||||||
}
|
}
|
||||||
|
|
||||||
static DO_NOT_DISTURB: Lazy<id::Toggler> = Lazy::new(id::Toggler::unique);
|
static DO_NOT_DISTURB: Lazy<id::Toggler> = Lazy::new(id::Toggler::unique);
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
struct Notifications {
|
struct Notifications {
|
||||||
applet_helper: CosmicAppletHelper,
|
applet_helper: CosmicAppletHelper,
|
||||||
|
|
@ -64,6 +68,7 @@ enum Message {
|
||||||
Config(NotificationsConfig),
|
Config(NotificationsConfig),
|
||||||
DbusEvent(subscriptions::dbus::Output),
|
DbusEvent(subscriptions::dbus::Output),
|
||||||
Dismissed(u32),
|
Dismissed(u32),
|
||||||
|
ClearAll,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Application for Notifications {
|
impl Application for Notifications {
|
||||||
|
|
@ -230,6 +235,21 @@ impl Application for Notifications {
|
||||||
Command::none()
|
Command::none()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
Message::ClearAll => {
|
||||||
|
for n in self.notifications.drain(..) {
|
||||||
|
if let Some(tx) = &self.dbus_sender {
|
||||||
|
let tx = tx.clone();
|
||||||
|
tokio::spawn(async move {
|
||||||
|
if let Err(err) =
|
||||||
|
tx.send(subscriptions::dbus::Input::Dismiss(n.id)).await
|
||||||
|
{
|
||||||
|
tracing::error!("{:?}", err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Command::none()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -254,28 +274,36 @@ impl Application for Notifications {
|
||||||
row_button(vec!["Notification Settings...".into()]).on_press(Message::Settings);
|
row_button(vec!["Notification Settings...".into()]).on_press(Message::Settings);
|
||||||
|
|
||||||
let notifications = if self.notifications.len() == 0 {
|
let notifications = if self.notifications.len() == 0 {
|
||||||
row![
|
row![container(
|
||||||
Space::with_width(Length::Fill),
|
|
||||||
column![text_icon(&self.icon_name, 40), "No Notifications"]
|
column![text_icon(&self.icon_name, 40), "No Notifications"]
|
||||||
.align_items(Alignment::Center),
|
.align_items(Alignment::Center)
|
||||||
Space::with_width(Length::Fill)
|
)
|
||||||
]
|
.width(Length::Fill)
|
||||||
|
.align_x(Horizontal::Center)]
|
||||||
.spacing(12)
|
.spacing(12)
|
||||||
} else {
|
} else {
|
||||||
let mut notifs = Vec::with_capacity(self.notifications.len());
|
let mut notifs: Vec<Element<_>> = Vec::with_capacity(self.notifications.len());
|
||||||
|
notifs.push(
|
||||||
|
column![cosmic::widget::button(Button::Text)
|
||||||
|
.custom(vec![text(fl!("clear-all")).size(14).into()])
|
||||||
|
.on_press(Message::ClearAll)]
|
||||||
|
.into(),
|
||||||
|
);
|
||||||
for n in &self.notifications {
|
for n in &self.notifications {
|
||||||
let summary = text(if n.summary.len() > 24 {
|
let app_name = text(if n.app_name.len() > 24 {
|
||||||
Cow::from(format!(
|
Cow::from(format!(
|
||||||
"{:.26}...",
|
"{:.26}...",
|
||||||
n.summary.lines().next().unwrap_or_default()
|
n.app_name.lines().next().unwrap_or_default()
|
||||||
))
|
))
|
||||||
} else {
|
} else {
|
||||||
Cow::from(&n.summary)
|
Cow::from(&n.app_name)
|
||||||
})
|
})
|
||||||
.size(18);
|
.size(12)
|
||||||
|
.width(Length::Fill);
|
||||||
let urgency = n.urgency();
|
let urgency = n.urgency();
|
||||||
|
|
||||||
|
let duration_since = text(duration_ago_msg(n)).size(12);
|
||||||
|
|
||||||
notifs.push(
|
notifs.push(
|
||||||
cosmic::widget::button(Button::Custom {
|
cosmic::widget::button(Button::Custom {
|
||||||
active: Box::new(move |t| {
|
active: Box::new(move |t| {
|
||||||
|
|
@ -310,12 +338,12 @@ impl Application for Notifications {
|
||||||
.custom(vec![column!(
|
.custom(vec![column!(
|
||||||
match n.image() {
|
match n.image() {
|
||||||
Some(cosmic_notifications_util::Image::File(path)) => {
|
Some(cosmic_notifications_util::Image::File(path)) => {
|
||||||
row![icon(path.as_path(), 32), summary]
|
row![icon(path.as_path(), 16), app_name, duration_since]
|
||||||
.spacing(8)
|
.spacing(8)
|
||||||
.align_items(Alignment::Center)
|
.align_items(Alignment::Center)
|
||||||
}
|
}
|
||||||
Some(cosmic_notifications_util::Image::Name(name)) => {
|
Some(cosmic_notifications_util::Image::Name(name)) => {
|
||||||
row![icon(name.as_str(), 32), summary]
|
row![icon(name.as_str(), 16), app_name, duration_since]
|
||||||
.spacing(8)
|
.spacing(8)
|
||||||
.align_items(Alignment::Center)
|
.align_items(Alignment::Center)
|
||||||
}
|
}
|
||||||
|
|
@ -326,21 +354,30 @@ impl Application for Notifications {
|
||||||
}) => {
|
}) => {
|
||||||
let handle =
|
let handle =
|
||||||
image::Handle::from_pixels(*width, *height, data.clone());
|
image::Handle::from_pixels(*width, *height, data.clone());
|
||||||
row![icon(handle, 32), summary]
|
row![icon(handle, 16), app_name, duration_since]
|
||||||
.spacing(8)
|
.spacing(8)
|
||||||
.align_items(Alignment::Center)
|
.align_items(Alignment::Center)
|
||||||
}
|
}
|
||||||
None => row![summary],
|
None => row![app_name, duration_since],
|
||||||
},
|
},
|
||||||
text(if n.body.len() > 38 {
|
text(if n.summary.len() > 77 {
|
||||||
Cow::from(format!(
|
Cow::from(format!(
|
||||||
"{:.40}...",
|
"{:.80}...",
|
||||||
n.body.lines().next().unwrap_or_default()
|
n.summary.lines().next().unwrap_or_default()
|
||||||
))
|
))
|
||||||
} else {
|
} else {
|
||||||
Cow::from(&n.summary)
|
Cow::from(&n.summary)
|
||||||
})
|
})
|
||||||
.size(14),
|
.size(14),
|
||||||
|
text(if n.body.len() > 77 {
|
||||||
|
Cow::from(format!(
|
||||||
|
"{:.80}...",
|
||||||
|
n.body.lines().next().unwrap_or_default()
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
Cow::from(&n.body)
|
||||||
|
})
|
||||||
|
.size(12),
|
||||||
horizontal_space(Length::Fixed(300.0)),
|
horizontal_space(Length::Fixed(300.0)),
|
||||||
)
|
)
|
||||||
.spacing(8)
|
.spacing(8)
|
||||||
|
|
@ -349,7 +386,6 @@ impl Application for Notifications {
|
||||||
.into(),
|
.into(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
row!(scrollable(
|
row!(scrollable(
|
||||||
Column::with_children(notifs)
|
Column::with_children(notifs)
|
||||||
.spacing(8)
|
.spacing(8)
|
||||||
|
|
@ -402,3 +438,17 @@ fn row_button(
|
||||||
fn text_icon(name: &str, size: u16) -> cosmic::widget::Icon {
|
fn text_icon(name: &str, size: u16) -> cosmic::widget::Icon {
|
||||||
icon(name, size).style(Svg::Symbolic)
|
icon(name, size).style(Svg::Symbolic)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn duration_ago_msg(notification: &Notification) -> String {
|
||||||
|
if let Some(d) = notification.duration_since() {
|
||||||
|
let min = d.as_secs() / 60;
|
||||||
|
let hrs = min / 60;
|
||||||
|
if hrs > 0 {
|
||||||
|
fl!("hours-ago", HashMap::from_iter(vec![("duration", hrs)]))
|
||||||
|
} else {
|
||||||
|
fl!("minutes-ago", HashMap::from_iter(vec![("duration", min)]))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
format!("")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue