mod parser;
use parser::header::Header;
use parser::link::Link;
use parser::plain::Plain;
use gtk::{
gio::SimpleAction,
glib::{GString, Propagation, Uri, UriFlags},
prelude::{ActionExt, StyleContextExt, ToVariant, WidgetExt},
Align, CssProvider, Label, STYLE_PROVIDER_PRIORITY_APPLICATION,
};
use std::sync::Arc;
pub struct Reader {
title: Option<GString>,
widget: Label,
}
impl Reader {
pub fn new(gemtext: &str, base: &Uri, action_page_open: Arc<SimpleAction>) -> Self {
let mut title = None;
let mut markup = String::new();
for line in gemtext.lines() {
if let Some(header) = Header::from(line) {
markup.push_str(header.markup());
if title == None {
title = Some(header.text().clone());
}
continue;
}
if let Some(link) = Link::from(line, base) {
markup.push_str(link.markup());
continue;
}
markup.push_str(Plain::from(line).markup())
}
let css = CssProvider::new();
css.load_from_data("label{caret-color: transparent;}");
let widget = Label::builder()
.halign(Align::Fill)
.valign(Align::Fill)
.hexpand(true)
.vexpand(true)
.xalign(0.0)
.yalign(0.0)
.margin_start(8)
.margin_end(8)
.wrap(true)
.selectable(true)
.use_markup(true)
.label(markup)
.build();
widget
.style_context()
.add_provider(&css, STYLE_PROVIDER_PRIORITY_APPLICATION);
widget.connect_activate_link(move |_, href| {
if let Ok(uri) = Uri::parse(&href, UriFlags::NONE) {
return match uri.scheme().as_str() {
"gemini" => {
action_page_open.activate(Some(&uri.to_str().to_variant()));
Propagation::Stop
}
_ => Propagation::Proceed,
};
}
Propagation::Proceed
});
Self {
title,
widget,
}
}
pub fn title(&self) -> &Option<GString> {
&self.title
}
pub fn widget(&self) -> &Label {
&self.widget
}
}