Compare commits

..

18 commits
v0.1.0 ... main

Author SHA1 Message Date
Breadway
7ece4fd762 CI: use /tmp for ecosystem clone, avoid permissions conflict
All checks were successful
Mirror to GitHub / mirror (push) Successful in 2s
2026-06-19 08:38:38 +08:00
Breadway
a33e979e5e Bump bread-theme to v0.2.8 (live-reload fix)
Some checks failed
Mirror to GitHub / mirror (push) Successful in 2s
Build and publish package / package (push) Failing after 2m39s
2026-06-17 12:55:52 +08:00
Breadway
83b9fd908e Fix illegible text on light pywal palettes + hot-reload
Use bread-theme 0.2.7's luminance-picked ink (@on-*): a selected/hovered row set
its background to `surface` but kept the pywal foreground, so the highlighted
item went unreadable when `surface` came out light. Colour is now set per surface
(panel, search box, row states) and labels inherit it.

Switch to bread_theme::gtk::apply_app_css so breadbox picks up `bread-theme
reload` (and rebuilds from the fresh palette on each launch).
2026-06-17 12:41:30 +08:00
Breadway
8c64ec1bf2 Release 0.2.4: shared bread-theme stylesheet
Some checks failed
Mirror to GitHub / mirror (push) Successful in 2s
Build and publish package / package (push) Failing after 3m13s
2026-06-16 18:32:48 +08:00
Breadway
732c3126ae Release 0.2.2: shared bread-theme stylesheet
Pin bread-theme v0.2.6 and load the shared ecosystem stylesheet.
2026-06-16 18:32:12 +08:00
Breadway
d464689a18 theme: load the shared bread-theme stylesheet
Apply bread_theme::gtk::apply_shared() before breadbox's own CSS so fonts,
palette, and generic widgets come from the shared ecosystem stylesheet. Keep
only launcher-specific rules (launcher-bg, searchentry, rows). Bump bread-theme
dep to v0.2.6.
2026-06-16 16:57:12 +08:00
Breadway
a6007e9a6a Disable debug package so the main package publishes correctly
Some checks failed
Mirror to GitHub / mirror (push) Successful in 2s
Build and publish package / package (push) Failing after 3m11s
makepkg's debug split produced a -debug pkg; the upload's head -1 could
grab it instead of the main package. !debug yields a single package.
2026-06-13 23:00:50 +08:00
Breadway
21392645cd Use REGISTRY_TOKEN (scoped write:package) for registry publish
Some checks failed
Mirror to GitHub / mirror (push) Successful in 2s
Build and publish package / package (push) Failing after 2m57s
2026-06-13 22:55:41 +08:00
Breadway
b80e06b253 Disable LTO in PKGBUILD (vendored ring/mlua static libs vs makepkg -flto) 2026-06-13 17:06:55 +08:00
Breadway
8bc185f40c Clone from public URL, not GITHUB_SERVER_URL (resolves to localhost in runner)
The Forgejo runner injects GITHUB_SERVER_URL as http://localhost:3002, which
is unreachable from inside the job container. Use the public URL instead.
2026-06-13 16:14:14 +08:00
Breadway
cbb1cf03d6 Rename mirror secret to MIRROR_TOKEN (GITHUB_ prefix is reserved)
Forgejo/gitea rejects user secret names starting with GITHUB_.
2026-06-13 16:10:50 +08:00
Breadway
43df888aa1 Fix Forgejo workflows for the actual server capabilities
- package.yml: correct Arch registry upload (octet-stream + binary body),
  drop --privileged, manual shell clone (archlinux image has no Node),
  built-in Actions token, --nocheck
- mirror.yml: clone --mirror + explicit refs push with --prune
2026-06-13 16:02:16 +08:00
Breadway
f93596adf2 Add packaging/arch PKGBUILD and Forgejo Actions workflows
- packaging/arch/PKGBUILD: builds and publishes breadbox to [breadway] repo
- .forgejo/workflows/mirror.yml: mirrors every push/tag to GitHub
- .forgejo/workflows/package.yml: builds on tag, publishes to Forgejo registry

