store optional page title

This commit is contained in:
yggverse 2025-03-11 19:58:09 +02:00
parent 24ee6bb28f
commit ce2b02569d
4 changed files with 46 additions and 30 deletions

View file

@ -78,7 +78,7 @@ impl Page {
pub fn bookmark(&self) { pub fn bookmark(&self) {
self.profile self.profile
.bookmark .bookmark
.toggle(&self.navigation.request()) .toggle(&self.navigation.request(), Some(&self.title()))
.unwrap(); // @TODO .unwrap(); // @TODO
} }

View file

@ -27,11 +27,8 @@ impl Bookmark {
// Build initial index // Build initial index
{ {
let mut memory = memory.borrow_mut(); let mut memory = memory.borrow_mut();
for record in database.records(None)? { for item in database.records(None, None)? {
memory.add(Item { memory.add(item);
id: record.id,
request: record.request,
});
} }
} }
@ -43,17 +40,18 @@ impl Bookmark {
/// Toggle bookmark in `database` and `memory` index /// Toggle bookmark in `database` and `memory` index
/// * return `true` on bookmark create, `false` on delete /// * return `true` on bookmark create, `false` on delete
pub fn toggle(&self, request: &str) -> Result<bool> { pub fn toggle(&self, request: &str, title: Option<&str>) -> Result<bool> {
let mut memory = self.memory.borrow_mut(); let mut memory = self.memory.borrow_mut();
Ok(match memory.delete_by_request(request) { Ok(match memory.delete_by_request(request) {
Some(record) => { Some(item) => {
self.database.delete(record.id)?; self.database.delete(item.id)?;
false false
} }
None => { None => {
memory.add(Item { memory.add(Item {
id: self.database.add(DateTime::now_local()?, request)?, id: self.database.add(DateTime::now_local()?, request, title)?,
request: request.into(), request: request.into(),
title: title.map(|t| t.to_string()),
}); });
true true
} }

View file

@ -23,20 +23,20 @@ impl Database {
// Getters // Getters
/// Get bookmark records from database with optional filter by `request` /// Get bookmark records from database with optional filter by `request`
pub fn records(&self, request: Option<&str>) -> Result<Vec<Item>> { pub fn records(&self, request: Option<&str>, title: Option<&str>) -> Result<Vec<Item>> {
let readable = self.connection.read().unwrap(); // @TODO let readable = self.connection.read().unwrap(); // @TODO
let tx = readable.unchecked_transaction()?; let tx = readable.unchecked_transaction()?;
select(&tx, *self.profile_id, request) select(&tx, *self.profile_id, request, title)
} }
// Setters // Setters
/// Create new bookmark record in database /// Create new bookmark record in database
/// * return last insert ID on success /// * return last insert ID on success
pub fn add(&self, time: DateTime, request: &str) -> Result<i64> { pub fn add(&self, time: DateTime, request: &str, title: Option<&str>) -> Result<i64> {
let mut writable = self.connection.write().unwrap(); // @TODO let mut writable = self.connection.write().unwrap(); // @TODO
let tx = writable.transaction()?; let tx = writable.transaction()?;
let id = insert(&tx, *self.profile_id, time, request)?; let id = insert(&tx, *self.profile_id, time, request, title)?;
tx.commit()?; tx.commit()?;
Ok(id) Ok(id)
} }
@ -61,40 +61,57 @@ pub fn init(tx: &Transaction) -> Result<usize> {
`profile_id` INTEGER NOT NULL, `profile_id` INTEGER NOT NULL,
`time` INTEGER NOT NULL, `time` INTEGER NOT NULL,
`request` TEXT NOT NULL, `request` TEXT NOT NULL,
`title` TEXT NULL,
FOREIGN KEY (`profile_id`) REFERENCES `profile`(`id`) FOREIGN KEY (`profile_id`) REFERENCES `profile` (`id`)
)", )",
[], [],
)?) )?)
} }
pub fn insert(tx: &Transaction, profile_id: i64, time: DateTime, request: &str) -> Result<i64> { pub fn insert(
tx: &Transaction,
profile_id: i64,
time: DateTime,
request: &str,
title: Option<&str>,
) -> Result<i64> {
tx.execute( tx.execute(
"INSERT INTO `profile_bookmark` ( "INSERT INTO `profile_bookmark` (
`profile_id`, `profile_id`,
`time`, `time`,
`request` `request`,
) VALUES (?, ?, ?)", `title`
(profile_id, time.to_unix(), request), ) VALUES (?, ?, ?, ?)",
(profile_id, time.to_unix(), request, title),
)?; )?;
Ok(tx.last_insert_rowid()) Ok(tx.last_insert_rowid())
} }
pub fn select(tx: &Transaction, profile_id: i64, request: Option<&str>) -> Result<Vec<Item>> { pub fn select(
tx: &Transaction,
profile_id: i64,
request: Option<&str>,
title: Option<&str>,
) -> Result<Vec<Item>> {
let mut stmt = tx.prepare( let mut stmt = tx.prepare(
"SELECT `id`, `profile_id`, `time`, `request` "SELECT `id`, `profile_id`, `time`, `request`, `title`
FROM `profile_bookmark` FROM `profile_bookmark`
WHERE `profile_id` = ? AND `request` LIKE ?", WHERE `profile_id` = ? AND (`request` LIKE ? OR `title` LIKE ?)",
)?; )?;
let result = stmt.query_map((profile_id, request.unwrap_or("%")), |row| { let result = stmt.query_map(
Ok(Item { (profile_id, request.unwrap_or("%"), title.unwrap_or("%")),
id: row.get(0)?, |row| {
//profile_id: row.get(1)?, Ok(Item {
//time: DateTime::from_unix_local(row.get(2)?).unwrap(), id: row.get(0)?,
request: row.get(3)?, //profile_id: row.get(1)?,
}) //time: DateTime::from_unix_local(row.get(2)?).unwrap(),
})?; request: row.get(3)?,
title: row.get(4)?,
})
},
)?;
let mut records = Vec::new(); let mut records = Vec::new();

View file

@ -2,4 +2,5 @@
pub struct Item { pub struct Item {
pub id: i64, pub id: i64,
pub request: String, pub request: String,
pub title: Option<String>,
} }