mirror of https://github.com/YGGverse/Yoda.git
yggverse
2 months ago
5 changed files with 0 additions and 384 deletions
@ -1,197 +0,0 @@
@@ -1,197 +0,0 @@
|
||||
mod database; |
||||
mod pin; |
||||
mod title; |
||||
mod widget; |
||||
|
||||
use database::Database; |
||||
use pin::Pin; |
||||
use sqlite::Transaction; |
||||
use title::Title; |
||||
use widget::Widget; |
||||
|
||||
use gtk::{gdk, glib::GString, prelude::WidgetExt, Box, GestureClick}; |
||||
use std::sync::{Arc, Mutex}; |
||||
|
||||
pub struct Label { |
||||
// Components
|
||||
pin: Arc<Pin>, |
||||
title: Arc<Title>, |
||||
// GTK
|
||||
widget: Arc<Widget>, |
||||
// Extras
|
||||
is_pinned: Mutex<bool>, |
||||
} |
||||
|
||||
impl Label { |
||||
// Construct
|
||||
pub fn new_arc(name: GString, is_pinned: bool) -> Arc<Self> { |
||||
// Init components
|
||||
let pin = Pin::new_arc(is_pinned); |
||||
let title = Title::new_arc(); |
||||
let widget = Widget::new_arc(name, pin.gobject(), title.gobject()); |
||||
|
||||
// Init label struct
|
||||
let label = Arc::new(Self { |
||||
pin: pin.clone(), |
||||
title: title.clone(), |
||||
widget: widget.clone(), |
||||
is_pinned: Mutex::new(is_pinned), |
||||
}); |
||||
|
||||
// Init events:
|
||||
|
||||
// Await for widget realize to continue this feature
|
||||
widget.gobject().connect_realize({ |
||||
let widget = widget.clone(); |
||||
let label = label.clone(); |
||||
move |_| { |
||||
// Init primary button listener
|
||||
let primary_button_controller = GestureClick::new(); |
||||
|
||||
// Listen for double primary button click
|
||||
primary_button_controller.connect_pressed({ |
||||
let label = label.clone(); |
||||
move |_, count, _, _| { |
||||
if count == 2 { |
||||
label.pin(!label.is_pinned()); |
||||
} |
||||
} |
||||
}); |
||||
|
||||
// Init middle button listener
|
||||
let middle_button_controller = |
||||
GestureClick::builder().button(gdk::BUTTON_MIDDLE).build(); |
||||
|
||||
// Listen for single middle button click
|
||||
middle_button_controller.connect_pressed({ |
||||
let label = label.clone(); |
||||
move |_, _, _, _| { |
||||
if label.is_pinned() { |
||||
label.pin(false); // unpin
|
||||
} else { |
||||
// @TODO close
|
||||
} |
||||
} |
||||
}); |
||||
|
||||
// Label's parent contain native GTK paddings, that makes click event ignored
|
||||
// try assign the controller to parent as the solution
|
||||
match widget.gobject().parent() { |
||||
// @TODO check for GtkGizmo type?
|
||||
Some(parent) => { |
||||
parent.add_controller(primary_button_controller); |
||||
parent.add_controller(middle_button_controller); |
||||
} |
||||
// Assign controller to the current widget, drop notice
|
||||
None => { |
||||
widget.gobject().add_controller(primary_button_controller); |
||||
widget.gobject().add_controller(middle_button_controller); |
||||
println!("Could not assign action to destination, please report"); |
||||
// @TODO
|
||||
} |
||||
} |
||||
} |
||||
}); |
||||
|
||||
// Result
|
||||
label |
||||
} |
||||
|
||||
// Actions
|
||||
pub fn clean( |
||||
&self, |
||||
transaction: &Transaction, |
||||
app_browser_window_tab_id: &i64, |
||||
) -> Result<(), String> { |
||||
match Database::records(transaction, app_browser_window_tab_id) { |
||||
Ok(records) => { |
||||
for record in records { |
||||
match Database::delete(transaction, &record.id) { |
||||
Ok(_) => { |
||||
// Delegate clean action to childs
|
||||
// nothing yet..
|
||||
} |
||||
Err(e) => return Err(e.to_string()), |
||||
} |
||||
} |
||||
} |
||||
Err(e) => return Err(e.to_string()), |
||||
} |
||||
|
||||
Ok(()) |
||||
} |
||||
|
||||
pub fn restore( |
||||
&self, |
||||
transaction: &Transaction, |
||||
app_browser_window_tab_id: &i64, |
||||
) -> Result<(), String> { |
||||
match Database::records(transaction, app_browser_window_tab_id) { |
||||
Ok(records) => { |
||||
for record in records { |
||||
self.pin(record.is_pinned); |
||||
|
||||
// Delegate restore action to childs
|
||||
// nothing yet..
|
||||
} |
||||
} |
||||
Err(e) => return Err(e.to_string()), |
||||
} |
||||
|
||||
Ok(()) |
||||
} |
||||
|
||||
pub fn save( |
||||
&self, |
||||
transaction: &Transaction, |
||||
app_browser_window_tab_id: &i64, |
||||
) -> Result<(), String> { |
||||
match Database::add(transaction, app_browser_window_tab_id, &self.is_pinned()) { |
||||
Ok(_) => { |
||||
// Delegate save action to childs
|
||||
// nothing yet..
|
||||
} |
||||
Err(e) => return Err(e.to_string()), |
||||
} |
||||
|
||||
Ok(()) |
||||
} |
||||
|
||||
pub fn update(&self, title: Option<&GString>) { |
||||
self.title.update(title); |
||||
self.widget.update(title); |
||||
} |
||||
|
||||
// Setters
|
||||
pub fn pin(&self, is_pinned: bool) { |
||||
// Update Self
|
||||
*self.is_pinned.lock().unwrap() = is_pinned; |
||||
|
||||
// Update child components
|
||||
self.pin.pin(is_pinned); |
||||
self.title.pin(is_pinned); |
||||
} |
||||
|
||||
// Getters
|
||||
pub fn is_pinned(&self) -> bool { |
||||
self.is_pinned.lock().unwrap().clone() |
||||
} |
||||
|
||||
pub fn gobject(&self) -> &Box { |
||||
&self.widget.gobject() |
||||
} |
||||
|
||||
// Tools
|
||||
pub fn migrate(tx: &Transaction) -> Result<(), String> { |
||||
// Migrate self components
|
||||
if let Err(e) = Database::init(&tx) { |
||||
return Err(e.to_string()); |
||||
} |
||||
|
||||
// Delegate migration to childs
|
||||
// nothing yet..
|
||||
|
||||
// Success
|
||||
Ok(()) |
||||
} |
||||
} |
@ -1,80 +0,0 @@
@@ -1,80 +0,0 @@
|
||||
use sqlite::{Error, Transaction}; |
||||
|
||||
pub struct Table { |
||||
pub id: i64, |
||||
// pub app_browser_window_tab_item_id: i64, not in use
|
||||
pub is_pinned: bool, |
||||
} |
||||
|
||||
pub struct Database { |
||||
// nothing yet..
|
||||
} |
||||
|
||||
impl Database { |
||||
pub fn init(tx: &Transaction) -> Result<usize, Error> { |
||||
tx.execute( |
||||
"CREATE TABLE IF NOT EXISTS `app_browser_window_tab_item_label` |
||||
( |
||||
`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, |
||||
`app_browser_window_tab_item_id` INTEGER NOT NULL, |
||||
`is_pinned` INTEGER NOT NULL |
||||
)", |
||||
[], |
||||
) |
||||
} |
||||
|
||||
pub fn add( |
||||
tx: &Transaction, |
||||
app_browser_window_tab_item_id: &i64, |
||||
is_pinned: &bool, |
||||
) -> Result<usize, Error> { |
||||
tx.execute( |
||||
"INSERT INTO `app_browser_window_tab_item_label` ( |
||||
`app_browser_window_tab_item_id`, |
||||
`is_pinned` |
||||
) VALUES (?,?)", |
||||
[app_browser_window_tab_item_id, &(*is_pinned as i64)], |
||||
) |
||||
} |
||||
|
||||
pub fn records( |
||||
tx: &Transaction, |
||||
app_browser_window_tab_item_id: &i64, |
||||
) -> Result<Vec<Table>, Error> { |
||||
let mut stmt = tx.prepare( |
||||
"SELECT `id`, |
||||
`app_browser_window_tab_item_id`, |
||||
`is_pinned` FROM `app_browser_window_tab_item_label` |
||||
WHERE `app_browser_window_tab_item_id` = ?", |
||||
)?; |
||||
|
||||
let result = stmt.query_map([app_browser_window_tab_item_id], |row| { |
||||
Ok(Table { |
||||
id: row.get(0)?, |
||||
// app_browser_window_tab_item_id: row.get(1)?, not in use
|
||||
is_pinned: row.get(2)?, |
||||
}) |
||||
})?; |
||||
|
||||
let mut records = Vec::new(); |
||||
|
||||
for record in result { |
||||
let table = record?; |
||||
records.push(table); |
||||
} |
||||
|
||||
Ok(records) |
||||
} |
||||
|
||||
pub fn delete(tx: &Transaction, id: &i64) -> Result<usize, Error> { |
||||
tx.execute( |
||||
"DELETE FROM `app_browser_window_tab_item_label` WHERE `id` = ?", |
||||
[id], |
||||
) |
||||
} |
||||
|
||||
/* not in use
|
||||
pub fn last_insert_id(tx: &Transaction) -> i64 { |
||||
tx.last_insert_rowid() |
||||
} */ |
||||
} |
@ -1,28 +0,0 @@
@@ -1,28 +0,0 @@
|
||||
use std::sync::Arc; |
||||
|
||||
use gtk::{prelude::WidgetExt, Image}; |
||||
|
||||
pub struct Pin { |
||||
gobject: Image, |
||||
} |
||||
|
||||
impl Pin { |
||||
// Construct
|
||||
pub fn new_arc(visible: bool) -> Arc<Pin> { |
||||
let gobject = Image::builder() |
||||
.icon_name("view-pin-symbolic") |
||||
.visible(visible) |
||||
.build(); |
||||
|
||||
Arc::new(Self { gobject }) |
||||
} |
||||
|
||||
pub fn pin(&self, is_pinned: bool) { |
||||
self.gobject().set_visible(is_pinned); |
||||
} |
||||
|
||||
// Getters
|
||||
pub fn gobject(&self) -> &Image { |
||||
&self.gobject |
||||
} |
||||
} |
@ -1,40 +0,0 @@
@@ -1,40 +0,0 @@
|
||||
use std::sync::Arc; |
||||
|
||||
use gtk::{glib::GString, pango::EllipsizeMode, prelude::WidgetExt, Label}; |
||||
|
||||
const DEFAULT_LABEL_TEXT: &str = "New page"; |
||||
|
||||
pub struct Title { |
||||
gobject: Label, |
||||
} |
||||
|
||||
impl Title { |
||||
// Construct
|
||||
pub fn new_arc() -> Arc<Self> { |
||||
Arc::new(Self { |
||||
gobject: Label::builder() |
||||
.label(DEFAULT_LABEL_TEXT) |
||||
.ellipsize(EllipsizeMode::End) |
||||
.width_chars(16) |
||||
.single_line_mode(true) |
||||
.build(), |
||||
}) |
||||
} |
||||
|
||||
// Actions
|
||||
pub fn update(&self, title: Option<&GString>) { |
||||
match title { |
||||
Some(title) => self.gobject.set_text(title), |
||||
None => self.gobject.set_text(DEFAULT_LABEL_TEXT), |
||||
} |
||||
} |
||||
|
||||
pub fn pin(&self, is_pinned: bool) { |
||||
self.gobject.set_visible(!is_pinned); |
||||
} |
||||
|
||||
// Getters
|
||||
pub fn gobject(&self) -> &Label { |
||||
&self.gobject |
||||
} |
||||
} |
@ -1,39 +0,0 @@
@@ -1,39 +0,0 @@
|
||||
use std::sync::Arc; |
||||
|
||||
use gtk::{ |
||||
glib::GString, prelude::BoxExt, prelude::WidgetExt, Align, Box, Image, Label, Orientation, |
||||
}; |
||||
|
||||
pub struct Widget { |
||||
gobject: Box, |
||||
} |
||||
|
||||
impl Widget { |
||||
// Construct
|
||||
pub fn new_arc(name: GString, pin: &Image, title: &Label) -> Arc<Self> { |
||||
let gobject = Box::builder() |
||||
.orientation(Orientation::Horizontal) |
||||
.halign(Align::Center) |
||||
.name(name) |
||||
.tooltip_text(title.text()) |
||||
.build(); |
||||
|
||||
gobject.append(pin); |
||||
gobject.append(title); |
||||
|
||||
Arc::new(Self { gobject }) |
||||
} |
||||
|
||||
// Action
|
||||
pub fn update(&self, title: Option<&GString>) { |
||||
match title { |
||||
Some(tooltip_text) => self.gobject.set_tooltip_text(Some(tooltip_text)), |
||||
None => self.gobject.set_tooltip_text(None), |
||||
} |
||||
} |
||||
|
||||
// Getters
|
||||
pub fn gobject(&self) -> &Box { |
||||
&self.gobject |
||||
} |
||||
} |
Loading…
Reference in new issue