Requires FORGEJO_TOKEN and GITHUB_MIRROR_TOKEN secrets in Forgejo.
2026-06-13 12:12:42 +08:00
Breadway
d2ef12551d chore: update Cargo.lock for v0.2.1
Some checks failed
release / build (push) Failing after 8s
2026-06-11 14:28:01 +08:00
Breadway
311f0d261f chore: bump version to 0.2.1 2026-06-11 14:21:31 +08:00
Breadway
c01cb67aa5 fix: add optional_system_deps (hyprland)
hyprland is used via IPC but not a linked dep — optional now.
2026-06-11 13:37:49 +08:00
Breadway
423d00383b feat: rank search results by match quality and launch frequency
Some checks failed
release / build (push) Failing after 4s
Track per-app launch counts in ~/.cache/breadbox/history.json. When a
query is active, sort visible results by fuzzy match quality (exact >
prefix > contains > subsequence) then by launch count descending, so
the most relevant and most-used app rises to the top. The base list
(no query) also surfaces most-launched apps above unvisited ones.
2026-06-07 14:35:06 +08:00
Breadway
fde5fe7c64 fix: use relative symlink for latest to work inside Docker containers 2026-06-07 09:02:38 +08:00
11 changed files with 238 additions and 53 deletions

View file

@ -0,0 +1,21 @@
name: Mirror to GitHub
on:
push:
branches: ['**']
tags: ['**']
jobs:
mirror:
runs-on: [self-hosted, hestia]
steps:
- name: Mirror to GitHub
run: |
set -euo pipefail
git clone --mirror "https://git.breadway.dev/${GITHUB_REPOSITORY}.git" repo.git
cd repo.git
# Mirror only branches and tags (not refs/pull/*, which GitHub rejects);
# --prune deletes GitHub refs that no longer exist on Forgejo.
git push --prune \
"https://x-access-token:${{ secrets.MIRROR_TOKEN }}@github.com/Breadway/breadbox.git" \
'+refs/heads/*:refs/heads/*' '+refs/tags/*:refs/tags/*'

View file

@ -0,0 +1,40 @@
name: Build and publish package
on:
push:
tags: ['v*']
jobs:
package:
runs-on: [self-hosted, hestia]
container:
image: archlinux:latest
steps:
# Note: no actions/checkout — the archlinux image has no Node, which JS
# actions require. Everything runs as shell steps and clones manually.
- name: Build and publish
env:
PUBLISH_TOKEN: ${{ secrets.REGISTRY_TOKEN }}
run: |
set -euo pipefail
VERSION="${GITHUB_REF_NAME#v}"
pacman -Syu --noconfirm base-devel git rust cargo gtk4 gtk4-layer-shell librsvg
useradd -m builder
git config --global --add safe.directory '*'
git clone --branch "${GITHUB_REF_NAME}" --depth 1 \
"https://git.breadway.dev/${GITHUB_REPOSITORY}.git" /home/builder/src
cd /home/builder/src
git archive --format=tar.gz --prefix="breadbox-${VERSION}/" HEAD \
> packaging/arch/breadbox-${VERSION}.tar.gz
SHA=$(sha256sum packaging/arch/breadbox-${VERSION}.tar.gz | awk '{print $1}')
sed -i "s/^pkgver=.*/pkgver=${VERSION}/" packaging/arch/PKGBUILD
sed -i "s/^sha256sums=.*/sha256sums=('${SHA}')/" packaging/arch/PKGBUILD
chown -R builder:builder /home/builder/src
# --nocheck: packaging builds the artifact; tests belong in a CI job.
su builder -c "cd /home/builder/src/packaging/arch && makepkg -f --noconfirm --nocheck"
PKG=$(find /home/builder/src/packaging/arch -name '*.pkg.tar.zst' | head -1)
curl -fsS -X PUT \
-H "Authorization: token ${PUBLISH_TOKEN}" \
-H "Content-Type: application/octet-stream" \
--data-binary "@${PKG}" \
"https://git.breadway.dev/api/packages/Breadway/arch/os"

View file

