Yoda/src/browser/main/tab/page/mod.rs
2024-09-25 22:25:12 +03:00

229 lines
8.8 KiB
Rust

mod content;
mod navigation;
use content::Content;
use navigation::Navigation;
use gtk::{
gio::{Cancellable, SocketClient, SocketProtocol, TlsCertificateFlags},
glib::{GString, Priority, Regex, RegexCompileFlags, RegexMatchFlags, Uri, UriFlags},
prelude::{
BoxExt, IOStreamExt, InputStreamExtManual, OutputStreamExtManual, SocketClientExt,
WidgetExt,
},
Box, Orientation,
};
use std::{cell::RefCell, sync::Arc};
// Extras
enum Mime {
Undefined,
TextGemini,
TextPlain,
}
struct Meta {
title: GString,
description: GString,
mime: Mime,
progress_fraction: f32,
}
// Main
pub struct Page {
// GTK
widget: Box,
// Components
navigation: Navigation,
content: Arc<Content>,
// Extras
meta: RefCell<Meta>,
}
impl Page {
// Construct
pub fn new(name: GString) -> Page {
// Init components
let content = Content::new();
let navigation = Navigation::new();
// Init widget
let widget = Box::builder()
.orientation(Orientation::Vertical)
.name(name)
.build();
widget.append(navigation.widget());
widget.append(content.widget());
// Init meta
let meta = RefCell::new(Meta {
title: GString::new(),
description: GString::new(),
mime: Mime::Undefined,
progress_fraction: 0.0,
});
// Result
Self {
widget,
content,
navigation,
meta,
}
}
// Actions
pub fn reload(&self) {
// Init globals
let request_text = self.navigation.request_text();
// Update
self.meta.borrow_mut().title = GString::from("Reload");
self.meta.borrow_mut().description = request_text.clone();
self.meta.borrow_mut().mime = Mime::Undefined;
self.meta.borrow_mut().progress_fraction = 0.0;
let _ = self.widget.activate_action("win.update", None);
/*let _uri = */
match Uri::parse(&request_text, UriFlags::NONE) {
Ok(uri) => {
// Route request by scheme
match uri.scheme().as_str() {
"file" => {
todo!()
}
"gemini" => {
let client = SocketClient::new();
client.set_timeout(10);
client.set_tls(true);
client.set_tls_validation_flags(TlsCertificateFlags::INSECURE);
client.set_protocol(SocketProtocol::Tcp);
client.connect_to_uri_async(
"gemini://geminiprotocol.net:1965/", // @TODO &uri.to_str(),
1965,
Some(&Cancellable::new()),
move |result| match result {
Ok(connection) => {
connection.output_stream().write_all_async(
"gemini://geminiprotocol.net:1965/\r\n", // @TODO
Priority::DEFAULT,
Some(&Cancellable::new()),
move |result| match result {
Ok(_) => {
connection.input_stream().read_all_async(
vec![0; 0xfffff], // 1Mb @TODO
Priority::DEFAULT,
Some(&Cancellable::new()),
move |result| match result {
Ok(response) => {
match GString::from_utf8_until_nul(
response.0,
) {
Ok(data) => {
// Detect page meta
let meta = Regex::split_simple(
r"^(\d+)?\s([\w]+\/[\w]+)?",
&data,
RegexCompileFlags::DEFAULT,
RegexMatchFlags::DEFAULT,
);
//println!("{:?}", meta);
//println!("Result: {}", data);
}
Err(e) => {
eprintln!(
"Failed to read buffer: {e}"
)
}
}
// @TODO connection.close(cancellable);
}
Err(e) => {
eprintln!(
"Failed to read response: {:?}",
e
);
}
},
);
}
Err(e) => {
eprintln!("Failed to write request: {:?}", e);
}
},
);
}
Err(e) => {
eprintln!("Failed to connect: {e}"); // @TODO
}
},
);
}
"nex" => {
todo!()
}
scheme => {
println!("Protocol {scheme} not supported");
}
}
}
Err(_) => {
// Try interpret URI manually
if Regex::match_simple(
r"^[^\/\s]+\.[\w]{2,}",
request_text.clone(),
RegexCompileFlags::DEFAULT,
RegexMatchFlags::DEFAULT,
) {
// Seems request contain some host, try append default scheme
let request_text = GString::from(format!("gemini://{request_text}"));
// Make sure new request conversible to valid URI
match Uri::parse(&request_text, UriFlags::NONE) {
Ok(_) => {
self.navigation.set_request_text(
&request_text,
true, // activate (page reload)
);
}
Err(_) => {
// @TODO any action here?
}
}
} else {
// Plain text given, make search request to default provider
self.navigation.set_request_text(
&GString::from(format!(
"gemini://tlgs.one/search?{}",
Uri::escape_string(&request_text, None, false)
)),
true, // activate (page reload)
);
}
}
};
}
pub fn update(&self) {
self.navigation.update();
// @TODO self.content.update();
}
// Getters
pub fn title(&self) -> GString {
self.meta.borrow().title.clone()
}
pub fn description(&self) -> GString {
self.meta.borrow().description.clone()
}
pub fn widget(&self) -> &Box {
&self.widget
}
}