mirror of
https://github.com/YGGverse/Yoda.git
synced 2025-01-15 17:20:08 +00:00
draft redirection features
This commit is contained in:
parent
aa44325dea
commit
e3a6796627
@ -217,9 +217,7 @@ impl Tab {
|
|||||||
|
|
||||||
// Update tab title on loading indicator inactive
|
// Update tab title on loading indicator inactive
|
||||||
if !item.page_is_loading() {
|
if !item.page_is_loading() {
|
||||||
if let Some(title) = item.page_meta_title() {
|
item.gobject().set_title(item.page_meta_title().as_str())
|
||||||
item.gobject().set_title(title.as_str())
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Update all tabs on ID not found @TODO change initial update method
|
// Update all tabs on ID not found @TODO change initial update method
|
||||||
@ -230,9 +228,7 @@ impl Tab {
|
|||||||
|
|
||||||
// Update tab title on loading indicator inactive
|
// Update tab title on loading indicator inactive
|
||||||
if !item.page_is_loading() {
|
if !item.page_is_loading() {
|
||||||
if let Some(title) = item.page_meta_title() {
|
item.gobject().set_title(item.page_meta_title().as_str())
|
||||||
item.gobject().set_title(title.as_str())
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -212,7 +212,7 @@ impl Item {
|
|||||||
self.page.is_loading()
|
self.page.is_loading()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn page_meta_title(&self) -> Option<GString> {
|
pub fn page_meta_title(&self) -> GString {
|
||||||
self.page.meta_title()
|
self.page.meta_title()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,7 +21,7 @@ use gtk::{
|
|||||||
},
|
},
|
||||||
glib::{
|
glib::{
|
||||||
gformat, uuid_string_random, Bytes, GString, Priority, Regex, RegexCompileFlags,
|
gformat, uuid_string_random, Bytes, GString, Priority, Regex, RegexCompileFlags,
|
||||||
RegexMatchFlags, Uri, UriFlags,
|
RegexMatchFlags, Uri, UriFlags, UriHideFlags,
|
||||||
},
|
},
|
||||||
prelude::{
|
prelude::{
|
||||||
ActionExt, IOStreamExt, OutputStreamExt, SocketClientExt, StaticVariantType, ToVariant,
|
ActionExt, IOStreamExt, OutputStreamExt, SocketClientExt, StaticVariantType, ToVariant,
|
||||||
@ -29,7 +29,7 @@ use gtk::{
|
|||||||
Box,
|
Box,
|
||||||
};
|
};
|
||||||
use sqlite::Transaction;
|
use sqlite::Transaction;
|
||||||
use std::{cell::RefCell, sync::Arc, time::Duration};
|
use std::{sync::Arc, time::Duration};
|
||||||
|
|
||||||
pub struct Page {
|
pub struct Page {
|
||||||
id: GString,
|
id: GString,
|
||||||
@ -42,7 +42,7 @@ pub struct Page {
|
|||||||
content: Arc<Content>,
|
content: Arc<Content>,
|
||||||
input: Arc<Input>,
|
input: Arc<Input>,
|
||||||
// Extras
|
// Extras
|
||||||
meta: Arc<RefCell<Meta>>,
|
meta: Arc<Meta>,
|
||||||
// GTK
|
// GTK
|
||||||
widget: Arc<Widget>,
|
widget: Arc<Widget>,
|
||||||
}
|
}
|
||||||
@ -84,7 +84,7 @@ impl Page {
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Init async mutable Meta object
|
// Init async mutable Meta object
|
||||||
let meta = Arc::new(RefCell::new(Meta::new()));
|
let meta = Meta::new_arc(Status::New, gformat!("New page"));
|
||||||
|
|
||||||
// Init events
|
// Init events
|
||||||
action_page_open.connect_activate({
|
action_page_open.connect_activate({
|
||||||
@ -105,7 +105,7 @@ impl Page {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Return activated structure
|
// Return activated `Self`
|
||||||
Arc::new(Self {
|
Arc::new(Self {
|
||||||
id,
|
id,
|
||||||
// Actions
|
// Actions
|
||||||
@ -164,11 +164,7 @@ impl Page {
|
|||||||
let id = self.id.to_variant();
|
let id = self.id.to_variant();
|
||||||
|
|
||||||
// Update
|
// Update
|
||||||
self.meta.replace(Meta {
|
self.meta.set_status(Status::Reload).set_title(&"Loading..");
|
||||||
status: Some(Status::Reload),
|
|
||||||
title: Some(gformat!("Loading..")),
|
|
||||||
});
|
|
||||||
|
|
||||||
self.action_update.activate(Some(&id));
|
self.action_update.activate(Some(&id));
|
||||||
|
|
||||||
// Route by request
|
// Route by request
|
||||||
@ -181,20 +177,18 @@ impl Page {
|
|||||||
scheme => {
|
scheme => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
let description = gformat!("Protocol `{scheme}` not supported");
|
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
self.content
|
self.content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(description.as_str()));
|
.set_description(Some(
|
||||||
|
gformat!("Protocol `{scheme}` not supported").as_str(),
|
||||||
|
));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
self.meta.replace(Meta {
|
self.meta.set_status(status).set_title(title);
|
||||||
status: Some(status),
|
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
self.action_update.activate(Some(&id));
|
self.action_update.activate(Some(&id));
|
||||||
@ -314,19 +308,19 @@ impl Page {
|
|||||||
// Getters
|
// Getters
|
||||||
pub fn progress_fraction(&self) -> Option<f64> {
|
pub fn progress_fraction(&self) -> Option<f64> {
|
||||||
// Interpret status to progress fraction
|
// Interpret status to progress fraction
|
||||||
match self.meta.borrow().status {
|
match self.meta.status() {
|
||||||
Some(Status::Reload) => Some(0.0),
|
Status::Reload => Some(0.0),
|
||||||
Some(Status::Resolving) => Some(0.1),
|
Status::Resolving => Some(0.1),
|
||||||
Some(Status::Resolved) => Some(0.2),
|
Status::Resolved => Some(0.2),
|
||||||
Some(Status::Connecting) => Some(0.3),
|
Status::Connecting => Some(0.3),
|
||||||
Some(Status::Connected) => Some(0.4),
|
Status::Connected => Some(0.4),
|
||||||
Some(Status::ProxyNegotiating) => Some(0.5),
|
Status::ProxyNegotiating => Some(0.5),
|
||||||
Some(Status::ProxyNegotiated) => Some(0.6),
|
Status::ProxyNegotiated => Some(0.6),
|
||||||
Some(Status::TlsHandshaking) => Some(0.7),
|
Status::TlsHandshaking => Some(0.7),
|
||||||
Some(Status::TlsHandshaked) => Some(0.8),
|
Status::TlsHandshaked => Some(0.8),
|
||||||
Some(Status::Complete) => Some(0.9),
|
Status::Complete => Some(0.9),
|
||||||
Some(Status::Failure | Status::Redirect | Status::Success | Status::Input) => Some(1.0),
|
Status::Failure | Status::Redirect | Status::Success | Status::Input => Some(1.0),
|
||||||
_ => None,
|
Status::New => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,8 +331,8 @@ impl Page {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn meta_title(&self) -> Option<GString> {
|
pub fn meta_title(&self) -> GString {
|
||||||
self.meta.borrow().title.clone()
|
self.meta.title()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn gobject(&self) -> &Box {
|
pub fn gobject(&self) -> &Box {
|
||||||
@ -366,6 +360,7 @@ impl Page {
|
|||||||
|
|
||||||
// Init shared objects (async)
|
// Init shared objects (async)
|
||||||
let action_page_open = self.action_page_open.clone();
|
let action_page_open = self.action_page_open.clone();
|
||||||
|
let action_page_reload = self.action_page_reload.clone();
|
||||||
let action_update = self.action_update.clone();
|
let action_update = self.action_update.clone();
|
||||||
let content = self.content.clone();
|
let content = self.content.clone();
|
||||||
let id = self.id.to_variant();
|
let id = self.id.to_variant();
|
||||||
@ -374,6 +369,31 @@ impl Page {
|
|||||||
let navigation = self.navigation.clone();
|
let navigation = self.navigation.clone();
|
||||||
let url = uri.clone().to_str();
|
let url = uri.clone().to_str();
|
||||||
|
|
||||||
|
// Check for page redirect pending
|
||||||
|
if meta.is_redirect() {
|
||||||
|
// Check for protocol limits
|
||||||
|
if meta.redirect_count().unwrap() > 5 {
|
||||||
|
// Update meta
|
||||||
|
meta.set_status(Status::Failure).set_title(&"Oops");
|
||||||
|
// Show placeholder with confirmation request to continue
|
||||||
|
content.to_text_gemini(
|
||||||
|
&uri,
|
||||||
|
&gformat!(
|
||||||
|
// @TODO status page?
|
||||||
|
"# Redirect issue\n\nRedirection limit reached\n\nContinue:\n\n=> {}",
|
||||||
|
meta.redirect_target().unwrap().to_string()
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
return; // @TODO
|
||||||
|
} else {
|
||||||
|
action_page_open.activate(Some(
|
||||||
|
&meta.redirect_target().unwrap().to_string().to_variant(),
|
||||||
|
));
|
||||||
|
// @TODO is_follow
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Add history record
|
// Add history record
|
||||||
match navigation.history_current() {
|
match navigation.history_current() {
|
||||||
Some(current) => {
|
Some(current) => {
|
||||||
@ -397,7 +417,7 @@ impl Page {
|
|||||||
let id = id.clone();
|
let id = id.clone();
|
||||||
let meta = meta.clone();
|
let meta = meta.clone();
|
||||||
move |_, event, _, _| {
|
move |_, event, _, _| {
|
||||||
meta.borrow_mut().status = Some(match event {
|
meta.set_status(match event {
|
||||||
SocketClientEvent::Resolving => Status::Resolving,
|
SocketClientEvent::Resolving => Status::Resolving,
|
||||||
SocketClientEvent::Resolved => Status::Resolved,
|
SocketClientEvent::Resolved => Status::Resolved,
|
||||||
SocketClientEvent::Connecting => Status::Connecting,
|
SocketClientEvent::Connecting => Status::Connecting,
|
||||||
@ -442,10 +462,10 @@ impl Page {
|
|||||||
gemini::client::response::meta::Status::SensitiveInput => {
|
gemini::client::response::meta::Status::SensitiveInput => {
|
||||||
// Format response
|
// Format response
|
||||||
let status = Status::Input;
|
let status = Status::Input;
|
||||||
let title = gformat!("Input expected");
|
let title = &"Input expected";
|
||||||
let description = match response.data() {
|
let description = match response.data() {
|
||||||
Some(data) => data.value(),
|
Some(data) => data.value().as_str(),
|
||||||
None => &title,
|
None => title,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Toggle input form variant
|
// Toggle input form variant
|
||||||
@ -467,10 +487,8 @@ impl Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update page
|
// Update page
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -493,12 +511,14 @@ impl Page {
|
|||||||
&buffer.data()
|
&buffer.data()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let title = match text_gemini.meta_title() {
|
||||||
|
Some(title) => title,
|
||||||
|
None => &uri_to_title(&uri)
|
||||||
|
};
|
||||||
|
|
||||||
// Update page meta
|
// Update page meta
|
||||||
meta.borrow_mut().status = Some(Status::Success);
|
meta.set_status(Status::Success)
|
||||||
meta.borrow_mut().title = Some(match text_gemini.meta_title() {
|
.set_title(title);
|
||||||
Some(title) => title.clone(),
|
|
||||||
None => uri_to_title(&uri)
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window components
|
// Update window components
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -506,7 +526,7 @@ impl Page {
|
|||||||
Err((reason, message)) => {
|
Err((reason, message)) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
let description = match reason {
|
let description = match reason {
|
||||||
gemini::client::response::data::text::Error::InputStream => match message {
|
gemini::client::response::data::text::Error::InputStream => match message {
|
||||||
Some(error) => gformat!("{error}"),
|
Some(error) => gformat!("{error}"),
|
||||||
@ -519,14 +539,12 @@ impl Page {
|
|||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(description.as_str()));
|
.set_description(Some(description.as_str()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -569,8 +587,8 @@ impl Page {
|
|||||||
match result {
|
match result {
|
||||||
Ok(buffer) => {
|
Ok(buffer) => {
|
||||||
// Update page meta
|
// Update page meta
|
||||||
meta.borrow_mut().status = Some(Status::Success);
|
meta.set_status(Status::Success)
|
||||||
meta.borrow_mut().title = Some(uri_to_title(&uri));
|
.set_title(uri_to_title(&uri).as_str());
|
||||||
|
|
||||||
// Update page content
|
// Update page content
|
||||||
content.to_image(&buffer);
|
content.to_image(&buffer);
|
||||||
@ -581,19 +599,17 @@ impl Page {
|
|||||||
Err(reason) => {
|
Err(reason) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(reason.message()));
|
.set_description(Some(reason.message()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -602,7 +618,7 @@ impl Page {
|
|||||||
Err((error, reason)) => {
|
Err((error, reason)) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
let description = match reason {
|
let description = match reason {
|
||||||
Some(message) => gformat!("{message}"),
|
Some(message) => gformat!("{message}"),
|
||||||
None => match error {
|
None => match error {
|
||||||
@ -614,14 +630,12 @@ impl Page {
|
|||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(description.as_str()));
|
.set_description(Some(description.as_str()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -643,20 +657,18 @@ impl Page {
|
|||||||
_ => {
|
_ => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
let description = gformat!("Content type not supported");
|
let description = gformat!("Content type not supported");
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(description.as_str()));
|
.set_description(Some(description.as_str()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -666,29 +678,87 @@ impl Page {
|
|||||||
// https://geminiprotocol.net/docs/protocol-specification.gmi#redirection
|
// https://geminiprotocol.net/docs/protocol-specification.gmi#redirection
|
||||||
gemini::client::response::meta::Status::Redirect |
|
gemini::client::response::meta::Status::Redirect |
|
||||||
gemini::client::response::meta::Status::PermanentRedirect => {
|
gemini::client::response::meta::Status::PermanentRedirect => {
|
||||||
|
// Extract redirection URL from response data
|
||||||
// @TODO ClientStatus::TemporaryRedirect
|
|
||||||
|
|
||||||
// Update meta
|
|
||||||
meta.borrow_mut().status = Some(Status::Redirect);
|
|
||||||
meta.borrow_mut().title = Some(gformat!("Redirect"));
|
|
||||||
|
|
||||||
// Build gemtext message for manual redirection @TODO use template?
|
|
||||||
match response.data() {
|
match response.data() {
|
||||||
Some(url) => {
|
Some(unresolved_url) => {
|
||||||
// @TODO URI can by relative, resolve to base
|
// New URL from server MAY to be relative (according to the protocol),
|
||||||
content.to_text_gemini(
|
// resolve to absolute URI using current request value as the base for parser
|
||||||
&uri,
|
// https://docs.gtk.org/glib/type_func.Uri.resolve_relative.html
|
||||||
&gformat!(
|
match Uri::resolve_relative(
|
||||||
"# Redirect\n\nAuto-follow not implemented, click on link below to continue\n\n=> {}",
|
Some(&uri.to_string()),
|
||||||
url.value()
|
&unresolved_url.value(),
|
||||||
)
|
UriFlags::NONE,
|
||||||
);
|
) {
|
||||||
|
Ok(resolved_url) => {
|
||||||
|
// Build valid URI (this conversion wanted to process query and fragment later)
|
||||||
|
match Uri::parse(resolved_url.as_str(), UriFlags::NONE) {
|
||||||
|
Ok(resolved_uri) => {
|
||||||
|
// Client MUST prevent external redirects
|
||||||
|
if is_external_uri(&resolved_uri, &uri) {
|
||||||
|
// Update meta
|
||||||
|
meta.set_status(Status::Failure)
|
||||||
|
.set_title(&"Oops");
|
||||||
|
|
||||||
|
// Show placeholder with confirmation request to continue
|
||||||
|
content.to_text_gemini(
|
||||||
|
&uri,
|
||||||
|
&gformat!( // @TODO status page?
|
||||||
|
"# Redirect issue\n\nExternal redirects not allowed by protocol\n\nContinue:\n\n=> {}",
|
||||||
|
resolved_uri.to_string()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// Update meta
|
||||||
|
meta.set_redirect(
|
||||||
|
match meta.redirect_count() {
|
||||||
|
Some(count) => count + 1,
|
||||||
|
None => 0
|
||||||
|
},
|
||||||
|
match response.status() {
|
||||||
|
gemini::client::response::meta::Status::PermanentRedirect => true,
|
||||||
|
_ => false
|
||||||
|
},
|
||||||
|
Uri::parse(
|
||||||
|
resolved_uri.to_string_partial(
|
||||||
|
UriHideFlags::FRAGMENT | UriHideFlags::QUERY // @TODO review fragment specification
|
||||||
|
).as_str(),
|
||||||
|
UriFlags::NONE
|
||||||
|
).unwrap()
|
||||||
|
)
|
||||||
|
.set_status(Status::Redirect)
|
||||||
|
.set_title(&"Redirect"); // @TODO is really wanted here?
|
||||||
|
|
||||||
|
// Reload page to apply redirect
|
||||||
|
action_page_reload.activate(None);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(reason) => {
|
||||||
|
meta.set_status(Status::Failure);
|
||||||
|
content
|
||||||
|
.to_status_failure()
|
||||||
|
.set_description(Some(reason.message()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(reason) => {
|
||||||
|
meta.set_status(Status::Failure);
|
||||||
|
content
|
||||||
|
.to_status_failure()
|
||||||
|
.set_description(Some(reason.message()));
|
||||||
|
},
|
||||||
|
}
|
||||||
},
|
},
|
||||||
None => {
|
None => {
|
||||||
|
let status = Status::Failure;
|
||||||
|
let title = &"Oops";
|
||||||
|
|
||||||
|
meta.set_status(status)
|
||||||
|
.set_title(title);
|
||||||
|
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_description(Some("Could not parse redirect meta"));
|
.set_title(title)
|
||||||
|
.set_description(Some("Redirection target not defined"));
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -697,19 +767,17 @@ impl Page {
|
|||||||
_ => {
|
_ => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some("Status code yet not supported"));
|
.set_description(Some("Status code yet not supported"));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -719,7 +787,7 @@ impl Page {
|
|||||||
Err((reason, message)) => {
|
Err((reason, message)) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
let description = match reason {
|
let description = match reason {
|
||||||
// Common
|
// Common
|
||||||
gemini::client::response::meta::Error::InputStream => match message {
|
gemini::client::response::meta::Error::InputStream => match message {
|
||||||
@ -770,15 +838,12 @@ impl Page {
|
|||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(description.as_str()));
|
.set_description(Some(description.as_str()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
//description: Some(description),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -789,19 +854,17 @@ impl Page {
|
|||||||
Err(reason) => {
|
Err(reason) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(reason.message()));
|
.set_description(Some(reason.message()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -812,19 +875,17 @@ impl Page {
|
|||||||
Err(reason) => {
|
Err(reason) => {
|
||||||
// Define common data
|
// Define common data
|
||||||
let status = Status::Failure;
|
let status = Status::Failure;
|
||||||
let title = gformat!("Oops");
|
let title = &"Oops";
|
||||||
|
|
||||||
// Update widget
|
// Update widget
|
||||||
content
|
content
|
||||||
.to_status_failure()
|
.to_status_failure()
|
||||||
.set_title(title.as_str())
|
.set_title(title)
|
||||||
.set_description(Some(reason.message()));
|
.set_description(Some(reason.message()));
|
||||||
|
|
||||||
// Update meta
|
// Update meta
|
||||||
meta.replace(Meta {
|
meta.set_status(status)
|
||||||
status: Some(status),
|
.set_title(title);
|
||||||
title: Some(title),
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update window
|
// Update window
|
||||||
action_update.activate(Some(&id));
|
action_update.activate(Some(&id));
|
||||||
@ -850,3 +911,16 @@ fn uri_to_title(uri: &Uri) -> GString {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Compare `subject` with `base`
|
||||||
|
///
|
||||||
|
/// Return `false` on scheme, port or host mismatch
|
||||||
|
fn is_external_uri(subject: &Uri, base: &Uri) -> bool {
|
||||||
|
if subject.scheme() != base.scheme() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if subject.port() != base.port() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
subject.host() != base.host()
|
||||||
|
}
|
||||||
|
@ -1,11 +1,17 @@
|
|||||||
use gtk::glib::GString;
|
mod redirect;
|
||||||
|
use redirect::Redirect;
|
||||||
|
|
||||||
|
use gtk::glib::{GString, Uri};
|
||||||
|
use std::{cell::RefCell, sync::Arc};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
pub enum Status {
|
pub enum Status {
|
||||||
Complete,
|
Complete,
|
||||||
|
Connected,
|
||||||
|
Connecting,
|
||||||
Failure,
|
Failure,
|
||||||
Input,
|
Input,
|
||||||
Connecting,
|
New,
|
||||||
Connected,
|
|
||||||
ProxyNegotiated,
|
ProxyNegotiated,
|
||||||
ProxyNegotiating,
|
ProxyNegotiating,
|
||||||
Redirect,
|
Redirect,
|
||||||
@ -18,17 +24,88 @@ pub enum Status {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct Meta {
|
pub struct Meta {
|
||||||
pub title: Option<GString>,
|
status: RefCell<Status>,
|
||||||
//pub description: Option<GString>,
|
title: RefCell<GString>,
|
||||||
pub status: Option<Status>,
|
redirect: RefCell<Option<Redirect>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Meta {
|
impl Meta {
|
||||||
pub fn new() -> Self {
|
// Constructors
|
||||||
Self {
|
|
||||||
title: None,
|
pub fn new_arc(status: Status, title: GString) -> Arc<Self> {
|
||||||
//description: None,
|
Arc::new(Self {
|
||||||
status: None,
|
status: RefCell::new(status),
|
||||||
|
title: RefCell::new(title),
|
||||||
|
redirect: RefCell::new(None),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setters
|
||||||
|
|
||||||
|
pub fn set_status(&self, status: Status) -> &Self {
|
||||||
|
match status {
|
||||||
|
Status::Redirect => {
|
||||||
|
if self.redirect.borrow().is_none() {
|
||||||
|
panic!("Set `redirect` before use this status")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
self.unset_redirect();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
self.status.replace(status);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_title(&self, title: &str) -> &Self {
|
||||||
|
self.title.replace(GString::from(title));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_redirect(&self, count: i8, is_follow: bool, target: Uri) -> &Self {
|
||||||
|
self.redirect
|
||||||
|
.replace(Some(Redirect::new(count, is_follow, target)));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unset_redirect(&self) -> &Self {
|
||||||
|
self.redirect.replace(None);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
|
||||||
|
pub fn status(&self) -> Status {
|
||||||
|
self.status.borrow().clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn title(&self) -> GString {
|
||||||
|
self.title.borrow().clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_redirect(&self) -> bool {
|
||||||
|
self.redirect.borrow().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn redirect_count(&self) -> Option<i8> {
|
||||||
|
match *self.redirect.borrow() {
|
||||||
|
Some(ref redirect) => Some(redirect.count().clone()),
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn redirect_target(&self) -> Option<Uri> {
|
||||||
|
match *self.redirect.borrow() {
|
||||||
|
Some(ref redirect) => Some(redirect.target().clone()),
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn redirect_is_follow(&self) -> Option<bool> {
|
||||||
|
match *self.redirect.borrow() {
|
||||||
|
Some(ref redirect) => Some(redirect.is_follow().clone()),
|
||||||
|
None => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
40
src/app/browser/window/tab/item/page/meta/redirect.rs
Normal file
40
src/app/browser/window/tab/item/page/meta/redirect.rs
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
use gtk::glib::Uri;
|
||||||
|
|
||||||
|
/// # Redirection data holder
|
||||||
|
///
|
||||||
|
/// This component does nothing,
|
||||||
|
/// but useful as the container for temporary redirection data
|
||||||
|
/// operated by external controller
|
||||||
|
///
|
||||||
|
/// ## Members
|
||||||
|
///
|
||||||
|
/// * `count` - to limit redirect attempts
|
||||||
|
/// * `is_follow` - indicates how to process this redirect exactly
|
||||||
|
/// * `target` - destination address
|
||||||
|
pub struct Redirect {
|
||||||
|
count: i8,
|
||||||
|
is_follow: bool,
|
||||||
|
target: Uri,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Redirect {
|
||||||
|
pub fn new(count: i8, is_follow: bool, target: Uri) -> Self {
|
||||||
|
Self {
|
||||||
|
count,
|
||||||
|
is_follow,
|
||||||
|
target,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn count(&self) -> &i8 {
|
||||||
|
&self.count
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_follow(&self) -> &bool {
|
||||||
|
&self.is_follow
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn target(&self) -> &Uri {
|
||||||
|
&self.target
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user