@ -9,7 +9,7 @@ permissions:
env: env:
DL_DIR: /srv/breadway-dl DL_DIR: /srv/breadway-dl
ECOSYSTEM_DIR: /home/breadway/Projects/bread-ecosystem ECOSYSTEM_DIR: /tmp/bread-ecosystem-ci
jobs: jobs:
build: build:
@ -37,16 +37,12 @@ jobs:
cp packaging/breadbox-sync.service "${PKG_DIR}/" cp packaging/breadbox-sync.service "${PKG_DIR}/"
cp config.example.toml "${PKG_DIR}/" cp config.example.toml "${PKG_DIR}/"
cp bakery.toml "${PKG_DIR}/bakery.toml" cp bakery.toml "${PKG_DIR}/bakery.toml"
ln -sfn "${PKG_DIR}" "${DL_DIR}/breadbox/latest" ln -sfn "${VERSION}" "${DL_DIR}/breadbox/latest"
- name: ensure bread-ecosystem - name: ensure bread-ecosystem
run: | run: |
if [[ -d "${ECOSYSTEM_DIR}/.git" ]]; then rm -rf "${ECOSYSTEM_DIR}"
git -C "${ECOSYSTEM_DIR}" pull --ff-only
else
mkdir -p "$(dirname "${ECOSYSTEM_DIR}")"
git clone https://github.com/Breadway/bread-ecosystem.git "${ECOSYSTEM_DIR}" git clone https://github.com/Breadway/bread-ecosystem.git "${ECOSYSTEM_DIR}"
fi
- name: regenerate index.json - name: regenerate index.json
run: bash "${ECOSYSTEM_DIR}/scripts/gen-index.sh" run: bash "${ECOSYSTEM_DIR}/scripts/gen-index.sh"

11
Cargo.lock generated
View file

@ -28,8 +28,8 @@ checksum = "b4388bee8683e3d04af747c73422af53102d2bd24d9eadb6cbc100baef4b43f8"
[[package]] [[package]]
name = "bread-theme" name = "bread-theme"
version = "0.1.0" version = "0.2.3"
source = "git+https://github.com/Breadway/bread-ecosystem?tag=v0.1.0#6b5f4f475f66a645b08cb865e6dda8228d23679b" source = "git+https://github.com/Breadway/bread-ecosystem?tag=v0.2.8#77417d552130281ff787e07d52541eb25e9d533b"
dependencies = [ dependencies = [
"dirs", "dirs",
"gtk4", "gtk4",
@ -39,7 +39,7 @@ dependencies = [
[[package]] [[package]]
name = "breadbox" name = "breadbox"
version = "0.1.0" version = "0.2.4"
dependencies = [ dependencies = [
"bread-theme", "bread-theme",
"breadbox-shared", "breadbox-shared",
@ -50,15 +50,16 @@ dependencies = [
[[package]] [[package]]
name = "breadbox-shared" name = "breadbox-shared"
version = "0.1.0" version = "0.2.4"
dependencies = [ dependencies = [
"serde", "serde",
"serde_json",
"toml 0.8.23", "toml 0.8.23",
] ]
[[package]] [[package]]
name = "breadbox-sync" name = "breadbox-sync"
version = "0.1.0" version = "0.2.4"
dependencies = [ dependencies = [
"breadbox-shared", "breadbox-shared",
"serde_json", "serde_json",

View file

@ -2,6 +2,7 @@ name = "breadbox"
description = "App launcher for Hyprland / Wayland" description = "App launcher for Hyprland / Wayland"
binaries = ["breadbox", "breadbox-sync"] binaries = ["breadbox", "breadbox-sync"]
system_deps = ["gtk4", "gtk4-layer-shell", "librsvg"] system_deps = ["gtk4", "gtk4-layer-shell", "librsvg"]
optional_system_deps = ["hyprland"]
bread_deps = [] bread_deps = []
[[service]] [[service]]

View file

@ -1,9 +1,10 @@
[package] [package]
name = "breadbox-shared" name = "breadbox-shared"
version = "0.1.0" version = "0.2.4"
edition = "2021" edition = "2021"
license = "MIT" license = "MIT"
[dependencies] [dependencies]
serde = { version = "1", features = ["derive"] } serde = { version = "1", features = ["derive"] }
serde_json = "1"
toml = "0.8" toml = "0.8"

View file

@ -1,4 +1,5 @@
use std::{ use std::{
collections::HashMap,
env, env,
fs::{self, File}, fs::{self, File},
io::{BufRead, BufReader}, io::{BufRead, BufReader},
@ -215,6 +216,38 @@ impl Default for IconCache {
} }
} }
// ---- Launch history ---------------------------------------------------------
pub struct LaunchHistory {
counts: HashMap<String, u32>,
path: PathBuf,
}
impl LaunchHistory {
pub fn load() -> Self {
let path = cache_dir().join("history.json");
let counts = fs::read_to_string(&path)
.ok()
.and_then(|s| serde_json::from_str(&s).ok())
.unwrap_or_default();
LaunchHistory { counts, path }
}
pub fn count(&self, name: &str) -> u32 {
self.counts.get(name).copied().unwrap_or(0)
}
pub fn increment(&mut self, name: &str) {
*self.counts.entry(name.to_string()).or_insert(0) += 1;
}
pub fn save(&self) {
if let Ok(json) = serde_json::to_string(&self.counts) {
let _ = fs::write(&self.path, json);
}
}
}
// ---- Config ----------------------------------------------------------------- // ---- Config -----------------------------------------------------------------
#[derive(Debug, Clone, Serialize, Deserialize, Default)] #[derive(Debug, Clone, Serialize, Deserialize, Default)]

View file

@ -1,6 +1,6 @@
[package] [package]
name = "breadbox-sync" name = "breadbox-sync"
version = "0.1.0" version = "0.2.4"
edition = "2021" edition = "2021"
license = "MIT" license = "MIT"

View file

@ -1,6 +1,6 @@
[package] [package]
name = "breadbox" name = "breadbox"
version = "0.1.0" version = "0.2.4"
edition = "2021" edition = "2021"
license = "MIT" license = "MIT"
@ -9,7 +9,7 @@ name = "breadbox"
path = "src/main.rs" path = "src/main.rs"
[dependencies] [dependencies]
bread-theme = { git = "https://github.com/Breadway/bread-ecosystem", tag = "v0.1.0", features = ["gtk"] } bread-theme = { git = "https://github.com/Breadway/bread-ecosystem", tag = "v0.2.8", features = ["gtk"] }
breadbox-shared = { path = "../breadbox-shared" } breadbox-shared = { path = "../breadbox-shared" }
gtk4 = { version = "0.11", features = ["v4_12"] } gtk4 = { version = "0.11", features = ["v4_12"] }
gtk4-layer-shell = "0.8" gtk4-layer-shell = "0.8"

View file

@ -1,5 +1,6 @@
use bread_theme::{hex_to_rgba, load_palette, Palette}; use bread_theme::{hex_to_rgba, ink_on, load_palette, Palette};
use std::{ use std::{
cell::RefCell,
collections::HashMap, collections::HashMap,
env, env,
fs, fs,
@ -11,10 +12,9 @@ use std::{
}; };
use breadbox_shared::{ use breadbox_shared::{
config_dir, load_all_desktop_entries, Config, DesktopEntry, IconCache, config_dir, load_all_desktop_entries, Config, DesktopEntry, IconCache, LaunchHistory,
}; };
use gtk4::{ use gtk4::{
gdk::Display,
glib, glib,
pango::EllipsizeMode, pango::EllipsizeMode,
prelude::*, prelude::*,
@ -58,6 +58,7 @@ fn load_manifest() -> HashMap<String, PathBuf> {
fn load_sorted_entries( fn load_sorted_entries(
manifest: &HashMap<String, PathBuf>, manifest: &HashMap<String, PathBuf>,
priority: &[String], priority: &[String],
history: &LaunchHistory,
) -> Vec<DesktopEntry> { ) -> Vec<DesktopEntry> {
let mut entries = load_all_desktop_entries(); let mut entries = load_all_desktop_entries();
@ -79,7 +80,11 @@ fn load_sorted_entries(
(Some(i), Some(j)) => i.cmp(&j), (Some(i), Some(j)) => i.cmp(&j),
(Some(_), None) => std::cmp::Ordering::Less, (Some(_), None) => std::cmp::Ordering::Less,
(None, Some(_)) => std::cmp::Ordering::Greater, (None, Some(_)) => std::cmp::Ordering::Greater,
(None, None) => a.name.to_lowercase().cmp(&b.name.to_lowercase()), (None, None) => {
// Most-launched first, then alphabetical
history.count(&b.name).cmp(&history.count(&a.name))
.then(a.name.to_lowercase().cmp(&b.name.to_lowercase()))
}
} }
}); });
@ -127,26 +132,32 @@ fn matches_term(field: &str, term: &str) -> bool {
fn build_css(p: &Palette) -> String { fn build_css(p: &Palette) -> String {
let bg_panel = hex_to_rgba(&p.background, 0.60); let bg_panel = hex_to_rgba(&p.background, 0.60);
// breadbox-specific rules only — fonts, palette, and generic widgets come
// from the shared ecosystem stylesheet (applied first in connect_activate).
// Colour is set on each surface (panel, search box, hovered/selected row) so
// child labels inherit the legible ink for that background. `on_*` are
// luminance-picked black/white — the pywal hues are untouched. Without this a
// light `surface` slot makes the selected row's text vanish.
format!( format!(
"* {{ font-family: 'Varela Round', sans-serif; font-size: 14px; }}\ "window {{ background-color: transparent; }}\
window {{ background-color: transparent; }}\ .launcher-bg {{ background-color: {bg_panel}; color: {on_bg}; border-radius: 8px;\
.launcher-bg {{ background-color: {bg_panel}; border-radius: 8px;\
box-shadow: 0 8px 32px rgba(0,0,0,0.6); }}\ box-shadow: 0 8px 32px rgba(0,0,0,0.6); }}\
searchentry {{ background-color: {surface}; color: {fg}; caret-color: {accent};\ searchentry {{ background-color: {surface}; color: {on_surface}; caret-color: {accent};\
border: none; outline: none; box-shadow: none;\ border: none; outline: none; box-shadow: none;\
padding: 12px 16px; border-radius: 6px 6px 0 0; }}\ padding: 12px 16px; border-radius: 6px 6px 0 0; }}\
listbox {{ background-color: transparent; padding: 4px; }}\ listbox {{ background-color: transparent; padding: 4px; }}\
row {{ padding: 8px 12px; color: {fg}; background-color: transparent;\ row {{ padding: 8px 12px; color: {on_bg}; background-color: transparent;\
border-radius: 6px; }}\ border-radius: 6px; }}\
row:hover {{ background-color: {surface}; }}\ row:hover {{ background-color: {surface}; color: {on_surface}; }}\
row:selected {{ background-color: {surface}; }}\ row:selected {{ background-color: {surface}; color: {on_surface}; }}\
.app-name {{ font-size: 14px; }}\ .app-name {{ font-size: 14px; }}\
.app-muted {{ color: {fg}; opacity: 0.6; font-size: 12px; }}\ .app-muted {{ opacity: 0.6; font-size: 12px; }}\
image {{ margin-right: 8px; }}", image {{ margin-right: 8px; }}",
bg_panel = bg_panel, bg_panel = bg_panel,
surface = p.color0, surface = p.color0,
fg = p.foreground,
accent = p.color4, accent = p.color4,
on_bg = ink_on(&p.background),
on_surface = ink_on(&p.color0),
) )
} }
@ -230,6 +241,17 @@ fn fuzzy_matches(pattern: &str, text: &str) -> bool {
true true
} }
fn fuzzy_score(query: &str, entry: &DesktopEntry) -> u32 {
let q = query.to_lowercase();
let name = entry.name.to_lowercase();
let wm = entry.wm_class.as_deref().unwrap_or("").to_lowercase();
if name == q || wm == q { return 0; }
if name.starts_with(&q) { return 1; }
if name.contains(&q) { return 2; }
if wm.starts_with(&q) || wm.contains(&q) { return 3; }
4 // subsequence match
}
// ---- PID file toggle -------------------------------------------------------- // ---- PID file toggle --------------------------------------------------------
fn pid_file() -> PathBuf { fn pid_file() -> PathBuf {
@ -273,24 +295,23 @@ fn get_row_entry(row: &gtk4::ListBoxRow) -> Option<DesktopEntry> {
} }
} }
fn run_ui(entries: Vec<DesktopEntry>, css: String) { fn run_ui(entries: Vec<DesktopEntry>, history: LaunchHistory) {
let app = Application::builder() let app = Application::builder()
.application_id("com.breadway.breadbox") .application_id("com.breadway.breadbox")
.build(); .build();
let history_rc = Rc::new(RefCell::new(history));
let query_rc: Rc<RefCell<String>> = Rc::new(RefCell::new(String::new()));
app.connect_activate(move |app| { app.connect_activate(move |app| {
// Base CSS // Shared ecosystem base (fonts, palette, generic widgets) first, then
let provider = CssProvider::new(); // breadbox-specific CSS layered on top — both hot-reload on
provider.load_from_string(&css); // `bread-theme reload` (the closure re-reads the pywal palette).
gtk4::style_context_add_provider_for_display( bread_theme::gtk::apply_shared();
&Display::default().expect("no display"), bread_theme::gtk::apply_app_css(|| build_css(&load_palette()));
&provider,
gtk4::STYLE_PROVIDER_PRIORITY_APPLICATION,
);
// User CSS override // User CSS override
{ {
use std::cell::RefCell;
let user_css_path = config_dir().join("style.css"); let user_css_path = config_dir().join("style.css");
let user_cell: RefCell<Option<CssProvider>> = RefCell::new(None); let user_cell: RefCell<Option<CssProvider>> = RefCell::new(None);
bread_theme::gtk::apply_user_css(&user_css_path, &user_cell); bread_theme::gtk::apply_user_css(&user_css_path, &user_cell);
@ -334,7 +355,7 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
let list = ListBox::new(); let list = ListBox::new();
list.set_selection_mode(SelectionMode::Browse); list.set_selection_mode(SelectionMode::Browse);
for entry in &entries { for (idx, entry) in entries.iter().enumerate() {
let row = gtk4::ListBoxRow::new(); let row = gtk4::ListBoxRow::new();
let hbox = GBox::new(Orientation::Horizontal, 0); let hbox = GBox::new(Orientation::Horizontal, 0);
hbox.set_margin_start(6); hbox.set_margin_start(6);
@ -360,9 +381,35 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
row.set_child(Some(&hbox)); row.set_child(Some(&hbox));
unsafe { row.set_data("entry", entry.clone()) }; unsafe { row.set_data("entry", entry.clone()) };
unsafe { row.set_data("initial_order", idx as u32) };
list.append(&row); list.append(&row);
} }
// Sort by match quality + launch count when a query is active;
// fall back to insertion order (priority + launch frequency) when empty.
let sort_query = Rc::clone(&query_rc);
let sort_history = Rc::clone(&history_rc);
list.set_sort_func(move |row_a, row_b| {
let query = sort_query.borrow();
if query.is_empty() {
let oa = unsafe { row_a.data::<u32>("initial_order").map_or(u32::MAX, |p| *p.as_ref()) };
let ob = unsafe { row_b.data::<u32>("initial_order").map_or(u32::MAX, |p| *p.as_ref()) };
return oa.cmp(&ob).into();
}
let (Some(ea), Some(eb)) = (get_row_entry(row_a), get_row_entry(row_b)) else {
return std::cmp::Ordering::Equal.into();
};
let sa = fuzzy_score(&query, &ea);
let sb = fuzzy_score(&query, &eb);
let history = sort_history.borrow();
let ca = history.count(&ea.name);
let cb = history.count(&eb.name);
sa.cmp(&sb)
.then(cb.cmp(&ca))
.then(ea.name.to_lowercase().cmp(&eb.name.to_lowercase()))
.into()
});
if let Some(first) = list.row_at_index(0) { if let Some(first) = list.row_at_index(0) {
list.select_row(Some(&first)); list.select_row(Some(&first));
} }
@ -373,10 +420,11 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
// Filter on keystroke // Filter on keystroke
let list_f = list.clone(); let list_f = list.clone();
let filter_query = Rc::clone(&query_rc);
search.connect_changed(move |entry| { search.connect_changed(move |entry| {
let text = entry.text(); let text = entry.text();
let query = text.as_str(); let query = text.as_str();
let mut first_vis: Option<gtk4::ListBoxRow> = None; *filter_query.borrow_mut() = query.to_string();
let mut i = 0i32; let mut i = 0i32;
while let Some(row) = list_f.row_at_index(i) { while let Some(row) = list_f.row_at_index(i) {
let vis = get_row_entry(&row) let vis = get_row_entry(&row)
@ -389,11 +437,12 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
}) })
.unwrap_or(false); .unwrap_or(false);
row.set_visible(vis); row.set_visible(vis);
if vis && first_vis.is_none() {
first_vis = Some(row);
}
i += 1; i += 1;
} }
list_f.invalidate_sort();
let first_vis = (0i32..).find_map(|j| {
list_f.row_at_index(j).filter(|r| r.is_visible())
});
list_f.select_row(first_vis.as_ref()); list_f.select_row(first_vis.as_ref());
}); });
@ -402,6 +451,7 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
key_ctrl.set_propagation_phase(gtk4::PropagationPhase::Capture); key_ctrl.set_propagation_phase(gtk4::PropagationPhase::Capture);
let close_k = Rc::clone(&close_all); let close_k = Rc::clone(&close_all);
let list_k = list.clone(); let list_k = list.clone();
let history_k = Rc::clone(&history_rc);
key_ctrl.connect_key_pressed(move |_, key, _, _| { key_ctrl.connect_key_pressed(move |_, key, _, _| {
use gtk4::gdk::Key; use gtk4::gdk::Key;
match key { match key {
@ -412,6 +462,8 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
Key::Return | Key::KP_Enter => { Key::Return | Key::KP_Enter => {
if let Some(row) = list_k.selected_row() { if let Some(row) = list_k.selected_row() {
if let Some(entry) = get_row_entry(&row) { if let Some(entry) = get_row_entry(&row) {
history_k.borrow_mut().increment(&entry.name);
history_k.borrow().save();
do_launch(&entry); do_launch(&entry);
close_k(); close_k();
} }
@ -458,8 +510,11 @@ fn run_ui(entries: Vec<DesktopEntry>, css: String) {
// Row click launches // Row click launches
let close_a = Rc::clone(&close_all); let close_a = Rc::clone(&close_all);
let history_a = Rc::clone(&history_rc);
list.connect_row_activated(move |_, row| { list.connect_row_activated(move |_, row| {
if let Some(entry) = get_row_entry(row) { if let Some(entry) = get_row_entry(row) {
history_a.borrow_mut().increment(&entry.name);
history_a.borrow().save();
do_launch(&entry); do_launch(&entry);
close_a(); close_a();
} }
@ -505,11 +560,9 @@ fn main() {
.map(|c| c.priority.clone()) .map(|c| c.priority.clone())
.unwrap_or_default(); .unwrap_or_default();
let history = LaunchHistory::load();
let manifest = load_manifest(); let manifest = load_manifest();
let entries = load_sorted_entries(&manifest, &priority); let entries = load_sorted_entries(&manifest, &priority, &history);
let palette = load_palette(); run_ui(entries, history);
let css = build_css(&palette);
run_ui(entries, css);
} }

39
packaging/arch/PKGBUILD Normal file
View file

@ -0,0 +1,39 @@
# Maintainer: Breadway <rileyhorsham@gmail.com>
pkgname=breadbox
pkgver=0.1.0
pkgrel=1
pkgdesc="App launcher for Hyprland / Wayland"
arch=('x86_64')
url="https://github.com/Breadway/breadbox"
license=('MIT')
# Some Rust deps (ring/mlua) build vendored C/asm into static archives; makepkg's
# default -flto=auto emits GCC LTO bitcode the Rust (lld) link cannot read,
# causing undefined-symbol errors. Disable LTO.
options=(!lto !debug)
depends=('gtk4' 'gtk4-layer-shell' 'librsvg')
optdepends=(
'hyprland: window and workspace integration'
)
makedepends=('rust' 'cargo')
source=("${pkgname}-${pkgver}.tar.gz")
sha256sums=('SKIP')
build() {
cd "${srcdir}/${pkgname}-${pkgver}"
cargo build --release --locked
}
check() {
cd "${srcdir}/${pkgname}-${pkgver}"
cargo test --release --locked --workspace
}
package() {
cd "${srcdir}/${pkgname}-${pkgver}"
install -Dm755 target/release/breadbox "${pkgdir}/usr/bin/breadbox"
install -Dm755 target/release/breadbox-sync "${pkgdir}/usr/bin/breadbox-sync"
install -Dm644 packaging/breadbox-sync.service \
"${pkgdir}/usr/lib/systemd/user/breadbox-sync.service"
install -Dm644 LICENSE "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE"
}