Compare commits
135 Commits
release-20
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
71456ff4d1 | ||
|
|
1a3099d926 | ||
|
|
4ef4ee8d45 | ||
|
|
253067b417 | ||
|
|
8d5edc659f | ||
|
|
c122290f37 | ||
|
|
999cef34a7 | ||
|
|
93a43d1b38 | ||
|
|
91ca6ff96f | ||
|
|
8c0d35cca5 | ||
|
|
444ea4326d | ||
|
|
401ccb69b2 | ||
|
|
9cfea2aed5 | ||
|
|
ce212cb2d6 | ||
|
|
851d79029a | ||
|
|
ab27cf3eb4 | ||
|
|
7b4c9ff336 | ||
|
|
743596d701 | ||
|
|
598f038ea6 | ||
|
|
7f8c721115 | ||
|
|
ba7f1ffd91 | ||
|
|
daa24dd7ec | ||
|
|
45271b5060 | ||
|
|
4d02bc655d | ||
|
|
3c2ba477f5 | ||
|
|
6106d1cd22 | ||
|
|
50b83ebacf | ||
|
|
566308d889 | ||
|
|
e878d1fe33 | ||
|
|
d88084fb9a | ||
|
|
f8639f2967 | ||
|
|
7129c9a8eb | ||
|
|
91202e7cf8 | ||
|
|
3c2fec8b8c | ||
|
|
ec23285a6a | ||
|
|
f075a87668 | ||
|
|
3ce980239c | ||
|
|
d00fc41010 | ||
|
|
0636020a86 | ||
|
|
322e0ab4a3 | ||
|
|
89f0a5423d | ||
|
|
80f9e5cda2 | ||
|
|
a12265573c | ||
|
|
e45ec46793 | ||
|
|
0e075d6092 | ||
|
|
dbbc722f50 | ||
|
|
dd3b3f3504 | ||
|
|
bb9e06c9ed | ||
|
|
a834d185e3 | ||
|
|
4e81565ab6 | ||
|
|
795eef4bda | ||
|
|
3ad8424d17 | ||
|
|
83feb5a5cf | ||
|
|
0dd97f3d7e | ||
|
|
bb32c1f7f6 | ||
|
|
3bb2d68a65 | ||
|
|
fe117cdaec | ||
|
|
e062a3c8cd | ||
|
|
ae2c9c934d | ||
|
|
f7e1356eae | ||
|
|
98b1f059c7 | ||
|
|
a3735d0931 | ||
|
|
55f00729ee | ||
|
|
275f4a91b2 | ||
|
|
025a0c4a57 | ||
|
|
b29f9f3cc2 | ||
|
|
feede5c5b4 | ||
|
|
c1306a32a9 | ||
|
|
ed5fba4b46 | ||
|
|
f149603ac8 | ||
|
|
89ad42f24d | ||
|
|
bec804131b | ||
|
|
79979e3e09 | ||
|
|
75efd877c4 | ||
|
|
52c6f45a91 | ||
|
|
f9a8fbccfd | ||
|
|
4a0ebf0cb1 | ||
|
|
e5a68fb630 | ||
|
|
155dd07193 | ||
|
|
e5f76fe548 | ||
|
|
5e098817f2 | ||
|
|
4dcbd8187e | ||
|
|
6c0c0a0919 | ||
|
|
3158a11229 | ||
|
|
45f5d1b678 | ||
|
|
c8e3caa4fc | ||
|
|
98555f16ca | ||
|
|
5449651db6 | ||
|
|
1156f0a111 | ||
|
|
9b8c075d41 | ||
|
|
c5679f043d | ||
|
|
0b3d4d1fd4 | ||
|
|
b2fb411bb1 | ||
|
|
1225c550b7 | ||
|
|
48193db81b | ||
|
|
48d8a8e0ee | ||
|
|
945f4718eb | ||
|
|
6a2952c6f3 | ||
|
|
03b63dd5d0 | ||
|
|
7717dffc56 | ||
|
|
3a2cab7ca7 | ||
|
|
e0b5411eb1 | ||
|
|
f85adfa007 | ||
|
|
88c3cd57c1 | ||
|
|
d67215a6eb | ||
|
|
5cc2fdd8b4 | ||
|
|
38bce3fecf | ||
|
|
f1c75c468a | ||
|
|
bfb152f0d8 | ||
|
|
8a7d9957aa | ||
|
|
56e8cc03d1 | ||
|
|
04cb7d51cb | ||
|
|
555505b80e | ||
|
|
fa07fd88dc | ||
|
|
bbb8e8dc98 | ||
|
|
d09ecd21b7 | ||
|
|
9a00e341af | ||
|
|
c78dcda55e | ||
|
|
57abbb3335 | ||
|
|
315a2421c4 | ||
|
|
c135c96d27 | ||
|
|
315a2f9a53 | ||
|
|
9d160a7ef5 | ||
|
|
a24e4101e8 | ||
|
|
7539307e18 | ||
|
|
907ae66a7f | ||
|
|
f35b119c0d | ||
|
|
920704ee72 | ||
|
|
d8ad9e62d8 | ||
|
|
ea99ac62bc | ||
|
|
af13b5af09 | ||
|
|
c8907e7999 | ||
|
|
714e2cb7d5 | ||
|
|
f35b716f93 | ||
|
|
47db9fa0c0 |
@@ -22,12 +22,32 @@ jobs:
|
|||||||
git fetch --depth=1 origin ${{ gitea.sha }}
|
git fetch --depth=1 origin ${{ gitea.sha }}
|
||||||
git checkout FETCH_HEAD
|
git checkout FETCH_HEAD
|
||||||
|
|
||||||
|
- name: Cache Cargo
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.cargo/bin/
|
||||||
|
~/.cargo/registry/index/
|
||||||
|
~/.cargo/registry/cache/
|
||||||
|
~/.cargo/git/db/
|
||||||
|
target/
|
||||||
|
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-cargo-
|
||||||
|
|
||||||
|
- name: Cache Node Modules
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: frontend/node_modules
|
||||||
|
key: ${{ runner.os }}-node-${{ hashFiles('frontend/package-lock.json') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-node-
|
||||||
|
|
||||||
- name: Build Frontend
|
- name: Build Frontend
|
||||||
run: |
|
run: |
|
||||||
cd frontend
|
cd frontend
|
||||||
npm install
|
npm install
|
||||||
npx @tailwindcss/cli -i input.css -o public/tailwind.css --minify --content './src/**/*.rs'
|
npx @tailwindcss/cli -i input.css -o public/tailwind.css --minify --content './src/**/*.rs'
|
||||||
# Trunk'ın optimizasyonunu kapalı (0) tutuyoruz çünkü Cargo.toml'daki opt-level='z' zaten o işi yapıyor.
|
|
||||||
trunk build --release
|
trunk build --release
|
||||||
|
|
||||||
- name: Build Backend (MIPS)
|
- name: Build Backend (MIPS)
|
||||||
|
|||||||
469
Cargo.lock
generated
469
Cargo.lock
generated
@@ -320,9 +320,11 @@ dependencies = [
|
|||||||
"dotenvy",
|
"dotenvy",
|
||||||
"futures",
|
"futures",
|
||||||
"governor",
|
"governor",
|
||||||
|
"icons",
|
||||||
"jsonwebtoken",
|
"jsonwebtoken",
|
||||||
"leptos",
|
"leptos",
|
||||||
"leptos_axum",
|
"leptos_axum",
|
||||||
|
"leptos_ui",
|
||||||
"mime_guess",
|
"mime_guess",
|
||||||
"openssl",
|
"openssl",
|
||||||
"quick-xml",
|
"quick-xml",
|
||||||
@@ -343,6 +345,7 @@ dependencies = [
|
|||||||
"tower_governor",
|
"tower_governor",
|
||||||
"tracing",
|
"tracing",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
|
"tw_merge",
|
||||||
"utoipa",
|
"utoipa",
|
||||||
"utoipa-swagger-ui",
|
"utoipa-swagger-ui",
|
||||||
"web-push",
|
"web-push",
|
||||||
@@ -1256,30 +1259,15 @@ dependencies = [
|
|||||||
"console_error_panic_hook",
|
"console_error_panic_hook",
|
||||||
"console_log",
|
"console_log",
|
||||||
"futures",
|
"futures",
|
||||||
|
"gloo-console",
|
||||||
"gloo-net",
|
"gloo-net",
|
||||||
"gloo-timers",
|
"gloo-timers",
|
||||||
|
"icons",
|
||||||
"js-sys",
|
"js-sys",
|
||||||
"leptos",
|
"leptos",
|
||||||
"leptos-shadcn-alert",
|
|
||||||
"leptos-shadcn-avatar",
|
|
||||||
"leptos-shadcn-badge",
|
|
||||||
"leptos-shadcn-button",
|
|
||||||
"leptos-shadcn-card",
|
|
||||||
"leptos-shadcn-context-menu",
|
|
||||||
"leptos-shadcn-dialog",
|
|
||||||
"leptos-shadcn-dropdown-menu",
|
|
||||||
"leptos-shadcn-input",
|
|
||||||
"leptos-shadcn-label",
|
|
||||||
"leptos-shadcn-progress",
|
|
||||||
"leptos-shadcn-scroll-area",
|
|
||||||
"leptos-shadcn-separator",
|
|
||||||
"leptos-shadcn-sheet",
|
|
||||||
"leptos-shadcn-skeleton",
|
|
||||||
"leptos-shadcn-tabs",
|
|
||||||
"leptos-shadcn-toast",
|
|
||||||
"leptos-shadcn-tooltip",
|
|
||||||
"leptos-use",
|
"leptos-use",
|
||||||
"leptos_router",
|
"leptos_router",
|
||||||
|
"leptos_ui",
|
||||||
"log",
|
"log",
|
||||||
"rmp-serde",
|
"rmp-serde",
|
||||||
"serde",
|
"serde",
|
||||||
@@ -1287,8 +1275,10 @@ dependencies = [
|
|||||||
"serde_json",
|
"serde_json",
|
||||||
"shared",
|
"shared",
|
||||||
"struct-patch",
|
"struct-patch",
|
||||||
|
"strum 0.26.3",
|
||||||
"tailwind_fuse",
|
"tailwind_fuse",
|
||||||
"thiserror 2.0.18",
|
"thiserror 2.0.18",
|
||||||
|
"tw_merge",
|
||||||
"uuid",
|
"uuid",
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
"wasm-bindgen-futures",
|
"wasm-bindgen-futures",
|
||||||
@@ -1440,6 +1430,19 @@ dependencies = [
|
|||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "gloo-console"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2a17868f56b4a24f677b17c8cb69958385102fa879418052d60b50bc1727e261"
|
||||||
|
dependencies = [
|
||||||
|
"gloo-utils",
|
||||||
|
"js-sys",
|
||||||
|
"serde",
|
||||||
|
"wasm-bindgen",
|
||||||
|
"web-sys",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "gloo-net"
|
name = "gloo-net"
|
||||||
version = "0.6.0"
|
version = "0.6.0"
|
||||||
@@ -1865,6 +1868,17 @@ dependencies = [
|
|||||||
"cc",
|
"cc",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "icons"
|
||||||
|
version = "0.18.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "75da3d473e24e0b541bf28cf31e67b262c379a4cbc2149f4865b7d99406711dc"
|
||||||
|
dependencies = [
|
||||||
|
"leptos",
|
||||||
|
"strum 0.27.2",
|
||||||
|
"tw_merge",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "icu_collections"
|
name = "icu_collections"
|
||||||
version = "2.1.1"
|
version = "2.1.1"
|
||||||
@@ -2156,334 +2170,6 @@ dependencies = [
|
|||||||
"web-sys",
|
"web-sys",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-node-ref"
|
|
||||||
version = "0.2.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "4f57b1ebc451fe9e7b6c7eba680fa8bc7313b410cc6c0f18481cb55a60ff3ac6"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"send_wrapper",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-alert"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "884551df61ade405bdb61b0d5a92a3a88b3a7af7a2d283b8d3e942ae0d71309c"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-avatar"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "8cb3c5b1f5ba02f7282b55fde1513cdfecef3b25bf5fa44e1eb29fcaf8b927c5"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"wasm-bindgen",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-badge"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "24578fb0bc21eb21be4e686e6719c7e183acb8fd071a4f81fb27fe452751c88a"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-button"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7d6d1a7b813b726be7920f7238c127a14129ba4a45fa879312cad3ed2f8a1745"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-card"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5b5cda16742d1e20284e5f6805eab88b6e54c1378d1548a8e15a5eedda1ea3eb"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-context-menu"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "e0f440e9a7517dfe6ba758080ddba1dfe42e4697008f60adfc112c5da02dca8d"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"wasm-bindgen",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-dialog"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a3fdbb636393b150c2db1e37d44a6832e9dde177ce2e81281932fefad8bde98e"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-dropdown-menu"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f50189623a176386a30443d281483c5aa6cc34dc45fa11c3e53bd187ffccde21"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-input"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b0939cdad5a878d920decda39a4b42ecf4eba15736a92bbd73b1b408807899b8"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"regex",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-label"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5e7cad4b5fae11df9bf3b1d4265a56509a9bb7d3a8580e7487f398b733eadf0c"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-progress"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a34ca41b8ebfd7f29126e4f8656987834f3613717016f11f3983da85a90669f6"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-scroll-area"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ef3d7bdcae4919ad495529ec2a5974036fb0b959580df310f36b2fd33f90860c"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-separator"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f5dfda49f059fd4d1549d663e6743e37a5c6c84d1ac2d6daec32caa3156bc268"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-sheet"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ba85819a0c94a7705ed92989442c64cc75d9ed3a4540e711e87c56b206431611"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-signal-management"
|
|
||||||
version = "0.1.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a5097c5171eb0be12bbf8fd736f4e669012657112865506a825480f2b013f6de"
|
|
||||||
dependencies = [
|
|
||||||
"chrono",
|
|
||||||
"js-sys",
|
|
||||||
"leptos",
|
|
||||||
"serde",
|
|
||||||
"serde_json",
|
|
||||||
"thiserror 1.0.69",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-skeleton"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "6c14b6bd0f2fe191e3e114a34cee889fc983546ad488e76e76511e3d75ea3f86"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-tabs"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "39f817c834e70a8359933b7b274564313be64105370611af96f05508541b661b"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-toast"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "3315f7ed844e3286704cc7b57db7209cad592c11eee770f5dc48ebdc92d66cfb"
|
|
||||||
dependencies = [
|
|
||||||
"gloo-timers",
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"uuid",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-shadcn-tooltip"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "3e41d37932d700444e1d3a21f10f198c3c9e76dde3fd78d58da4b5a099939fd7"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-node-ref",
|
|
||||||
"leptos-shadcn-signal-management",
|
|
||||||
"leptos-struct-component",
|
|
||||||
"leptos-style",
|
|
||||||
"tailwind_fuse",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-struct-component"
|
|
||||||
version = "0.2.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "c32085b37b67e61e69e0949d94e36c40e4fde83867681cbb884f9cd40a43881e"
|
|
||||||
dependencies = [
|
|
||||||
"leptos",
|
|
||||||
"leptos-struct-component-macro",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-struct-component-macro"
|
|
||||||
version = "0.2.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a40efd792acc28a115605b84ecb39e89397a278950bc8f2aad1bdcc7af2033af"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn 2.0.114",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "leptos-style"
|
|
||||||
version = "0.2.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "c65408961a0bd8e70f317de8973d532a0cb9ffbac910c488d97f9c5a2e4411e2"
|
|
||||||
dependencies = [
|
|
||||||
"indexmap",
|
|
||||||
"leptos",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "leptos-use"
|
name = "leptos-use"
|
||||||
version = "0.16.3"
|
version = "0.16.3"
|
||||||
@@ -2689,6 +2375,17 @@ dependencies = [
|
|||||||
"tachys",
|
"tachys",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "leptos_ui"
|
||||||
|
version = "0.3.20"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c7c30ca85b1aac5637bc59a9201a6aeb648452679bf0ef0e451a8f30acf153f7"
|
||||||
|
dependencies = [
|
||||||
|
"leptos",
|
||||||
|
"paste",
|
||||||
|
"tw_merge",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "libc"
|
name = "libc"
|
||||||
version = "0.2.180"
|
version = "0.2.180"
|
||||||
@@ -3981,6 +3678,7 @@ dependencies = [
|
|||||||
"inventory",
|
"inventory",
|
||||||
"js-sys",
|
"js-sys",
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
|
"rmp-serde",
|
||||||
"rustc_version",
|
"rustc_version",
|
||||||
"rustversion",
|
"rustversion",
|
||||||
"send_wrapper",
|
"send_wrapper",
|
||||||
@@ -4066,6 +3764,7 @@ dependencies = [
|
|||||||
"bcrypt",
|
"bcrypt",
|
||||||
"bytes",
|
"bytes",
|
||||||
"cookie",
|
"cookie",
|
||||||
|
"http 1.4.0",
|
||||||
"jsonwebtoken",
|
"jsonwebtoken",
|
||||||
"leptos",
|
"leptos",
|
||||||
"leptos_axum",
|
"leptos_axum",
|
||||||
@@ -4447,6 +4146,49 @@ dependencies = [
|
|||||||
"syn 2.0.114",
|
"syn 2.0.114",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "strum"
|
||||||
|
version = "0.26.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8fec0f0aef304996cf250b31b5a10dee7980c85da9d759361292b8bca5a18f06"
|
||||||
|
dependencies = [
|
||||||
|
"strum_macros 0.26.4",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "strum"
|
||||||
|
version = "0.27.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "af23d6f6c1a224baef9d3f61e287d2761385a5b88fdab4eb4c6f11aeb54c4bcf"
|
||||||
|
dependencies = [
|
||||||
|
"strum_macros 0.27.2",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "strum_macros"
|
||||||
|
version = "0.26.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "4c6bee85a5a24955dc440386795aa378cd9cf82acd5f764469152d2270e581be"
|
||||||
|
dependencies = [
|
||||||
|
"heck",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"rustversion",
|
||||||
|
"syn 2.0.114",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "strum_macros"
|
||||||
|
version = "0.27.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7695ce3845ea4b33927c055a39dc438a45b059f7c1b3d91d38d10355fb8cbca7"
|
||||||
|
dependencies = [
|
||||||
|
"heck",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn 2.0.114",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "subtle"
|
name = "subtle"
|
||||||
version = "2.6.1"
|
version = "2.6.1"
|
||||||
@@ -4557,19 +4299,6 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "7ca71fb01735fbc6fa13e9390d7a3037dde97053c0b65c0c72c0159cd009d26b"
|
checksum = "7ca71fb01735fbc6fa13e9390d7a3037dde97053c0b65c0c72c0159cd009d26b"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"nom",
|
"nom",
|
||||||
"tailwind_fuse_macro",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "tailwind_fuse_macro"
|
|
||||||
version = "0.3.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "efa51b9ff80b5533001f8452d254a688bc7bb39c6bb77f9e0a19c1664d035888"
|
|
||||||
dependencies = [
|
|
||||||
"darling",
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn 2.0.114",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -4994,6 +4723,28 @@ dependencies = [
|
|||||||
"utf-8",
|
"utf-8",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tw_merge"
|
||||||
|
version = "0.1.17"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "25e4ae38c226104e3c821c60b311bca321f45dcf46e48b683a0db2fac9e2c6e2"
|
||||||
|
dependencies = [
|
||||||
|
"nom",
|
||||||
|
"tw_merge_variants",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tw_merge_variants"
|
||||||
|
version = "0.1.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "03de956478d5562138828bb736cc066949bda33dbb99c55ef77b2bb5438868e4"
|
||||||
|
dependencies = [
|
||||||
|
"darling",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn 2.0.114",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "typed-builder"
|
name = "typed-builder"
|
||||||
version = "0.21.2"
|
version = "0.21.2"
|
||||||
|
|||||||
@@ -2,6 +2,9 @@
|
|||||||
members = ["backend", "frontend", "shared"]
|
members = ["backend", "frontend", "shared"]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
|
|
||||||
|
[[workspace.metadata.leptos]]
|
||||||
|
tailwind-input-file = "frontend/input.css"
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
# En küçük binary boyutu
|
# En küçük binary boyutu
|
||||||
opt-level = "z"
|
opt-level = "z"
|
||||||
|
|||||||
@@ -46,4 +46,7 @@ governor = "0.10.4"
|
|||||||
# Leptos
|
# Leptos
|
||||||
leptos = { version = "0.8.15", features = ["nightly"] }
|
leptos = { version = "0.8.15", features = ["nightly"] }
|
||||||
leptos_axum = { version = "0.8.7" }
|
leptos_axum = { version = "0.8.7" }
|
||||||
jsonwebtoken = "9"
|
jsonwebtoken = "9"
|
||||||
|
tw_merge = { version = "0.1.17", features = ["variant"] }
|
||||||
|
icons = { version = "0.18.0", features = ["leptos"] }
|
||||||
|
leptos_ui = "0.3.20"
|
||||||
|
|||||||
83
backend/src/bin/test_trackers.rs
Normal file
83
backend/src/bin/test_trackers.rs
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
use shared::xmlrpc::{RtorrentClient, RpcParam, parse_multicall_response};
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() {
|
||||||
|
let mut client = None;
|
||||||
|
for path in ["127.0.0.1:8000", "0.0.0.0:8000", "localhost:8000"] {
|
||||||
|
let test_client = RtorrentClient::new(path);
|
||||||
|
match test_client.call("system.client_version", &[]).await {
|
||||||
|
Ok(res) => {
|
||||||
|
println!("SUCCESS: Connected to rTorrent at {} (Version: {})", path, res);
|
||||||
|
client = Some(test_client);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
Err(_) => {
|
||||||
|
// println!("Failed to connect to {}", path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let client = match client {
|
||||||
|
Some(c) => c,
|
||||||
|
None => {
|
||||||
|
println!("Could not connect to rTorrent on port 8000.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut hash = String::new();
|
||||||
|
match client.call("d.multicall2", &[RpcParam::from(""), RpcParam::from("main"), RpcParam::from("d.hash=")]).await {
|
||||||
|
Ok(xml) => {
|
||||||
|
if let Ok(rows) = parse_multicall_response(&xml) {
|
||||||
|
if let Some(row) = rows.first() {
|
||||||
|
if let Some(h) = row.first() {
|
||||||
|
hash = h.clone();
|
||||||
|
println!("Using torrent hash: {}", hash);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
println!("Error getting torrents: {:?}", e);
|
||||||
|
return;
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if hash.is_empty() {
|
||||||
|
println!("No torrents found to test trackers.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now test Tracker fields one by one to see which one is failing
|
||||||
|
let fields = vec![
|
||||||
|
"t.url=",
|
||||||
|
"t.is_enabled=",
|
||||||
|
"t.group=",
|
||||||
|
"t.scrape_complete=",
|
||||||
|
"t.scrape_incomplete=",
|
||||||
|
"t.scrape_downloaded=",
|
||||||
|
"t.activity_date_last=",
|
||||||
|
"t.normal_interval=",
|
||||||
|
"t.message=",
|
||||||
|
];
|
||||||
|
|
||||||
|
for field in &fields {
|
||||||
|
let params = vec![
|
||||||
|
RpcParam::from(hash.as_str()),
|
||||||
|
RpcParam::from(""),
|
||||||
|
RpcParam::from(*field),
|
||||||
|
];
|
||||||
|
|
||||||
|
print!("Testing field {:<22} : ", field);
|
||||||
|
match client.call("t.multicall", ¶ms).await {
|
||||||
|
Ok(xml) => {
|
||||||
|
if xml.contains("faultCode") {
|
||||||
|
println!("FAILED");
|
||||||
|
} else {
|
||||||
|
println!("SUCCESS");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(e) => println!("ERROR: {:?}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,154 +1,2 @@
|
|||||||
use crate::AppState;
|
// This file is intentionally empty as authentication is now handled by Server Functions.
|
||||||
use axum::{
|
// See shared/src/server_fns/auth.rs
|
||||||
extract::{State, Json},
|
|
||||||
http::StatusCode,
|
|
||||||
response::IntoResponse,
|
|
||||||
};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use utoipa::ToSchema;
|
|
||||||
use axum_extra::extract::cookie::{Cookie, CookieJar, SameSite};
|
|
||||||
use time::Duration;
|
|
||||||
|
|
||||||
#[derive(Deserialize, ToSchema)]
|
|
||||||
pub struct LoginRequest {
|
|
||||||
username: String,
|
|
||||||
password: String,
|
|
||||||
#[serde(default)]
|
|
||||||
remember_me: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, ToSchema)]
|
|
||||||
pub struct UserResponse {
|
|
||||||
username: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[utoipa::path(
|
|
||||||
post,
|
|
||||||
path = "/api/auth/login",
|
|
||||||
request_body = LoginRequest,
|
|
||||||
responses(
|
|
||||||
(status = 200, description = "Login successful"),
|
|
||||||
(status = 401, description = "Invalid credentials"),
|
|
||||||
(status = 500, description = "Internal server error")
|
|
||||||
)
|
|
||||||
)]
|
|
||||||
pub async fn login_handler(
|
|
||||||
State(state): State<AppState>,
|
|
||||||
jar: CookieJar,
|
|
||||||
Json(payload): Json<LoginRequest>,
|
|
||||||
) -> impl IntoResponse {
|
|
||||||
tracing::info!("Login attempt for user: {}", payload.username);
|
|
||||||
|
|
||||||
let user = match state.db.get_user_by_username(&payload.username).await {
|
|
||||||
Ok(Some(u)) => u,
|
|
||||||
Ok(None) => {
|
|
||||||
tracing::warn!("Login failed: User not found for {}", payload.username);
|
|
||||||
return (StatusCode::UNAUTHORIZED, "Invalid credentials").into_response();
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("DB error during login for {}: {}", payload.username, e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Database error").into_response();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let (user_id, password_hash) = user;
|
|
||||||
|
|
||||||
match bcrypt::verify(&payload.password, &password_hash) {
|
|
||||||
Ok(true) => {
|
|
||||||
tracing::info!("Password verified for user: {}", payload.username);
|
|
||||||
|
|
||||||
// Create session
|
|
||||||
let token: String = (0..32).map(|_| {
|
|
||||||
use rand::{distributions::Alphanumeric, Rng};
|
|
||||||
rand::thread_rng().sample(Alphanumeric) as char
|
|
||||||
}).collect();
|
|
||||||
|
|
||||||
// Expiration: 30 days if remember_me is true, else 1 day
|
|
||||||
let expires_in = if payload.remember_me {
|
|
||||||
60 * 60 * 24 * 30
|
|
||||||
} else {
|
|
||||||
60 * 60 * 24
|
|
||||||
};
|
|
||||||
|
|
||||||
let expires_at = time::OffsetDateTime::now_utc().unix_timestamp() + expires_in;
|
|
||||||
|
|
||||||
if let Err(e) = state.db.create_session(user_id, &token, expires_at).await {
|
|
||||||
tracing::error!("Failed to create session for {}: {}", payload.username, e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Failed to create session").into_response();
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut cookie = Cookie::build(("auth_token", token))
|
|
||||||
.path("/")
|
|
||||||
.http_only(true)
|
|
||||||
.same_site(SameSite::Lax)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
cookie.set_max_age(Duration::seconds(expires_in));
|
|
||||||
|
|
||||||
tracing::info!("Session created and cookie set for user: {}", payload.username);
|
|
||||||
(StatusCode::OK, jar.add(cookie), Json(UserResponse { username: payload.username })).into_response()
|
|
||||||
}
|
|
||||||
Ok(false) => {
|
|
||||||
tracing::warn!("Login failed: Invalid password for {}", payload.username);
|
|
||||||
(StatusCode::UNAUTHORIZED, "Invalid credentials").into_response()
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("Bcrypt error for {}: {}", payload.username, e);
|
|
||||||
(StatusCode::INTERNAL_SERVER_ERROR, "Auth error").into_response()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[utoipa::path(
|
|
||||||
post,
|
|
||||||
path = "/api/auth/logout",
|
|
||||||
responses(
|
|
||||||
(status = 200, description = "Logged out")
|
|
||||||
)
|
|
||||||
)]
|
|
||||||
pub async fn logout_handler(
|
|
||||||
State(state): State<AppState>,
|
|
||||||
jar: CookieJar,
|
|
||||||
) -> impl IntoResponse {
|
|
||||||
if let Some(token) = jar.get("auth_token") {
|
|
||||||
let _ = state.db.delete_session(token.value()).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
let cookie = Cookie::build(("auth_token", ""))
|
|
||||||
.path("/")
|
|
||||||
.http_only(true)
|
|
||||||
.max_age(Duration::seconds(-1)) // Expire immediately
|
|
||||||
.build();
|
|
||||||
|
|
||||||
(StatusCode::OK, jar.add(cookie), "Logged out").into_response()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[utoipa::path(
|
|
||||||
get,
|
|
||||||
path = "/api/auth/check",
|
|
||||||
responses(
|
|
||||||
(status = 200, description = "Authenticated", body = UserResponse),
|
|
||||||
(status = 401, description = "Not authenticated")
|
|
||||||
)
|
|
||||||
)]
|
|
||||||
pub async fn check_auth_handler(
|
|
||||||
State(state): State<AppState>,
|
|
||||||
jar: CookieJar,
|
|
||||||
) -> impl IntoResponse {
|
|
||||||
if let Some(token) = jar.get("auth_token") {
|
|
||||||
match state.db.get_session_user(token.value()).await {
|
|
||||||
Ok(Some(user_id)) => {
|
|
||||||
// Fetch username
|
|
||||||
// We need a helper in db.rs to get username by id, or we can use a direct query here if we don't want to change db.rs interface yet.
|
|
||||||
// But better to add `get_username_by_id` to db.rs
|
|
||||||
// For now let's query directly or via a new db method.
|
|
||||||
if let Ok(Some(username)) = state.db.get_username_by_id(user_id).await {
|
|
||||||
return (StatusCode::OK, Json(UserResponse { username })).into_response();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
_ => {} // Invalid session
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
StatusCode::UNAUTHORIZED.into_response()
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ use rust_embed::RustEmbed;
|
|||||||
|
|
||||||
pub mod auth;
|
pub mod auth;
|
||||||
pub mod setup;
|
pub mod setup;
|
||||||
|
pub mod notifications;
|
||||||
|
|
||||||
#[derive(RustEmbed)]
|
#[derive(RustEmbed)]
|
||||||
#[folder = "../frontend/dist"]
|
#[folder = "../frontend/dist"]
|
||||||
|
|||||||
54
backend/src/handlers/notifications.rs
Normal file
54
backend/src/handlers/notifications.rs
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
use axum::{
|
||||||
|
extract::{State, Query},
|
||||||
|
http::StatusCode,
|
||||||
|
};
|
||||||
|
use serde::Deserialize;
|
||||||
|
use shared::{AppEvent, SystemNotification, NotificationLevel};
|
||||||
|
use crate::AppState;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
pub struct TorrentFinishedQuery {
|
||||||
|
pub name: String,
|
||||||
|
pub hash: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn torrent_finished_handler(
|
||||||
|
State(state): State<AppState>,
|
||||||
|
Query(params): Query<TorrentFinishedQuery>,
|
||||||
|
) -> StatusCode {
|
||||||
|
tracing::info!("WEBHOOK: Received notification from rTorrent. Name: {:?}, Hash: {:?}", params.name, params.hash);
|
||||||
|
|
||||||
|
let torrent_name = if params.name.is_empty() || params.name == "$d.name=" {
|
||||||
|
"Bilinmeyen Torrent".to_string()
|
||||||
|
} else {
|
||||||
|
params.name.clone()
|
||||||
|
};
|
||||||
|
|
||||||
|
let message = format!("Torrent tamamlandı: {}", torrent_name);
|
||||||
|
|
||||||
|
// 1. Send to active SSE clients (for Toast)
|
||||||
|
let notification = SystemNotification {
|
||||||
|
level: NotificationLevel::Success,
|
||||||
|
message: message.clone(),
|
||||||
|
};
|
||||||
|
let _ = state.event_bus.send(AppEvent::Notification(notification));
|
||||||
|
|
||||||
|
// 2. Send Web Push Notification (for Background)
|
||||||
|
#[cfg(feature = "push-notifications")]
|
||||||
|
{
|
||||||
|
let push_store = state.push_store.clone();
|
||||||
|
let title = "Torrent Tamamlandı".to_string();
|
||||||
|
let body = message;
|
||||||
|
let name_for_log = torrent_name.clone();
|
||||||
|
|
||||||
|
tokio::spawn(async move {
|
||||||
|
tracing::info!("Attempting to send Web Push notification for torrent: {}", name_for_log);
|
||||||
|
match crate::push::send_push_notification(&push_store, &title, &body).await {
|
||||||
|
Ok(_) => tracing::info!("Web Push notification task completed for: {}", name_for_log),
|
||||||
|
Err(e) => tracing::error!("Failed to send Web Push notification for {}: {:?}", name_for_log, e),
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
StatusCode::OK
|
||||||
|
}
|
||||||
@@ -1,125 +1,2 @@
|
|||||||
use crate::AppState;
|
// This file is intentionally empty as setup is now handled by Server Functions.
|
||||||
use axum::{
|
// See shared/src/server_fns/auth.rs
|
||||||
extract::{State, Json},
|
|
||||||
http::StatusCode,
|
|
||||||
response::IntoResponse,
|
|
||||||
};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use utoipa::ToSchema;
|
|
||||||
use axum_extra::extract::cookie::{Cookie, CookieJar, SameSite};
|
|
||||||
use time::Duration;
|
|
||||||
|
|
||||||
#[derive(Deserialize, ToSchema)]
|
|
||||||
pub struct SetupRequest {
|
|
||||||
username: String,
|
|
||||||
password: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, ToSchema)]
|
|
||||||
pub struct SetupStatusResponse {
|
|
||||||
completed: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[utoipa::path(
|
|
||||||
get,
|
|
||||||
path = "/api/setup/status",
|
|
||||||
responses(
|
|
||||||
(status = 200, description = "Setup status", body = SetupStatusResponse)
|
|
||||||
)
|
|
||||||
)]
|
|
||||||
pub async fn get_setup_status_handler(State(state): State<AppState>) -> impl IntoResponse {
|
|
||||||
let completed = match state.db.has_users().await {
|
|
||||||
Ok(has) => has,
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("DB error checking users: {}", e);
|
|
||||||
false
|
|
||||||
}
|
|
||||||
};
|
|
||||||
Json(SetupStatusResponse { completed }).into_response()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[utoipa::path(
|
|
||||||
post,
|
|
||||||
path = "/api/setup",
|
|
||||||
request_body = SetupRequest,
|
|
||||||
responses(
|
|
||||||
(status = 200, description = "Setup completed and logged in"),
|
|
||||||
(status = 400, description = "Invalid request"),
|
|
||||||
(status = 403, description = "Setup already completed"),
|
|
||||||
(status = 500, description = "Internal server error")
|
|
||||||
)
|
|
||||||
)]
|
|
||||||
pub async fn setup_handler(
|
|
||||||
State(state): State<AppState>,
|
|
||||||
jar: CookieJar,
|
|
||||||
Json(payload): Json<SetupRequest>,
|
|
||||||
) -> impl IntoResponse {
|
|
||||||
// 1. Check if setup is already completed (i.e., users exist)
|
|
||||||
match state.db.has_users().await {
|
|
||||||
Ok(true) => return (StatusCode::FORBIDDEN, "Setup already completed").into_response(),
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("DB error checking users: {}", e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Database error").into_response();
|
|
||||||
}
|
|
||||||
Ok(false) => {} // Proceed
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. Validate input
|
|
||||||
if payload.username.len() < 3 || payload.password.len() < 6 {
|
|
||||||
return (StatusCode::BAD_REQUEST, "Username must be at least 3 chars, password at least 6").into_response();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 3. Create User
|
|
||||||
// Lower cost for faster login on low-power devices (MIPS routers etc.)
|
|
||||||
let password_hash = match bcrypt::hash(&payload.password, 6) {
|
|
||||||
Ok(h) => h,
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("Failed to hash password: {}", e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Failed to process password").into_response();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Err(e) = state.db.create_user(&payload.username, &password_hash).await {
|
|
||||||
tracing::error!("Failed to create user: {}", e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Failed to create user").into_response();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. Auto-Login (Create Session)
|
|
||||||
// Get the created user's ID
|
|
||||||
let user = match state.db.get_user_by_username(&payload.username).await {
|
|
||||||
Ok(Some(u)) => u,
|
|
||||||
Ok(None) => return (StatusCode::INTERNAL_SERVER_ERROR, "User created but not found").into_response(),
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("DB error fetching new user: {}", e);
|
|
||||||
return (StatusCode::INTERNAL_SERVER_ERROR, "Database error").into_response();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
let (user_id, _) = user;
|
|
||||||
|
|
||||||
// Create session token
|
|
||||||
let token: String = (0..32).map(|_| {
|
|
||||||
use rand::{distributions::Alphanumeric, Rng};
|
|
||||||
rand::thread_rng().sample(Alphanumeric) as char
|
|
||||||
}).collect();
|
|
||||||
|
|
||||||
// Default expiration: 1 day (since it's not "remember me")
|
|
||||||
let expires_in = 60 * 60 * 24;
|
|
||||||
let expires_at = time::OffsetDateTime::now_utc().unix_timestamp() + expires_in;
|
|
||||||
|
|
||||||
if let Err(e) = state.db.create_session(user_id, &token, expires_at).await {
|
|
||||||
tracing::error!("Failed to create session for new user: {}", e);
|
|
||||||
// Even if session fails, setup is technically complete, but login failed.
|
|
||||||
// We return OK but user will have to login manually.
|
|
||||||
return (StatusCode::OK, "Setup completed, please login").into_response();
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut cookie = Cookie::build(("auth_token", token))
|
|
||||||
.path("/")
|
|
||||||
.http_only(true)
|
|
||||||
.same_site(SameSite::Lax)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
cookie.set_max_age(Duration::seconds(expires_in));
|
|
||||||
|
|
||||||
(StatusCode::OK, jar.add(cookie), "Setup completed and logged in").into_response()
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ use std::sync::Arc;
|
|||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use tokio::sync::{broadcast, watch};
|
use tokio::sync::{broadcast, watch};
|
||||||
use tower::ServiceBuilder;
|
use tower::ServiceBuilder;
|
||||||
use tower_governor::GovernorLayer;
|
|
||||||
use tower_http::{
|
use tower_http::{
|
||||||
compression::{CompressionLayer, CompressionLevel},
|
compression::{CompressionLayer, CompressionLevel},
|
||||||
cors::CorsLayer,
|
cors::CorsLayer,
|
||||||
@@ -48,23 +47,28 @@ pub struct AppState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn auth_middleware(
|
async fn auth_middleware(
|
||||||
state: axum::extract::State<AppState>,
|
_state: axum::extract::State<AppState>,
|
||||||
jar: CookieJar,
|
jar: CookieJar,
|
||||||
request: Request<Body>,
|
request: Request<Body>,
|
||||||
next: Next,
|
next: Next,
|
||||||
) -> Result<Response, StatusCode> {
|
) -> Result<Response, StatusCode> {
|
||||||
// Skip auth for public paths
|
// Skip auth for public server functions
|
||||||
let path = request.uri().path();
|
let path = request.uri().path();
|
||||||
if path.starts_with("/api/server_fns/Login") // Login server fn
|
if path.starts_with("/api/server_fns/Login")
|
||||||
|
|| path.starts_with("/api/server_fns/login")
|
||||||
|| path.starts_with("/api/server_fns/GetSetupStatus")
|
|| path.starts_with("/api/server_fns/GetSetupStatus")
|
||||||
|
|| path.starts_with("/api/server_fns/get_setup_status")
|
||||||
|| path.starts_with("/api/server_fns/Setup")
|
|| path.starts_with("/api/server_fns/Setup")
|
||||||
|
|| path.starts_with("/api/server_fns/setup")
|
||||||
|
|| path.starts_with("/api/internal/")
|
||||||
|| path.starts_with("/swagger-ui")
|
|| path.starts_with("/swagger-ui")
|
||||||
|| path.starts_with("/api-docs")
|
|| path.starts_with("/api-docs")
|
||||||
|| !path.starts_with("/api/") // Allow static files (frontend)
|
|| !path.starts_with("/api/")
|
||||||
{
|
{
|
||||||
return Ok(next.run(request).await);
|
return Ok(next.run(request).await);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Check token
|
// Check token
|
||||||
if let Some(token) = jar.get("auth_token") {
|
if let Some(token) = jar.get("auth_token") {
|
||||||
use jsonwebtoken::{decode, Validation, DecodingKey};
|
use jsonwebtoken::{decode, Validation, DecodingKey};
|
||||||
@@ -113,13 +117,6 @@ struct Args {
|
|||||||
#[cfg(feature = "swagger")]
|
#[cfg(feature = "swagger")]
|
||||||
#[derive(OpenApi)]
|
#[derive(OpenApi)]
|
||||||
#[openapi(
|
#[openapi(
|
||||||
paths(
|
|
||||||
handlers::auth::login_handler,
|
|
||||||
handlers::auth::logout_handler,
|
|
||||||
handlers::auth::check_auth_handler,
|
|
||||||
handlers::setup::setup_handler,
|
|
||||||
handlers::setup::get_setup_status_handler
|
|
||||||
),
|
|
||||||
components(
|
components(
|
||||||
schemas(
|
schemas(
|
||||||
shared::AddTorrentRequest,
|
shared::AddTorrentRequest,
|
||||||
@@ -132,10 +129,6 @@ struct Args {
|
|||||||
shared::SetFilePriorityRequest,
|
shared::SetFilePriorityRequest,
|
||||||
shared::SetLabelRequest,
|
shared::SetLabelRequest,
|
||||||
shared::GlobalLimitRequest,
|
shared::GlobalLimitRequest,
|
||||||
handlers::auth::LoginRequest,
|
|
||||||
handlers::setup::SetupRequest,
|
|
||||||
handlers::setup::SetupStatusResponse,
|
|
||||||
handlers::auth::UserResponse
|
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
tags(
|
tags(
|
||||||
@@ -144,6 +137,7 @@ struct Args {
|
|||||||
)]
|
)]
|
||||||
struct ApiDoc;
|
struct ApiDoc;
|
||||||
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
// Load .env file
|
// Load .env file
|
||||||
@@ -232,17 +226,31 @@ async fn main() {
|
|||||||
tracing::info!("Socket: {}", args.socket);
|
tracing::info!("Socket: {}", args.socket);
|
||||||
tracing::info!("Port: {}", args.port);
|
tracing::info!("Port: {}", args.port);
|
||||||
|
|
||||||
|
// Force linking of server functions from shared crate for registration on Mac
|
||||||
|
{
|
||||||
|
use shared::server_fns::auth::*;
|
||||||
|
let _ = get_setup_status;
|
||||||
|
let _ = setup;
|
||||||
|
let _ = login;
|
||||||
|
let _ = logout;
|
||||||
|
let _ = get_user;
|
||||||
|
tracing::info!("Server functions linked successfully.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// ... rest of the main function ...
|
// ... rest of the main function ...
|
||||||
// Startup Health Check
|
// Startup Health Check
|
||||||
let socket_path = std::path::Path::new(&args.socket);
|
let socket_path = std::path::Path::new(&args.socket);
|
||||||
if !socket_path.exists() {
|
if !socket_path.exists() {
|
||||||
tracing::error!("CRITICAL: rTorrent socket not found at {:?}.", socket_path);
|
tracing::error!("CRITICAL: rTorrent socket not found at {:?}.", socket_path);
|
||||||
tracing::warn!(
|
tracing::error!(
|
||||||
"HINT: Make sure rTorrent is running and the SCGI socket is enabled in .rtorrent.rc"
|
"HINT: Make sure rTorrent is running and the SCGI socket is enabled in .rtorrent.rc"
|
||||||
);
|
);
|
||||||
tracing::warn!(
|
tracing::error!(
|
||||||
"HINT: You can configure the socket path via --socket ARG or RTORRENT_SOCKET ENV."
|
"HINT: You can configure the socket path via --socket ARG or RTORRENT_SOCKET ENV."
|
||||||
);
|
);
|
||||||
|
tracing::error!("FATAL: VibeTorrent cannot start without a running rTorrent instance. Exiting.");
|
||||||
|
std::process::exit(1);
|
||||||
} else {
|
} else {
|
||||||
tracing::info!("Socket file exists. Testing connection...");
|
tracing::info!("Socket file exists. Testing connection...");
|
||||||
let client = xmlrpc::RtorrentClient::new(&args.socket);
|
let client = xmlrpc::RtorrentClient::new(&args.socket);
|
||||||
@@ -253,7 +261,11 @@ async fn main() {
|
|||||||
let version = xmlrpc::parse_string_response(&xml).unwrap_or(xml);
|
let version = xmlrpc::parse_string_response(&xml).unwrap_or(xml);
|
||||||
tracing::info!("Connected to rTorrent successfully. Version: {}", version);
|
tracing::info!("Connected to rTorrent successfully. Version: {}", version);
|
||||||
}
|
}
|
||||||
Err(e) => tracing::error!("Socket exists but failed to connect to rTorrent: {}", e),
|
Err(e) => {
|
||||||
|
tracing::error!("CRITICAL: Socket exists but failed to connect to rTorrent: {}", e);
|
||||||
|
tracing::error!("FATAL: Ensure rTorrent is fully started and the socket has correct permissions. Exiting.");
|
||||||
|
std::process::exit(1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -308,7 +320,7 @@ async fn main() {
|
|||||||
let loop_interval = if active_clients > 0 {
|
let loop_interval = if active_clients > 0 {
|
||||||
Duration::from_secs(1)
|
Duration::from_secs(1)
|
||||||
} else {
|
} else {
|
||||||
Duration::from_secs(30)
|
Duration::from_secs(60)
|
||||||
};
|
};
|
||||||
|
|
||||||
// 1. Fetch Torrents
|
// 1. Fetch Torrents
|
||||||
@@ -423,11 +435,13 @@ async fn main() {
|
|||||||
#[cfg(feature = "swagger")]
|
#[cfg(feature = "swagger")]
|
||||||
let app = app.merge(SwaggerUi::new("/swagger-ui").url("/api-docs/openapi.json", ApiDoc::openapi()));
|
let app = app.merge(SwaggerUi::new("/swagger-ui").url("/api-docs/openapi.json", ApiDoc::openapi()));
|
||||||
|
|
||||||
|
// Setup & Auth Routes (cookie-based, stay as REST)
|
||||||
// Setup & Auth Routes (cookie-based, stay as REST)
|
// Setup & Auth Routes (cookie-based, stay as REST)
|
||||||
let scgi_path_for_ctx = args.socket.clone();
|
let scgi_path_for_ctx = args.socket.clone();
|
||||||
let db_for_ctx = db.clone();
|
let db_for_ctx = db.clone();
|
||||||
let app = app
|
let app = app
|
||||||
.route("/api/events", get(sse::sse_handler))
|
.route("/api/events", get(sse::sse_handler))
|
||||||
|
.route("/api/internal/torrent-finished", post(handlers::notifications::torrent_finished_handler))
|
||||||
.route("/api/server_fns/{*fn_name}", post({
|
.route("/api/server_fns/{*fn_name}", post({
|
||||||
let scgi_path = scgi_path_for_ctx.clone();
|
let scgi_path = scgi_path_for_ctx.clone();
|
||||||
let db = db_for_ctx.clone();
|
let db = db_for_ctx.clone();
|
||||||
|
|||||||
@@ -191,11 +191,21 @@ pub async fn send_push_notification(
|
|||||||
tracing::debug!("Push notification sent to: {}", subscription.endpoint);
|
tracing::debug!("Push notification sent to: {}", subscription.endpoint);
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
tracing::error!("Failed to send push notification to {}: {}", subscription.endpoint, e);
|
let err_msg = format!("{:?}", e);
|
||||||
|
tracing::error!("Delivery failed for {}: {}", subscription.endpoint, err_msg);
|
||||||
|
// Always remove on delivery failure (Gone, Unauthorized, etc.)
|
||||||
|
tracing::info!("Removing problematic subscription after delivery failure: {}", subscription.endpoint);
|
||||||
|
let _ = store.remove_subscription(&subscription.endpoint).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(e) => tracing::error!("Failed to build push message: {}", e),
|
Err(e) => {
|
||||||
|
let err_msg = format!("{:?}", e);
|
||||||
|
tracing::error!("Encryption/Build failed for {}: {}", subscription.endpoint, err_msg);
|
||||||
|
// Always remove on encryption failure
|
||||||
|
tracing::info!("Removing problematic subscription after encryption failure: {}", subscription.endpoint);
|
||||||
|
let _ = store.remove_subscription(&subscription.endpoint).await;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(e) => tracing::error!("Failed to build VAPID signature: {}", e),
|
Err(e) => tracing::error!("Failed to build VAPID signature: {}", e),
|
||||||
|
|||||||
@@ -1,16 +1,3 @@
|
|||||||
use governor::clock::QuantaInstant;
|
// This file can be removed or repurposed if rate limiting is needed for other endpoints.
|
||||||
use governor::middleware::NoOpMiddleware;
|
// Login rate limiting is now handled within the server function or needs to be reimplemented
|
||||||
use tower_governor::governor::GovernorConfig;
|
// as a middleware for the server function endpoint.
|
||||||
use tower_governor::governor::GovernorConfigBuilder;
|
|
||||||
use tower_governor::key_extractor::SmartIpKeyExtractor;
|
|
||||||
|
|
||||||
pub fn get_login_rate_limit_config() -> GovernorConfig<SmartIpKeyExtractor, NoOpMiddleware<QuantaInstant>> {
|
|
||||||
// 5 yanlış denemeden sonra bloklanır.
|
|
||||||
// Her yeni hak için 60 saniye (1 dakika) bekleme süresi.
|
|
||||||
GovernorConfigBuilder::default()
|
|
||||||
.key_extractor(SmartIpKeyExtractor)
|
|
||||||
.per_second(60)
|
|
||||||
.burst_size(5)
|
|
||||||
.finish()
|
|
||||||
.unwrap()
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ use shared::xmlrpc::{
|
|||||||
use crate::AppState;
|
use crate::AppState;
|
||||||
use axum::extract::State;
|
use axum::extract::State;
|
||||||
use axum::response::sse::{Event, Sse};
|
use axum::response::sse::{Event, Sse};
|
||||||
use futures::stream::{self, Stream};
|
use futures::stream::{self};
|
||||||
use shared::{AppEvent, GlobalStats, Torrent, TorrentStatus};
|
use shared::{AppEvent, GlobalStats, Torrent, TorrentStatus};
|
||||||
use std::convert::Infallible;
|
use std::convert::Infallible;
|
||||||
use tokio_stream::StreamExt;
|
use tokio_stream::StreamExt;
|
||||||
@@ -51,6 +51,21 @@ mod fields {
|
|||||||
|
|
||||||
pub const IDX_LABEL: usize = 12;
|
pub const IDX_LABEL: usize = 12;
|
||||||
pub const CMD_LABEL: &str = "d.custom1=";
|
pub const CMD_LABEL: &str = "d.custom1=";
|
||||||
|
|
||||||
|
pub const IDX_RATIO: usize = 13;
|
||||||
|
pub const CMD_RATIO: &str = "d.ratio=";
|
||||||
|
|
||||||
|
pub const IDX_UPLOADED: usize = 14;
|
||||||
|
pub const CMD_UPLOADED: &str = "d.up.total=";
|
||||||
|
|
||||||
|
pub const IDX_WASTED: usize = 15;
|
||||||
|
pub const CMD_WASTED: &str = "d.skip.total=";
|
||||||
|
|
||||||
|
pub const IDX_SAVE_PATH: usize = 16;
|
||||||
|
pub const CMD_SAVE_PATH: &str = "d.base_path=";
|
||||||
|
|
||||||
|
pub const IDX_FREE_DISK: usize = 17;
|
||||||
|
pub const CMD_FREE_DISK: &str = "d.free_diskspace=";
|
||||||
}
|
}
|
||||||
|
|
||||||
use fields::*;
|
use fields::*;
|
||||||
@@ -72,6 +87,11 @@ const RTORRENT_FIELDS: &[&str] = &[
|
|||||||
CMD_CREATION_DATE,
|
CMD_CREATION_DATE,
|
||||||
CMD_HASHING,
|
CMD_HASHING,
|
||||||
CMD_LABEL,
|
CMD_LABEL,
|
||||||
|
CMD_RATIO,
|
||||||
|
CMD_UPLOADED,
|
||||||
|
CMD_WASTED,
|
||||||
|
CMD_SAVE_PATH,
|
||||||
|
CMD_FREE_DISK,
|
||||||
];
|
];
|
||||||
|
|
||||||
fn parse_long(s: Option<&String>) -> i64 {
|
fn parse_long(s: Option<&String>) -> i64 {
|
||||||
@@ -98,6 +118,11 @@ fn from_rtorrent_row(row: Vec<String>) -> Torrent {
|
|||||||
let added_date = parse_long(row.get(IDX_CREATION_DATE));
|
let added_date = parse_long(row.get(IDX_CREATION_DATE));
|
||||||
let is_hashing = parse_long(row.get(IDX_HASHING));
|
let is_hashing = parse_long(row.get(IDX_HASHING));
|
||||||
let label_raw = parse_string(row.get(IDX_LABEL));
|
let label_raw = parse_string(row.get(IDX_LABEL));
|
||||||
|
let ratio = parse_long(row.get(IDX_RATIO)) as f64 / 1000.0;
|
||||||
|
let uploaded = parse_long(row.get(IDX_UPLOADED));
|
||||||
|
let wasted = parse_long(row.get(IDX_WASTED));
|
||||||
|
let save_path = parse_string(row.get(IDX_SAVE_PATH));
|
||||||
|
let free_disk_space = parse_long(row.get(IDX_FREE_DISK));
|
||||||
|
|
||||||
let label = if label_raw.is_empty() {
|
let label = if label_raw.is_empty() {
|
||||||
None
|
None
|
||||||
@@ -144,6 +169,11 @@ fn from_rtorrent_row(row: Vec<String>) -> Torrent {
|
|||||||
error_message: message,
|
error_message: message,
|
||||||
added_date,
|
added_date,
|
||||||
label,
|
label,
|
||||||
|
ratio,
|
||||||
|
uploaded,
|
||||||
|
wasted,
|
||||||
|
save_path,
|
||||||
|
free_disk_space,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ edition = "2021"
|
|||||||
crate-type = ["cdylib", "rlib"]
|
crate-type = ["cdylib", "rlib"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
leptos = { version = "0.8.15", features = ["csr", "msgpack"] }
|
leptos = { version = "0.8.15", features = ["csr", "msgpack", "nightly"] }
|
||||||
leptos_router = { version = "0.8.11" }
|
leptos_router = { version = "0.8.11" }
|
||||||
|
|
||||||
console_error_panic_hook = "0.1"
|
console_error_panic_hook = "0.1"
|
||||||
@@ -17,6 +17,7 @@ serde = { version = "1", features = ["derive"] }
|
|||||||
serde_json = "1"
|
serde_json = "1"
|
||||||
gloo-net = "0.6"
|
gloo-net = "0.6"
|
||||||
gloo-timers = { version = "0.3", features = ["futures"] }
|
gloo-timers = { version = "0.3", features = ["futures"] }
|
||||||
|
gloo-console = "0.3"
|
||||||
wasm-bindgen = "0.2"
|
wasm-bindgen = "0.2"
|
||||||
wasm-bindgen-futures = "0.4"
|
wasm-bindgen-futures = "0.4"
|
||||||
uuid = { version = "1", features = ["v4", "js"] }
|
uuid = { version = "1", features = ["v4", "js"] }
|
||||||
@@ -34,22 +35,11 @@ thiserror = "2.0"
|
|||||||
rmp-serde = "1.3"
|
rmp-serde = "1.3"
|
||||||
struct-patch = "0.5"
|
struct-patch = "0.5"
|
||||||
|
|
||||||
# ShadCN UI Components (Individual)
|
# Rust/UI Components
|
||||||
leptos-shadcn-button = "0.8"
|
leptos_ui = "0.3"
|
||||||
leptos-shadcn-input = "0.8"
|
tw_merge = "0.1"
|
||||||
leptos-shadcn-card = "0.8"
|
strum = { version = "0.26", features = ["derive"] }
|
||||||
leptos-shadcn-badge = "0.8"
|
icons = { version = "0.18.0", features = ["leptos"] }
|
||||||
leptos-shadcn-context-menu = "0.8"
|
|
||||||
leptos-shadcn-separator = "0.8"
|
[package.metadata.leptos]
|
||||||
leptos-shadcn-progress = "0.8"
|
tailwind-input-file = "input.css"
|
||||||
leptos-shadcn-avatar = "0.8"
|
|
||||||
leptos-shadcn-sheet = "0.8"
|
|
||||||
leptos-shadcn-tabs = "0.8"
|
|
||||||
leptos-shadcn-scroll-area = "0.8"
|
|
||||||
leptos-shadcn-dialog = "0.8"
|
|
||||||
leptos-shadcn-label = "0.8"
|
|
||||||
leptos-shadcn-alert = "0.8"
|
|
||||||
leptos-shadcn-toast = "0.8"
|
|
||||||
leptos-shadcn-dropdown-menu = "0.8"
|
|
||||||
leptos-shadcn-tooltip = "0.8"
|
|
||||||
leptos-shadcn-skeleton = "0.8"
|
|
||||||
|
|||||||
@@ -7,6 +7,11 @@ stage = "build"
|
|||||||
command = "sh"
|
command = "sh"
|
||||||
command_arguments = ["-c", "npx @tailwindcss/cli -i input.css -o public/tailwind.css"]
|
command_arguments = ["-c", "npx @tailwindcss/cli -i input.css -o public/tailwind.css"]
|
||||||
|
|
||||||
|
[[hooks]]
|
||||||
|
stage = "post_build"
|
||||||
|
command = "sh"
|
||||||
|
command_arguments = ["-c", "sed -i '' -e 's/<link rel=\"modulepreload\"[^>]*>//g' -e 's/<link rel=\"preload\"[^>]*>//g' \"$TRUNK_STAGING_DIR/index.html\" || sed -i -e 's/<link rel=\"modulepreload\"[^>]*>//g' -e 's/<link rel=\"preload\"[^>]*>//g' \"$TRUNK_STAGING_DIR/index.html\""]
|
||||||
|
|
||||||
[build]
|
[build]
|
||||||
target = "index.html"
|
target = "index.html"
|
||||||
dist = "dist"
|
dist = "dist"
|
||||||
|
|||||||
@@ -25,6 +25,7 @@
|
|||||||
<link data-trunk rel="copy-file" href="manifest.json" />
|
<link data-trunk rel="copy-file" href="manifest.json" />
|
||||||
<link data-trunk rel="copy-file" href="icon-192.png" />
|
<link data-trunk rel="copy-file" href="icon-192.png" />
|
||||||
<link data-trunk rel="copy-file" href="icon-512.png" />
|
<link data-trunk rel="copy-file" href="icon-512.png" />
|
||||||
|
<link data-trunk rel="copy-file" href="public/lock_scroll.js" />
|
||||||
<link data-trunk rel="copy-file" href="sw.js" />
|
<link data-trunk rel="copy-file" href="sw.js" />
|
||||||
<script>
|
<script>
|
||||||
(function () {
|
(function () {
|
||||||
|
|||||||
@@ -1,177 +1,99 @@
|
|||||||
@import "tailwindcss";
|
@import "tailwindcss";
|
||||||
@config "./tailwind.config.js";
|
@import "tw-animate-css";
|
||||||
@source "../src/**/*.rs";
|
|
||||||
@source "/Users/bilal/.cargo/registry/src/index.crates.io-1949cf8c6b5b557f/leptos-shadcn-*/src/**/*.rs";
|
|
||||||
|
|
||||||
@theme {
|
|
||||||
/* Shadcn Colors */
|
|
||||||
--color-border: hsl(var(--border));
|
|
||||||
--color-input: hsl(var(--input));
|
|
||||||
--color-ring: hsl(var(--ring));
|
|
||||||
--color-background: hsl(var(--background));
|
|
||||||
--color-foreground: hsl(var(--foreground));
|
|
||||||
|
|
||||||
--color-primary: hsl(var(--primary));
|
:root {
|
||||||
--color-primary-foreground: hsl(var(--primary-foreground));
|
--radius: 0.625rem;
|
||||||
|
--background: oklch(1 0 0);
|
||||||
|
--foreground: oklch(0.145 0 0);
|
||||||
|
--card: oklch(1 0 0);
|
||||||
|
--card-foreground: oklch(0.145 0 0);
|
||||||
|
--popover: oklch(1 0 0);
|
||||||
|
--popover-foreground: oklch(0.145 0 0);
|
||||||
|
--primary: oklch(0.205 0 0);
|
||||||
|
--primary-foreground: oklch(0.985 0 0);
|
||||||
|
--secondary: oklch(0.97 0 0);
|
||||||
|
--secondary-foreground: oklch(0.205 0 0);
|
||||||
|
--muted: oklch(0.97 0 0);
|
||||||
|
--muted-foreground: oklch(0.556 0 0);
|
||||||
|
--accent: oklch(0.97 0 0);
|
||||||
|
--accent-foreground: oklch(0.205 0 0);
|
||||||
|
--destructive: oklch(0.577 0.245 27.325);
|
||||||
|
--border: oklch(0.922 0 0);
|
||||||
|
--input: oklch(0.922 0 0);
|
||||||
|
--ring: oklch(0.708 0 0);
|
||||||
|
}
|
||||||
|
|
||||||
--color-secondary: hsl(var(--secondary));
|
.dark {
|
||||||
--color-secondary-foreground: hsl(var(--secondary-foreground));
|
--background: oklch(0.145 0 0);
|
||||||
|
--foreground: oklch(0.985 0 0);
|
||||||
|
--card: oklch(0.205 0 0);
|
||||||
|
--card-foreground: oklch(0.985 0 0);
|
||||||
|
--popover: oklch(0.205 0 0);
|
||||||
|
--popover-foreground: oklch(0.985 0 0);
|
||||||
|
--primary: oklch(0.922 0 0);
|
||||||
|
--primary-foreground: oklch(0.205 0 0);
|
||||||
|
--secondary: oklch(0.269 0 0);
|
||||||
|
--secondary-foreground: oklch(0.985 0 0);
|
||||||
|
--muted: oklch(0.269 0 0);
|
||||||
|
--muted-foreground: oklch(0.708 0 0);
|
||||||
|
--accent: oklch(0.269 0 0);
|
||||||
|
--accent-foreground: oklch(0.985 0 0);
|
||||||
|
--destructive: oklch(0.704 0.191 22.216);
|
||||||
|
--border: oklch(1 0 0 / 10%);
|
||||||
|
--input: oklch(1 0 0 / 15%);
|
||||||
|
--ring: oklch(0.556 0 0);
|
||||||
|
}
|
||||||
|
|
||||||
--color-destructive: hsl(var(--destructive));
|
@theme inline {
|
||||||
--color-destructive-foreground: hsl(var(--destructive-foreground));
|
--animate-shimmer: shimmer 2s infinite;
|
||||||
|
|
||||||
--color-muted: hsl(var(--muted));
|
@keyframes shimmer {
|
||||||
--color-muted-foreground: hsl(var(--muted-foreground));
|
100% {
|
||||||
|
transform: translateX(100%);
|
||||||
--color-accent: hsl(var(--accent));
|
|
||||||
--color-accent-foreground: hsl(var(--accent-foreground));
|
|
||||||
|
|
||||||
--color-popover: hsl(var(--popover));
|
|
||||||
--color-popover-foreground: hsl(var(--popover-foreground));
|
|
||||||
|
|
||||||
--color-card: hsl(var(--card));
|
|
||||||
--color-card-foreground: hsl(var(--card-foreground));
|
|
||||||
|
|
||||||
--radius-lg: var(--radius);
|
|
||||||
--radius-md: calc(var(--radius) - 2px);
|
|
||||||
--radius-sm: calc(var(--radius) - 4px);
|
|
||||||
|
|
||||||
--animate-accordion-down: accordion-down 0.2s ease-out;
|
|
||||||
--animate-accordion-up: accordion-up 0.2s ease-out;
|
|
||||||
|
|
||||||
@keyframes accordion-down {
|
|
||||||
from {
|
|
||||||
height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
to {
|
|
||||||
height: var(--radix-accordion-content-height);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@keyframes accordion-up {
|
--color-background: var(--background);
|
||||||
from {
|
--color-foreground: var(--foreground);
|
||||||
height: var(--radix-accordion-content-height);
|
--color-card: var(--card);
|
||||||
}
|
--color-card-foreground: var(--card-foreground);
|
||||||
|
--color-popover: var(--popover);
|
||||||
to {
|
--color-popover-foreground: var(--popover-foreground);
|
||||||
height: 0;
|
--color-primary: var(--primary);
|
||||||
}
|
--color-primary-foreground: var(--primary-foreground);
|
||||||
}
|
--color-secondary: var(--secondary);
|
||||||
|
--color-secondary-foreground: var(--secondary-foreground);
|
||||||
|
--color-muted: var(--muted);
|
||||||
|
--color-muted-foreground: var(--muted-foreground);
|
||||||
|
--color-accent: var(--accent);
|
||||||
|
--color-accent-foreground: var(--accent-foreground);
|
||||||
|
--color-destructive: var(--destructive);
|
||||||
|
--color-destructive-foreground: var(--destructive-foreground);
|
||||||
|
--color-border: var(--border);
|
||||||
|
--color-input: var(--input);
|
||||||
|
--color-ring: var(--ring);
|
||||||
|
--radius-sm: calc(var(--radius) - 4px);
|
||||||
|
--radius-md: calc(var(--radius) - 2px);
|
||||||
|
--radius-lg: var(--radius);
|
||||||
|
--radius-xl: calc(var(--radius) + 4px);
|
||||||
}
|
}
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
:root {
|
* {
|
||||||
--background: 0 0% 100%;
|
@apply border-border outline-ring/50;
|
||||||
--foreground: 222.2 84% 4.9%;
|
}
|
||||||
|
|
||||||
--card: 0 0% 100%;
|
body {
|
||||||
--card-foreground: 222.2 84% 4.9%;
|
@apply bg-background text-foreground;
|
||||||
|
}
|
||||||
|
|
||||||
--popover: 0 0% 100%;
|
button:not(:disabled),
|
||||||
--popover-foreground: 222.2 84% 4.9%;
|
[role="button"]:not(:disabled) {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
--primary: 222.2 47.4% 11.2%;
|
dialog {
|
||||||
--primary-foreground: 210 40% 98%;
|
margin: auto;
|
||||||
|
}
|
||||||
--secondary: 210 40% 96.1%;
|
|
||||||
--secondary-foreground: 222.2 47.4% 11.2%;
|
|
||||||
|
|
||||||
--muted: 210 40% 96.1%;
|
|
||||||
--muted-foreground: 215.4 16.3% 46.9%;
|
|
||||||
|
|
||||||
--accent: 210 40% 96.1%;
|
|
||||||
--accent-foreground: 222.2 47.4% 11.2%;
|
|
||||||
|
|
||||||
--destructive: 0 84.2% 60.2%;
|
|
||||||
--destructive-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--border: 214.3 31.8% 91.4%;
|
|
||||||
--input: 214.3 31.8% 91.4%;
|
|
||||||
--ring: 222.2 84% 4.9%;
|
|
||||||
|
|
||||||
--radius: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dark {
|
|
||||||
--background: 222.2 84% 4.9%;
|
|
||||||
--foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--card: 222.2 84% 4.9%;
|
|
||||||
--card-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--popover: 222.2 84% 4.9%;
|
|
||||||
--popover-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--primary: 210 40% 98%;
|
|
||||||
--primary-foreground: 222.2 47.4% 11.2%;
|
|
||||||
|
|
||||||
--secondary: 217.2 32.6% 17.5%;
|
|
||||||
--secondary-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--muted: 217.2 32.6% 17.5%;
|
|
||||||
--muted-foreground: 215 20.2% 65.1%;
|
|
||||||
|
|
||||||
--accent: 217.2 32.6% 17.5%;
|
|
||||||
--accent-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--destructive: 0 62.8% 30.6%;
|
|
||||||
--destructive-foreground: 210 40% 98%;
|
|
||||||
|
|
||||||
--border: 217.2 32.6% 17.5%;
|
|
||||||
--input: 217.2 32.6% 17.5%;
|
|
||||||
--ring: 212.7 26.8% 83.9%;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@layer base {
|
|
||||||
* {
|
|
||||||
@apply border-border;
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
|
||||||
@apply bg-background text-foreground;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Ensure Shadcn Utilities are always available */
|
|
||||||
.bg-popover {
|
|
||||||
background-color: hsl(var(--popover));
|
|
||||||
}
|
|
||||||
|
|
||||||
.text-popover-foreground {
|
|
||||||
color: hsl(var(--popover-foreground));
|
|
||||||
}
|
|
||||||
|
|
||||||
.border-border {
|
|
||||||
border-color: hsl(var(--border));
|
|
||||||
}
|
|
||||||
|
|
||||||
.shadow-md {
|
|
||||||
box-shadow: 0 4px 6px -1px rgb(0 0 0 / 0.1), 0 2px 4px -2px rgb(0 0 0 / 0.1);
|
|
||||||
}
|
|
||||||
|
|
||||||
.z-50 {
|
|
||||||
z-index: 50;
|
|
||||||
}
|
|
||||||
|
|
||||||
.z-100 {
|
|
||||||
z-index: 100;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Fix for iOS click/blur events */
|
|
||||||
@media (hover: none) {
|
|
||||||
body {
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Remove default focus outline/tap highlight */
|
|
||||||
* {
|
|
||||||
-webkit-tap-highlight-color: transparent;
|
|
||||||
outline: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
:focus {
|
|
||||||
outline: none !important;
|
|
||||||
}
|
}
|
||||||
56
frontend/package-lock.json
generated
56
frontend/package-lock.json
generated
@@ -9,7 +9,12 @@
|
|||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@tailwindcss/cli": "^4.1.18"
|
"@tailwindcss/cli": "^4.1.18",
|
||||||
|
"class-variance-authority": "^0.7.1",
|
||||||
|
"clsx": "^2.1.1",
|
||||||
|
"tailwind-merge": "^3.4.0",
|
||||||
|
"tailwindcss-animate": "^1.0.7",
|
||||||
|
"tw-animate-css": "^1.4.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@tailwindcss/postcss": "^4.1.18",
|
"@tailwindcss/postcss": "^4.1.18",
|
||||||
@@ -1958,6 +1963,18 @@
|
|||||||
"fsevents": "~2.3.2"
|
"fsevents": "~2.3.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/class-variance-authority": {
|
||||||
|
"version": "0.7.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/class-variance-authority/-/class-variance-authority-0.7.1.tgz",
|
||||||
|
"integrity": "sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"clsx": "^2.1.1"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://polar.sh/cva"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/cliui": {
|
"node_modules/cliui": {
|
||||||
"version": "8.0.1",
|
"version": "8.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz",
|
||||||
@@ -1973,6 +1990,15 @@
|
|||||||
"node": ">=12"
|
"node": ">=12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/clsx": {
|
||||||
|
"version": "2.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz",
|
||||||
|
"integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/color-convert": {
|
"node_modules/color-convert": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
|
||||||
@@ -3637,12 +3663,31 @@
|
|||||||
"node": ">=8"
|
"node": ">=8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/tailwind-merge": {
|
||||||
|
"version": "3.4.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/tailwind-merge/-/tailwind-merge-3.4.0.tgz",
|
||||||
|
"integrity": "sha512-uSaO4gnW+b3Y2aWoWfFpX62vn2sR3skfhbjsEnaBI81WD1wBLlHZe5sWf0AqjksNdYTbGBEd0UasQMT3SNV15g==",
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": {
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/dcastil"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/tailwindcss": {
|
"node_modules/tailwindcss": {
|
||||||
"version": "4.1.18",
|
"version": "4.1.18",
|
||||||
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.1.18.tgz",
|
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.1.18.tgz",
|
||||||
"integrity": "sha512-4+Z+0yiYyEtUVCScyfHCxOYP06L5Ne+JiHhY2IjR2KWMIWhJOYZKLSGZaP5HkZ8+bY0cxfzwDE5uOmzFXyIwxw==",
|
"integrity": "sha512-4+Z+0yiYyEtUVCScyfHCxOYP06L5Ne+JiHhY2IjR2KWMIWhJOYZKLSGZaP5HkZ8+bY0cxfzwDE5uOmzFXyIwxw==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/tailwindcss-animate": {
|
||||||
|
"version": "1.0.7",
|
||||||
|
"resolved": "https://registry.npmjs.org/tailwindcss-animate/-/tailwindcss-animate-1.0.7.tgz",
|
||||||
|
"integrity": "sha512-bl6mpH3T7I3UFxuvDEXLxy/VuFxBk5bbzplh7tXI68mwMokNYd1t9qPBHlnyTwfa4JGC4zP516I1hYYtQ/vspA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"peerDependencies": {
|
||||||
|
"tailwindcss": ">=3.0.0 || insiders"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/tapable": {
|
"node_modules/tapable": {
|
||||||
"version": "2.3.0",
|
"version": "2.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz",
|
||||||
@@ -3693,6 +3738,15 @@
|
|||||||
"node": ">=8.0"
|
"node": ">=8.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/tw-animate-css": {
|
||||||
|
"version": "1.4.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/tw-animate-css/-/tw-animate-css-1.4.0.tgz",
|
||||||
|
"integrity": "sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/Wombosvideo"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/universalify": {
|
"node_modules/universalify": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz",
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
{
|
{
|
||||||
"name": "frontend",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "",
|
|
||||||
"main": "tailwind.config.js",
|
|
||||||
"scripts": {
|
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
|
||||||
},
|
|
||||||
"keywords": [],
|
|
||||||
"author": "",
|
"author": "",
|
||||||
"license": "ISC",
|
"dependencies": {
|
||||||
|
"@tailwindcss/cli": "^4.1.18",
|
||||||
|
"class-variance-authority": "^0.7.1",
|
||||||
|
"clsx": "^2.1.1",
|
||||||
|
"tailwind-merge": "^3.4.0",
|
||||||
|
"tailwindcss-animate": "^1.0.7",
|
||||||
|
"tw-animate-css": "^1.4.0"
|
||||||
|
},
|
||||||
|
"description": "",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@tailwindcss/postcss": "^4.1.18",
|
"@tailwindcss/postcss": "^4.1.18",
|
||||||
"autoprefixer": "^10.4.23",
|
"autoprefixer": "^10.4.23",
|
||||||
@@ -17,7 +17,13 @@
|
|||||||
"postcss-preset-env": "^10.1.3",
|
"postcss-preset-env": "^10.1.3",
|
||||||
"tailwindcss": "^4.1.18"
|
"tailwindcss": "^4.1.18"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"keywords": [],
|
||||||
"@tailwindcss/cli": "^4.1.18"
|
"license": "ISC",
|
||||||
}
|
"main": "tailwind.config.js",
|
||||||
|
"name": "frontend",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"type": "module",
|
||||||
|
"version": "1.0.0"
|
||||||
}
|
}
|
||||||
|
|||||||
253
frontend/public/lock_scroll.js
Normal file
253
frontend/public/lock_scroll.js
Normal file
@@ -0,0 +1,253 @@
|
|||||||
|
/**
|
||||||
|
* Scroll Lock Utility
|
||||||
|
* Handles locking and unlocking scroll for both window and all scrollable containers
|
||||||
|
* Similar to react-remove-scroll but in vanilla JavaScript
|
||||||
|
*/
|
||||||
|
|
||||||
|
(() => {
|
||||||
|
// Prevent multiple initializations
|
||||||
|
if (window.ScrollLock) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScrollLock {
|
||||||
|
constructor() {
|
||||||
|
this.locked = false;
|
||||||
|
this.scrollableElements = [];
|
||||||
|
this.scrollPositions = new Map();
|
||||||
|
this.originalStyles = new Map();
|
||||||
|
this.fixedElements = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find all scrollable elements in the DOM (optimized)
|
||||||
|
* Uses more targeted selectors instead of querying all elements
|
||||||
|
*/
|
||||||
|
findScrollableElements() {
|
||||||
|
const scrollables = [];
|
||||||
|
|
||||||
|
// More targeted query - only look for elements with overflow properties
|
||||||
|
const candidates = document.querySelectorAll(
|
||||||
|
'[style*="overflow"], [class*="overflow"], [class*="scroll"], main, aside, section, div',
|
||||||
|
);
|
||||||
|
|
||||||
|
// Batch all style reads first to minimize reflows
|
||||||
|
const elementsToCheck = [];
|
||||||
|
for (const el of candidates) {
|
||||||
|
// Skip the element itself or if it's inside these containers
|
||||||
|
const dataName = el.getAttribute("data-name");
|
||||||
|
const isExcludedElement =
|
||||||
|
dataName === "ScrollArea" ||
|
||||||
|
dataName === "CommandList" ||
|
||||||
|
dataName === "SelectContent" ||
|
||||||
|
dataName === "MultiSelectContent" ||
|
||||||
|
dataName === "DropdownMenuContent" ||
|
||||||
|
dataName === "ContextMenuContent";
|
||||||
|
|
||||||
|
if (
|
||||||
|
el !== document.body &&
|
||||||
|
el !== document.documentElement &&
|
||||||
|
!isExcludedElement &&
|
||||||
|
!el.closest('[data-name="ScrollArea"]') &&
|
||||||
|
!el.closest('[data-name="CommandList"]') &&
|
||||||
|
!el.closest('[data-name="SelectContent"]') &&
|
||||||
|
!el.closest('[data-name="MultiSelectContent"]') &&
|
||||||
|
!el.closest('[data-name="DropdownMenuContent"]') &&
|
||||||
|
!el.closest('[data-name="ContextMenuContent"]')
|
||||||
|
) {
|
||||||
|
elementsToCheck.push(el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now batch read all computed styles and dimensions
|
||||||
|
elementsToCheck.forEach((el) => {
|
||||||
|
const style = window.getComputedStyle(el);
|
||||||
|
const hasOverflow =
|
||||||
|
style.overflow === "auto" ||
|
||||||
|
style.overflow === "scroll" ||
|
||||||
|
style.overflowY === "auto" ||
|
||||||
|
style.overflowY === "scroll";
|
||||||
|
|
||||||
|
// Only check scrollHeight if overflow is set
|
||||||
|
if (hasOverflow && el.scrollHeight > el.clientHeight) {
|
||||||
|
scrollables.push(el);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return scrollables;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lock scrolling on all scrollable elements (optimized)
|
||||||
|
* Batches all DOM reads before DOM writes to prevent forced reflows
|
||||||
|
*/
|
||||||
|
lock() {
|
||||||
|
if (this.locked) return;
|
||||||
|
|
||||||
|
this.locked = true;
|
||||||
|
|
||||||
|
// Find all scrollable elements
|
||||||
|
this.scrollableElements = this.findScrollableElements();
|
||||||
|
|
||||||
|
// ===== BATCH 1: READ PHASE - Read all layout properties first =====
|
||||||
|
const windowScrollY = window.scrollY;
|
||||||
|
const scrollbarWidth = window.innerWidth - document.body.clientWidth;
|
||||||
|
|
||||||
|
// Store window scroll position
|
||||||
|
this.scrollPositions.set("window", windowScrollY);
|
||||||
|
|
||||||
|
// Store original body styles
|
||||||
|
this.originalStyles.set("body", {
|
||||||
|
position: document.body.style.position,
|
||||||
|
top: document.body.style.top,
|
||||||
|
width: document.body.style.width,
|
||||||
|
overflow: document.body.style.overflow,
|
||||||
|
paddingRight: document.body.style.paddingRight,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Read all fixed-position elements and their padding (only if we have scrollbar)
|
||||||
|
if (scrollbarWidth > 0) {
|
||||||
|
// Use more targeted query for fixed elements
|
||||||
|
const fixedCandidates = document.querySelectorAll(
|
||||||
|
'[style*="fixed"], [class*="fixed"], header, nav, aside, [role="dialog"], [role="alertdialog"]',
|
||||||
|
);
|
||||||
|
|
||||||
|
this.fixedElements = Array.from(fixedCandidates).filter((el) => {
|
||||||
|
const style = window.getComputedStyle(el);
|
||||||
|
return (
|
||||||
|
style.position === "fixed" &&
|
||||||
|
!el.closest('[data-name="DropdownMenuContent"]') &&
|
||||||
|
!el.closest('[data-name="MultiSelectContent"]') &&
|
||||||
|
!el.closest('[data-name="ContextMenuContent"]')
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Batch read all padding values
|
||||||
|
this.fixedElements.forEach((el) => {
|
||||||
|
const computedStyle = window.getComputedStyle(el);
|
||||||
|
const currentPadding = Number.parseInt(computedStyle.paddingRight, 10) || 0;
|
||||||
|
|
||||||
|
this.originalStyles.set(el, {
|
||||||
|
paddingRight: el.style.paddingRight,
|
||||||
|
computedPadding: currentPadding,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read scrollable elements info
|
||||||
|
const scrollableInfo = this.scrollableElements.map((el) => {
|
||||||
|
const scrollTop = el.scrollTop;
|
||||||
|
const elementScrollbarWidth = el.offsetWidth - el.clientWidth;
|
||||||
|
const computedStyle = window.getComputedStyle(el);
|
||||||
|
const currentPadding = Number.parseInt(computedStyle.paddingRight, 10) || 0;
|
||||||
|
|
||||||
|
this.scrollPositions.set(el, scrollTop);
|
||||||
|
this.originalStyles.set(el, {
|
||||||
|
overflow: el.style.overflow,
|
||||||
|
overflowY: el.style.overflowY,
|
||||||
|
paddingRight: el.style.paddingRight,
|
||||||
|
});
|
||||||
|
|
||||||
|
return { el, elementScrollbarWidth, currentPadding };
|
||||||
|
});
|
||||||
|
|
||||||
|
// ===== BATCH 2: WRITE PHASE - Apply all styles at once =====
|
||||||
|
|
||||||
|
// Apply body lock
|
||||||
|
document.body.style.position = "fixed";
|
||||||
|
document.body.style.top = `-${windowScrollY}px`;
|
||||||
|
document.body.style.width = "100%";
|
||||||
|
document.body.style.overflow = "hidden";
|
||||||
|
|
||||||
|
if (scrollbarWidth > 0) {
|
||||||
|
document.body.style.paddingRight = `${scrollbarWidth}px`;
|
||||||
|
|
||||||
|
// Apply padding compensation to fixed elements
|
||||||
|
this.fixedElements.forEach((el) => {
|
||||||
|
const stored = this.originalStyles.get(el);
|
||||||
|
if (stored) {
|
||||||
|
el.style.paddingRight = `${stored.computedPadding + scrollbarWidth}px`;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lock all scrollable containers
|
||||||
|
scrollableInfo.forEach(({ el, elementScrollbarWidth, currentPadding }) => {
|
||||||
|
el.style.overflow = "hidden";
|
||||||
|
|
||||||
|
if (elementScrollbarWidth > 0) {
|
||||||
|
el.style.paddingRight = `${currentPadding + elementScrollbarWidth}px`;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unlock scrolling on all elements (optimized)
|
||||||
|
* @param {number} delay - Delay in milliseconds before unlocking (for animations)
|
||||||
|
*/
|
||||||
|
unlock(delay = 0) {
|
||||||
|
if (!this.locked) return;
|
||||||
|
|
||||||
|
const performUnlock = () => {
|
||||||
|
// Restore body scroll
|
||||||
|
const bodyStyles = this.originalStyles.get("body");
|
||||||
|
if (bodyStyles) {
|
||||||
|
document.body.style.position = bodyStyles.position;
|
||||||
|
document.body.style.top = bodyStyles.top;
|
||||||
|
document.body.style.width = bodyStyles.width;
|
||||||
|
document.body.style.overflow = bodyStyles.overflow;
|
||||||
|
document.body.style.paddingRight = bodyStyles.paddingRight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore window scroll position
|
||||||
|
const windowScrollY = this.scrollPositions.get("window") || 0;
|
||||||
|
window.scrollTo(0, windowScrollY);
|
||||||
|
|
||||||
|
// Restore all scrollable containers
|
||||||
|
this.scrollableElements.forEach((el) => {
|
||||||
|
const originalStyles = this.originalStyles.get(el);
|
||||||
|
if (originalStyles) {
|
||||||
|
el.style.overflow = originalStyles.overflow;
|
||||||
|
el.style.overflowY = originalStyles.overflowY;
|
||||||
|
el.style.paddingRight = originalStyles.paddingRight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore scroll position
|
||||||
|
const scrollPosition = this.scrollPositions.get(el) || 0;
|
||||||
|
el.scrollTop = scrollPosition;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Restore fixed-position elements padding
|
||||||
|
this.fixedElements.forEach((el) => {
|
||||||
|
const styles = this.originalStyles.get(el);
|
||||||
|
if (styles && styles.paddingRight !== undefined) {
|
||||||
|
el.style.paddingRight = styles.paddingRight;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Clear storage
|
||||||
|
this.scrollableElements = [];
|
||||||
|
this.fixedElements = [];
|
||||||
|
this.scrollPositions.clear();
|
||||||
|
this.originalStyles.clear();
|
||||||
|
this.locked = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (delay > 0) {
|
||||||
|
setTimeout(performUnlock, delay);
|
||||||
|
} else {
|
||||||
|
performUnlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if scrolling is currently locked
|
||||||
|
*/
|
||||||
|
isLocked() {
|
||||||
|
return this.locked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Export as singleton
|
||||||
|
window.ScrollLock = new ScrollLock();
|
||||||
|
})();
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,21 +1,39 @@
|
|||||||
use crate::components::layout::protected::Protected;
|
use crate::components::layout::protected::Protected;
|
||||||
|
use crate::components::ui::skeleton::Skeleton;
|
||||||
use crate::components::torrent::table::TorrentTable;
|
use crate::components::torrent::table::TorrentTable;
|
||||||
use crate::components::torrent::detail::TorrentDetail;
|
|
||||||
use crate::components::auth::login::Login;
|
use crate::components::auth::login::Login;
|
||||||
use crate::components::auth::setup::Setup;
|
use crate::components::auth::setup::Setup;
|
||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use leptos_router::components::{Router, Routes, Route};
|
use leptos_router::components::{Router, Routes, Route};
|
||||||
use leptos_router::hooks::use_navigate;
|
use leptos_router::hooks::{use_navigate, use_location};
|
||||||
use leptos_shadcn_skeleton::Skeleton;
|
use crate::components::ui::toast::Toaster;
|
||||||
use leptos_shadcn_toast::SonnerProvider;
|
use crate::components::hooks::use_theme_mode::ThemeMode;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn App() -> impl IntoView {
|
pub fn App() -> impl IntoView {
|
||||||
|
crate::components::ui::toast::provide_toaster();
|
||||||
|
let theme_mode = ThemeMode::init();
|
||||||
|
|
||||||
|
// Sync theme with document
|
||||||
|
Effect::new(move |_| {
|
||||||
|
let is_dark = theme_mode.get();
|
||||||
|
if let Some(doc) = document().document_element() {
|
||||||
|
if is_dark {
|
||||||
|
let _ = doc.class_list().add_1("dark");
|
||||||
|
let _ = doc.set_attribute("data-theme", "dark");
|
||||||
|
} else {
|
||||||
|
let _ = doc.class_list().remove_1("dark");
|
||||||
|
let _ = doc.set_attribute("data-theme", "light");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<SonnerProvider>
|
<Toaster />
|
||||||
|
<Router>
|
||||||
<InnerApp />
|
<InnerApp />
|
||||||
</SonnerProvider>
|
</Router>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -23,6 +41,7 @@ pub fn App() -> impl IntoView {
|
|||||||
fn InnerApp() -> impl IntoView {
|
fn InnerApp() -> impl IntoView {
|
||||||
crate::store::provide_torrent_store();
|
crate::store::provide_torrent_store();
|
||||||
let store = use_context::<crate::store::TorrentStore>();
|
let store = use_context::<crate::store::TorrentStore>();
|
||||||
|
let _loc = use_location();
|
||||||
|
|
||||||
let is_loading = signal(true);
|
let is_loading = signal(true);
|
||||||
let is_authenticated = signal(false);
|
let is_authenticated = signal(false);
|
||||||
@@ -30,13 +49,10 @@ fn InnerApp() -> impl IntoView {
|
|||||||
|
|
||||||
Effect::new(move |_| {
|
Effect::new(move |_| {
|
||||||
spawn_local(async move {
|
spawn_local(async move {
|
||||||
log::info!("App initialization started...");
|
|
||||||
|
|
||||||
// Check if setup is needed via Server Function
|
// Check if setup is needed via Server Function
|
||||||
match shared::server_fns::auth::get_setup_status().await {
|
match shared::server_fns::auth::get_setup_status().await {
|
||||||
Ok(status) => {
|
Ok(status) => {
|
||||||
if !status.completed {
|
if !status.completed {
|
||||||
log::info!("Setup not completed");
|
|
||||||
needs_setup.1.set(true);
|
needs_setup.1.set(true);
|
||||||
is_loading.1.set(false);
|
is_loading.1.set(false);
|
||||||
return;
|
return;
|
||||||
@@ -48,21 +64,19 @@ fn InnerApp() -> impl IntoView {
|
|||||||
// Check authentication via GetUser Server Function
|
// Check authentication via GetUser Server Function
|
||||||
match shared::server_fns::auth::get_user().await {
|
match shared::server_fns::auth::get_user().await {
|
||||||
Ok(Some(user_info)) => {
|
Ok(Some(user_info)) => {
|
||||||
log::info!("Authenticated as {}", user_info.username);
|
|
||||||
if let Some(s) = store {
|
if let Some(s) = store {
|
||||||
s.user.set(Some(user_info.username));
|
s.user.set(Some(user_info.username));
|
||||||
}
|
}
|
||||||
is_authenticated.1.set(true);
|
is_authenticated.1.set(true);
|
||||||
}
|
}
|
||||||
Ok(None) => {
|
Ok(None) => {}
|
||||||
log::info!("Not authenticated");
|
|
||||||
}
|
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Auth check failed: {:?}", e);
|
log::error!("Auth check failed: {:?}", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
is_loading.1.set(false);
|
is_loading.1.set(false);
|
||||||
|
crate::store::toast_success("VibeTorrent'e Hoşgeldiniz");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -80,53 +94,51 @@ fn InnerApp() -> impl IntoView {
|
|||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div class="relative w-full h-screen" style="height: 100dvh;">
|
<div class="relative w-full h-screen" style="height: 100dvh;">
|
||||||
<Router>
|
<Routes fallback=|| view! { <div class="p-4">"404 Not Found"</div> }>
|
||||||
<Routes fallback=|| view! { <div class="p-4">"404 Not Found"</div> }>
|
<Route path=leptos_router::path!("/login") view=move || {
|
||||||
<Route path=leptos_router::path!("/login") view=move || {
|
let authenticated = is_authenticated.0.get();
|
||||||
let authenticated = is_authenticated.0.get();
|
let setup_needed = needs_setup.0.get();
|
||||||
let setup_needed = needs_setup.0.get();
|
|
||||||
|
Effect::new(move |_| {
|
||||||
Effect::new(move |_| {
|
if setup_needed {
|
||||||
if setup_needed {
|
let navigate = use_navigate();
|
||||||
let navigate = use_navigate();
|
navigate("/setup", Default::default());
|
||||||
navigate("/setup", Default::default());
|
} else if authenticated {
|
||||||
} else if authenticated {
|
let navigate = use_navigate();
|
||||||
log::info!("Already authenticated, redirecting to home");
|
navigate("/", Default::default());
|
||||||
let navigate = use_navigate();
|
}
|
||||||
navigate("/", Default::default());
|
});
|
||||||
}
|
|
||||||
});
|
view! { <Login /> }
|
||||||
|
} />
|
||||||
view! { <Login /> }
|
<Route path=leptos_router::path!("/setup") view=move || {
|
||||||
} />
|
Effect::new(move |_| {
|
||||||
<Route path=leptos_router::path!("/setup") view=move || {
|
if is_authenticated.0.get() {
|
||||||
Effect::new(move |_| {
|
let navigate = use_navigate();
|
||||||
if is_authenticated.0.get() {
|
navigate("/", Default::default());
|
||||||
let navigate = use_navigate();
|
}
|
||||||
navigate("/", Default::default());
|
});
|
||||||
}
|
|
||||||
});
|
view! { <Setup /> }
|
||||||
|
} />
|
||||||
view! { <Setup /> }
|
|
||||||
} />
|
|
||||||
|
|
||||||
<Route path=leptos_router::path!("/") view=move || {
|
<Route path=leptos_router::path!("/") view=move || {
|
||||||
let navigate = use_navigate();
|
let navigate = use_navigate();
|
||||||
Effect::new(move |_| {
|
Effect::new(move |_| {
|
||||||
if !is_loading.0.get() {
|
if !is_loading.0.get() {
|
||||||
if needs_setup.0.get() {
|
if needs_setup.0.get() {
|
||||||
log::info!("Setup not completed, redirecting to setup");
|
navigate("/setup", Default::default());
|
||||||
navigate("/setup", Default::default());
|
} else if !is_authenticated.0.get() {
|
||||||
} else if !is_authenticated.0.get() {
|
navigate("/login", Default::default());
|
||||||
log::info!("Not authenticated, redirecting to login");
|
|
||||||
navigate("/login", Default::default());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
});
|
||||||
view! {
|
|
||||||
<Show when=move || !is_loading.0.get() fallback=|| view! {
|
view! {
|
||||||
<div class="flex h-screen bg-background">
|
<Show when=move || !is_loading.0.get() fallback=|| {
|
||||||
|
// Standard 1: Always show Dashboard Skeleton
|
||||||
|
view! {
|
||||||
|
<div class="flex h-screen bg-background text-foreground overflow-hidden">
|
||||||
// Sidebar skeleton
|
// Sidebar skeleton
|
||||||
<div class="w-56 border-r border-border p-4 space-y-4">
|
<div class="w-56 border-r border-border p-4 space-y-4">
|
||||||
<Skeleton class="h-8 w-3/4" />
|
<Skeleton class="h-8 w-3/4" />
|
||||||
@@ -140,14 +152,12 @@ fn InnerApp() -> impl IntoView {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
// Main content skeleton
|
// Main content skeleton
|
||||||
<div class="flex-1 flex flex-col">
|
<div class="flex-1 flex flex-col min-w-0">
|
||||||
// Header skeleton
|
|
||||||
<div class="border-b border-border p-4 flex items-center gap-4">
|
<div class="border-b border-border p-4 flex items-center gap-4">
|
||||||
<Skeleton class="h-8 w-48" />
|
<Skeleton class="h-8 w-48" />
|
||||||
<Skeleton class="h-8 w-64" />
|
<Skeleton class="h-8 w-64" />
|
||||||
<div class="ml-auto"><Skeleton class="h-8 w-24" /></div>
|
<div class="ml-auto"><Skeleton class="h-8 w-24" /></div>
|
||||||
</div>
|
</div>
|
||||||
// Table skeleton rows
|
|
||||||
<div class="flex-1 p-4 space-y-3">
|
<div class="flex-1 p-4 space-y-3">
|
||||||
<Skeleton class="h-10 w-full" />
|
<Skeleton class="h-10 w-full" />
|
||||||
<Skeleton class="h-10 w-full" />
|
<Skeleton class="h-10 w-full" />
|
||||||
@@ -156,47 +166,46 @@ fn InnerApp() -> impl IntoView {
|
|||||||
<Skeleton class="h-10 w-full" />
|
<Skeleton class="h-10 w-full" />
|
||||||
<Skeleton class="h-10 w-3/4" />
|
<Skeleton class="h-10 w-3/4" />
|
||||||
</div>
|
</div>
|
||||||
// Status bar skeleton
|
|
||||||
<div class="border-t border-border p-3">
|
<div class="border-t border-border p-3">
|
||||||
<Skeleton class="h-5 w-96" />
|
<Skeleton class="h-5 w-96" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}.into_any()>
|
}.into_any()
|
||||||
<Show when=move || is_authenticated.0.get() fallback=|| ()>
|
}>
|
||||||
<Protected>
|
<Show when=move || is_authenticated.0.get() fallback=|| ()>
|
||||||
<div class="flex flex-col h-full overflow-hidden">
|
<Protected>
|
||||||
<div class="flex-1 overflow-hidden">
|
<div class="flex flex-col h-full overflow-hidden">
|
||||||
<TorrentTable />
|
<div class="flex-1 overflow-hidden">
|
||||||
</div>
|
<TorrentTable />
|
||||||
<TorrentDetail />
|
|
||||||
</div>
|
</div>
|
||||||
</Protected>
|
</div>
|
||||||
</Show>
|
</Protected>
|
||||||
</Show>
|
</Show>
|
||||||
}.into_any()
|
</Show>
|
||||||
}/>
|
}.into_any()
|
||||||
|
}/>
|
||||||
|
|
||||||
<Route path=leptos_router::path!("/settings") view=move || {
|
<Route path=leptos_router::path!("/settings") view=move || {
|
||||||
Effect::new(move |_| {
|
let authenticated = is_authenticated.0.get();
|
||||||
if !is_authenticated.0.get() {
|
Effect::new(move |_| {
|
||||||
let navigate = use_navigate();
|
if !authenticated {
|
||||||
navigate("/login", Default::default());
|
let navigate = use_navigate();
|
||||||
}
|
navigate("/login", Default::default());
|
||||||
});
|
|
||||||
|
|
||||||
view! {
|
|
||||||
<Show when=move || !is_loading.0.get() fallback=|| ()>
|
|
||||||
<Show when=move || is_authenticated.0.get() fallback=|| ()>
|
|
||||||
<Protected>
|
|
||||||
<div class="p-4">"Settings Page (Coming Soon)"</div>
|
|
||||||
</Protected>
|
|
||||||
</Show>
|
|
||||||
</Show>
|
|
||||||
}
|
}
|
||||||
}/>
|
});
|
||||||
</Routes>
|
|
||||||
</Router>
|
view! {
|
||||||
|
<Show when=move || !is_loading.0.get() fallback=|| ()>
|
||||||
|
<Show when=move || authenticated fallback=|| ()>
|
||||||
|
<Protected>
|
||||||
|
<div class="p-4">"Settings Page (Coming Soon)"</div>
|
||||||
|
</Protected>
|
||||||
|
</Show>
|
||||||
|
</Show>
|
||||||
|
}
|
||||||
|
}/>
|
||||||
|
</Routes>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use leptos_shadcn_card::{Card, CardHeader, CardContent};
|
use crate::components::ui::card::{Card, CardHeader, CardContent};
|
||||||
use leptos_shadcn_input::Input;
|
use crate::components::ui::input::{Input, InputType};
|
||||||
use leptos_shadcn_button::Button;
|
|
||||||
use leptos_shadcn_label::Label;
|
use crate::components::ui::button::Button;
|
||||||
use leptos_shadcn_alert::{Alert, AlertDescription, AlertVariant};
|
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Login() -> impl IntoView {
|
pub fn Login() -> impl IntoView {
|
||||||
let username = signal(String::new());
|
let username = RwSignal::new(String::new());
|
||||||
let password = signal(String::new());
|
let password = RwSignal::new(String::new());
|
||||||
let error = signal(Option::<String>::None);
|
let error = signal(Option::<String>::None);
|
||||||
let loading = signal(false);
|
let loading = signal(false);
|
||||||
|
|
||||||
@@ -18,8 +17,8 @@ pub fn Login() -> impl IntoView {
|
|||||||
loading.1.set(true);
|
loading.1.set(true);
|
||||||
error.1.set(None);
|
error.1.set(None);
|
||||||
|
|
||||||
let user = username.0.get();
|
let user = username.get();
|
||||||
let pass = password.0.get();
|
let pass = password.get();
|
||||||
|
|
||||||
spawn_local(async move {
|
spawn_local(async move {
|
||||||
match shared::server_fns::auth::login(user, pass).await {
|
match shared::server_fns::auth::login(user, pass).await {
|
||||||
@@ -52,40 +51,37 @@ pub fn Login() -> impl IntoView {
|
|||||||
<CardContent class="pt-4">
|
<CardContent class="pt-4">
|
||||||
<form on:submit=handle_login class="space-y-4">
|
<form on:submit=handle_login class="space-y-4">
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<Label>"Kullanıcı Adı"</Label>
|
<label class="text-sm font-medium leading-none">"Kullanıcı Adı"</label>
|
||||||
<Input
|
<Input
|
||||||
input_type="text"
|
r#type=InputType::Text
|
||||||
placeholder="Kullanıcı adınız"
|
placeholder="Kullanıcı adınız"
|
||||||
value=MaybeProp::derive(move || Some(username.0.get()))
|
bind_value=username
|
||||||
on_change=Callback::new(move |val: String| username.1.set(val))
|
disabled=loading.0.get()
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<Label>"Şifre"</Label>
|
<label class="text-sm font-medium leading-none">"Şifre"</label>
|
||||||
<Input
|
<Input
|
||||||
input_type="password"
|
r#type=InputType::Password
|
||||||
placeholder="******"
|
placeholder="******"
|
||||||
value=MaybeProp::derive(move || Some(password.0.get()))
|
bind_value=password
|
||||||
on_change=Callback::new(move |val: String| password.1.set(val))
|
disabled=loading.0.get()
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Show when=move || error.0.get().is_some()>
|
<Show when=move || error.0.get().is_some()>
|
||||||
<Alert variant=AlertVariant::Destructive>
|
<div class="rounded-lg border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
|
||||||
<AlertDescription>
|
{move || error.0.get().unwrap_or_default()}
|
||||||
{move || error.0.get().unwrap_or_default()}
|
</div>
|
||||||
</AlertDescription>
|
|
||||||
</Alert>
|
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<div class="pt-2">
|
<div class="pt-2">
|
||||||
<Button
|
<Button
|
||||||
class="w-full"
|
class="w-full"
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
attr:r#type="submit"
|
||||||
|
attr:disabled=move || loading.0.get()
|
||||||
>
|
>
|
||||||
<Show when=move || loading.0.get() fallback=|| "Giriş Yap">
|
<Show when=move || loading.0.get() fallback=|| view! { "Giriş Yap" }.into_any()>
|
||||||
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
||||||
"Giriş Yapılıyor..."
|
"Giriş Yapılıyor..."
|
||||||
</Show>
|
</Show>
|
||||||
|
|||||||
@@ -1,24 +1,23 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use leptos_shadcn_card::{Card, CardHeader, CardContent};
|
use crate::components::ui::card::{Card, CardHeader, CardContent};
|
||||||
use leptos_shadcn_input::Input;
|
use crate::components::ui::input::{Input, InputType};
|
||||||
use leptos_shadcn_button::Button;
|
|
||||||
use leptos_shadcn_label::Label;
|
use crate::components::ui::button::Button;
|
||||||
use leptos_shadcn_alert::{Alert, AlertDescription, AlertVariant};
|
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Setup() -> impl IntoView {
|
pub fn Setup() -> impl IntoView {
|
||||||
let username = signal(String::new());
|
let username = RwSignal::new(String::new());
|
||||||
let password = signal(String::new());
|
let password = RwSignal::new(String::new());
|
||||||
let confirm_password = signal(String::new());
|
let confirm_password = RwSignal::new(String::new());
|
||||||
let error = signal(Option::<String>::None);
|
let error = signal(Option::<String>::None);
|
||||||
let loading = signal(false);
|
let loading = signal(false);
|
||||||
|
|
||||||
let handle_setup = move |ev: web_sys::SubmitEvent| {
|
let handle_setup = move |ev: web_sys::SubmitEvent| {
|
||||||
ev.prevent_default();
|
ev.prevent_default();
|
||||||
|
|
||||||
let pass = password.0.get();
|
let pass = password.get();
|
||||||
let confirm = confirm_password.0.get();
|
let confirm = confirm_password.get();
|
||||||
|
|
||||||
if pass != confirm {
|
if pass != confirm {
|
||||||
error.1.set(Some("Şifreler eşleşmiyor".to_string()));
|
error.1.set(Some("Şifreler eşleşmiyor".to_string()));
|
||||||
@@ -33,7 +32,7 @@ pub fn Setup() -> impl IntoView {
|
|||||||
loading.1.set(true);
|
loading.1.set(true);
|
||||||
error.1.set(None);
|
error.1.set(None);
|
||||||
|
|
||||||
let user = username.0.get();
|
let user = username.get();
|
||||||
|
|
||||||
spawn_local(async move {
|
spawn_local(async move {
|
||||||
match shared::server_fns::auth::setup(user, pass).await {
|
match shared::server_fns::auth::setup(user, pass).await {
|
||||||
@@ -67,50 +66,46 @@ pub fn Setup() -> impl IntoView {
|
|||||||
<CardContent class="pt-4">
|
<CardContent class="pt-4">
|
||||||
<form on:submit=handle_setup class="space-y-4">
|
<form on:submit=handle_setup class="space-y-4">
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<Label>"Yönetici Kullanıcı Adı"</Label>
|
<label class="text-sm font-medium leading-none">"Yönetici Kullanıcı Adı"</label>
|
||||||
<Input
|
<Input
|
||||||
input_type="text"
|
r#type=InputType::Text
|
||||||
placeholder="admin"
|
placeholder="admin"
|
||||||
value=MaybeProp::derive(move || Some(username.0.get()))
|
bind_value=username
|
||||||
on_change=Callback::new(move |val: String| username.1.set(val))
|
disabled=loading.0.get()
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<Label>"Şifre"</Label>
|
<label class="text-sm font-medium leading-none">"Şifre"</label>
|
||||||
<Input
|
<Input
|
||||||
input_type="password"
|
r#type=InputType::Password
|
||||||
placeholder="******"
|
placeholder="******"
|
||||||
value=MaybeProp::derive(move || Some(password.0.get()))
|
bind_value=password
|
||||||
on_change=Callback::new(move |val: String| password.1.set(val))
|
disabled=loading.0.get()
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<Label>"Şifre Onay"</Label>
|
<label class="text-sm font-medium leading-none">"Şifre Onay"</label>
|
||||||
<Input
|
<Input
|
||||||
input_type="password"
|
r#type=InputType::Password
|
||||||
placeholder="******"
|
placeholder="******"
|
||||||
value=MaybeProp::derive(move || Some(confirm_password.0.get()))
|
bind_value=confirm_password
|
||||||
on_change=Callback::new(move |val: String| confirm_password.1.set(val))
|
disabled=loading.0.get()
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Show when=move || error.0.get().is_some() fallback=|| ()>
|
<Show when=move || error.0.get().is_some() fallback=|| ()>
|
||||||
<Alert variant=AlertVariant::Destructive>
|
<div class="rounded-lg border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
|
||||||
<AlertDescription>
|
<span>{move || error.0.get().unwrap_or_default()}</span>
|
||||||
<span>{move || error.0.get().unwrap_or_default()}</span>
|
</div>
|
||||||
</AlertDescription>
|
|
||||||
</Alert>
|
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<div class="pt-2">
|
<div class="pt-2">
|
||||||
<Button
|
<Button
|
||||||
class="w-full"
|
class="w-full"
|
||||||
disabled=Signal::derive(move || loading.0.get())
|
attr:r#type="submit"
|
||||||
|
attr:disabled=move || loading.0.get()
|
||||||
>
|
>
|
||||||
<Show when=move || loading.0.get() fallback=|| "Kurulumu Tamamla">
|
<Show when=move || loading.0.get() fallback=|| view! { "Kurulumu Tamamla" }.into_any()>
|
||||||
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
||||||
"Kuruluyor..."
|
"Kuruluyor..."
|
||||||
</Show>
|
</Show>
|
||||||
|
|||||||
@@ -1,12 +1,9 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use web_sys::MouseEvent;
|
use crate::components::ui::context_menu::{
|
||||||
use wasm_bindgen::prelude::*;
|
ContextMenu, ContextMenuContent, ContextMenuItem, ContextMenuTrigger,
|
||||||
use wasm_bindgen::JsCast;
|
};
|
||||||
|
use crate::components::ui::button_action::ButtonAction;
|
||||||
// ── Kendi reaktif Context Menu implementasyonumuz ──
|
use crate::components::ui::button::ButtonVariant;
|
||||||
// leptos-shadcn-context-menu v0.8.1'de ContextMenuContent'te
|
|
||||||
// `if open.get()` statik kontrolü reaktif değil. Aşağıda
|
|
||||||
// `Show` bileşeni ile düzgün reaktif versiyon yer alıyor.
|
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn TorrentContextMenu(
|
pub fn TorrentContextMenu(
|
||||||
@@ -14,145 +11,61 @@ pub fn TorrentContextMenu(
|
|||||||
torrent_hash: String,
|
torrent_hash: String,
|
||||||
on_action: Callback<(String, String)>,
|
on_action: Callback<(String, String)>,
|
||||||
) -> impl IntoView {
|
) -> impl IntoView {
|
||||||
let hash = StoredValue::new(torrent_hash);
|
let hash_c1 = torrent_hash.clone();
|
||||||
let on_action = StoredValue::new(on_action);
|
let hash_c2 = torrent_hash.clone();
|
||||||
|
let hash_c3 = torrent_hash.clone();
|
||||||
let open = RwSignal::new(false);
|
let hash_c4 = torrent_hash.clone();
|
||||||
let position = RwSignal::new((0i32, 0i32));
|
|
||||||
|
let on_action_stored = StoredValue::new(on_action);
|
||||||
// Sağ tıklama handler
|
|
||||||
let on_contextmenu = move |e: MouseEvent| {
|
|
||||||
e.prevent_default();
|
|
||||||
e.stop_propagation();
|
|
||||||
position.set((e.client_x(), e.client_y()));
|
|
||||||
open.set(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Menü dışına tıklandığında kapanma
|
|
||||||
Effect::new(move |_| {
|
|
||||||
if open.get() {
|
|
||||||
let cb = Closure::wrap(Box::new(move |_: MouseEvent| {
|
|
||||||
open.set(false);
|
|
||||||
}) as Box<dyn Fn(MouseEvent)>);
|
|
||||||
|
|
||||||
let window = web_sys::window().unwrap();
|
|
||||||
let document = window.document().unwrap();
|
|
||||||
let _ = document.add_event_listener_with_callback(
|
|
||||||
"click",
|
|
||||||
cb.as_ref().unchecked_ref(),
|
|
||||||
);
|
|
||||||
|
|
||||||
// Cleanup: tek sefer dinleyici — click yakalandığında otomatik kapanıp listener kalıyor
|
|
||||||
// ama open=false olduğunda effect tekrar çalışmaz, böylece sorun yok.
|
|
||||||
cb.forget();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let menu_action = move |action: &'static str| {
|
|
||||||
open.set(false);
|
|
||||||
on_action.get_value().run((action.to_string(), hash.get_value()));
|
|
||||||
};
|
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div
|
<ContextMenu>
|
||||||
class="w-full"
|
<ContextMenuTrigger>
|
||||||
on:contextmenu=on_contextmenu
|
{children()}
|
||||||
>
|
</ContextMenuTrigger>
|
||||||
{children()}
|
<ContextMenuContent class="w-56 p-1.5">
|
||||||
</div>
|
<ContextMenuItem on:click={let h = hash_c1; move |_| {
|
||||||
|
on_action_stored.get_value().run(("start".to_string(), h.clone()));
|
||||||
<Show when=move || open.get()>
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
{
|
}}>
|
||||||
let (x, y) = position.get();
|
"Başlat"
|
||||||
// Menü yaklaşık boyutları
|
</ContextMenuItem>
|
||||||
let menu_width = 200;
|
<ContextMenuItem on:click={let h = hash_c2; move |_| {
|
||||||
let menu_height = 220;
|
on_action_stored.get_value().run(("stop".to_string(), h.clone()));
|
||||||
let window = web_sys::window().unwrap();
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
let vw = window.inner_width().unwrap().as_f64().unwrap() as i32;
|
}}>
|
||||||
let vh = window.inner_height().unwrap().as_f64().unwrap() as i32;
|
"Durdur"
|
||||||
// Sağa taşarsa sola aç, alta taşarsa yukarı aç
|
</ContextMenuItem>
|
||||||
let final_x = if x + menu_width > vw { x - menu_width } else { x };
|
|
||||||
let final_y = if y + menu_height > vh { y - menu_height } else { y };
|
<div class="my-1.5 h-px bg-border/50" />
|
||||||
let final_x = final_x.max(0);
|
|
||||||
let final_y = final_y.max(0);
|
// --- Modern Hold-to-Action Buttons ---
|
||||||
view! {
|
<div class="space-y-1">
|
||||||
<div
|
<ButtonAction
|
||||||
class="fixed inset-0 z-[99]"
|
variant=ButtonVariant::Ghost
|
||||||
on:click=move |e: MouseEvent| {
|
class="w-full justify-start h-8 px-2 text-xs text-destructive hover:bg-destructive/10 hover:text-destructive transition-none"
|
||||||
e.stop_propagation();
|
hold_duration=800
|
||||||
open.set(false);
|
on_action={let h = hash_c3; move || {
|
||||||
}
|
on_action_stored.get_value().run(("delete".to_string(), h.clone()));
|
||||||
on:contextmenu=move |e: MouseEvent| {
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
e.prevent_default();
|
}}
|
||||||
e.stop_propagation();
|
|
||||||
open.set(false);
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<div
|
|
||||||
class="fixed z-[100] min-w-[12rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-md animate-in fade-in-0 zoom-in-95"
|
|
||||||
style=format!("left: {}px; top: {}px;", final_x, final_y)
|
|
||||||
on:click=move |e: MouseEvent| e.stop_propagation()
|
|
||||||
>
|
>
|
||||||
// Start
|
"Sil (Basılı Tut)"
|
||||||
<div
|
</ButtonAction>
|
||||||
class="relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground"
|
|
||||||
on:click=move |_| menu_action("start")
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="mr-2 h-4 w-4 opacity-70">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M5.25 5.653c0-.856.917-1.398 1.667-.986l11.54 6.348a1.125 1.125 0 010 1.971l-11.54 6.347a1.125 1.125 0 01-1.667-.985V5.653z" />
|
|
||||||
</svg>
|
|
||||||
"Start"
|
|
||||||
</div>
|
|
||||||
|
|
||||||
// Stop
|
<ButtonAction
|
||||||
<div
|
variant=ButtonVariant::Destructive
|
||||||
class="relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground"
|
class="w-full justify-start h-8 px-2 text-xs font-bold"
|
||||||
on:click=move |_| menu_action("stop")
|
hold_duration=1200
|
||||||
>
|
on_action={let h = hash_c4; move || {
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="mr-2 h-4 w-4 opacity-70">
|
on_action_stored.get_value().run(("delete_with_data".to_string(), h.clone()));
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 5.25v13.5m-7.5-13.5v13.5" />
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
</svg>
|
}}
|
||||||
"Stop"
|
>
|
||||||
</div>
|
"Verilerle Sil (Basılı Tut)"
|
||||||
|
</ButtonAction>
|
||||||
// Recheck
|
</div>
|
||||||
<div
|
</ContextMenuContent>
|
||||||
class="relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground"
|
</ContextMenu>
|
||||||
on:click=move |_| menu_action("recheck")
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="mr-2 h-4 w-4 opacity-70">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0l3.181 3.183a8.25 8.25 0 0013.803-3.7M4.031 9.865a8.25 8.25 0 0113.803-3.7l3.181 3.182m0-4.991v4.99" />
|
|
||||||
</svg>
|
|
||||||
"Recheck"
|
|
||||||
</div>
|
|
||||||
|
|
||||||
// Separator
|
|
||||||
<div class="-mx-1 my-1 h-px bg-border" />
|
|
||||||
|
|
||||||
// Remove
|
|
||||||
<div
|
|
||||||
class="relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors text-destructive hover:bg-destructive hover:text-destructive-foreground"
|
|
||||||
on:click=move |_| menu_action("delete")
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="mr-2 h-4 w-4 opacity-70">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.164h-2.34c-1.18 0-2.09.984-2.09 2.164v.916m7.5 0a48.667 48.667 0 00-7.5 0" />
|
|
||||||
</svg>
|
|
||||||
"Remove"
|
|
||||||
</div>
|
|
||||||
|
|
||||||
// Remove with Data
|
|
||||||
<div
|
|
||||||
class="relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors text-destructive hover:bg-destructive hover:text-destructive-foreground"
|
|
||||||
on:click=move |_| menu_action("delete_with_data")
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="mr-2 h-4 w-4 opacity-70">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M20.25 7.5l-.625 10.632a2.25 2.25 0 01-2.247 2.118H6.622a2.25 2.25 0 01-2.247-2.118L3.75 7.5m6 4.125l2.25 2.25m0 0l2.25 2.25M12 13.875l2.25-2.25M12 13.875l-2.25-2.25M3.375 7.5h17.25c.621 0 1.125-.504 1.125-1.125v-1.5c0-.621-.504-1.125-1.125-1.125H3.375c-.621 0-1.125.504-1.125 1.125v1.5c0 .621.504 1.125 1.125 1.125z" />
|
|
||||||
</svg>
|
|
||||||
"Remove with Data"
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</Show>
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
91
frontend/src/components/demos/demo_shimmer.rs
Normal file
91
frontend/src/components/demos/demo_shimmer.rs
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos::task::spawn_local;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::components::ui::button::{Button, ButtonVariant};
|
||||||
|
use crate::components::ui::card::{Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle};
|
||||||
|
use crate::components::ui::shimmer::Shimmer;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
|
pub struct CardData {
|
||||||
|
pub title: String,
|
||||||
|
pub description: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Simulates a database fetch with 1 second delay
|
||||||
|
#[server]
|
||||||
|
pub async fn fetch_card_data() -> Result<CardData, ServerFnError> {
|
||||||
|
// Simulate network/database latency (only on server)
|
||||||
|
#[cfg(feature = "ssr")]
|
||||||
|
tokio::time::sleep(std::time::Duration::from_secs(1)).await;
|
||||||
|
|
||||||
|
Ok(CardData {
|
||||||
|
title: "Fetched Title".to_string(),
|
||||||
|
description: "This content was fetched from the server after a 1 second simulated delay. The shimmer effect automatically showed during the loading period.".to_string(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DemoShimmer() -> impl IntoView {
|
||||||
|
// Loading state
|
||||||
|
let loading = RwSignal::new(false);
|
||||||
|
|
||||||
|
// Store fetched data
|
||||||
|
let card_data = RwSignal::new(None::<CardData>);
|
||||||
|
|
||||||
|
// Fetch handler using spawn_local for reliable repeated calls
|
||||||
|
let on_fetch = move |_| {
|
||||||
|
spawn_local(async move {
|
||||||
|
loading.set(true);
|
||||||
|
let result = fetch_card_data().await;
|
||||||
|
if let Ok(data) = result {
|
||||||
|
card_data.set(Some(data));
|
||||||
|
}
|
||||||
|
loading.set(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class="flex flex-col gap-4">
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<Button variant=ButtonVariant::Outline on:click=move |_| loading.set(!loading.get())>
|
||||||
|
"Toggle Loading"
|
||||||
|
</Button>
|
||||||
|
<Button variant=ButtonVariant::Default on:click=on_fetch>
|
||||||
|
"Fetch Data (1s)"
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Shimmer loading=Signal::from(loading)>
|
||||||
|
<Card class="max-w-lg lg:max-w-2xl">
|
||||||
|
<CardHeader>
|
||||||
|
<CardTitle>
|
||||||
|
{move || {
|
||||||
|
card_data.get().map(|data| data.title).unwrap_or_else(|| "Card Title".to_string())
|
||||||
|
}}
|
||||||
|
</CardTitle>
|
||||||
|
</CardHeader>
|
||||||
|
|
||||||
|
<CardContent>
|
||||||
|
<CardDescription>
|
||||||
|
{move || {
|
||||||
|
card_data
|
||||||
|
.get()
|
||||||
|
.map(|data| data.description)
|
||||||
|
.unwrap_or_else(|| {
|
||||||
|
"Click 'Toggle Loading' for manual control, or 'Fetch Data' to simulate a real server call with 1 second delay."
|
||||||
|
.to_string()
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
</CardDescription>
|
||||||
|
</CardContent>
|
||||||
|
|
||||||
|
<CardFooter class="justify-end">
|
||||||
|
<Button variant=ButtonVariant::Outline>"Cancel"</Button>
|
||||||
|
<Button>"Confirm"</Button>
|
||||||
|
</CardFooter>
|
||||||
|
</Card>
|
||||||
|
</Shimmer>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
1
frontend/src/components/demos/mod.rs
Normal file
1
frontend/src/components/demos/mod.rs
Normal file
@@ -0,0 +1 @@
|
|||||||
|
pub mod demo_shimmer;
|
||||||
3
frontend/src/components/hooks/mod.rs
Normal file
3
frontend/src/components/hooks/mod.rs
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
pub mod use_random;
|
||||||
|
pub mod use_theme_mode;
|
||||||
|
pub mod use_can_scroll_vertical;
|
||||||
25
frontend/src/components/hooks/use_can_scroll_vertical.rs
Normal file
25
frontend/src/components/hooks/use_can_scroll_vertical.rs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use wasm_bindgen::JsCast;
|
||||||
|
|
||||||
|
/// Hook to determine if an element can scroll vertically.
|
||||||
|
///
|
||||||
|
/// Returns (on_scroll_callback, can_scroll_up_signal, can_scroll_down_signal)
|
||||||
|
pub fn use_can_scroll_vertical() -> (Callback<web_sys::Event>, ReadSignal<bool>, ReadSignal<bool>) {
|
||||||
|
let can_scroll_up = RwSignal::new(false);
|
||||||
|
let can_scroll_down = RwSignal::new(false);
|
||||||
|
|
||||||
|
let on_scroll = Callback::new(move |ev: web_sys::Event| {
|
||||||
|
if let Some(target) = ev.target() {
|
||||||
|
if let Some(el) = target.dyn_ref::<web_sys::HtmlElement>() {
|
||||||
|
let scroll_top = el.scroll_top();
|
||||||
|
let scroll_height = el.scroll_height();
|
||||||
|
let client_height = el.client_height();
|
||||||
|
|
||||||
|
can_scroll_up.set(scroll_top > 0);
|
||||||
|
can_scroll_down.set(scroll_top + client_height < scroll_height - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
(on_scroll, can_scroll_up.read_only(), can_scroll_down.read_only())
|
||||||
|
}
|
||||||
31
frontend/src/components/hooks/use_random.rs
Normal file
31
frontend/src/components/hooks/use_random.rs
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
use std::collections::hash_map::DefaultHasher;
|
||||||
|
use std::hash::{Hash, Hasher};
|
||||||
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
|
||||||
|
const PREFIX: &str = "rust_ui"; // Must NOT contain "/" or "-"
|
||||||
|
|
||||||
|
pub fn use_random_id() -> String {
|
||||||
|
format!("_{PREFIX}_{}", generate_hash())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn use_random_id_for(element: &str) -> String {
|
||||||
|
format!("{}_{PREFIX}_{}", element, generate_hash())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn use_random_transition_name() -> String {
|
||||||
|
let random_id = use_random_id();
|
||||||
|
format!("view-transition-name: {random_id}")
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
static COUNTER: AtomicUsize = AtomicUsize::new(1);
|
||||||
|
|
||||||
|
fn generate_hash() -> u64 {
|
||||||
|
let mut hasher = DefaultHasher::new();
|
||||||
|
let counter = COUNTER.fetch_add(1, Ordering::SeqCst);
|
||||||
|
counter.hash(&mut hasher);
|
||||||
|
hasher.finish()
|
||||||
|
}
|
||||||
108
frontend/src/components/hooks/use_theme_mode.rs
Normal file
108
frontend/src/components/hooks/use_theme_mode.rs
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use web_sys::Storage;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct ThemeMode {
|
||||||
|
state: RwSignal<bool>,
|
||||||
|
}
|
||||||
|
|
||||||
|
const LOCALSTORAGE_KEY: &str = "darkmode";
|
||||||
|
|
||||||
|
/// Hook to access the dark mode context
|
||||||
|
///
|
||||||
|
/// Returns the ThemeMode instance from context for easy access
|
||||||
|
pub fn use_theme_mode() -> ThemeMode {
|
||||||
|
expect_context::<ThemeMode>()
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
impl ThemeMode {
|
||||||
|
#[must_use]
|
||||||
|
/// Initializes a new ThemeMode instance.
|
||||||
|
pub fn init() -> Self {
|
||||||
|
let theme_mode = Self { state: RwSignal::new(false) };
|
||||||
|
|
||||||
|
provide_context(theme_mode);
|
||||||
|
|
||||||
|
// Use Effect to handle browser-only initialization
|
||||||
|
Effect::new(move |_| {
|
||||||
|
let initial = Self::get_storage_state().unwrap_or(Self::prefers_dark_mode());
|
||||||
|
theme_mode.state.set(initial);
|
||||||
|
});
|
||||||
|
|
||||||
|
theme_mode
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn toggle(&self) {
|
||||||
|
self.state.update(|state| {
|
||||||
|
*state = !*state;
|
||||||
|
Self::set_storage_state(*state);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_dark(&self) {
|
||||||
|
self.set(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_light(&self) {
|
||||||
|
self.set(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// - `dark`: Set to `true` for dark mode, and `false` for light mode.
|
||||||
|
pub fn set(&self, dark: bool) {
|
||||||
|
self.state.set(dark);
|
||||||
|
Self::set_storage_state(dark);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
pub fn get(&self) -> bool {
|
||||||
|
self.state.get()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_dark(&self) -> bool {
|
||||||
|
self.state.get()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_light(&self) -> bool {
|
||||||
|
!self.state.get()
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
/// Retrieves the local storage object, if available.
|
||||||
|
fn get_storage() -> Option<Storage> {
|
||||||
|
window().local_storage().ok().flatten()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Retrieves the dark mode state from local storage, if available.
|
||||||
|
fn get_storage_state() -> Option<bool> {
|
||||||
|
Self::get_storage()
|
||||||
|
.and_then(|storage| storage.get(LOCALSTORAGE_KEY).ok())
|
||||||
|
.flatten()
|
||||||
|
.and_then(|entry| entry.parse::<bool>().ok())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Checks whether the user's system prefers dark mode based on media queries.
|
||||||
|
fn prefers_dark_mode() -> bool {
|
||||||
|
window()
|
||||||
|
.match_media("(prefers-color-scheme: dark)")
|
||||||
|
.ok()
|
||||||
|
.flatten()
|
||||||
|
.map(|media| media.matches())
|
||||||
|
.unwrap_or_default()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stores the dark mode state in local storage.
|
||||||
|
fn set_storage_state(state: bool) {
|
||||||
|
if let Some(storage) = Self::get_storage() {
|
||||||
|
storage.set(LOCALSTORAGE_KEY, state.to_string().as_str()).ok();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
18
frontend/src/components/layout/footer.rs
Normal file
18
frontend/src/components/layout/footer.rs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use crate::components::ui::separator::Separator;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Footer() -> impl IntoView {
|
||||||
|
let year = chrono::Local::now().format("%Y").to_string();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<footer class="mt-auto pb-6 px-4">
|
||||||
|
<Separator class="mb-4 opacity-30" />
|
||||||
|
<div class="flex items-center justify-center gap-2 text-[10px] uppercase tracking-widest text-muted-foreground/60 font-medium">
|
||||||
|
<span>{format!("© {} VibeTorrent", year)}</span>
|
||||||
|
<span class="size-1 rounded-full bg-muted-foreground/30" />
|
||||||
|
<span>"v3.0.0-beta"</span>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
pub mod sidebar;
|
pub mod sidebar;
|
||||||
pub mod statusbar;
|
|
||||||
pub mod toolbar;
|
pub mod toolbar;
|
||||||
|
pub mod footer;
|
||||||
pub mod protected;
|
pub mod protected;
|
||||||
|
|||||||
@@ -1,52 +1,31 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use crate::components::layout::sidebar::Sidebar;
|
use crate::components::layout::sidebar::Sidebar;
|
||||||
use crate::components::layout::toolbar::Toolbar;
|
use crate::components::layout::toolbar::Toolbar;
|
||||||
use crate::components::layout::statusbar::StatusBar;
|
use crate::components::layout::footer::Footer;
|
||||||
|
use crate::components::ui::sidenav::{SidenavWrapper, Sidenav, SidenavInset};
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Protected(children: Children) -> impl IntoView {
|
pub fn Protected(children: Children) -> impl IntoView {
|
||||||
// Mobil menü durumu için bir sinyal oluşturuyoruz (RwSignal for easier passing)
|
|
||||||
let is_mobile_menu_open = RwSignal::new(false);
|
|
||||||
|
|
||||||
// Sinyali context olarak sağlıyoruz ki Toolbar ve Sidebar buna erişebilsin
|
|
||||||
provide_context(is_mobile_menu_open);
|
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div class="flex h-screen w-full overflow-hidden bg-background">
|
<SidenavWrapper attr:style="--sidenav-width:16rem; --sidenav-width-icon:3rem;">
|
||||||
|
// Masaüstü Sidenav
|
||||||
// --- SIDEBAR (Desktop: Sabit, Mobil: Overlay) ---
|
<Sidenav>
|
||||||
<aside class=move || {
|
|
||||||
let base = "fixed inset-y-0 left-0 z-50 w-64 transform transition-transform duration-300 ease-in-out border-r border-border bg-card lg:relative lg:translate-x-0";
|
|
||||||
if is_mobile_menu_open.get() {
|
|
||||||
format!("{} translate-x-0", base)
|
|
||||||
} else {
|
|
||||||
format!("{} -translate-x-full", base)
|
|
||||||
}
|
|
||||||
}>
|
|
||||||
<Sidebar />
|
<Sidebar />
|
||||||
</aside>
|
</Sidenav>
|
||||||
|
|
||||||
// Mobil arka plan karartma (Overlay)
|
// İçerik Alanı
|
||||||
<Show when=move || is_mobile_menu_open.get()>
|
<SidenavInset class="flex flex-col h-screen overflow-hidden">
|
||||||
<div
|
// Toolbar (Üst Bar)
|
||||||
class="fixed inset-0 z-40 bg-background/80 backdrop-blur-sm lg:hidden"
|
|
||||||
on:click=move |_| is_mobile_menu_open.set(false)
|
|
||||||
></div>
|
|
||||||
</Show>
|
|
||||||
|
|
||||||
// --- MAIN CONTENT AREA ---
|
|
||||||
<div class="flex flex-1 flex-col overflow-hidden">
|
|
||||||
// --- TOOLBAR (TOP) ---
|
|
||||||
<Toolbar />
|
<Toolbar />
|
||||||
|
|
||||||
// --- MAIN CONTENT ---
|
// Ana İçerik
|
||||||
<main class="flex-1 overflow-hidden relative bg-background">
|
<main class="flex-1 overflow-y-auto relative bg-background flex flex-col">
|
||||||
{children()}
|
<div class="flex-1">
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
<Footer />
|
||||||
</main>
|
</main>
|
||||||
|
</SidenavInset>
|
||||||
// --- STATUS BAR (BOTTOM) ---
|
</SidenavWrapper>
|
||||||
<StatusBar />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,13 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use leptos_shadcn_button::{Button, ButtonVariant, ButtonSize};
|
use crate::components::ui::sidenav::*;
|
||||||
use leptos_shadcn_avatar::{Avatar, AvatarFallback};
|
use crate::components::ui::button::{Button, ButtonVariant, ButtonSize};
|
||||||
use leptos_shadcn_separator::Separator;
|
use crate::components::ui::theme_toggle::ThemeToggle;
|
||||||
|
use crate::components::ui::switch::Switch;
|
||||||
use leptos_use::storage::use_local_storage;
|
|
||||||
use ::codee::string::FromToStringCodec;
|
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Sidebar() -> impl IntoView {
|
pub fn Sidebar() -> impl IntoView {
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
||||||
let is_mobile_menu_open = use_context::<RwSignal<bool>>().expect("mobile menu state not provided");
|
|
||||||
|
|
||||||
let total_count = move || store.torrents.with(|map| map.len());
|
let total_count = move || store.torrents.with(|map| map.len());
|
||||||
let downloading_count = move || {
|
let downloading_count = move || {
|
||||||
@@ -57,7 +54,6 @@ pub fn Sidebar() -> impl IntoView {
|
|||||||
|
|
||||||
let set_filter = move |f: crate::store::FilterStatus| {
|
let set_filter = move |f: crate::store::FilterStatus| {
|
||||||
store.filter.set(f);
|
store.filter.set(f);
|
||||||
is_mobile_menu_open.set(false);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
let is_active = move |f: crate::store::FilterStatus| store.filter.get() == f;
|
let is_active = move |f: crate::store::FilterStatus| store.filter.get() == f;
|
||||||
@@ -70,179 +66,160 @@ pub fn Sidebar() -> impl IntoView {
|
|||||||
username().chars().next().unwrap_or('?').to_uppercase().to_string()
|
username().chars().next().unwrap_or('?').to_uppercase().to_string()
|
||||||
};
|
};
|
||||||
|
|
||||||
// --- THEME LOGIC START ---
|
let on_push_toggle = move |checked: bool| {
|
||||||
let (current_theme, set_current_theme, _) = use_local_storage::<String, FromToStringCodec>("vibetorrent_theme");
|
spawn_local(async move {
|
||||||
|
if checked {
|
||||||
// Initialize with default if empty
|
crate::store::subscribe_to_push_notifications().await;
|
||||||
let current_theme_val = current_theme.get();
|
|
||||||
if current_theme_val.is_empty() {
|
|
||||||
set_current_theme.set("dark".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Automatically sync theme to document attribute
|
|
||||||
Effect::new(move |_| {
|
|
||||||
let theme = current_theme.get().to_lowercase();
|
|
||||||
if let Some(doc) = document().document_element() {
|
|
||||||
let _ = doc.set_attribute("data-theme", &theme);
|
|
||||||
// Also set class for Shadcn dark mode support
|
|
||||||
if theme == "dark" || theme == "dracula" || theme == "dim" || theme == "abyss" || theme == "sunset" || theme == "cyberpunk" || theme == "nord" || theme == "business" || theme == "night" || theme == "black" || theme == "luxury" || theme == "coffee" || theme == "forest" || theme == "halloween" || theme == "synthwave" {
|
|
||||||
let _ = doc.class_list().add_1("dark");
|
|
||||||
} else {
|
} else {
|
||||||
let _ = doc.class_list().remove_1("dark");
|
crate::store::unsubscribe_from_push_notifications().await;
|
||||||
}
|
}
|
||||||
}
|
if let Ok(enabled) = crate::store::is_push_subscribed().await {
|
||||||
});
|
store.push_enabled.set(enabled);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
let toggle_theme = move |_| {
|
|
||||||
let new_theme = if current_theme.get() == "dark" { "light" } else { "dark" };
|
|
||||||
set_current_theme.set(new_theme.to_string());
|
|
||||||
};
|
};
|
||||||
// --- THEME LOGIC END ---
|
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div class="w-full h-full flex flex-col bg-card" style="padding-top: env(safe-area-inset-top);">
|
<SidenavHeader>
|
||||||
<div class="p-4 flex-1 overflow-y-auto">
|
<div class="flex items-center gap-2 px-2 py-4">
|
||||||
<div class="mb-4 px-2 text-lg font-semibold tracking-tight text-foreground">
|
<div class="flex size-8 items-center justify-center rounded-lg bg-primary text-primary-foreground shadow-sm">
|
||||||
"VibeTorrent"
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="size-5">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M15.362 5.214A8.252 8.252 0 0112 21 8.25 8.25 0 016.038 7.048 8.287 8.287 0 009 9.6a8.983 8.983 0 013.361-6.867 8.21 8.25 0 003 2.48z" />
|
||||||
|
</svg>
|
||||||
</div>
|
</div>
|
||||||
<div class="space-y-1">
|
<div class="grid flex-1 text-left text-sm leading-tight overflow-hidden">
|
||||||
<h4 class="mb-1 rounded-md px-2 py-1 text-sm font-semibold text-muted-foreground">"Filters"</h4>
|
<span class="truncate font-semibold text-foreground text-base">"VibeTorrent"</span>
|
||||||
|
<span class="truncate text-[10px] text-muted-foreground opacity-70">"v3.0.0"</span>
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::All) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::All))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M3.75 6.75h16.5M3.75 12h16.5m-16.5 5.25h16.5" />
|
|
||||||
</svg>
|
|
||||||
"All"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{total_count}</span>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::Downloading) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::Downloading))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M3 16.5v2.25A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75V16.5M16.5 12L12 16.5m0 0L7.5 12m4.5 4.5V3" />
|
|
||||||
</svg>
|
|
||||||
"Downloading"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{downloading_count}</span>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::Seeding) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::Seeding))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M3 16.5v2.25A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75V16.5m-13.5-9L12 3m0 0l4.5 4.5M12 3v13.5" />
|
|
||||||
</svg>
|
|
||||||
"Seeding"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{seeding_count}</span>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::Completed) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::Completed))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M9 12.75L11.25 15 15 9.75M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
||||||
</svg>
|
|
||||||
"Completed"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{completed_count}</span>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::Paused) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::Paused))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 5.25v13.5m-7.5-13.5v13.5" />
|
|
||||||
</svg>
|
|
||||||
"Paused"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{paused_count}</span>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
|
||||||
variant=MaybeProp::derive(move || Some(if is_active(crate::store::FilterStatus::Inactive) { ButtonVariant::Secondary } else { ButtonVariant::Ghost }))
|
|
||||||
size=ButtonSize::Sm
|
|
||||||
class="w-full justify-start gap-2"
|
|
||||||
on_click=Callback::new(move |()| set_filter(crate::store::FilterStatus::Inactive))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M18.364 18.364A9 9 0 005.636 5.636m12.728 12.728A9 9 0 015.636 5.636m12.728 12.728L5.636 5.636" />
|
|
||||||
</svg>
|
|
||||||
"Inactive"
|
|
||||||
<span class="ml-auto text-xs font-mono opacity-70">{inactive_count}</span>
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</SidenavHeader>
|
||||||
|
|
||||||
<Separator />
|
<SidenavContent>
|
||||||
|
<SidenavGroup>
|
||||||
|
<SidenavGroupLabel>"Filtreler"</SidenavGroupLabel>
|
||||||
|
<SidenavGroupContent>
|
||||||
|
<SidenavMenu>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::All))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::All)
|
||||||
|
icon="M3.75 6.75h16.5M3.75 12h16.5m-16.5 5.25h16.5"
|
||||||
|
label="Tümü"
|
||||||
|
count=Signal::derive(total_count)
|
||||||
|
/>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::Downloading))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::Downloading)
|
||||||
|
icon="M3 16.5v2.25A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75V16.5M16.5 12L12 16.5m0 0L7.5 12m4.5 4.5V3"
|
||||||
|
label="İndirilenler"
|
||||||
|
count=Signal::derive(downloading_count)
|
||||||
|
/>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::Seeding))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::Seeding)
|
||||||
|
icon="M3 16.5v2.25A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75V16.5m-13.5-9L12 3m0 0l4.5 4.5M12 3v13.5"
|
||||||
|
label="Gönderilenler"
|
||||||
|
count=Signal::derive(seeding_count)
|
||||||
|
/>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::Completed))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::Completed)
|
||||||
|
icon="M9 12.75L11.25 15 15 9.75M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
||||||
|
label="Tamamlananlar"
|
||||||
|
count=Signal::derive(completed_count)
|
||||||
|
/>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::Paused))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::Paused)
|
||||||
|
icon="M15.75 5.25v13.5m-7.5-13.5v13.5"
|
||||||
|
label="Durdurulanlar"
|
||||||
|
count=Signal::derive(paused_count)
|
||||||
|
/>
|
||||||
|
<SidebarItem
|
||||||
|
active=Signal::derive(move || is_active(crate::store::FilterStatus::Inactive))
|
||||||
|
on_click=move |_| set_filter(crate::store::FilterStatus::Inactive)
|
||||||
|
icon="M18.364 18.364A9 9 0 005.636 5.636m12.728 12.728A9 9 0 015.636 5.636m12.728 12.728L5.636 5.636"
|
||||||
|
label="Pasif"
|
||||||
|
count=Signal::derive(inactive_count)
|
||||||
|
/>
|
||||||
|
</SidenavMenu>
|
||||||
|
</SidenavGroupContent>
|
||||||
|
</SidenavGroup>
|
||||||
|
</SidenavContent>
|
||||||
|
|
||||||
<div class="p-4 bg-card" style="padding-bottom: calc(1rem + env(safe-area-inset-bottom));">
|
<SidenavFooter>
|
||||||
<div class="flex items-center gap-3">
|
<div class="flex flex-col gap-4 p-4">
|
||||||
<Avatar class="h-8 w-8">
|
// Push Notification Toggle
|
||||||
<AvatarFallback class="bg-primary text-primary-foreground text-xs font-medium">
|
<div class="flex items-center justify-between px-2 py-1 bg-muted/20 rounded-md border border-border/50">
|
||||||
{first_letter}
|
<div class="flex flex-col gap-0.5">
|
||||||
</AvatarFallback>
|
<span class="text-[10px] font-bold uppercase tracking-wider text-foreground/70">"Bildirimler"</span>
|
||||||
</Avatar>
|
<span class="text-[9px] text-muted-foreground">"Web Push"</span>
|
||||||
<div class="flex-1 overflow-hidden">
|
|
||||||
<div class="font-medium text-sm truncate text-foreground">{username}</div>
|
|
||||||
<div class="text-[10px] text-muted-foreground truncate">"Online"</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<Switch
|
||||||
|
checked=Signal::from(store.push_enabled)
|
||||||
|
on_checked_change=Callback::new(on_push_toggle)
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
// --- THEME BUTTON ---
|
<div class="flex items-center gap-3 p-2 rounded-lg border bg-muted/30 shadow-xs overflow-hidden">
|
||||||
<Button
|
<div class="h-8 w-8 rounded-full bg-primary text-primary-foreground flex items-center justify-center text-xs font-medium shrink-0 border border-primary-foreground/10">
|
||||||
variant=ButtonVariant::Ghost
|
{first_letter}
|
||||||
size=ButtonSize::Icon
|
</div>
|
||||||
class="h-8 w-8 text-muted-foreground hover:text-foreground"
|
<div class="flex-1 overflow-hidden">
|
||||||
on_click=Callback::new(toggle_theme)
|
<div class="font-medium text-[11px] truncate text-foreground leading-tight">{username}</div>
|
||||||
>
|
<div class="text-[9px] text-muted-foreground truncate opacity-70">"Yönetici"</div>
|
||||||
// Sun icon for dark mode (to switch to light), Moon for light (to switch to dark)
|
</div>
|
||||||
// Actually show current state or action? Usually action.
|
|
||||||
// If dark, show Sun. If light, show Moon.
|
<div class="flex items-center gap-1">
|
||||||
<Show when=move || current_theme.get() == "dark" fallback=|| view! {
|
<ThemeToggle />
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M21.752 15.002A9.718 9.718 0 0118 15.75c-5.385 0-9.75-4.365-9.75-9.75 0-1.33.266-2.597.748-3.752A9.753 9.753 0 003 11.25C3 16.635 7.365 21 12.75 21a9.753 9.753 0 009.002-5.998z" />
|
<Button
|
||||||
|
variant=ButtonVariant::Ghost
|
||||||
|
size=ButtonSize::Icon
|
||||||
|
class="size-7 text-destructive hover:bg-destructive/10"
|
||||||
|
on:click=move |_| {
|
||||||
|
spawn_local(async move {
|
||||||
|
if shared::server_fns::auth::logout().await.is_ok() {
|
||||||
|
let window = web_sys::window().expect("window should exist");
|
||||||
|
let _ = window.location().set_href("/login");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="size-4">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 9V5.25A2.25 2.25 0 0013.5 3h-6a2.25 2.25 0 00-2.25 2.25v13.5A2.25 2.25 0 007.5 21h6a2.25 2.25 0 002.25-2.25V15M12 9l-3 3m0 0l3 3m-3-3h12.75" />
|
||||||
</svg>
|
</svg>
|
||||||
}>
|
</Button>
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
</div>
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M12 3v2.25m6.364.386l-1.591 1.591M21 12h-2.25m-.386 6.364l-1.591-1.591M12 18.75V21m-4.773-4.227l-1.591 1.591M5.25 12H3m4.227-4.773L5.636 5.636M15.75 12a3.75 3.75 0 11-7.5 0 3.75 3.75 0 017.5 0z" />
|
|
||||||
</svg>
|
|
||||||
</Show>
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant=ButtonVariant::Ghost
|
|
||||||
size=ButtonSize::Icon
|
|
||||||
class="text-destructive h-8 w-8"
|
|
||||||
on_click=Callback::new(move |()| {
|
|
||||||
spawn_local(async move {
|
|
||||||
if shared::server_fns::auth::logout().await.is_ok() {
|
|
||||||
let window = web_sys::window().expect("window should exist");
|
|
||||||
let _ = window.location().set_href("/login");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 9V5.25A2.25 2.25 0 0013.5 3h-6a2.25 2.25 0 00-2.25 2.25v13.5A2.25 2.25 0 007.5 21h6a2.25 2.25 0 002.25-2.25V15M12 9l-3 3m0 0l3 3m-3-3h12.75" />
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</SidenavFooter>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn SidebarItem(
|
||||||
|
active: Signal<bool>,
|
||||||
|
on_click: impl Fn(web_sys::MouseEvent) + 'static + Send,
|
||||||
|
#[prop(into)] icon: String,
|
||||||
|
#[prop(into)] label: &'static str,
|
||||||
|
count: Signal<usize>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let variant = move || if active.get() { SidenavMenuButtonVariant::Outline } else { SidenavMenuButtonVariant::Default };
|
||||||
|
let class = move || if active.get() { "bg-accent/50 border-accent text-foreground".to_string() } else { "text-muted-foreground hover:text-foreground".to_string() };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<SidenavMenuItem>
|
||||||
|
<SidenavMenuButton
|
||||||
|
variant=Signal::derive(variant)
|
||||||
|
class=Signal::derive(class)
|
||||||
|
on:click=on_click
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="size-4 shrink-0">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d=icon.clone() />
|
||||||
|
</svg>
|
||||||
|
<span class="flex-1 truncate">{label}</span>
|
||||||
|
<span class="text-[10px] font-mono opacity-50">{count}</span>
|
||||||
|
</SidenavMenuButton>
|
||||||
|
</SidenavMenuItem>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,201 +0,0 @@
|
|||||||
use leptos::prelude::*;
|
|
||||||
use leptos::html;
|
|
||||||
use shared::GlobalLimitRequest;
|
|
||||||
use crate::api;
|
|
||||||
|
|
||||||
fn format_bytes(bytes: i64) -> String {
|
|
||||||
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
|
||||||
if bytes < 1024 {
|
|
||||||
return format!("{} B", bytes);
|
|
||||||
}
|
|
||||||
let i = (bytes as f64).log2().div_euclid(10.0) as usize;
|
|
||||||
format!(
|
|
||||||
"{:.1} {}",
|
|
||||||
(bytes as f64) / 1024_f64.powi(i as i32),
|
|
||||||
UNITS[i]
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn format_speed(bytes_per_sec: i64) -> String {
|
|
||||||
if bytes_per_sec == 0 {
|
|
||||||
return "0 B/s".to_string();
|
|
||||||
}
|
|
||||||
format!("{}/s", format_bytes(bytes_per_sec))
|
|
||||||
}
|
|
||||||
|
|
||||||
#[component]
|
|
||||||
pub fn StatusBar() -> impl IntoView {
|
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
|
||||||
let stats = store.global_stats;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Preset limits in bytes/s
|
|
||||||
let limits: Vec<(i64, &str)> = vec!(
|
|
||||||
(0, "Unlimited"),
|
|
||||||
(100 * 1024, "100 KB/s"),
|
|
||||||
(500 * 1024, "500 KB/s"),
|
|
||||||
(1024 * 1024, "1 MB/s"),
|
|
||||||
(2 * 1024 * 1024, "2 MB/s"),
|
|
||||||
(5 * 1024 * 1024, "5 MB/s"),
|
|
||||||
(10 * 1024 * 1024, "10 MB/s"),
|
|
||||||
(20 * 1024 * 1024, "20 MB/s"),
|
|
||||||
);
|
|
||||||
|
|
||||||
let set_limit = move |limit_type: &str, val: i64| {
|
|
||||||
let limit_type = limit_type.to_string();
|
|
||||||
log::info!("Setting {} limit to {}", limit_type, val);
|
|
||||||
|
|
||||||
let req = if limit_type == "down" {
|
|
||||||
GlobalLimitRequest {
|
|
||||||
max_download_rate: Some(val),
|
|
||||||
max_upload_rate: None,
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
GlobalLimitRequest {
|
|
||||||
max_download_rate: None,
|
|
||||||
max_upload_rate: Some(val),
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
leptos::task::spawn_local(async move {
|
|
||||||
if let Err(e) = api::settings::set_global_limits(&req).await {
|
|
||||||
log::error!("Failed to set limit: {:?}", e);
|
|
||||||
} else {
|
|
||||||
log::info!("Limit set successfully");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
let down_details_ref = NodeRef::<html::Details>::new();
|
|
||||||
let up_details_ref = NodeRef::<html::Details>::new();
|
|
||||||
|
|
||||||
let close_details = move |node_ref: NodeRef<html::Details>| {
|
|
||||||
if let Some(el) = node_ref.get_untracked() {
|
|
||||||
el.set_open(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
view! {
|
|
||||||
<div class="fixed bottom-0 left-0 right-0 h-8 min-h-8 bg-muted border-t border-border flex items-center px-4 text-xs gap-4 text-muted-foreground z-[99] cursor-pointer">
|
|
||||||
|
|
||||||
// --- DOWNLOAD SPEED DROPDOWN ---
|
|
||||||
<details class="group relative" node_ref=down_details_ref>
|
|
||||||
<summary class="flex items-center gap-2 cursor-pointer hover:text-foreground transition-colors select-none list-none [&::-webkit-details-marker]:hidden outline-none">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M19.5 13.5L12 21m0 0l-7.5-7.5M12 21V3" />
|
|
||||||
</svg>
|
|
||||||
<span class="font-mono">{move || format_speed(stats.get().down_rate)}</span>
|
|
||||||
<Show when=move || { stats.get().down_limit.unwrap_or(0) > 0 } fallback=|| ()>
|
|
||||||
<span class="text-[10px] opacity-60">
|
|
||||||
{move || format!("(Limit: {})", format_speed(stats.get().down_limit.unwrap_or(0)))}
|
|
||||||
</span>
|
|
||||||
</Show>
|
|
||||||
</summary>
|
|
||||||
|
|
||||||
<div class="absolute bottom-full left-0 mb-2 z-[100] min-w-[8rem] overflow-hidden rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md hidden group-open:block animate-in fade-in-0 zoom-in-95 slide-in-from-bottom-2">
|
|
||||||
<ul class="w-full">
|
|
||||||
{
|
|
||||||
limits.clone().into_iter().map(|(val, label)| {
|
|
||||||
let is_active = move || {
|
|
||||||
let current = stats.get().down_limit.unwrap_or(0);
|
|
||||||
(current - val).abs() < 1024
|
|
||||||
};
|
|
||||||
view! {
|
|
||||||
<li>
|
|
||||||
<button
|
|
||||||
class=move || {
|
|
||||||
let base = "relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-xs outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50 hover:bg-accent hover:text-accent-foreground";
|
|
||||||
if is_active() { format!("{} bg-accent text-accent-foreground font-medium", base) } else { base.to_string() }
|
|
||||||
}
|
|
||||||
on:click=move |_| {
|
|
||||||
set_limit("down", val);
|
|
||||||
close_details(down_details_ref);
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span class="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
|
||||||
<Show when=is_active fallback=|| ()>
|
|
||||||
<span>"✓"</span>
|
|
||||||
</Show>
|
|
||||||
</span>
|
|
||||||
{label}
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
}).collect::<Vec<_>>()
|
|
||||||
}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</details>
|
|
||||||
|
|
||||||
// --- UPLOAD SPEED DROPDOWN ---
|
|
||||||
<details class="group relative" node_ref=up_details_ref>
|
|
||||||
<summary class="flex items-center gap-2 cursor-pointer hover:text-foreground transition-colors select-none list-none [&::-webkit-details-marker]:hidden outline-none">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M4.5 10.5L12 3m0 0l7.5 7.5M12 3v18" />
|
|
||||||
</svg>
|
|
||||||
<span class="font-mono">{move || format_speed(stats.get().up_rate)}</span>
|
|
||||||
<Show when=move || { stats.get().up_limit.unwrap_or(0) > 0 } fallback=|| ()>
|
|
||||||
<span class="text-[10px] opacity-60">
|
|
||||||
{move || format!("(Limit: {})", format_speed(stats.get().up_limit.unwrap_or(0)))}
|
|
||||||
</span>
|
|
||||||
</Show>
|
|
||||||
</summary>
|
|
||||||
|
|
||||||
<div class="absolute bottom-full left-0 mb-2 z-[100] min-w-[8rem] overflow-hidden rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md hidden group-open:block animate-in fade-in-0 zoom-in-95 slide-in-from-bottom-2">
|
|
||||||
<ul class="w-full">
|
|
||||||
{
|
|
||||||
limits.clone().into_iter().map(|(val, label)| {
|
|
||||||
let is_active = move || {
|
|
||||||
let current = stats.get().up_limit.unwrap_or(0);
|
|
||||||
(current - val).abs() < 1024
|
|
||||||
};
|
|
||||||
view! {
|
|
||||||
<li>
|
|
||||||
<button
|
|
||||||
class=move || {
|
|
||||||
let base = "relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-xs outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50 hover:bg-accent hover:text-accent-foreground";
|
|
||||||
if is_active() { format!("{} bg-accent text-accent-foreground font-medium", base) } else { base.to_string() }
|
|
||||||
}
|
|
||||||
on:click=move |_| {
|
|
||||||
set_limit("up", val);
|
|
||||||
close_details(up_details_ref);
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span class="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
|
||||||
<Show when=is_active fallback=|| ()>
|
|
||||||
<span>"✓"</span>
|
|
||||||
</Show>
|
|
||||||
</span>
|
|
||||||
{label}
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
}).collect::<Vec<_>>()
|
|
||||||
}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<div class="ml-auto flex items-center gap-4">
|
|
||||||
<button
|
|
||||||
class="inline-flex items-center justify-center rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 hover:bg-accent hover:text-accent-foreground h-7 w-7"
|
|
||||||
title="Settings & Notification Permissions"
|
|
||||||
on:click=move |_| {
|
|
||||||
// Request push notification permission
|
|
||||||
leptos::task::spawn_local(async {
|
|
||||||
// ... existing logic ...
|
|
||||||
crate::store::subscribe_to_push_notifications().await;
|
|
||||||
// ... existing logic ...
|
|
||||||
});
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M9.594 3.94c.09-.542.56-.94 1.11-.94h2.593c.55 0 1.02.398 1.11.94l.213 1.281c.063.374.313.686.645.87.074.04.147.083.22.127.324.196.72.257 1.075.124l1.217-.456a1.125 1.125 0 011.37.49l1.296 2.247a1.125 1.125 0 01-.26 1.431l-1.003.827c-.293.24-.438.613-.431.992a6.759 6.759 0 010 .255c-.007.378.138.75.43.99l1.005.828c.424.35.534.954.26 1.43l-1.298 2.247a1.125 1.125 0 01-1.369.491l-1.217-.456c-.355-.133-.75-.072-1.076.124a6.57 6.57 0 01-.22.128c-.331.183-.581.495-.644.869l-.212 1.28c-.09.543-.56.941-1.11.941h-2.594c-.55 0-1.02-.398-1.11-.94l-.213-1.281c-.062-.374-.312-.686-.644-.87a6.52 6.52 0 01-.22-.127c-.325-.196-.72-.257-1.076-.124l-1.217.456a1.125 1.125 0 01-1.369-.49l-1.297-2.247a1.125 1.125 0 012.6-1.431l1.004-.827c.292-.24.437-.613.43-.992a6.932 6.932 0 010-.255c.007-.378-.138-.75-.43-.99l-1.004-.828a1.125 1.125 0 01-.26-1.43l1.297-2.247a1.125 1.125 0 011.37-.491l1.216.456c.356.133.751.072 1.076-.124.072-.044.146-.087.22-.128.332-.183.582-.495.644-.869l.214-1.281z" />
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,61 +1,55 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos_shadcn_input::Input;
|
use icons::{PanelLeft, Plus};
|
||||||
use leptos_shadcn_button::{Button, ButtonVariant, ButtonSize};
|
use crate::components::torrent::add_torrent::AddTorrentDialogContent;
|
||||||
use crate::components::torrent::add_torrent::AddTorrentDialog;
|
use crate::components::ui::button::{ButtonVariant, ButtonSize};
|
||||||
|
use crate::components::ui::sheet::{Sheet, SheetContent, SheetTrigger, SheetDirection};
|
||||||
|
use crate::components::ui::dialog::{Dialog, DialogContent, DialogTrigger};
|
||||||
|
use crate::components::layout::sidebar::Sidebar;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Toolbar() -> impl IntoView {
|
pub fn Toolbar() -> impl IntoView {
|
||||||
let show_add_modal = signal(false);
|
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
|
||||||
let is_mobile_menu_open = use_context::<RwSignal<bool>>().expect("mobile menu state not provided");
|
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div class="flex min-h-14 h-auto items-center border-b border-border bg-background px-4" style="padding-top: env(safe-area-inset-top);">
|
<div class="flex min-h-14 h-auto items-center border-b border-border bg-background px-4" style="padding-top: env(safe-area-inset-top);">
|
||||||
// Sol kısım: Menü butonu + Add Torrent
|
// Sol kısım: Menü butonu (Mobil) + Add Torrent
|
||||||
<div class="flex items-center gap-3">
|
<div class="flex items-center gap-3">
|
||||||
// Mobile Menu Trigger
|
|
||||||
<Button
|
|
||||||
variant=ButtonVariant::Ghost
|
|
||||||
size=ButtonSize::Icon
|
|
||||||
class="lg:hidden"
|
|
||||||
on_click=Callback::new(move |()| is_mobile_menu_open.update(|v| *v = !*v))
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" class="w-5 h-5 stroke-current"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16"></path></svg>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
// --- MOBILE SHEET (SIDEBAR) ---
|
||||||
class="gap-2 shadow"
|
<div class="lg:hidden">
|
||||||
on_click=Callback::new(move |()| show_add_modal.1.set(true))
|
<Sheet>
|
||||||
>
|
<SheetTrigger variant=ButtonVariant::Ghost size=ButtonSize::Icon class="size-9">
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" class="w-4 h-4 md:w-5 md:h-5">
|
<PanelLeft class="size-5" />
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M12 4.5v15m7.5-7.5h-15" />
|
<span class="hidden">"Menüyü Aç"</span>
|
||||||
</svg>
|
</SheetTrigger>
|
||||||
<span class="hidden sm:inline">"Add Torrent"</span>
|
<SheetContent
|
||||||
<span class="sm:hidden">"Add"</span>
|
direction=SheetDirection::Left
|
||||||
</Button>
|
class="p-0 w-[18rem] bg-card border-r border-border"
|
||||||
</div>
|
hide_close_button=true
|
||||||
|
>
|
||||||
// Sağ kısım: Search kutusu
|
<div class="flex flex-col h-full overflow-hidden">
|
||||||
<div class="flex flex-1 items-center justify-end gap-2">
|
<Sidebar />
|
||||||
<div class="hidden md:flex items-center gap-2 w-full max-w-xs">
|
</div>
|
||||||
<div class="relative flex-1">
|
</SheetContent>
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="absolute left-2.5 top-1/2 -translate-y-1/2 w-4 h-4 text-muted-foreground pointer-events-none">
|
</Sheet>
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="m21 21-5.197-5.197m0 0A7.5 7.5 0 1 0 5.196 5.196a7.5 7.5 0 0 0 10.607 10.607Z" />
|
|
||||||
</svg>
|
|
||||||
<Input
|
|
||||||
input_type="search"
|
|
||||||
placeholder="Search..."
|
|
||||||
value=MaybeProp::derive(move || Some(store.search_query.get()))
|
|
||||||
on_change=Callback::new(move |val: String| store.search_query.set(val))
|
|
||||||
class="pl-8 h-9"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<Dialog>
|
||||||
|
<DialogTrigger
|
||||||
|
variant=ButtonVariant::Default
|
||||||
|
class="gap-2"
|
||||||
|
>
|
||||||
|
<Plus class="w-4 h-4 md:w-5 md:h-5" />
|
||||||
|
<span class="hidden sm:inline">"Add Torrent"</span>
|
||||||
|
<span class="sm:hidden">"Add"</span>
|
||||||
|
</DialogTrigger>
|
||||||
|
<DialogContent id="add-torrent-dialog" class="sm:max-w-[425px]">
|
||||||
|
<AddTorrentDialogContent />
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Show when=move || show_add_modal.0.get()>
|
// Sağ kısım boş
|
||||||
<AddTorrentDialog on_close=Callback::new(move |()| show_add_modal.1.set(false)) />
|
<div class="flex flex-1 items-center justify-end gap-2">
|
||||||
</Show>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,8 @@
|
|||||||
|
pub mod hooks;
|
||||||
pub mod context_menu;
|
pub mod context_menu;
|
||||||
pub mod layout;
|
pub mod layout;
|
||||||
pub mod torrent;
|
pub mod torrent;
|
||||||
pub mod auth;
|
pub mod auth;
|
||||||
|
// pub mod toast; (Removed)
|
||||||
|
pub mod ui;
|
||||||
|
pub mod demos;
|
||||||
|
|||||||
111
frontend/src/components/toast.rs
Normal file
111
frontend/src/components/toast.rs
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use uuid::Uuid;
|
||||||
|
use shared::NotificationLevel;
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
|
pub struct Toast {
|
||||||
|
pub id: String,
|
||||||
|
pub message: String,
|
||||||
|
pub level: NotificationLevel,
|
||||||
|
pub visible: RwSignal<bool>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy)]
|
||||||
|
pub struct ToastContext {
|
||||||
|
pub toasts: RwSignal<HashMap<String, Toast>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToastContext {
|
||||||
|
pub fn add(&self, message: impl Into<String>, level: NotificationLevel) {
|
||||||
|
let id = Uuid::new_v4().to_string();
|
||||||
|
let message = message.into();
|
||||||
|
let toast = Toast {
|
||||||
|
id: id.clone(),
|
||||||
|
message,
|
||||||
|
level,
|
||||||
|
visible: RwSignal::new(true),
|
||||||
|
};
|
||||||
|
|
||||||
|
self.toasts.update(|m| {
|
||||||
|
m.insert(id.clone(), toast);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Auto remove after 5 seconds
|
||||||
|
let toasts = self.toasts;
|
||||||
|
let id_clone = id.clone();
|
||||||
|
leptos::task::spawn_local(async move {
|
||||||
|
gloo_timers::future::TimeoutFuture::new(5000).await;
|
||||||
|
toasts.update(|m| {
|
||||||
|
if let Some(t) = m.get(&id_clone) {
|
||||||
|
t.visible.set(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// Wait for animation
|
||||||
|
gloo_timers::future::TimeoutFuture::new(300).await;
|
||||||
|
toasts.update(|m| {
|
||||||
|
m.remove(&id_clone);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn provide_toast_context() {
|
||||||
|
let toasts = RwSignal::new(HashMap::new());
|
||||||
|
provide_context(ToastContext { toasts });
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Toaster() -> impl IntoView {
|
||||||
|
let context = expect_context::<ToastContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class="fixed top-4 right-4 z-[100] flex flex-col gap-2 w-full max-w-sm pointer-events-none">
|
||||||
|
{move || {
|
||||||
|
context.toasts.get().into_values().map(|toast| {
|
||||||
|
view! { <ToastItem toast=toast /> }
|
||||||
|
}).collect::<Vec<_>>()
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn ToastItem(toast: Toast) -> impl IntoView {
|
||||||
|
let (visible, set_visible) = (toast.visible, toast.visible.write_only());
|
||||||
|
|
||||||
|
let base_classes = "pointer-events-auto relative w-full rounded-lg border p-4 shadow-lg transition-all duration-300 ease-in-out";
|
||||||
|
let color_classes = match toast.level {
|
||||||
|
NotificationLevel::Success => "bg-green-50 text-green-900 border-green-200 dark:bg-green-900 dark:text-green-100 dark:border-green-800",
|
||||||
|
NotificationLevel::Error => "bg-red-50 text-red-900 border-red-200 dark:bg-red-900 dark:text-red-100 dark:border-red-800",
|
||||||
|
NotificationLevel::Warning => "bg-yellow-50 text-yellow-900 border-yellow-200 dark:bg-yellow-900 dark:text-yellow-100 dark:border-yellow-800",
|
||||||
|
NotificationLevel::Info => "bg-blue-50 text-blue-900 border-blue-200 dark:bg-blue-900 dark:text-blue-100 dark:border-blue-800",
|
||||||
|
};
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
class=move || format!("{} {} {}",
|
||||||
|
base_classes,
|
||||||
|
color_classes,
|
||||||
|
if visible.get() { "opacity-100 translate-x-0" } else { "opacity-0 translate-x-full" }
|
||||||
|
)
|
||||||
|
role="alert"
|
||||||
|
>
|
||||||
|
<div class="flex items-start gap-4">
|
||||||
|
<div class="flex-1">
|
||||||
|
<p class="text-sm font-medium">{toast.message.clone()}</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
class="inline-flex shrink-0 opacity-50 hover:opacity-100 focus:opacity-100 focus:outline-none"
|
||||||
|
on:click=move |_| set_visible.set(false)
|
||||||
|
>
|
||||||
|
<span class="sr-only">"Kapat"</span>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="h-4 w-4">
|
||||||
|
<line x1="18" x2="6" y1="6" y2="18"></line>
|
||||||
|
<line x1="6" x2="18" y1="6" y2="18"></line>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,40 +1,48 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use leptos_shadcn_input::Input;
|
use wasm_bindgen::JsCast;
|
||||||
use leptos_shadcn_button::{Button, ButtonVariant};
|
use crate::components::ui::input::{Input, InputType};
|
||||||
use leptos_shadcn_alert::{Alert, AlertDescription, AlertVariant};
|
|
||||||
use crate::store::TorrentStore;
|
|
||||||
use crate::api;
|
use crate::api;
|
||||||
|
use crate::components::ui::button::Button;
|
||||||
|
use crate::components::ui::dialog::{
|
||||||
|
DialogBody, DialogHeader, DialogTitle, DialogDescription, DialogFooter, DialogClose
|
||||||
|
};
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn AddTorrentDialog(
|
pub fn AddTorrentDialogContent() -> impl IntoView {
|
||||||
on_close: Callback<()>,
|
let uri = RwSignal::new(String::new());
|
||||||
) -> impl IntoView {
|
|
||||||
let store = use_context::<TorrentStore>().expect("TorrentStore not provided");
|
|
||||||
|
|
||||||
let uri = signal(String::new());
|
|
||||||
let is_loading = signal(false);
|
let is_loading = signal(false);
|
||||||
let error_msg = signal(Option::<String>::None);
|
let error_msg = signal(Option::<String>::None);
|
||||||
|
|
||||||
let handle_submit = move |ev: web_sys::SubmitEvent| {
|
let handle_submit = move |ev: web_sys::SubmitEvent| {
|
||||||
ev.prevent_default();
|
ev.prevent_default();
|
||||||
let uri_val = uri.0.get();
|
let uri_val = uri.get();
|
||||||
|
|
||||||
if uri_val.is_empty() {
|
if uri_val.is_empty() {
|
||||||
error_msg.1.set(Some("Please enter a Magnet URI or URL".to_string()));
|
error_msg.1.set(Some("Lütfen bir Magnet URI veya URL girin".to_string()));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
is_loading.1.set(true);
|
is_loading.1.set(true);
|
||||||
error_msg.1.set(None);
|
error_msg.1.set(None);
|
||||||
|
|
||||||
let on_close = on_close.clone();
|
|
||||||
spawn_local(async move {
|
spawn_local(async move {
|
||||||
match api::torrent::add(&uri_val).await {
|
match api::torrent::add(&uri_val).await {
|
||||||
Ok(_) => {
|
Ok(_) => {
|
||||||
log::info!("Torrent added successfully");
|
log::info!("Torrent added successfully");
|
||||||
crate::store::toast_success("Torrent başarıyla eklendi");
|
crate::store::toast_success("Torrent başarıyla eklendi");
|
||||||
on_close.run(());
|
|
||||||
|
// Programmatically close the dialog by triggering the close button
|
||||||
|
if let Some(doc) = web_sys::window().and_then(|w| w.document()) {
|
||||||
|
if let Some(el) = doc.get_element_by_id("add-torrent-dialog") {
|
||||||
|
if let Some(close_btn) = el.query_selector("[data-dialog-close]").ok().flatten() {
|
||||||
|
let _ = close_btn.dyn_into::<web_sys::HtmlElement>().map(|btn| btn.click());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uri.set(String::new());
|
||||||
|
is_loading.1.set(false);
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Failed to add torrent: {:?}", e);
|
log::error!("Failed to add torrent: {:?}", e);
|
||||||
@@ -45,53 +53,37 @@ pub fn AddTorrentDialog(
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
let handle_backdrop = {
|
|
||||||
let on_close = on_close.clone();
|
|
||||||
move |e: web_sys::MouseEvent| {
|
|
||||||
e.stop_propagation();
|
|
||||||
on_close.run(());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
// Backdrop overlay
|
<DialogBody>
|
||||||
<div
|
<DialogHeader>
|
||||||
class="fixed inset-0 z-50 bg-background/80 backdrop-blur-sm"
|
<DialogTitle>"Add Torrent"</DialogTitle>
|
||||||
on:click=handle_backdrop
|
<DialogDescription>
|
||||||
/>
|
"Enter a Magnet link or a .torrent file URL."
|
||||||
// Dialog panel
|
</DialogDescription>
|
||||||
<div class="fixed left-1/2 top-1/2 z-50 grid w-full max-w-lg -translate-x-1/2 -translate-y-1/2 gap-4 border bg-card p-6 shadow-lg rounded-lg sm:max-w-[425px]">
|
</DialogHeader>
|
||||||
// Header
|
|
||||||
<div class="flex flex-col space-y-1.5 text-center sm:text-left">
|
|
||||||
<h2 class="text-lg font-semibold leading-none tracking-tight">"Add Torrent"</h2>
|
|
||||||
<p class="text-sm text-muted-foreground">"Enter a Magnet link or a .torrent file URL."</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<form on:submit=handle_submit class="space-y-4">
|
<form on:submit=handle_submit class="space-y-4 pt-4">
|
||||||
<Input
|
<Input
|
||||||
input_type="text"
|
r#type=InputType::Text
|
||||||
placeholder="magnet:?xt=urn:btih:..."
|
placeholder="magnet:?xt=urn:btih:..."
|
||||||
value=MaybeProp::derive(move || Some(uri.0.get()))
|
bind_value=uri
|
||||||
on_change=Callback::new(move |val: String| uri.1.set(val))
|
disabled=is_loading.0.get()
|
||||||
disabled=Signal::derive(move || is_loading.0.get())
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{move || error_msg.0.get().map(|msg| view! {
|
{move || error_msg.0.get().map(|msg| view! {
|
||||||
<Alert variant=AlertVariant::Destructive>
|
<div class="rounded-lg border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
|
||||||
<AlertDescription>{msg}</AlertDescription>
|
{msg}
|
||||||
</Alert>
|
</div>
|
||||||
})}
|
})}
|
||||||
|
|
||||||
<div class="flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2">
|
<DialogFooter class="pt-2">
|
||||||
<Button
|
<DialogClose>
|
||||||
variant=ButtonVariant::Ghost
|
|
||||||
on_click=Callback::new(move |()| {
|
|
||||||
on_close.run(());
|
|
||||||
})
|
|
||||||
>
|
|
||||||
"Cancel"
|
"Cancel"
|
||||||
</Button>
|
</DialogClose>
|
||||||
<Button disabled=Signal::derive(move || is_loading.0.get())>
|
<Button
|
||||||
|
attr:r#type="submit"
|
||||||
|
attr:disabled=move || is_loading.0.get()
|
||||||
|
>
|
||||||
{move || if is_loading.0.get() {
|
{move || if is_loading.0.get() {
|
||||||
leptos::either::Either::Left(view! {
|
leptos::either::Either::Left(view! {
|
||||||
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
<span class="animate-spin mr-2 h-4 w-4 border-2 border-current border-t-transparent rounded-full"></span>
|
||||||
@@ -101,20 +93,8 @@ pub fn AddTorrentDialog(
|
|||||||
leptos::either::Either::Right(view! { "Add" })
|
leptos::either::Either::Right(view! { "Add" })
|
||||||
}}
|
}}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</DialogFooter>
|
||||||
</form>
|
</form>
|
||||||
|
</DialogBody>
|
||||||
// Close button (X)
|
|
||||||
<button
|
|
||||||
class="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none"
|
|
||||||
on:click=move |_| on_close.run(())
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="h-4 w-4">
|
|
||||||
<path d="M18 6 6 18"></path>
|
|
||||||
<path d="m6 6 12 12"></path>
|
|
||||||
</svg>
|
|
||||||
<span class="sr-only">"Close"</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,156 +0,0 @@
|
|||||||
use leptos::prelude::*;
|
|
||||||
use leptos_shadcn_tabs::{Tabs, TabsList, TabsTrigger, TabsContent};
|
|
||||||
|
|
||||||
fn format_bytes(bytes: i64) -> String {
|
|
||||||
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
|
||||||
if bytes < 1024 { return format!("{} B", bytes); }
|
|
||||||
let i = (bytes as f64).log2().div_euclid(10.0) as usize;
|
|
||||||
format!("{:.1} {}", (bytes as f64) / 1024_f64.powi(i as i32), UNITS[i])
|
|
||||||
}
|
|
||||||
|
|
||||||
fn format_speed(bytes_per_sec: i64) -> String {
|
|
||||||
if bytes_per_sec == 0 { return "0 B/s".to_string(); }
|
|
||||||
format!("{}/s", format_bytes(bytes_per_sec))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn format_date(timestamp: i64) -> String {
|
|
||||||
if timestamp <= 0 { return "N/A".to_string(); }
|
|
||||||
let dt = chrono::DateTime::from_timestamp(timestamp, 0);
|
|
||||||
match dt { Some(dt) => dt.format("%d/%m/%Y %H:%M").to_string(), None => "N/A".to_string() }
|
|
||||||
}
|
|
||||||
|
|
||||||
fn format_duration(seconds: i64) -> String {
|
|
||||||
if seconds <= 0 { return "∞".to_string(); }
|
|
||||||
let days = seconds / 86400;
|
|
||||||
let hours = (seconds % 86400) / 3600;
|
|
||||||
let minutes = (seconds % 3600) / 60;
|
|
||||||
let secs = seconds % 60;
|
|
||||||
if days > 0 { format!("{}d {}h", days, hours) }
|
|
||||||
else if hours > 0 { format!("{}h {}m", hours, minutes) }
|
|
||||||
else if minutes > 0 { format!("{}m {}s", minutes, secs) }
|
|
||||||
else { format!("{}s", secs) }
|
|
||||||
}
|
|
||||||
|
|
||||||
#[component]
|
|
||||||
pub fn TorrentDetail() -> impl IntoView {
|
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
|
||||||
|
|
||||||
let torrent = Memo::new(move |_| {
|
|
||||||
let hash = store.selected_torrent.get()?;
|
|
||||||
store.torrents.with(|map| map.get(&hash).cloned())
|
|
||||||
});
|
|
||||||
|
|
||||||
let close = move |_| {
|
|
||||||
store.selected_torrent.set(None);
|
|
||||||
};
|
|
||||||
|
|
||||||
view! {
|
|
||||||
<Show when=move || torrent.get().is_some()>
|
|
||||||
{move || {
|
|
||||||
let t = torrent.get().unwrap();
|
|
||||||
let name = t.name.clone();
|
|
||||||
let status_color = match t.status {
|
|
||||||
shared::TorrentStatus::Seeding => "text-green-500",
|
|
||||||
shared::TorrentStatus::Downloading => "text-blue-500",
|
|
||||||
shared::TorrentStatus::Paused => "text-yellow-500",
|
|
||||||
shared::TorrentStatus::Error => "text-red-500",
|
|
||||||
_ => "text-muted-foreground",
|
|
||||||
};
|
|
||||||
let status_text = format!("{:?}", t.status);
|
|
||||||
|
|
||||||
view! {
|
|
||||||
<div class="border-t border-border bg-card flex flex-col" style="height: 280px; min-height: 200px;">
|
|
||||||
// Header
|
|
||||||
<div class="flex items-center justify-between px-4 py-2 border-b border-border bg-muted/30">
|
|
||||||
<div class="flex items-center gap-3 min-w-0 flex-1">
|
|
||||||
<h3 class="text-sm font-semibold truncate">{name}</h3>
|
|
||||||
<span class={format!("text-xs font-medium {}", status_color)}>{status_text}</span>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
class="inline-flex items-center justify-center rounded-md text-sm font-medium hover:bg-accent hover:text-accent-foreground h-7 w-7 text-muted-foreground shrink-0"
|
|
||||||
on:click=close
|
|
||||||
title="Close"
|
|
||||||
>
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" class="w-4 h-4">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
// Tabs
|
|
||||||
<Tabs default_value="general" class="flex-1 flex flex-col overflow-hidden">
|
|
||||||
<div class="px-4 pt-2">
|
|
||||||
<TabsList class="w-full">
|
|
||||||
<TabsTrigger value="general">"General"</TabsTrigger>
|
|
||||||
<TabsTrigger value="transfer">"Transfer"</TabsTrigger>
|
|
||||||
<TabsTrigger value="files">"Files"</TabsTrigger>
|
|
||||||
<TabsTrigger value="peers">"Peers"</TabsTrigger>
|
|
||||||
</TabsList>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<TabsContent value="general" class="flex-1 overflow-y-auto px-4 pb-3">
|
|
||||||
<div class="grid grid-cols-2 md:grid-cols-4 gap-x-6 gap-y-2 text-sm">
|
|
||||||
<DetailItem label="Size" value=format_bytes(t.size) />
|
|
||||||
<DetailItem label="Downloaded" value=format_bytes(t.completed) />
|
|
||||||
<DetailItem label="Progress" value=format!("{:.1}%", t.percent_complete) />
|
|
||||||
<DetailItem label="Added" value=format_date(t.added_date) />
|
|
||||||
<DetailItem label="Hash" value={
|
|
||||||
let hash = store.selected_torrent.get().unwrap_or_default();
|
|
||||||
format!("{}…", &hash[..std::cmp::min(16, hash.len())])
|
|
||||||
} />
|
|
||||||
<DetailItem label="Label" value=t.label.clone().unwrap_or_else(|| "—".to_string()) />
|
|
||||||
<DetailItem label="Error" value={
|
|
||||||
if t.error_message.is_empty() { "None".to_string() } else { t.error_message.clone() }
|
|
||||||
} />
|
|
||||||
</div>
|
|
||||||
</TabsContent>
|
|
||||||
|
|
||||||
<TabsContent value="transfer" class="flex-1 overflow-y-auto px-4 pb-3">
|
|
||||||
<div class="grid grid-cols-2 md:grid-cols-4 gap-x-6 gap-y-2 text-sm">
|
|
||||||
<DetailItem label="Download Speed" value=format_speed(t.down_rate) />
|
|
||||||
<DetailItem label="Upload Speed" value=format_speed(t.up_rate) />
|
|
||||||
<DetailItem label="ETA" value=format_duration(t.eta) />
|
|
||||||
<DetailItem label="Downloaded" value=format_bytes(t.completed) />
|
|
||||||
<DetailItem label="Total Size" value=format_bytes(t.size) />
|
|
||||||
<DetailItem label="Remaining" value=format_bytes(t.size - t.completed) />
|
|
||||||
</div>
|
|
||||||
</TabsContent>
|
|
||||||
|
|
||||||
<TabsContent value="files" class="flex-1 overflow-y-auto px-4 pb-3">
|
|
||||||
<div class="text-sm text-muted-foreground flex items-center gap-2 py-4">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M2.25 12.75V12A2.25 2.25 0 014.5 9.75h15A2.25 2.25 0 0121.75 12v.75m-8.69-6.44l-2.12-2.12a1.5 1.5 0 00-1.061-.44H4.5A2.25 2.25 0 002.25 6v12a2.25 2.25 0 002.25 2.25h15A2.25 2.25 0 0021.75 18V9a2.25 2.25 0 00-2.25-2.25h-5.379a1.5 1.5 0 01-1.06-.44z" />
|
|
||||||
</svg>
|
|
||||||
"File list will be available when file API is connected."
|
|
||||||
</div>
|
|
||||||
</TabsContent>
|
|
||||||
|
|
||||||
<TabsContent value="peers" class="flex-1 overflow-y-auto px-4 pb-3">
|
|
||||||
<div class="text-sm text-muted-foreground flex items-center gap-2 py-4">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" />
|
|
||||||
</svg>
|
|
||||||
"Peer list will be available when peer API is connected."
|
|
||||||
</div>
|
|
||||||
</TabsContent>
|
|
||||||
</Tabs>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
</Show>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[component]
|
|
||||||
fn DetailItem(
|
|
||||||
#[prop(into)] label: String,
|
|
||||||
#[prop(into)] value: String,
|
|
||||||
) -> impl IntoView {
|
|
||||||
let title = value.clone();
|
|
||||||
view! {
|
|
||||||
<div class="flex flex-col gap-0.5 py-1">
|
|
||||||
<span class="text-[10px] uppercase tracking-wider text-muted-foreground font-medium">{label}</span>
|
|
||||||
<span class="text-foreground font-mono text-xs truncate" title=title>{value}</span>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
260
frontend/src/components/torrent/details.rs
Normal file
260
frontend/src/components/torrent/details.rs
Normal file
@@ -0,0 +1,260 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use crate::components::ui::tabs::*;
|
||||||
|
use crate::components::ui::skeleton::*;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TorrentDetailsPanel() -> impl IntoView {
|
||||||
|
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
||||||
|
|
||||||
|
let selected_torrent = Memo::new(move |_| {
|
||||||
|
let hash = store.selected_torrent.get()?;
|
||||||
|
store.torrents.with(|map| map.get(&hash).cloned())
|
||||||
|
});
|
||||||
|
|
||||||
|
let is_open = Signal::derive(move || store.selected_torrent.get().is_some());
|
||||||
|
|
||||||
|
view! {
|
||||||
|
// Mobil overlay backdrop
|
||||||
|
<div
|
||||||
|
class=move || if is_open.get() {
|
||||||
|
"fixed inset-0 bg-black/40 z-30 md:hidden backdrop-blur-sm transition-opacity duration-300 opacity-100"
|
||||||
|
} else {
|
||||||
|
"fixed inset-0 bg-black/0 z-30 md:hidden pointer-events-none transition-opacity duration-300 opacity-0"
|
||||||
|
}
|
||||||
|
on:click=move |_| store.selected_torrent.set(None)
|
||||||
|
/>
|
||||||
|
|
||||||
|
// Panel — masaüstünde sağ kolonda sabit, mobilde sağdan açılan overlay
|
||||||
|
<div class=move || {
|
||||||
|
if is_open.get() {
|
||||||
|
// Açık: masaüstünde görünür, mobilde sağdan gelir
|
||||||
|
"w-full md:w-[380px] md:min-w-[380px] shrink-0 \
|
||||||
|
flex flex-col border-l border-border bg-card \
|
||||||
|
fixed top-0 right-0 bottom-0 z-40 \
|
||||||
|
translate-x-0 \
|
||||||
|
md:static md:z-auto md:translate-x-0 \
|
||||||
|
transition-transform duration-300 ease-out shadow-2xl md:shadow-none"
|
||||||
|
} else {
|
||||||
|
// Kapalı: masaüstünde gizli, mobilde sağa kayar
|
||||||
|
"w-full md:w-0 shrink-0 overflow-hidden border-none \
|
||||||
|
fixed top-0 right-0 bottom-0 z-40 \
|
||||||
|
translate-x-full \
|
||||||
|
md:static md:z-auto md:translate-x-0 \
|
||||||
|
transition-transform duration-300 ease-in pointer-events-none"
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
// İpucu: panel kapalıyken içeriği render etme
|
||||||
|
<Show when=move || is_open.get()>
|
||||||
|
// Başlık
|
||||||
|
<div class="px-4 py-3 border-b flex items-center justify-between shrink-0 bg-card">
|
||||||
|
<div class="flex flex-col gap-0.5 min-w-0 flex-1">
|
||||||
|
<Show
|
||||||
|
when=move || selected_torrent.get().is_some()
|
||||||
|
fallback=move || view! { <Skeleton class="h-5 w-40" /> }
|
||||||
|
>
|
||||||
|
<h2 class="font-bold text-sm truncate leading-tight">
|
||||||
|
{move || selected_torrent.get().map(|t| t.name).unwrap_or_default()}
|
||||||
|
</h2>
|
||||||
|
</Show>
|
||||||
|
<Show
|
||||||
|
when=move || selected_torrent.get().is_some()
|
||||||
|
fallback=move || view! { <Skeleton class="h-3 w-20 mt-1" /> }
|
||||||
|
>
|
||||||
|
<p class="text-[10px] text-muted-foreground uppercase tracking-widest font-semibold flex items-center gap-1.5">
|
||||||
|
{move || selected_torrent.get().map(|t| format!("{:?}", t.status)).unwrap_or_default()}
|
||||||
|
<span class="bg-primary/20 text-primary px-1 py-0.5 rounded text-[9px] lowercase">
|
||||||
|
{move || selected_torrent.get().map(|t| format!("{:.1}%", t.percent_complete)).unwrap_or_default()}
|
||||||
|
</span>
|
||||||
|
</p>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
// Kapat butonu
|
||||||
|
<button
|
||||||
|
class="rounded-full p-1.5 hover:bg-muted transition-colors text-muted-foreground hover:text-foreground shrink-0 ml-2"
|
||||||
|
on:click=move |_| store.selected_torrent.set(None)
|
||||||
|
>
|
||||||
|
<icons::X class="size-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
// Sekmeler + içerik
|
||||||
|
<div class="flex-1 overflow-hidden flex flex-col min-h-0">
|
||||||
|
<Tabs default_value="general" class="flex-1 h-full min-h-0 flex flex-col">
|
||||||
|
<TabsList class="w-full justify-start rounded-none border-b bg-transparent p-0 shrink-0 px-2">
|
||||||
|
<TabsTrigger
|
||||||
|
value="general"
|
||||||
|
class="data-[state=active]:bg-transparent data-[state=active]:shadow-none data-[state=active]:border-b-2 data-[state=active]:border-primary rounded-none text-xs h-9"
|
||||||
|
>
|
||||||
|
"Genel"
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger
|
||||||
|
value="files"
|
||||||
|
class="data-[state=active]:bg-transparent data-[state=active]:shadow-none data-[state=active]:border-b-2 data-[state=active]:border-primary rounded-none text-xs h-9"
|
||||||
|
>
|
||||||
|
"Dosyalar"
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger
|
||||||
|
value="trackers"
|
||||||
|
class="data-[state=active]:bg-transparent data-[state=active]:shadow-none data-[state=active]:border-b-2 data-[state=active]:border-primary rounded-none text-xs h-9"
|
||||||
|
>
|
||||||
|
"İzleyiciler"
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger
|
||||||
|
value="peers"
|
||||||
|
class="data-[state=active]:bg-transparent data-[state=active]:shadow-none data-[state=active]:border-b-2 data-[state=active]:border-primary rounded-none text-xs h-9"
|
||||||
|
>
|
||||||
|
"Eşler"
|
||||||
|
</TabsTrigger>
|
||||||
|
</TabsList>
|
||||||
|
|
||||||
|
<crate::components::ui::scroll_area::ScrollArea class="flex-1 min-h-0">
|
||||||
|
<TabsContent value="general" class="p-4 space-y-5 animate-in fade-in duration-200">
|
||||||
|
<crate::components::ui::shimmer::Shimmer
|
||||||
|
loading=Signal::derive(move || selected_torrent.get().is_none())
|
||||||
|
shimmer_color="rgba(0,0,0,0.06)"
|
||||||
|
background_color="rgba(0,0,0,0.04)"
|
||||||
|
>
|
||||||
|
{move || {
|
||||||
|
let t = selected_torrent.get().unwrap_or_else(|| shared::Torrent {
|
||||||
|
hash: "----------------------------------------".to_string(),
|
||||||
|
name: "Yükleniyor...".to_string(),
|
||||||
|
size: 0,
|
||||||
|
completed: 0,
|
||||||
|
down_rate: 0,
|
||||||
|
up_rate: 0,
|
||||||
|
eta: 0,
|
||||||
|
percent_complete: 0.0,
|
||||||
|
status: shared::TorrentStatus::Downloading,
|
||||||
|
error_message: "".to_string(),
|
||||||
|
added_date: 0,
|
||||||
|
label: None,
|
||||||
|
ratio: 0.0,
|
||||||
|
uploaded: 0,
|
||||||
|
wasted: 0,
|
||||||
|
save_path: "Yükleniyor...".to_string(),
|
||||||
|
free_disk_space: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class="flex flex-col gap-5">
|
||||||
|
// Aktarım
|
||||||
|
<div>
|
||||||
|
<h3 class="text-[10px] font-bold border-b pb-1.5 mb-3 uppercase tracking-widest text-muted-foreground">"Aktarım"</h3>
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<InfoItem label="Kalan" value=format_duration(t.eta) />
|
||||||
|
<InfoItem label="Paylaşım Oranı" value=format!("{:.3}", t.ratio) />
|
||||||
|
<InfoItem label="İndirilen" value=format_bytes(t.completed) />
|
||||||
|
<InfoItem label="İndirme Hızı" value=format_speed(t.down_rate) class="text-blue-500" />
|
||||||
|
<InfoItem label="Gönderilen" value=format_bytes(t.uploaded) />
|
||||||
|
<InfoItem label="Gönderme Hızı" value=format_speed(t.up_rate) class="text-green-500" />
|
||||||
|
<InfoItem label="Boşa Giden" value=format_bytes(t.wasted) />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
// Genel
|
||||||
|
<div>
|
||||||
|
<h3 class="text-[10px] font-bold border-b pb-1.5 mb-3 uppercase tracking-widest text-muted-foreground">"Genel"</h3>
|
||||||
|
<div class="flex flex-col gap-3">
|
||||||
|
<InfoItem label="Kaydedilen Yer" value=t.save_path class="break-all font-mono text-xs" />
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<InfoItem label="Boş Disk Alanı" value=format_bytes(t.free_disk_space) />
|
||||||
|
<InfoItem label="Oluşturulma Tarihi" value=format_date(t.added_date) />
|
||||||
|
</div>
|
||||||
|
<InfoItem label="Hash" value=t.hash class="break-all font-mono text-[10px]" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
</crate::components::ui::shimmer::Shimmer>
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
|
<TabsContent value="files" class="h-full">
|
||||||
|
{move || match selected_torrent.get() {
|
||||||
|
Some(t) => leptos::either::Either::Left(view! {
|
||||||
|
<div class="h-full">
|
||||||
|
<crate::components::torrent::files::TorrentFilesTab hash=t.hash />
|
||||||
|
</div>
|
||||||
|
}),
|
||||||
|
None => leptos::either::Either::Right(view! {
|
||||||
|
<div class="flex flex-col items-center justify-center h-48 opacity-60 gap-2">
|
||||||
|
<icons::File class="size-10 text-muted-foreground" />
|
||||||
|
<p class="text-sm font-medium">"Dosya yükleniyor..."</p>
|
||||||
|
</div>
|
||||||
|
}),
|
||||||
|
}}
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
|
<TabsContent value="trackers" class="h-full">
|
||||||
|
{move || match selected_torrent.get() {
|
||||||
|
Some(t) => leptos::either::Either::Left(view! {
|
||||||
|
<div class="h-full">
|
||||||
|
<crate::components::torrent::trackers::TorrentTrackersTab hash=t.hash />
|
||||||
|
</div>
|
||||||
|
}),
|
||||||
|
None => leptos::either::Either::Right(view! {
|
||||||
|
<div class="flex flex-col items-center justify-center h-48 opacity-60 gap-2">
|
||||||
|
<icons::Settings2 class="size-10 text-muted-foreground" />
|
||||||
|
<p class="text-sm font-medium">"İzleyici yükleniyor..."</p>
|
||||||
|
</div>
|
||||||
|
}),
|
||||||
|
}}
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
|
<TabsContent value="peers" class="h-full">
|
||||||
|
<div class="flex flex-col items-center justify-center h-48 opacity-60 gap-2">
|
||||||
|
<icons::Users class="size-10 text-muted-foreground" />
|
||||||
|
<p class="text-sm font-medium">"Eş listesi yakında eklenecek"</p>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</crate::components::ui::scroll_area::ScrollArea>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn InfoItem(
|
||||||
|
label: &'static str,
|
||||||
|
value: String,
|
||||||
|
#[prop(optional)] class: &'static str
|
||||||
|
) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<div class=tailwind_fuse::tw_merge!("flex flex-col gap-0.5", class)>
|
||||||
|
<span class="text-[9px] font-semibold text-muted-foreground uppercase tracking-wider opacity-70">{label}</span>
|
||||||
|
<span class="text-xs font-medium leading-tight">{value}</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_bytes(bytes: i64) -> String {
|
||||||
|
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
||||||
|
if bytes < 1024 { return format!("{} B", bytes); }
|
||||||
|
let i = (bytes as f64).log2().div_euclid(10.0) as usize;
|
||||||
|
format!("{:.1} {}", (bytes as f64) / 1024_f64.powi(i as i32), UNITS[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_speed(bytes_per_sec: i64) -> String {
|
||||||
|
if bytes_per_sec == 0 { return "0 B/s".to_string(); }
|
||||||
|
format!("{}/s", format_bytes(bytes_per_sec))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_duration(seconds: i64) -> String {
|
||||||
|
if seconds <= 0 { return "∞".to_string(); }
|
||||||
|
let days = seconds / 86400;
|
||||||
|
let hours = (seconds % 86400) / 3600;
|
||||||
|
let minutes = (seconds % 3600) / 60;
|
||||||
|
let secs = seconds % 60;
|
||||||
|
if days > 0 { format!("{}g {}s", days, hours) }
|
||||||
|
else if hours > 0 { format!("{}s {}d", hours, minutes) }
|
||||||
|
else if minutes > 0 { format!("{}d {}sn", minutes, secs) }
|
||||||
|
else { format!("{}sn", secs) }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_date(timestamp: i64) -> String {
|
||||||
|
if timestamp <= 0 { return "N/A".to_string(); }
|
||||||
|
let dt = chrono::DateTime::from_timestamp(timestamp, 0);
|
||||||
|
match dt { Some(dt) => dt.format("%d/%m/%Y %H:%M").to_string(), None => "N/A".to_string() }
|
||||||
|
}
|
||||||
216
frontend/src/components/torrent/files.rs
Normal file
216
frontend/src/components/torrent/files.rs
Normal file
@@ -0,0 +1,216 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use crate::components::ui::table::*;
|
||||||
|
use crate::components::ui::badge::*;
|
||||||
|
use crate::components::ui::shimmer::*;
|
||||||
|
use crate::components::ui::context_menu::*;
|
||||||
|
use shared::TorrentFile;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TorrentFilesTab(hash: String) -> impl IntoView {
|
||||||
|
let hash_clone = hash.clone();
|
||||||
|
|
||||||
|
// Fetch files resource
|
||||||
|
let files_resource = Resource::new(
|
||||||
|
move || hash_clone.clone(),
|
||||||
|
|h| async move { shared::server_fns::torrent::get_files(h).await.unwrap_or_default() }
|
||||||
|
);
|
||||||
|
|
||||||
|
// Callback to trigger a refetch — safe, doesn't destroy existing components
|
||||||
|
let on_refresh = Callback::new(move |_: ()| {
|
||||||
|
files_resource.refetch();
|
||||||
|
});
|
||||||
|
|
||||||
|
let stored_hash = StoredValue::new(hash);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Suspense fallback=move || view! { <FilesFallback /> }>
|
||||||
|
{move || {
|
||||||
|
let files = files_resource.get().unwrap_or_default();
|
||||||
|
|
||||||
|
if files.is_empty() {
|
||||||
|
return view! {
|
||||||
|
<div class="flex flex-col items-center justify-center h-48 opacity-60">
|
||||||
|
<icons::File class="size-12 mb-3 text-muted-foreground" />
|
||||||
|
<p class="text-sm font-medium">"Bu torrent için dosya bulunamadı."</p>
|
||||||
|
</div>
|
||||||
|
}.into_any();
|
||||||
|
}
|
||||||
|
|
||||||
|
let files_len = files.len();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class="space-y-4">
|
||||||
|
<TableWrapper class="bg-card/50">
|
||||||
|
<Table>
|
||||||
|
<TableHeader class="sticky top-0 bg-muted/80 backdrop-blur-sm z-10">
|
||||||
|
<TableRow class="hover:bg-transparent">
|
||||||
|
<TableHead class="w-12 text-center text-xs">"#"</TableHead>
|
||||||
|
<TableHead class="text-xs">"Dosya Adı"</TableHead>
|
||||||
|
<TableHead class="w-24 text-right text-xs">"Boyut"</TableHead>
|
||||||
|
<TableHead class="w-24 text-right text-xs">"Tamamlanan"</TableHead>
|
||||||
|
<TableHead class="w-24 text-center text-xs">"Öncelik"</TableHead>
|
||||||
|
</TableRow>
|
||||||
|
</TableHeader>
|
||||||
|
<TableBody>
|
||||||
|
<For
|
||||||
|
each=move || files.clone()
|
||||||
|
key=|f| f.index
|
||||||
|
children={move |f| {
|
||||||
|
let p_hash = stored_hash.get_value();
|
||||||
|
view! {
|
||||||
|
<FileRow
|
||||||
|
file=f
|
||||||
|
hash=p_hash
|
||||||
|
on_refresh=on_refresh.clone()
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</TableBody>
|
||||||
|
</Table>
|
||||||
|
</TableWrapper>
|
||||||
|
|
||||||
|
<div class="flex items-center justify-between text-xs text-muted-foreground px-1">
|
||||||
|
<span>{format!("Toplam {} dosya", files_len)}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
|
}}
|
||||||
|
</Suspense>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn FileRow(file: TorrentFile, hash: String, on_refresh: Callback<()>) -> impl IntoView {
|
||||||
|
let f_idx = file.index;
|
||||||
|
let path_clone = file.path.clone();
|
||||||
|
|
||||||
|
// on_refresh is called AFTER the server responds, not before
|
||||||
|
let on_refresh_stored = StoredValue::new(on_refresh);
|
||||||
|
|
||||||
|
let set_priority = Action::new(move |req: &(String, u32, u8)| {
|
||||||
|
let (h, idx, p) = req.clone();
|
||||||
|
async move {
|
||||||
|
let res = shared::server_fns::torrent::set_file_priority(h, idx, p).await;
|
||||||
|
if let Err(e) = &res {
|
||||||
|
crate::store::show_toast(shared::NotificationLevel::Error, format!("Öncelik değiştirilemedi: {:?}", e));
|
||||||
|
} else {
|
||||||
|
crate::store::show_toast(shared::NotificationLevel::Success, "Dosya önceliği güncellendi.".to_string());
|
||||||
|
// Refetch AFTER the server has saved the priority
|
||||||
|
on_refresh_stored.get_value().run(());
|
||||||
|
}
|
||||||
|
res
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<FileContextMenu
|
||||||
|
torrent_hash=hash
|
||||||
|
file_index=f_idx
|
||||||
|
set_priority=set_priority
|
||||||
|
>
|
||||||
|
<TableRow class="hover:bg-muted/50 transition-colors group">
|
||||||
|
<TableCell class="text-center text-xs text-muted-foreground">{file.index}</TableCell>
|
||||||
|
<TableCell class="font-medium text-xs break-all max-w-[200px] md:max-w-md" attr:title=move || path_clone.clone()>
|
||||||
|
{file.path.clone()}
|
||||||
|
</TableCell>
|
||||||
|
<TableCell class="text-right text-xs text-muted-foreground whitespace-nowrap">
|
||||||
|
{format_bytes(file.size)}
|
||||||
|
</TableCell>
|
||||||
|
<TableCell class="text-right text-xs whitespace-nowrap">
|
||||||
|
<span class="text-primary font-medium">{format_bytes(file.completed_chunks)}</span>
|
||||||
|
</TableCell>
|
||||||
|
<TableCell class="text-center">
|
||||||
|
{
|
||||||
|
let (variant, label) = match file.priority {
|
||||||
|
0 => (BadgeVariant::Destructive, "İndirme"),
|
||||||
|
2 => (BadgeVariant::Success, "Yüksek"),
|
||||||
|
_ => (BadgeVariant::Secondary, "Normal"),
|
||||||
|
};
|
||||||
|
view! { <Badge variant=variant class="text-[10px] uppercase">{label}</Badge> }
|
||||||
|
}
|
||||||
|
</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
</FileContextMenu>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn FileContextMenu(
|
||||||
|
children: Children,
|
||||||
|
torrent_hash: String,
|
||||||
|
file_index: u32,
|
||||||
|
set_priority: Action<(String, u32, u8), Result<(), ServerFnError>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let hash_c1 = torrent_hash.clone();
|
||||||
|
let hash_c2 = torrent_hash.clone();
|
||||||
|
let hash_c3 = torrent_hash.clone();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<ContextMenu>
|
||||||
|
<ContextMenuTrigger>
|
||||||
|
{children()}
|
||||||
|
</ContextMenuTrigger>
|
||||||
|
|
||||||
|
<ContextMenuContent class="w-48">
|
||||||
|
<ContextMenuLabel>"Dosya Önceliği"</ContextMenuLabel>
|
||||||
|
<ContextMenuGroup>
|
||||||
|
<ContextMenuItem on:click={
|
||||||
|
let h = hash_c1;
|
||||||
|
let sp = set_priority.clone();
|
||||||
|
move |_| {
|
||||||
|
sp.dispatch((h.clone(), file_index, 2));
|
||||||
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
<icons::ChevronsUp class="text-green-500" />
|
||||||
|
<span>"Yüksek"</span>
|
||||||
|
</ContextMenuItem>
|
||||||
|
|
||||||
|
<ContextMenuItem on:click={
|
||||||
|
let h = hash_c2;
|
||||||
|
let sp = set_priority.clone();
|
||||||
|
move |_| {
|
||||||
|
sp.dispatch((h.clone(), file_index, 1));
|
||||||
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
<icons::Minus class="text-blue-500" />
|
||||||
|
<span>"Normal"</span>
|
||||||
|
</ContextMenuItem>
|
||||||
|
|
||||||
|
<ContextMenuItem class="text-destructive focus:bg-destructive/10" on:click={
|
||||||
|
let h = hash_c3;
|
||||||
|
let sp = set_priority.clone();
|
||||||
|
move |_| {
|
||||||
|
sp.dispatch((h.clone(), file_index, 0));
|
||||||
|
crate::components::ui::context_menu::close_context_menu();
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
<icons::X />
|
||||||
|
<span>"İndirme (Kapalı)"</span>
|
||||||
|
</ContextMenuItem>
|
||||||
|
</ContextMenuGroup>
|
||||||
|
</ContextMenuContent>
|
||||||
|
</ContextMenu>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn FilesFallback() -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<Shimmer loading=Signal::derive(|| true) class="space-y-2">
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
</Shimmer>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_bytes(bytes: i64) -> String {
|
||||||
|
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
||||||
|
if bytes < 1024 { return format!("{} B", bytes); }
|
||||||
|
let i = (bytes as f64).log2().div_euclid(10.0) as usize;
|
||||||
|
format!("{:.1} {}", (bytes as f64) / 1024_f64.powi(i as i32), UNITS[i])
|
||||||
|
}
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
pub mod table;
|
pub mod table;
|
||||||
pub mod add_torrent;
|
pub mod add_torrent;
|
||||||
pub mod detail;
|
pub mod details;
|
||||||
|
pub mod files;
|
||||||
|
pub mod trackers;
|
||||||
|
|||||||
@@ -1,10 +1,43 @@
|
|||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
|
use wasm_bindgen::JsCast;
|
||||||
|
use std::collections::HashSet;
|
||||||
|
use icons::{ArrowUpDown, Inbox, Settings2, Play, Square, Trash2, Ellipsis, ArrowUp, ArrowDown, Check, ListFilter};
|
||||||
use crate::store::{get_action_messages, show_toast};
|
use crate::store::{get_action_messages, show_toast};
|
||||||
use crate::api;
|
use crate::api;
|
||||||
use shared::NotificationLevel;
|
use shared::NotificationLevel;
|
||||||
use crate::components::context_menu::TorrentContextMenu;
|
use crate::components::context_menu::TorrentContextMenu;
|
||||||
use leptos_shadcn_card::{Card, CardHeader, CardTitle, CardContent};
|
use crate::components::ui::data_table::*;
|
||||||
|
use crate::components::ui::checkbox::Checkbox;
|
||||||
|
use crate::components::ui::badge::{Badge, BadgeVariant};
|
||||||
|
use crate::components::ui::button::{Button, ButtonVariant};
|
||||||
|
use crate::components::ui::empty::*;
|
||||||
|
use crate::components::ui::input::Input;
|
||||||
|
use crate::components::ui::multi_select::*;
|
||||||
|
use crate::components::ui::dropdown_menu::*;
|
||||||
|
use crate::components::ui::alert_dialog::{
|
||||||
|
AlertDialog,
|
||||||
|
AlertDialogBody,
|
||||||
|
AlertDialogClose,
|
||||||
|
AlertDialogContent,
|
||||||
|
AlertDialogDescription,
|
||||||
|
AlertDialogFooter,
|
||||||
|
AlertDialogHeader,
|
||||||
|
AlertDialogTitle,
|
||||||
|
AlertDialogTrigger,
|
||||||
|
};
|
||||||
|
use tailwind_fuse::tw_merge;
|
||||||
|
|
||||||
|
const ALL_COLUMNS: [(&str, &str); 8] = [
|
||||||
|
("Name", "Name"),
|
||||||
|
("Size", "Size"),
|
||||||
|
("Progress", "Progress"),
|
||||||
|
("Status", "Status"),
|
||||||
|
("DownSpeed", "DL Speed"),
|
||||||
|
("UpSpeed", "UP Speed"),
|
||||||
|
("ETA", "ETA"),
|
||||||
|
("AddedDate", "Date"),
|
||||||
|
];
|
||||||
|
|
||||||
fn format_bytes(bytes: i64) -> String {
|
fn format_bytes(bytes: i64) -> String {
|
||||||
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
const UNITS: [&str; 6] = ["B", "KB", "MB", "GB", "TB", "PB"];
|
||||||
@@ -49,8 +82,16 @@ pub fn TorrentTable() -> impl IntoView {
|
|||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
||||||
let sort_col = signal(SortColumn::AddedDate);
|
let sort_col = signal(SortColumn::AddedDate);
|
||||||
let sort_dir = signal(SortDirection::Descending);
|
let sort_dir = signal(SortDirection::Descending);
|
||||||
|
|
||||||
|
let selected_hashes = RwSignal::new(HashSet::<String>::new());
|
||||||
|
|
||||||
|
let visible_columns = RwSignal::new(HashSet::from([
|
||||||
|
"Name".to_string(), "Size".to_string(), "Progress".to_string(),
|
||||||
|
"Status".to_string(), "DownSpeed".to_string(), "UpSpeed".to_string(),
|
||||||
|
"ETA".to_string(), "AddedDate".to_string()
|
||||||
|
]));
|
||||||
|
|
||||||
let filtered_hashes = Memo::new(move |_| {
|
let sorted_hashes_data = Memo::new(move |_| {
|
||||||
let torrents_map = store.torrents.get();
|
let torrents_map = store.torrents.get();
|
||||||
let filter = store.filter.get();
|
let filter = store.filter.get();
|
||||||
let search = store.search_query.get();
|
let search = store.search_query.get();
|
||||||
@@ -89,7 +130,30 @@ pub fn TorrentTable() -> impl IntoView {
|
|||||||
};
|
};
|
||||||
if dir == SortDirection::Descending { cmp.reverse() } else { cmp }
|
if dir == SortDirection::Descending { cmp.reverse() } else { cmp }
|
||||||
});
|
});
|
||||||
torrents.into_iter().map(|t| t.hash.clone()).collect::<Vec<String>>()
|
torrents
|
||||||
|
});
|
||||||
|
|
||||||
|
let filtered_hashes = Memo::new(move |_| {
|
||||||
|
sorted_hashes_data.get().into_iter().map(|t| t.hash.clone()).collect::<Vec<String>>()
|
||||||
|
});
|
||||||
|
|
||||||
|
let selected_count = Signal::derive(move || {
|
||||||
|
let current_hashes: HashSet<String> = filtered_hashes.get().into_iter().collect();
|
||||||
|
selected_hashes.with(|selected| {
|
||||||
|
selected.iter().filter(|h| current_hashes.contains(*h)).count()
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
let has_selection = Signal::derive(move || selected_count.get() > 0);
|
||||||
|
|
||||||
|
let handle_select_all = Callback::new(move |checked: bool| {
|
||||||
|
selected_hashes.update(|selected| {
|
||||||
|
let hashes = filtered_hashes.get_untracked();
|
||||||
|
for h in hashes {
|
||||||
|
if checked { selected.insert(h); }
|
||||||
|
else { selected.remove(&h); }
|
||||||
|
}
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
let handle_sort = move |col: SortColumn| {
|
let handle_sort = move |col: SortColumn| {
|
||||||
@@ -103,13 +167,35 @@ pub fn TorrentTable() -> impl IntoView {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let sort_arrow = move |col: SortColumn| {
|
let sort_icon = move |col: SortColumn| {
|
||||||
if sort_col.0.get() == col {
|
let is_active = sort_col.0.get() == col;
|
||||||
match sort_dir.0.get() {
|
let class = if is_active { "size-3 text-primary" } else { "size-3 opacity-30 group-hover:opacity-100 transition-opacity" };
|
||||||
SortDirection::Ascending => view! { <span class="ml-1 text-xs">"▲"</span> }.into_any(),
|
view! { <ArrowUpDown class=class.to_string() /> }.into_any()
|
||||||
SortDirection::Descending => view! { <span class="ml-1 text-xs">"▼"</span> }.into_any(),
|
};
|
||||||
|
|
||||||
|
let bulk_action = move |action: &'static str| {
|
||||||
|
let hashes: Vec<String> = selected_hashes.get().into_iter().collect();
|
||||||
|
if hashes.is_empty() { return; }
|
||||||
|
|
||||||
|
spawn_local(async move {
|
||||||
|
let mut success = true;
|
||||||
|
for hash in hashes {
|
||||||
|
let res = match action {
|
||||||
|
"start" => api::torrent::start(&hash).await,
|
||||||
|
"stop" => api::torrent::stop(&hash).await,
|
||||||
|
"delete" => api::torrent::delete(&hash).await,
|
||||||
|
"delete_with_data" => api::torrent::delete_with_data(&hash).await,
|
||||||
|
_ => Ok(()),
|
||||||
|
};
|
||||||
|
if res.is_err() { success = false; }
|
||||||
}
|
}
|
||||||
} else { view! { <span class="ml-1 text-xs opacity-0 group-hover:opacity-50">"▲"</span> }.into_any() }
|
if success {
|
||||||
|
show_toast(NotificationLevel::Success, format!("Toplu işlem başarıyla tamamlandı: {}", action));
|
||||||
|
selected_hashes.update(|s| s.clear());
|
||||||
|
} else {
|
||||||
|
show_toast(NotificationLevel::Error, "Bazı işlemler başarısız oldu.");
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
let on_action = Callback::new(move |(action, hash): (String, String)| {
|
let on_action = Callback::new(move |(action, hash): (String, String)| {
|
||||||
@@ -132,78 +218,356 @@ pub fn TorrentTable() -> impl IntoView {
|
|||||||
});
|
});
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div class="h-full bg-background relative flex flex-col overflow-hidden">
|
<div class="h-full bg-background flex flex-row overflow-hidden">
|
||||||
// --- DESKTOP VIEW ---
|
// Sol: liste alanı
|
||||||
<div class="hidden md:flex flex-col h-full overflow-hidden">
|
<div class="flex-1 min-w-0 flex flex-col overflow-hidden px-4 py-4 gap-4">
|
||||||
// Header
|
// --- TOPBAR ---
|
||||||
<div class="flex items-center text-xs uppercase text-muted-foreground border-b border-border bg-muted/50 h-9 shrink-0 px-2 font-medium">
|
<div class="flex items-center justify-between gap-4">
|
||||||
<div class="flex-1 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::Name)>
|
<div class="flex items-center gap-2 flex-1 max-w-md">
|
||||||
"Name" {move || sort_arrow(SortColumn::Name)}
|
<Input
|
||||||
</div>
|
class="h-9"
|
||||||
<div class="w-24 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::Size)>
|
placeholder="Torrent ara..."
|
||||||
"Size" {move || sort_arrow(SortColumn::Size)}
|
bind_value=store.search_query
|
||||||
</div>
|
/>
|
||||||
<div class="w-48 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::Progress)>
|
|
||||||
"Progress" {move || sort_arrow(SortColumn::Progress)}
|
|
||||||
</div>
|
|
||||||
<div class="w-24 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::Status)>
|
|
||||||
"Status" {move || sort_arrow(SortColumn::Status)}
|
|
||||||
</div>
|
|
||||||
<div class="w-24 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::DownSpeed)>
|
|
||||||
"DL Speed" {move || sort_arrow(SortColumn::DownSpeed)}
|
|
||||||
</div>
|
|
||||||
<div class="w-24 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::UpSpeed)>
|
|
||||||
"Up Speed" {move || sort_arrow(SortColumn::UpSpeed)}
|
|
||||||
</div>
|
|
||||||
<div class="w-24 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::ETA)>
|
|
||||||
"ETA" {move || sort_arrow(SortColumn::ETA)}
|
|
||||||
</div>
|
|
||||||
<div class="w-32 px-2 cursor-pointer hover:text-foreground group select-none flex items-center" on:click=move |_| handle_sort(SortColumn::AddedDate)>
|
|
||||||
"Date" {move || sort_arrow(SortColumn::AddedDate)}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
// Regular List
|
<div class="flex items-center gap-2">
|
||||||
<div class="flex-1 overflow-y-auto min-h-0">
|
<Show when=move || has_selection.get()>
|
||||||
<For each=move || filtered_hashes.get() key=|hash| hash.clone() children={
|
<div class="flex items-center gap-2">
|
||||||
let on_action = on_action.clone();
|
<DropdownMenu>
|
||||||
move |hash| {
|
<DropdownMenuTrigger class="w-[140px] h-9 gap-2">
|
||||||
let h = hash.clone();
|
<Ellipsis class="size-4" />
|
||||||
view! {
|
{move || format!("Toplu İşlem ({})", selected_count.get())}
|
||||||
<TorrentContextMenu torrent_hash=h on_action=on_action.clone()>
|
</DropdownMenuTrigger>
|
||||||
<TorrentRow hash=hash.clone() />
|
<DropdownMenuContent class="w-48">
|
||||||
</TorrentContextMenu>
|
<DropdownMenuLabel>"Seçili Torrentler"</DropdownMenuLabel>
|
||||||
}
|
<DropdownMenuGroup class="mt-2">
|
||||||
}
|
<DropdownMenuItem on:click=move |_| bulk_action("start")>
|
||||||
} />
|
<Play class="mr-2 size-4" /> "Başlat"
|
||||||
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem on:click=move |_| bulk_action("stop")>
|
||||||
|
<Square class="mr-2 size-4" /> "Durdur"
|
||||||
|
</DropdownMenuItem>
|
||||||
|
|
||||||
|
<div class="my-1 h-px bg-border" />
|
||||||
|
|
||||||
|
<DropdownMenuItem class="text-destructive focus:bg-destructive/10 cursor-pointer" on:click=move |_| {
|
||||||
|
if let Some(trigger) = document().get_element_by_id("bulk-delete-trigger") {
|
||||||
|
let _ = trigger.dyn_into::<web_sys::HtmlElement>().map(|el: web_sys::HtmlElement| el.click());
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
<Trash2 class="mr-2 size-4" /> "Toplu Sil..."
|
||||||
|
</DropdownMenuItem>
|
||||||
|
</DropdownMenuGroup>
|
||||||
|
</DropdownMenuContent>
|
||||||
|
</DropdownMenu>
|
||||||
|
|
||||||
|
<AlertDialog>
|
||||||
|
<AlertDialogTrigger attr:id="bulk-delete-trigger" class="hidden">""</AlertDialogTrigger>
|
||||||
|
<AlertDialogContent class="sm:max-w-[425px]">
|
||||||
|
<AlertDialogBody>
|
||||||
|
<AlertDialogHeader class="space-y-3">
|
||||||
|
<AlertDialogTitle class="text-destructive flex items-center gap-2 text-xl">
|
||||||
|
<Trash2 class="size-6" />
|
||||||
|
"Toplu Silme Onayı"
|
||||||
|
</AlertDialogTitle>
|
||||||
|
<AlertDialogDescription class="text-sm leading-relaxed text-left">
|
||||||
|
{move || format!("Seçili {} adet torrent silinecek. Lütfen silme yöntemini seçin:", selected_count.get())}
|
||||||
|
<div class="mt-4 p-4 bg-destructive/5 rounded-lg border border-destructive/10 text-xs text-destructive/80 font-medium">
|
||||||
|
"⚠️ Dikkat: Verilerle birlikte silme işlemi dosyaları diskten de kalıcı olarak kaldıracaktır."
|
||||||
|
</div>
|
||||||
|
</AlertDialogDescription>
|
||||||
|
</AlertDialogHeader>
|
||||||
|
<AlertDialogFooter class="mt-6">
|
||||||
|
<div class="flex flex-col-reverse sm:flex-row gap-3 w-full sm:justify-end">
|
||||||
|
<AlertDialogClose class="sm:flex-1 md:flex-none">"Vazgeç"</AlertDialogClose>
|
||||||
|
<div class="flex flex-col sm:flex-row gap-2">
|
||||||
|
<Button
|
||||||
|
variant=ButtonVariant::Secondary
|
||||||
|
class="w-full sm:w-auto font-medium"
|
||||||
|
on:click=move |_| bulk_action("delete")
|
||||||
|
>
|
||||||
|
"Sadece Sil"
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant=ButtonVariant::Destructive
|
||||||
|
class="w-full sm:w-auto font-bold"
|
||||||
|
on:click=move |_| bulk_action("delete_with_data")
|
||||||
|
>
|
||||||
|
"Verilerle Sil"
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</AlertDialogFooter>
|
||||||
|
</AlertDialogBody>
|
||||||
|
</AlertDialogContent>
|
||||||
|
</AlertDialog>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
// Mobile Sort Menu
|
||||||
|
<div class="block md:hidden">
|
||||||
|
<DropdownMenu>
|
||||||
|
<DropdownMenuTrigger class="w-[100px] h-9 gap-2 text-xs">
|
||||||
|
<ListFilter class="size-4" />
|
||||||
|
"Sırala"
|
||||||
|
</DropdownMenuTrigger>
|
||||||
|
<DropdownMenuContent class="w-56">
|
||||||
|
<DropdownMenuLabel>"Sıralama Ölçütü"</DropdownMenuLabel>
|
||||||
|
<DropdownMenuGroup class="mt-2">
|
||||||
|
{move || {
|
||||||
|
let current_col = sort_col.0.get();
|
||||||
|
let current_dir = sort_dir.0.get();
|
||||||
|
|
||||||
|
let sort_items = vec![
|
||||||
|
(SortColumn::Name, "İsim"),
|
||||||
|
(SortColumn::Size, "Boyut"),
|
||||||
|
(SortColumn::Progress, "İlerleme"),
|
||||||
|
(SortColumn::Status, "Durum"),
|
||||||
|
(SortColumn::DownSpeed, "DL Hızı"),
|
||||||
|
(SortColumn::UpSpeed, "UP Hızı"),
|
||||||
|
(SortColumn::ETA, "Kalan Süre"),
|
||||||
|
(SortColumn::AddedDate, "Tarih"),
|
||||||
|
];
|
||||||
|
|
||||||
|
sort_items.into_iter().map(|(col, label)| {
|
||||||
|
let is_active = current_col == col;
|
||||||
|
view! {
|
||||||
|
<DropdownMenuItem on:click=move |_| handle_sort(col)>
|
||||||
|
<div class="flex items-center justify-between w-full">
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
{if is_active { view! { <Check class="size-4 text-primary" /> }.into_any() } else { view! { <div class="size-4" /> }.into_any() }}
|
||||||
|
<span class=if is_active { "font-bold text-primary" } else { "" }>{label}</span>
|
||||||
|
</div>
|
||||||
|
{if is_active {
|
||||||
|
match current_dir {
|
||||||
|
SortDirection::Ascending => view! { <ArrowUp class="size-3 opacity-50" /> }.into_any(),
|
||||||
|
SortDirection::Descending => view! { <ArrowDown class="size-3 opacity-50" /> }.into_any(),
|
||||||
|
}
|
||||||
|
} else { view! { "" }.into_any() }}
|
||||||
|
</div>
|
||||||
|
</DropdownMenuItem>
|
||||||
|
}.into_any()
|
||||||
|
}).collect_view()
|
||||||
|
}}
|
||||||
|
</DropdownMenuGroup>
|
||||||
|
</DropdownMenuContent>
|
||||||
|
</DropdownMenu>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
// Desktop Columns Menu
|
||||||
|
<div class="hidden md:flex">
|
||||||
|
<MultiSelect values=visible_columns>
|
||||||
|
<MultiSelectTrigger class="w-[140px] h-9">
|
||||||
|
<div class="flex items-center gap-2 text-xs">
|
||||||
|
<Settings2 class="size-4" />
|
||||||
|
"Sütunlar"
|
||||||
|
</div>
|
||||||
|
</MultiSelectTrigger>
|
||||||
|
<MultiSelectContent>
|
||||||
|
<MultiSelectGroup>
|
||||||
|
{ALL_COLUMNS.into_iter().map(|(id, label)| {
|
||||||
|
let id_val = id.to_string();
|
||||||
|
view! {
|
||||||
|
<MultiSelectItem>
|
||||||
|
<MultiSelectOption value=id_val.clone() attr:disabled=move || id_val == "Name">
|
||||||
|
{label}
|
||||||
|
</MultiSelectOption>
|
||||||
|
</MultiSelectItem>
|
||||||
|
}.into_any()
|
||||||
|
}).collect_view()}
|
||||||
|
</MultiSelectGroup>
|
||||||
|
</MultiSelectContent>
|
||||||
|
</MultiSelect>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
// --- MOBILE VIEW ---
|
// --- MAIN CONTENT ---
|
||||||
<div class="md:hidden flex flex-col h-full bg-muted/10 relative overflow-hidden">
|
<div class="flex-1 min-h-0 overflow-hidden">
|
||||||
<div class="flex-1 overflow-y-auto p-3 min-h-0">
|
// Desktop Table View
|
||||||
<For each=move || filtered_hashes.get() key=|hash| hash.clone() children={
|
<DataTableWrapper class="hidden md:block h-full bg-card/50">
|
||||||
let on_action = on_action.clone();
|
<div class="h-full overflow-auto">
|
||||||
move |hash| {
|
<DataTable>
|
||||||
let h = hash.clone();
|
<DataTableHeader class="sticky top-0 bg-muted/80 backdrop-blur-sm z-10">
|
||||||
view! {
|
<DataTableRow class="hover:bg-transparent">
|
||||||
<div class="pb-3">
|
<DataTableHead class="w-12 px-4">
|
||||||
<TorrentContextMenu torrent_hash=h on_action=on_action.clone()>
|
<Checkbox
|
||||||
<TorrentCard hash=hash.clone() />
|
checked=Signal::derive(move || {
|
||||||
</TorrentContextMenu>
|
let hashes = filtered_hashes.get();
|
||||||
</div>
|
!hashes.is_empty() && selected_count.get() == hashes.len()
|
||||||
|
})
|
||||||
|
on_checked_change=handle_select_all
|
||||||
|
/>
|
||||||
|
</DataTableHead>
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Name").then(|| view! {
|
||||||
|
<DataTableHead class="cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground" on:click=move |_| handle_sort(SortColumn::Name)>
|
||||||
|
<div class="flex items-center gap-2">"Name" {move || sort_icon(SortColumn::Name)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Size").then(|| view! {
|
||||||
|
<DataTableHead class="w-24 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground" on:click=move |_| handle_sort(SortColumn::Size)>
|
||||||
|
<div class="flex items-center gap-2">"Size" {move || sort_icon(SortColumn::Size)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Progress").then(|| view! {
|
||||||
|
<DataTableHead class="w-48 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground" on:click=move |_| handle_sort(SortColumn::Progress)>
|
||||||
|
<div class="flex items-center gap-2">"Progress" {move || sort_icon(SortColumn::Progress)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Status").then(|| view! {
|
||||||
|
<DataTableHead class="w-24 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground" on:click=move |_| handle_sort(SortColumn::Status)>
|
||||||
|
<div class="flex items-center gap-2">"Status" {move || sort_icon(SortColumn::Status)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("DownSpeed").then(|| view! {
|
||||||
|
<DataTableHead class="w-24 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground text-right" on:click=move |_| handle_sort(SortColumn::DownSpeed)>
|
||||||
|
<div class="flex items-center justify-end gap-2">"DL Speed" {move || sort_icon(SortColumn::DownSpeed)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("UpSpeed").then(|| view! {
|
||||||
|
<DataTableHead class="w-24 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground text-right" on:click=move |_| handle_sort(SortColumn::UpSpeed)>
|
||||||
|
<div class="flex items-center justify-end gap-2">"UP Speed" {move || sort_icon(SortColumn::UpSpeed)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("ETA").then(|| view! {
|
||||||
|
<DataTableHead class="w-24 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground text-right" on:click=move |_| handle_sort(SortColumn::ETA)>
|
||||||
|
<div class="flex items-center justify-end gap-2">"ETA" {move || sort_icon(SortColumn::ETA)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("AddedDate").then(|| view! {
|
||||||
|
<DataTableHead class="w-32 cursor-pointer group select-none transition-all duration-100 active:scale-[0.98] hover:bg-muted/30 hover:text-foreground text-right" on:click=move |_| handle_sort(SortColumn::AddedDate)>
|
||||||
|
<div class="flex items-center justify-end gap-2">"Date" {move || sort_icon(SortColumn::AddedDate)}</div>
|
||||||
|
</DataTableHead>
|
||||||
|
}).into_any()}
|
||||||
|
</DataTableRow>
|
||||||
|
</DataTableHeader>
|
||||||
|
<DataTableBody>
|
||||||
|
<Show
|
||||||
|
when=move || !filtered_hashes.get().is_empty()
|
||||||
|
fallback=move || view! {
|
||||||
|
<DataTableRow class="hover:bg-transparent">
|
||||||
|
<DataTableCell attr:colspan="10" class="h-[400px]">
|
||||||
|
<Empty class="h-full">
|
||||||
|
<EmptyHeader>
|
||||||
|
<EmptyMedia variant=EmptyMediaVariant::Icon>
|
||||||
|
<Inbox class="size-10 text-muted-foreground" />
|
||||||
|
</EmptyMedia>
|
||||||
|
<EmptyTitle>"Torrent Bulunamadı"</EmptyTitle>
|
||||||
|
<EmptyDescription>
|
||||||
|
{move || {
|
||||||
|
let query = store.search_query.get();
|
||||||
|
if query.is_empty() { "Henüz torrent bulunmuyor.".to_string() }
|
||||||
|
else { "Arama kriterlerinize uygun sonuç bulunamadı.".to_string() }
|
||||||
|
}}
|
||||||
|
</EmptyDescription>
|
||||||
|
</EmptyHeader>
|
||||||
|
</Empty>
|
||||||
|
</DataTableCell>
|
||||||
|
</DataTableRow>
|
||||||
|
}.into_any()
|
||||||
|
>
|
||||||
|
<For each=move || filtered_hashes.get() key=|hash| hash.clone() children={
|
||||||
|
let on_action = on_action.clone();
|
||||||
|
move |hash| {
|
||||||
|
let h = hash.clone();
|
||||||
|
let is_selected = Signal::derive(move || {
|
||||||
|
selected_hashes.with(|selected| selected.contains(&h))
|
||||||
|
});
|
||||||
|
let h_for_change = hash.clone();
|
||||||
|
view! {
|
||||||
|
<TorrentRow
|
||||||
|
hash=hash.clone()
|
||||||
|
on_action=on_action.clone()
|
||||||
|
is_selected=is_selected
|
||||||
|
visible_columns=visible_columns
|
||||||
|
on_select=Callback::new(move |checked| {
|
||||||
|
selected_hashes.update(|selected| {
|
||||||
|
if checked { selected.insert(h_for_change.clone()); }
|
||||||
|
else { selected.remove(&h_for_change); }
|
||||||
|
});
|
||||||
|
})
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} />
|
||||||
|
</Show>
|
||||||
|
</DataTableBody>
|
||||||
|
</DataTable>
|
||||||
|
</div>
|
||||||
|
</DataTableWrapper>
|
||||||
|
|
||||||
|
// Mobile Card View
|
||||||
|
<div class="block md:hidden h-full overflow-y-auto space-y-4 pb-32 px-1">
|
||||||
|
<Show
|
||||||
|
when=move || !filtered_hashes.get().is_empty()
|
||||||
|
fallback=move || view! {
|
||||||
|
<div class="flex flex-col items-center justify-center h-64 opacity-50 text-muted-foreground">
|
||||||
|
<Inbox class="size-12 mb-2" />
|
||||||
|
<p>"Torrent Bulunamadı"</p>
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
|
>
|
||||||
|
<For each=move || filtered_hashes.get() key=|hash| hash.clone() children={
|
||||||
|
let on_action = on_action.clone();
|
||||||
|
move |hash| {
|
||||||
|
let h = hash.clone();
|
||||||
|
let is_selected = Signal::derive(move || {
|
||||||
|
selected_hashes.with(|selected| selected.contains(&h))
|
||||||
|
});
|
||||||
|
let h_for_change = hash.clone();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<TorrentCard
|
||||||
|
hash=hash.clone()
|
||||||
|
on_action=on_action.clone()
|
||||||
|
is_selected=is_selected
|
||||||
|
on_select=Callback::new(move |checked| {
|
||||||
|
selected_hashes.update(|selected| {
|
||||||
|
if checked { selected.insert(h_for_change.clone()); }
|
||||||
|
else { selected.remove(&h_for_change); }
|
||||||
|
});
|
||||||
|
})
|
||||||
|
/>
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
} />
|
||||||
} />
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="flex items-center justify-between px-2 py-1.5 text-[10px] md:text-[11px] text-muted-foreground bg-muted/20 border rounded-md">
|
||||||
|
<div class="flex gap-3 md:gap-4">
|
||||||
|
<span>{move || format!("Toplam: {} torrent", filtered_hashes.get().len())}</span>
|
||||||
|
<Show when=move || has_selection.get()>
|
||||||
|
<span class="text-primary font-bold">{move || format!("{} seçili", selected_count.get())}</span>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
<div class="opacity-50">"VibeTorrent v3"</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
|
||||||
|
// Sağ: sabit detay paneli
|
||||||
|
<crate::components::torrent::details::TorrentDetailsPanel />
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
fn TorrentRow(
|
fn TorrentRow(
|
||||||
hash: String,
|
hash: String,
|
||||||
|
on_action: Callback<(String, String)>,
|
||||||
|
is_selected: Signal<bool>,
|
||||||
|
visible_columns: RwSignal<HashSet<String>>,
|
||||||
|
on_select: Callback<bool>,
|
||||||
) -> impl IntoView {
|
) -> impl IntoView {
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
||||||
let h = hash.clone();
|
let h = hash.clone();
|
||||||
@@ -214,50 +578,127 @@ fn TorrentRow(
|
|||||||
view! {
|
view! {
|
||||||
<Show when=move || torrent.get().is_some() fallback=|| ()>
|
<Show when=move || torrent.get().is_some() fallback=|| ()>
|
||||||
{
|
{
|
||||||
|
let on_action = on_action.clone();
|
||||||
move || {
|
move || {
|
||||||
let t = torrent.get().unwrap();
|
let t = torrent.get().unwrap();
|
||||||
let t_name = t.name.clone();
|
let t_name = t.name.clone();
|
||||||
let status_color = match t.status { shared::TorrentStatus::Seeding => "text-green-500", shared::TorrentStatus::Downloading => "text-blue-500", shared::TorrentStatus::Paused => "text-yellow-500", shared::TorrentStatus::Error => "text-red-500", _ => "text-muted-foreground" };
|
|
||||||
|
|
||||||
|
let is_active_selection = Memo::new(move |_| {
|
||||||
|
let selected = store.selected_torrent.get();
|
||||||
|
selected.as_deref() == Some(stored_hash.get_value().as_str())
|
||||||
|
});
|
||||||
|
|
||||||
|
let t_name_stored = StoredValue::new(t_name.clone());
|
||||||
|
let h_for_menu = stored_hash.get_value();
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div
|
<TorrentContextMenu torrent_hash=h_for_menu on_action=on_action.clone()>
|
||||||
class=move || {
|
<DataTableRow
|
||||||
let selected = store.selected_torrent.get();
|
class="cursor-pointer group h-10"
|
||||||
let is_selected = selected.as_deref() == Some(stored_hash.get_value().as_str());
|
attr:data-state=move || if is_selected.get() || is_active_selection.get() { "selected" } else { "" }
|
||||||
if is_selected {
|
on:click=move |_| store.selected_torrent.set(Some(stored_hash.get_value()))
|
||||||
"flex items-center text-sm bg-primary/10 border-b border-border h-[48px] px-2 select-none cursor-pointer transition-colors w-full"
|
>
|
||||||
} else {
|
<DataTableCell class="w-12 px-4">
|
||||||
"flex items-center text-sm hover:bg-muted/50 border-b border-border h-[48px] px-2 select-none cursor-pointer transition-colors w-full"
|
<div on:click=move |e| e.stop_propagation()>
|
||||||
}
|
<Checkbox
|
||||||
}
|
checked=is_selected
|
||||||
on:click=move |_| store.selected_torrent.set(Some(stored_hash.get_value()))
|
on_checked_change=on_select
|
||||||
>
|
/>
|
||||||
<div class="flex-1 min-w-0 px-2 font-medium truncate" title=t_name.clone()>{t_name.clone()}</div>
|
|
||||||
<div class="w-24 px-2 font-mono text-xs text-muted-foreground">{format_bytes(t.size)}</div>
|
|
||||||
<div class="w-48 px-2">
|
|
||||||
<div class="flex items-center gap-2">
|
|
||||||
<div class="h-2 w-full bg-secondary rounded-full overflow-hidden">
|
|
||||||
<div class="h-full bg-primary transition-all duration-500" style=format!("width: {}%", t.percent_complete)></div>
|
|
||||||
</div>
|
</div>
|
||||||
<span class="text-[10px] text-muted-foreground w-10 text-right">{format!("{:.1}%", t.percent_complete)}</span>
|
</DataTableCell>
|
||||||
</div>
|
|
||||||
</div>
|
{move || visible_columns.get().contains("Name").then({
|
||||||
<div class={format!("w-24 px-2 text-xs font-medium {}", status_color)}>{format!("{:?}", t.status)}</div>
|
move || view! {
|
||||||
<div class="w-24 px-2 text-right font-mono text-xs text-green-600 dark:text-green-500">{format_speed(t.down_rate)}</div>
|
<DataTableCell class="font-medium truncate max-w-[200px] lg:max-w-md" attr:title=t_name_stored.get_value()>
|
||||||
<div class="w-24 px-2 text-right font-mono text-xs text-blue-600 dark:text-blue-500">{format_speed(t.up_rate)}</div>
|
{t_name_stored.get_value()}
|
||||||
<div class="w-24 px-2 text-right font-mono text-xs text-muted-foreground">{format_duration(t.eta)}</div>
|
</DataTableCell>
|
||||||
<div class="w-32 px-2 text-right font-mono text-xs text-muted-foreground">{format_date(t.added_date)}</div>
|
}
|
||||||
</div>
|
}).into_any()}
|
||||||
}
|
|
||||||
|
{move || visible_columns.get().contains("Size").then({
|
||||||
|
let size_bytes = t.size;
|
||||||
|
move || {
|
||||||
|
let size_str = format_bytes(size_bytes);
|
||||||
|
view! { <DataTableCell class="font-mono text-xs text-muted-foreground whitespace-nowrap">{size_str}</DataTableCell> }
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Progress").then({
|
||||||
|
let percent = t.percent_complete;
|
||||||
|
move || view! {
|
||||||
|
<DataTableCell>
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
<div class="h-1.5 w-full bg-secondary rounded-full overflow-hidden min-w-[80px]">
|
||||||
|
<div class="h-full bg-primary transition-all duration-500" style=format!("width: {}%", percent)></div>
|
||||||
|
</div>
|
||||||
|
<span class="text-[10px] text-muted-foreground w-10 text-right">{format!("{:.1}%", percent)}</span>
|
||||||
|
</div>
|
||||||
|
</DataTableCell>
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("Status").then({
|
||||||
|
let status_text = format!("{:?}", t.status);
|
||||||
|
let variant = match t.status {
|
||||||
|
shared::TorrentStatus::Seeding => BadgeVariant::Success,
|
||||||
|
shared::TorrentStatus::Downloading => BadgeVariant::Info,
|
||||||
|
shared::TorrentStatus::Paused => BadgeVariant::Warning,
|
||||||
|
shared::TorrentStatus::Error => BadgeVariant::Destructive,
|
||||||
|
_ => BadgeVariant::Secondary,
|
||||||
|
};
|
||||||
|
move || view! {
|
||||||
|
<DataTableCell class="whitespace-nowrap">
|
||||||
|
<Badge variant=variant>{status_text.clone()}</Badge>
|
||||||
|
</DataTableCell>
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("DownSpeed").then({
|
||||||
|
let rate = t.down_rate;
|
||||||
|
move || {
|
||||||
|
let speed_str = format_speed(rate);
|
||||||
|
view! { <DataTableCell class="text-right font-mono text-xs text-green-600 dark:text-green-500 whitespace-nowrap">{speed_str}</DataTableCell> }
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("UpSpeed").then({
|
||||||
|
let rate = t.up_rate;
|
||||||
|
move || {
|
||||||
|
let speed_str = format_speed(rate);
|
||||||
|
view! { <DataTableCell class="text-right font-mono text-xs text-blue-600 dark:text-blue-500 whitespace-nowrap">{speed_str}</DataTableCell> }
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("ETA").then({
|
||||||
|
let eta = t.eta;
|
||||||
|
move || {
|
||||||
|
let eta_str = format_duration(eta);
|
||||||
|
view! { <DataTableCell class="text-right font-mono text-xs text-muted-foreground whitespace-nowrap">{eta_str}</DataTableCell> }
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
|
||||||
|
{move || visible_columns.get().contains("AddedDate").then({
|
||||||
|
let date = t.added_date;
|
||||||
|
move || {
|
||||||
|
let date_str = format_date(date);
|
||||||
|
view! { <DataTableCell class="text-right font-mono text-xs text-muted-foreground whitespace-nowrap">{date_str}</DataTableCell> }
|
||||||
|
}
|
||||||
|
}).into_any()}
|
||||||
|
</DataTableRow>
|
||||||
|
</TorrentContextMenu>
|
||||||
|
}.into_any()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</Show>
|
</Show>
|
||||||
}
|
}.into_any()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
fn TorrentCard(
|
fn TorrentCard(
|
||||||
hash: String,
|
hash: String,
|
||||||
|
on_action: Callback<(String, String)>,
|
||||||
|
is_selected: Signal<bool>,
|
||||||
|
on_select: Callback<bool>,
|
||||||
) -> impl IntoView {
|
) -> impl IntoView {
|
||||||
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
let store = use_context::<crate::store::TorrentStore>().expect("store not provided");
|
||||||
let h = hash.clone();
|
let h = hash.clone();
|
||||||
@@ -268,53 +709,88 @@ fn TorrentCard(
|
|||||||
view! {
|
view! {
|
||||||
<Show when=move || torrent.get().is_some() fallback=|| ()>
|
<Show when=move || torrent.get().is_some() fallback=|| ()>
|
||||||
{
|
{
|
||||||
|
let on_action = on_action.clone();
|
||||||
move || {
|
move || {
|
||||||
let t = torrent.get().unwrap();
|
let t = torrent.get().unwrap();
|
||||||
let t_name = t.name.clone();
|
let t_name = t.name.clone();
|
||||||
let status_badge_class = match t.status { shared::TorrentStatus::Seeding => "bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400 border-green-200 dark:border-green-800", shared::TorrentStatus::Downloading => "bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-400 border-blue-200 dark:border-blue-800", shared::TorrentStatus::Paused => "bg-yellow-100 text-yellow-700 dark:bg-yellow-900/30 dark:text-yellow-400 border-yellow-200 dark:border-yellow-800", shared::TorrentStatus::Error => "bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-400 border-red-200 dark:border-red-800", _ => "bg-muted text-muted-foreground" };
|
let status_variant = match t.status {
|
||||||
|
shared::TorrentStatus::Seeding => BadgeVariant::Success,
|
||||||
|
shared::TorrentStatus::Downloading => BadgeVariant::Info,
|
||||||
|
shared::TorrentStatus::Paused => BadgeVariant::Warning,
|
||||||
|
shared::TorrentStatus::Error => BadgeVariant::Destructive,
|
||||||
|
_ => BadgeVariant::Secondary
|
||||||
|
};
|
||||||
|
let h_for_menu = stored_hash.get_value();
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
<div
|
<TorrentContextMenu torrent_hash=h_for_menu on_action=on_action.clone()>
|
||||||
class=move || {
|
<div
|
||||||
let selected = store.selected_torrent.get();
|
class=move || tw_merge!(
|
||||||
let is_selected = selected.as_deref() == Some(stored_hash.get_value().as_str());
|
"rounded-lg transition-all duration-200 border cursor-pointer select-none overflow-hidden active:scale-[0.98]",
|
||||||
if is_selected {
|
if is_selected.get() {
|
||||||
"ring-2 ring-primary rounded-lg transition-all"
|
"bg-primary/10 border-primary shadow-sm"
|
||||||
} else {
|
} else {
|
||||||
"transition-all"
|
"bg-card border-border hover:border-primary/50"
|
||||||
|
}
|
||||||
|
)
|
||||||
|
on:click=move |_| {
|
||||||
|
store.selected_torrent.set(Some(stored_hash.get_value()));
|
||||||
}
|
}
|
||||||
}
|
>
|
||||||
on:click=move |_| store.selected_torrent.set(Some(stored_hash.get_value()))
|
<div class="p-4 space-y-3">
|
||||||
>
|
<div class="flex justify-between items-start gap-3">
|
||||||
<Card class="h-full select-none cursor-pointer hover:border-primary transition-colors">
|
<div class="flex items-start gap-3 flex-1 min-w-0">
|
||||||
<CardHeader class="p-3 pb-0">
|
<div on:click=move |e| e.stop_propagation() class="mt-0.5">
|
||||||
<div class="flex justify-between items-start gap-2">
|
<Checkbox
|
||||||
<CardTitle class="text-sm font-medium leading-tight line-clamp-2">{t_name.clone()}</CardTitle>
|
checked=is_selected
|
||||||
<div class={format!("inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 {}", status_badge_class)}>{format!("{:?}", t.status)}</div>
|
on_checked_change=on_select
|
||||||
</div>
|
/>
|
||||||
</CardHeader>
|
</div>
|
||||||
<CardContent class="p-3 pt-2 gap-3 flex flex-col">
|
<div class="flex-1 min-w-0">
|
||||||
<div class="flex flex-col gap-1">
|
<h3 class="text-sm font-bold leading-tight line-clamp-2 break-all">{t_name.clone()}</h3>
|
||||||
<div class="flex justify-between text-[10px] text-muted-foreground">
|
</div>
|
||||||
<span>{format_bytes(t.size)}</span>
|
</div>
|
||||||
<span>{format!("{:.1}%", t.percent_complete)}</span>
|
<Badge variant=status_variant class="uppercase tracking-wider text-[10px] shrink-0">
|
||||||
|
{format!("{:?}", t.status)}
|
||||||
|
</Badge>
|
||||||
</div>
|
</div>
|
||||||
<div class="h-1.5 w-full bg-secondary rounded-full overflow-hidden">
|
|
||||||
<div class="h-full bg-primary transition-all duration-500" style=format!("width: {}%", t.percent_complete)></div>
|
<div class="space-y-1.5">
|
||||||
|
<div class="flex justify-between text-[10px] font-medium text-muted-foreground">
|
||||||
|
<span class="flex items-center gap-1">
|
||||||
|
<span class="opacity-70">"Boyut:"</span> {format_bytes(t.size)}
|
||||||
|
</span>
|
||||||
|
<span class="font-bold text-primary">{format!("{:.1}%", t.percent_complete)}</span>
|
||||||
|
</div>
|
||||||
|
<div class="h-2 w-full bg-secondary rounded-full overflow-hidden">
|
||||||
|
<div class="h-full bg-primary transition-all duration-500 ease-out" style=format!("width: {}%", t.percent_complete)></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="grid grid-cols-2 gap-y-2 gap-x-4 text-[10px] font-mono pt-2 border-t border-border/40">
|
||||||
|
<div class="flex flex-col gap-0.5">
|
||||||
|
<span class="text-muted-foreground uppercase text-[8px] tracking-tighter">"İndirme"</span>
|
||||||
|
<span class="text-blue-600 dark:text-blue-400 font-bold">{format_speed(t.down_rate)}</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col gap-0.5">
|
||||||
|
<span class="text-muted-foreground uppercase text-[8px] tracking-tighter">"Gönderme"</span>
|
||||||
|
<span class="text-green-600 dark:text-green-400 font-bold">{format_speed(t.up_rate)}</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col gap-0.5">
|
||||||
|
<span class="text-muted-foreground uppercase text-[8px] tracking-tighter">"Kalan Süre"</span>
|
||||||
|
<span class="text-foreground font-medium">{format_duration(t.eta)}</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col gap-0.5 items-end text-right">
|
||||||
|
<span class="text-muted-foreground uppercase text-[8px] tracking-tighter">"Eklenme"</span>
|
||||||
|
<span class="text-foreground/70">{format_date(t.added_date)}</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="grid grid-cols-4 gap-2 text-[10px] font-mono text-muted-foreground pt-1 border-t border-border/50">
|
</div>
|
||||||
<div class="flex flex-col text-blue-600 dark:text-blue-500"><span>"DL"</span><span>{format_speed(t.down_rate)}</span></div>
|
</TorrentContextMenu>
|
||||||
<div class="flex flex-col text-green-600 dark:text-green-500"><span>"UP"</span><span>{format_speed(t.up_rate)}</span></div>
|
}.into_any()
|
||||||
<div class="flex flex-col"><span>"ETA"</span><span>{format_duration(t.eta)}</span></div>
|
|
||||||
<div class="flex flex-col text-right"><span>"DATE"</span><span>{format_date(t.added_date)}</span></div>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</Show>
|
</Show>
|
||||||
}
|
}.into_any()
|
||||||
}
|
}
|
||||||
|
|||||||
121
frontend/src/components/torrent/trackers.rs
Normal file
121
frontend/src/components/torrent/trackers.rs
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use crate::components::ui::table::*;
|
||||||
|
use crate::components::ui::shimmer::*;
|
||||||
|
use shared::TorrentTracker;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TorrentTrackersTab(hash: String) -> impl IntoView {
|
||||||
|
let hash_clone = hash.clone();
|
||||||
|
|
||||||
|
let trackers_resource = Resource::new(
|
||||||
|
move || hash_clone.clone(),
|
||||||
|
|h| async move { shared::server_fns::torrent::get_trackers(h).await.unwrap_or_default() }
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Suspense fallback=move || view! { <TrackersFallback /> }>
|
||||||
|
{move || {
|
||||||
|
let trackers = trackers_resource.get().unwrap_or_default();
|
||||||
|
|
||||||
|
if trackers.is_empty() {
|
||||||
|
return view! {
|
||||||
|
<div class="flex flex-col items-center justify-center h-48 opacity-60">
|
||||||
|
<icons::Settings2 class="size-12 mb-3 text-muted-foreground" />
|
||||||
|
<p class="text-sm font-medium">"Bu torrent için izleyici bulunamadı."</p>
|
||||||
|
</div>
|
||||||
|
}.into_any();
|
||||||
|
}
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class="h-full overflow-auto">
|
||||||
|
<TableWrapper class="bg-card/50 whitespace-nowrap">
|
||||||
|
<Table>
|
||||||
|
<TableHeader class="sticky top-0 bg-muted/80 backdrop-blur-sm z-10">
|
||||||
|
<TableRow class="hover:bg-transparent text-xs">
|
||||||
|
<TableHead>"İsim"</TableHead>
|
||||||
|
<TableHead class="text-center">"Tür"</TableHead>
|
||||||
|
<TableHead class="text-center">"Etkin"</TableHead>
|
||||||
|
<TableHead class="text-center">"Grup"</TableHead>
|
||||||
|
<TableHead class="text-center">"Ortaklar"</TableHead>
|
||||||
|
<TableHead class="text-center">"Eşler"</TableHead>
|
||||||
|
<TableHead class="text-center">"İndirilen"</TableHead>
|
||||||
|
<TableHead class="text-center">"Son Güncelleme"</TableHead>
|
||||||
|
<TableHead class="text-center">"Sıklık"</TableHead>
|
||||||
|
<TableHead class="text-center">"Özel"</TableHead>
|
||||||
|
</TableRow>
|
||||||
|
</TableHeader>
|
||||||
|
<TableBody>
|
||||||
|
<For
|
||||||
|
each=move || trackers.clone()
|
||||||
|
key=|t| t.url.clone()
|
||||||
|
children=move |t| {
|
||||||
|
let t_type = if t.url.starts_with("http") { "http" }
|
||||||
|
else if t.url.starts_with("udp") { "udp" }
|
||||||
|
else if t.url.starts_with("dht") { "dht" }
|
||||||
|
else { "diğer" };
|
||||||
|
let is_enabled = if t.is_enabled { "evet" } else { "hayır" };
|
||||||
|
|
||||||
|
// Format timestamp difference for last update
|
||||||
|
let now = chrono::Utc::now().timestamp();
|
||||||
|
let diff = now - t.last_updated;
|
||||||
|
let last_update_str = if t.last_updated == 0 {
|
||||||
|
"Güncellenmedi".to_string()
|
||||||
|
} else if diff >= 0 {
|
||||||
|
format_duration_short(diff)
|
||||||
|
} else {
|
||||||
|
"N/A".to_string()
|
||||||
|
};
|
||||||
|
|
||||||
|
let url_clone = t.url.clone();
|
||||||
|
view! {
|
||||||
|
<TableRow class="hover:bg-muted/50 transition-colors group text-xs text-muted-foreground">
|
||||||
|
<TableCell class="font-medium text-foreground max-w-[200px] md:max-w-md truncate" attr:title=url_clone>
|
||||||
|
{t.url.clone()}
|
||||||
|
</TableCell>
|
||||||
|
<TableCell class="text-center">{t_type}</TableCell>
|
||||||
|
<TableCell class="text-center">{is_enabled}</TableCell>
|
||||||
|
<TableCell class="text-center">{t.group}</TableCell>
|
||||||
|
<TableCell class="text-center">{t.seeders}</TableCell>
|
||||||
|
<TableCell class="text-center">{t.peers}</TableCell>
|
||||||
|
<TableCell class="text-center">{t.downloaded}</TableCell>
|
||||||
|
<TableCell class="text-center">{last_update_str}</TableCell>
|
||||||
|
<TableCell class="text-center">{format_duration_short(t.interval)}</TableCell>
|
||||||
|
<TableCell class="text-center">"bilinmiyor"</TableCell> // Özel flag isn't cleanly via XMLRPC per tracker usually
|
||||||
|
</TableRow>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</TableBody>
|
||||||
|
</Table>
|
||||||
|
</TableWrapper>
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
|
}}
|
||||||
|
</Suspense>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn TrackersFallback() -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<Shimmer loading=Signal::derive(|| true) class="space-y-2">
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
<div class="h-10 w-full bg-muted/20 rounded-md"></div>
|
||||||
|
</Shimmer>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_duration_short(seconds: i64) -> String {
|
||||||
|
if seconds <= 0 { return "0sn".to_string(); }
|
||||||
|
let days = seconds / 86400;
|
||||||
|
let hours = (seconds % 86400) / 3600;
|
||||||
|
let minutes = (seconds % 3600) / 60;
|
||||||
|
let secs = seconds % 60;
|
||||||
|
|
||||||
|
if days > 0 { format!("{}g {}s", days, hours) }
|
||||||
|
else if hours > 0 { format!("{}s {}dk", hours, minutes) }
|
||||||
|
else if minutes > 0 { format!("{}dk {}sn", minutes, secs) }
|
||||||
|
else { format!("{}sn", secs) }
|
||||||
|
}
|
||||||
39
frontend/src/components/ui/accordion.rs
Normal file
39
frontend/src/components/ui/accordion.rs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Accordion(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("w-full", class);
|
||||||
|
view! { <div class=class>{children()}</div> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AccordionItem(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("border-b", class);
|
||||||
|
view! { <div class=class>{children()}</div> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AccordionHeader(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("flex", class);
|
||||||
|
view! { <div class=class>{children()}</div> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AccordionTrigger(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!(
|
||||||
|
"flex flex-1 items-center justify-between py-4 font-medium transition-all hover:underline [&[data-state=open]>svg]:rotate-180",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
view! {
|
||||||
|
<button type="button" class=class>
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AccordionContent(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("overflow-hidden text-sm transition-all", class);
|
||||||
|
view! { <div class=class>{children()}</div> }
|
||||||
|
}
|
||||||
94
frontend/src/components/ui/alert_dialog.rs
Normal file
94
frontend/src/components/ui/alert_dialog.rs
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
|
||||||
|
use crate::components::ui::button::{ButtonSize, ButtonVariant};
|
||||||
|
use crate::components::ui::dialog::{
|
||||||
|
Dialog, DialogBody, DialogClose, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialog(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! { <Dialog class=class>{children()}</Dialog> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogTrigger class=class variant=variant size=size>
|
||||||
|
{children()}
|
||||||
|
</DialogTrigger>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogContent(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogContent class=class close_on_backdrop_click=false data_name_prefix="AlertDialog">
|
||||||
|
{children()}
|
||||||
|
</DialogContent>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogBody(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogBody class=class attr:data-name="AlertDialogBody">
|
||||||
|
{children()}
|
||||||
|
</DialogBody>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogHeader(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogHeader class=class attr:data-name="AlertDialogHeader">
|
||||||
|
{children()}
|
||||||
|
</DialogHeader>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogTitle(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogTitle class=class attr:data-name="AlertDialogTitle">
|
||||||
|
{children()}
|
||||||
|
</DialogTitle>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogDescription(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogDescription class=class attr:data-name="AlertDialogDescription">
|
||||||
|
{children()}
|
||||||
|
</DialogDescription>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogFooter(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogFooter class=class attr:data-name="AlertDialogFooter">
|
||||||
|
{children()}
|
||||||
|
</DialogFooter>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn AlertDialogClose(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<DialogClose class=class variant=variant size=size>
|
||||||
|
{children()}
|
||||||
|
</DialogClose>
|
||||||
|
}
|
||||||
|
}
|
||||||
43
frontend/src/components/ui/badge.rs
Normal file
43
frontend/src/components/ui/badge.rs
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tailwind_fuse::tw_merge;
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display)]
|
||||||
|
pub enum BadgeVariant {
|
||||||
|
#[default]
|
||||||
|
Default,
|
||||||
|
Secondary,
|
||||||
|
Outline,
|
||||||
|
Destructive,
|
||||||
|
Success,
|
||||||
|
Warning,
|
||||||
|
Info,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Badge(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, default = BadgeVariant::Default)] variant: BadgeVariant,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let variant_classes = match variant {
|
||||||
|
BadgeVariant::Default => "border-transparent bg-primary text-primary-foreground hover:bg-primary/80",
|
||||||
|
BadgeVariant::Secondary => "border-transparent bg-secondary text-secondary-foreground hover:bg-secondary/80",
|
||||||
|
BadgeVariant::Outline => "text-foreground",
|
||||||
|
BadgeVariant::Destructive => "border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80",
|
||||||
|
BadgeVariant::Success => "border-transparent bg-green-500/10 text-green-600 dark:text-green-400 border-green-500/20",
|
||||||
|
BadgeVariant::Warning => "border-transparent bg-yellow-500/10 text-yellow-600 dark:text-yellow-400 border-yellow-500/20",
|
||||||
|
BadgeVariant::Info => "border-transparent bg-blue-500/10 text-blue-600 dark:text-blue-400 border-blue-500/20",
|
||||||
|
};
|
||||||
|
|
||||||
|
let class = tw_merge!(
|
||||||
|
"inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
|
||||||
|
variant_classes,
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div class=class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
39
frontend/src/components/ui/button.rs
Normal file
39
frontend/src/components/ui/button.rs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::variants;
|
||||||
|
|
||||||
|
// TODO 💪 Loading state (demo_use_timeout_fn.rs and demo_button.rs)
|
||||||
|
|
||||||
|
variants! {
|
||||||
|
Button {
|
||||||
|
base: "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive w-fit hover:cursor-pointer active:scale-[0.98] active:opacity-100 touch-manipulation [-webkit-tap-highlight-color:transparent] select-none [-webkit-touch-callout:none]", // Using hover:cursor-pointer as workaround for href_support.
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
Default: "bg-primary text-primary-foreground shadow-xs hover:bg-primary/90",
|
||||||
|
Destructive: "bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
|
||||||
|
Outline: "border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/5",
|
||||||
|
Secondary: "bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80",
|
||||||
|
Ghost: "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
|
||||||
|
Accent: "bg-accent text-accent-foreground hover:bg-accent/80",
|
||||||
|
Link: "text-primary underline-offset-4 hover:underline",
|
||||||
|
//
|
||||||
|
Warning: "bg-warning text-warning-foreground hover:bg-warning/90",
|
||||||
|
Success: "bg-success text-success-foreground hover:bg-success/90",
|
||||||
|
Bordered: "bg-transparent border border-zinc-200 text-muted-foreground",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
Default: "h-9 px-4 py-2 has-[>svg]:px-3",
|
||||||
|
Sm: "h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5",
|
||||||
|
Lg: "h-10 rounded-md px-6 has-[>svg]:px-4",
|
||||||
|
Icon: "size-9",
|
||||||
|
//
|
||||||
|
Mobile: "px-6 py-3 rounded-[24px]",
|
||||||
|
Badge: "px-2.5 py-0.5 text-xs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
component: {
|
||||||
|
element: button,
|
||||||
|
support_href: true,
|
||||||
|
support_aria_current: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
72
frontend/src/components/ui/button_action.rs
Normal file
72
frontend/src/components/ui/button_action.rs
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tailwind_fuse::tw_merge;
|
||||||
|
use crate::components::ui::button::{Button, ButtonVariant};
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ButtonAction(
|
||||||
|
children: Children,
|
||||||
|
#[prop(into)] on_action: Callback<()>,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = 1000)] hold_duration: u64,
|
||||||
|
#[prop(default = ButtonVariant::Default)] variant: ButtonVariant,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let is_holding = RwSignal::new(false);
|
||||||
|
let generation = StoredValue::new(0u64);
|
||||||
|
|
||||||
|
let on_down = move |_| {
|
||||||
|
generation.update_value(|g| *g += 1);
|
||||||
|
is_holding.set(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
let on_up = move |_| is_holding.set(false);
|
||||||
|
|
||||||
|
Effect::new(move |_| {
|
||||||
|
if is_holding.get() {
|
||||||
|
let current_gen = generation.get_value();
|
||||||
|
leptos::task::spawn_local(async move {
|
||||||
|
gloo_timers::future::TimeoutFuture::new(hold_duration as u32).await;
|
||||||
|
// Double validation: Is user still holding AND is it the SAME hold attempt?
|
||||||
|
if is_holding.get_untracked() && generation.get_value() == current_gen {
|
||||||
|
on_action.run(());
|
||||||
|
is_holding.set(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let merged_class = move || tw_merge!(
|
||||||
|
"relative overflow-hidden transition-all active:scale-[0.98]",
|
||||||
|
class.clone()
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<style>
|
||||||
|
"@keyframes button-hold-progress {
|
||||||
|
from { width: 0%; }
|
||||||
|
to { width: 100%; }
|
||||||
|
}
|
||||||
|
.animate-button-hold {
|
||||||
|
animation: button-hold-progress var(--button-hold-duration) linear forwards;
|
||||||
|
}"
|
||||||
|
</style>
|
||||||
|
<Button
|
||||||
|
variant=variant
|
||||||
|
class=merged_class()
|
||||||
|
attr:style=format!("--button-hold-duration: {}ms", hold_duration)
|
||||||
|
on:mousedown=on_down
|
||||||
|
on:mouseup=on_up
|
||||||
|
on:mouseleave=on_up
|
||||||
|
on:touchstart=move |_| is_holding.set(true)
|
||||||
|
on:touchend=move |_| is_holding.set(false)
|
||||||
|
>
|
||||||
|
// Progress Overlay
|
||||||
|
<Show when=move || is_holding.get()>
|
||||||
|
<div class="absolute inset-0 bg-white/20 dark:bg-black/20 pointer-events-none animate-button-hold" />
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<span class="relative z-10 flex items-center justify-center gap-2">
|
||||||
|
{children()}
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
19
frontend/src/components/ui/card.rs
Normal file
19
frontend/src/components/ui/card.rs
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {Card, div, "bg-card text-card-foreground flex flex-col gap-4 rounded-xl border py-6 shadow-sm"}
|
||||||
|
// TODO. Change data-slot=card-action by data-name="CardAction".
|
||||||
|
clx! {CardHeader, div, "@container/card-header flex flex-col items-start gap-1.5 px-6 [.border-b]:pb-6 sm:grid sm:auto-rows-min sm:grid-rows-[auto_auto] has-data-[slot=card-action]:sm:grid-cols-[1fr_auto]"}
|
||||||
|
clx! {CardTitle, h2, "leading-none font-semibold"}
|
||||||
|
clx! {CardContent, div, "px-6"}
|
||||||
|
clx! {CardDescription, p, "text-muted-foreground text-sm"}
|
||||||
|
clx! {CardFooter, footer, "flex items-center px-6 [.border-t]:pt-6", "gap-2"}
|
||||||
|
|
||||||
|
clx! {CardAction, div, "self-start sm:col-start-2 sm:row-span-2 sm:row-start-1 sm:justify-self-end"}
|
||||||
|
clx! {CardList, ul, "flex flex-col gap-4"}
|
||||||
|
clx! {CardItem, li, "flex items-center [&_svg:not([class*='size-'])]:size-4 [&_svg]:shrink-0"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
43
frontend/src/components/ui/checkbox.rs
Normal file
43
frontend/src/components/ui/checkbox.rs
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
use icons::Check;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Checkbox(
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
#[prop(into, optional)] checked: Signal<bool>,
|
||||||
|
#[prop(into, optional)] disabled: Signal<bool>,
|
||||||
|
#[prop(into, optional)] on_checked_change: Option<Callback<bool>>,
|
||||||
|
#[prop(into, optional, default = "Checkbox".to_string())] aria_label: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let checked_state = move || if checked.get() { "checked" } else { "unchecked" };
|
||||||
|
|
||||||
|
let checkbox_class = tw_merge!(
|
||||||
|
"peer border-input dark:bg-input/30 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground dark:data-[state=checked]:bg-primary data-[state=checked]:border-primary focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive size-4 shrink-0 rounded-[4px] border shadow-xs transition-shadow outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
data-name="Checkbox"
|
||||||
|
class=checkbox_class
|
||||||
|
data-state=checked_state
|
||||||
|
type="button"
|
||||||
|
role="checkbox"
|
||||||
|
aria-checked=move || checked.get().to_string()
|
||||||
|
aria-label=aria_label
|
||||||
|
disabled=move || disabled.get()
|
||||||
|
on:click=move |_| {
|
||||||
|
if !disabled.get() {
|
||||||
|
if let Some(callback) = on_checked_change {
|
||||||
|
callback.run(!checked.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<span data-name="CheckboxIndicator" class="flex justify-center items-center text-current transition-none">
|
||||||
|
{move || { checked.get().then(|| view! { <Check class="size-3.5".to_string() /> }) }}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
419
frontend/src/components/ui/context_menu.rs
Normal file
419
frontend/src/components/ui/context_menu.rs
Normal file
@@ -0,0 +1,419 @@
|
|||||||
|
use icons::ChevronRight;
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
use tw_merge::*;
|
||||||
|
use wasm_bindgen::JsCast;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
|
||||||
|
/// Programmatically close any open context menu.
|
||||||
|
pub fn close_context_menu() {
|
||||||
|
let Some(document) = window().document() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
let Some(menu) = document.query_selector("[data-target='target__context'][data-state='open']").ok().flatten()
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
let _ = menu.set_attribute("data-state", "closed");
|
||||||
|
if let Some(el) = menu.dyn_ref::<web_sys::HtmlElement>() {
|
||||||
|
let _ = el.style().set_property("pointer-events", "none");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {ContextMenuLabel, span, "px-2 py-1.5 text-sm font-medium data-inset:pl-8", "mb-1"}
|
||||||
|
clx! {ContextMenuGroup, ul, "group"}
|
||||||
|
clx! {ContextMenuItem, li, "inline-flex gap-2 items-center w-full rounded-sm px-2 py-1.5 text-sm no-underline transition-colors duration-200 text-popover-foreground hover:bg-accent hover:text-accent-foreground [&_svg:not([class*='size-'])]:size-4"}
|
||||||
|
clx! {ContextMenuSubContent, ul, "context__menu_sub_content", "rounded-md border bg-card shadow-lg p-1 absolute z-[100] min-w-[160px] opacity-0 invisible translate-x-[-8px] transition-all duration-200 ease-out pointer-events-none"}
|
||||||
|
clx! {ContextMenuLink, a, "w-full inline-flex gap-2 items-center"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuAction(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] aria_selected: Option<Signal<bool>>,
|
||||||
|
#[prop(optional, into)] href: Option<String>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let _ctx = expect_context::<ContextMenuContext>();
|
||||||
|
|
||||||
|
let class = tw_merge!(
|
||||||
|
"inline-flex gap-2 items-center w-full text-sm text-left transition-colors duration-200 focus:outline-none focus-visible:outline-none text-popover-foreground [&_svg:not([class*='size-'])]:size-4",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let aria_selected_attr = move || aria_selected.map(|s| s.get()).unwrap_or(false).to_string();
|
||||||
|
|
||||||
|
if let Some(href) = href {
|
||||||
|
view! {
|
||||||
|
<a
|
||||||
|
data-name="ContextMenuAction"
|
||||||
|
class=class
|
||||||
|
href=href
|
||||||
|
aria-selected=aria_selected_attr
|
||||||
|
data-context-close="true"
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
} else {
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-name="ContextMenuAction"
|
||||||
|
class=class
|
||||||
|
data-context-close="true"
|
||||||
|
aria-selected=aria_selected_attr
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct ContextMenuContext {
|
||||||
|
target_id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenu(children: Children) -> impl IntoView {
|
||||||
|
let context_target_id = use_random_id_for("context");
|
||||||
|
|
||||||
|
let ctx = ContextMenuContext { target_id: context_target_id.clone() };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<style>
|
||||||
|
"
|
||||||
|
/* Submenu Styles */
|
||||||
|
.context__menu_sub_content {
|
||||||
|
position: absolute;
|
||||||
|
inset-inline-start: calc(100% + 8px);
|
||||||
|
inset-block-start: -4px;
|
||||||
|
z-index: 100;
|
||||||
|
min-inline-size: 160px;
|
||||||
|
opacity: 0;
|
||||||
|
visibility: hidden;
|
||||||
|
transform: translateX(-8px);
|
||||||
|
transition: all 0.2s ease-out;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.context__menu_sub_trigger:hover .context__menu_sub_content {
|
||||||
|
opacity: 1;
|
||||||
|
visibility: visible;
|
||||||
|
transform: translateX(0);
|
||||||
|
pointer-events: auto;
|
||||||
|
}
|
||||||
|
"
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div data-name="ContextMenu" class="contents">
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wrapper that triggers the context menu on right-click.
|
||||||
|
/// The `on_open` callback is triggered when the context menu opens (right-click).
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional)] on_open: Option<Callback<()>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<ContextMenuContext>();
|
||||||
|
let trigger_class = tw_merge!("contents", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
class=trigger_class
|
||||||
|
data-name="ContextMenuTrigger"
|
||||||
|
data-context-trigger=ctx.target_id
|
||||||
|
on:contextmenu=move |_e: web_sys::MouseEvent| {
|
||||||
|
if let Some(cb) = on_open {
|
||||||
|
cb.run(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Content of the context menu that appears on right-click.
|
||||||
|
/// The `on_close` callback is triggered when the menu closes (click outside, ESC key, or action click).
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuContent(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional)] on_close: Option<Callback<()>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<ContextMenuContext>();
|
||||||
|
|
||||||
|
let base_classes = "fixed z-50 min-w-[12rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-md transition-all duration-200 data-[state=closed]:opacity-0 data-[state=closed]:scale-95 data-[state=open]:opacity-100 data-[state=open]:scale-100";
|
||||||
|
|
||||||
|
let class = tw_merge!(base_classes, class);
|
||||||
|
|
||||||
|
let target_id_for_script = ctx.target_id.clone();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="ContextMenuContent"
|
||||||
|
class=class
|
||||||
|
// Listen for custom 'contextmenuclose' event dispatched by JS when menu closes
|
||||||
|
on:contextmenuclose=move |_: web_sys::CustomEvent| {
|
||||||
|
if let Some(cb) = on_close {
|
||||||
|
cb.run(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
id=ctx.target_id
|
||||||
|
data-target="target__context"
|
||||||
|
data-state="closed"
|
||||||
|
style="pointer-events: none;"
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupContextMenu = () => {{
|
||||||
|
const menu = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-context-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!menu || !trigger) {{
|
||||||
|
setTimeout(setupContextMenu, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (menu.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
menu.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
let isOpen = false;
|
||||||
|
|
||||||
|
const updatePosition = (x, y) => {{
|
||||||
|
const menuRect = menu.getBoundingClientRect();
|
||||||
|
const viewportHeight = window.innerHeight;
|
||||||
|
const viewportWidth = window.innerWidth;
|
||||||
|
|
||||||
|
// Calculate position, ensuring menu stays within viewport
|
||||||
|
let left = x;
|
||||||
|
let top = y;
|
||||||
|
|
||||||
|
// Adjust if menu would go off right edge
|
||||||
|
if (x + menuRect.width > viewportWidth) {{
|
||||||
|
left = Math.max(0, x - menuRect.width);
|
||||||
|
}}
|
||||||
|
|
||||||
|
// Adjust if menu would go off bottom edge
|
||||||
|
if (y + menuRect.height > viewportHeight) {{
|
||||||
|
top = Math.max(0, y - menuRect.height);
|
||||||
|
}}
|
||||||
|
|
||||||
|
// Adjust for CSS transformed containing block
|
||||||
|
const offsetParent = menu.offsetParent;
|
||||||
|
if (offsetParent && offsetParent !== document.body && offsetParent !== document.documentElement) {{
|
||||||
|
const parentRect = offsetParent.getBoundingClientRect();
|
||||||
|
left -= parentRect.left;
|
||||||
|
top -= parentRect.top;
|
||||||
|
}}
|
||||||
|
|
||||||
|
menu.style.left = `${{left}}px`;
|
||||||
|
menu.style.top = `${{top}}px`;
|
||||||
|
menu.style.transformOrigin = 'top left';
|
||||||
|
}};
|
||||||
|
|
||||||
|
const openMenu = (x, y) => {{
|
||||||
|
isOpen = true;
|
||||||
|
|
||||||
|
// Close any other open context menus
|
||||||
|
const allMenus = document.querySelectorAll('[data-target="target__context"]');
|
||||||
|
allMenus.forEach(m => {{
|
||||||
|
if (m !== menu && m.getAttribute('data-state') === 'open') {{
|
||||||
|
m.setAttribute('data-state', 'closed');
|
||||||
|
m.style.pointerEvents = 'none';
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
menu.setAttribute('data-state', 'open');
|
||||||
|
menu.style.visibility = 'hidden';
|
||||||
|
menu.style.pointerEvents = 'auto';
|
||||||
|
|
||||||
|
// Force reflow
|
||||||
|
menu.offsetHeight;
|
||||||
|
|
||||||
|
updatePosition(x, y);
|
||||||
|
menu.style.visibility = 'visible';
|
||||||
|
|
||||||
|
// Lock scroll
|
||||||
|
if (window.ScrollLock) {{
|
||||||
|
window.ScrollLock.lock();
|
||||||
|
}}
|
||||||
|
|
||||||
|
setTimeout(() => {{
|
||||||
|
document.addEventListener('click', handleClickOutside);
|
||||||
|
document.addEventListener('contextmenu', handleContextOutside);
|
||||||
|
}}, 0);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeMenu = () => {{
|
||||||
|
isOpen = false;
|
||||||
|
menu.setAttribute('data-state', 'closed');
|
||||||
|
menu.style.pointerEvents = 'none';
|
||||||
|
document.removeEventListener('click', handleClickOutside);
|
||||||
|
document.removeEventListener('contextmenu', handleContextOutside);
|
||||||
|
|
||||||
|
// Dispatch custom event for Leptos to listen to
|
||||||
|
menu.dispatchEvent(new CustomEvent('contextmenuclose', {{ bubbles: false }}));
|
||||||
|
|
||||||
|
if (window.ScrollLock) {{
|
||||||
|
window.ScrollLock.unlock(200);
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
const handleClickOutside = (e) => {{
|
||||||
|
if (!menu.contains(e.target)) {{
|
||||||
|
closeMenu();
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
const handleContextOutside = (e) => {{
|
||||||
|
if (!trigger.contains(e.target)) {{
|
||||||
|
closeMenu();
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
// Right-click on trigger (desktop)
|
||||||
|
trigger.addEventListener('contextmenu', (e) => {{
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
if (isOpen) {{
|
||||||
|
closeMenu();
|
||||||
|
}}
|
||||||
|
openMenu(e.clientX, e.clientY);
|
||||||
|
}});
|
||||||
|
|
||||||
|
// Long-press on trigger (mobile)
|
||||||
|
let touchTimer = null;
|
||||||
|
let touchStartX = 0;
|
||||||
|
let touchStartY = 0;
|
||||||
|
const LONG_PRESS_DURATION = 500;
|
||||||
|
const MOVE_THRESHOLD = 10;
|
||||||
|
|
||||||
|
trigger.addEventListener('touchstart', (e) => {{
|
||||||
|
const touch = e.touches[0];
|
||||||
|
touchStartX = touch.clientX;
|
||||||
|
touchStartY = touch.clientY;
|
||||||
|
|
||||||
|
touchTimer = setTimeout(() => {{
|
||||||
|
e.preventDefault();
|
||||||
|
if (isOpen) {{
|
||||||
|
closeMenu();
|
||||||
|
}}
|
||||||
|
openMenu(touchStartX, touchStartY);
|
||||||
|
}}, LONG_PRESS_DURATION);
|
||||||
|
}}, {{ passive: false }});
|
||||||
|
|
||||||
|
trigger.addEventListener('touchmove', (e) => {{
|
||||||
|
if (touchTimer) {{
|
||||||
|
const touch = e.touches[0];
|
||||||
|
const dx = Math.abs(touch.clientX - touchStartX);
|
||||||
|
const dy = Math.abs(touch.clientY - touchStartY);
|
||||||
|
if (dx > MOVE_THRESHOLD || dy > MOVE_THRESHOLD) {{
|
||||||
|
clearTimeout(touchTimer);
|
||||||
|
touchTimer = null;
|
||||||
|
}}
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
trigger.addEventListener('touchend', () => {{
|
||||||
|
if (touchTimer) {{
|
||||||
|
clearTimeout(touchTimer);
|
||||||
|
touchTimer = null;
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
trigger.addEventListener('touchcancel', () => {{
|
||||||
|
if (touchTimer) {{
|
||||||
|
clearTimeout(touchTimer);
|
||||||
|
touchTimer = null;
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
// Close when action is clicked
|
||||||
|
const actions = menu.querySelectorAll('[data-context-close]');
|
||||||
|
actions.forEach(action => {{
|
||||||
|
action.addEventListener('click', () => {{
|
||||||
|
closeMenu();
|
||||||
|
}});
|
||||||
|
}});
|
||||||
|
|
||||||
|
// Handle ESC key
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && isOpen) {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeMenu();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
|
||||||
|
if (document.readyState === 'loading') {{
|
||||||
|
document.addEventListener('DOMContentLoaded', setupContextMenu);
|
||||||
|
}} else {{
|
||||||
|
setupContextMenu();
|
||||||
|
}}
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
target_id_for_script,
|
||||||
|
target_id_for_script,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuSub(children: Children) -> impl IntoView {
|
||||||
|
clx! {ContextMenuSubRoot, li, "context__menu_sub_trigger", " relative inline-flex relative gap-2 items-center py-1.5 px-2 w-full text-sm no-underline rounded-sm transition-colors duration-200 cursor-pointer text-popover-foreground [&_svg:not([class*='size-'])]:size-4 hover:bg-accent hover:text-accent-foreground"}
|
||||||
|
|
||||||
|
view! { <ContextMenuSubRoot>{children()}</ContextMenuSubRoot> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuSubTrigger(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("flex items-center justify-between w-full", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<span data-name="ContextMenuSubTrigger" class=class>
|
||||||
|
<span class="flex gap-2 items-center">{children()}</span>
|
||||||
|
<ChevronRight class="opacity-70 size-4" />
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ContextMenuSubItem(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!(
|
||||||
|
"inline-flex gap-2 items-center w-full rounded-sm px-3 py-2 text-sm transition-all duration-150 ease text-popover-foreground hover:bg-accent hover:text-accent-foreground cursor-pointer hover:translate-x-[2px]",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<li data-name="ContextMenuSubItem" class=class data-context-close="true">
|
||||||
|
{children()}
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
}
|
||||||
6
frontend/src/components/ui/data_table.rs
Normal file
6
frontend/src/components/ui/data_table.rs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
// * Reuse @table.rs
|
||||||
|
pub use crate::components::ui::table::{
|
||||||
|
Table as DataTable, TableBody as DataTableBody, TableCaption as DataTableCaption, TableCell as DataTableCell,
|
||||||
|
TableFooter as DataTableFooter, TableHead as DataTableHead, TableHeader as DataTableHeader,
|
||||||
|
TableRow as DataTableRow, TableWrapper as DataTableWrapper,
|
||||||
|
};
|
||||||
238
frontend/src/components/ui/dialog.rs
Normal file
238
frontend/src/components/ui/dialog.rs
Normal file
@@ -0,0 +1,238 @@
|
|||||||
|
use icons::X;
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
use crate::components::ui::button::{Button, ButtonSize, ButtonVariant};
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {DialogBody, div, "flex flex-col gap-4"}
|
||||||
|
clx! {DialogHeader, div, "flex flex-col gap-2 text-center sm:text-left"}
|
||||||
|
clx! {DialogTitle, h3, "text-lg leading-none font-semibold"}
|
||||||
|
clx! {DialogDescription, p, "text-muted-foreground text-sm"}
|
||||||
|
clx! {DialogFooter, footer, "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct DialogContext {
|
||||||
|
target_id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Dialog(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let dialog_target_id = use_random_id_for("dialog");
|
||||||
|
|
||||||
|
let ctx = DialogContext { target_id: dialog_target_id.clone() };
|
||||||
|
|
||||||
|
let merged_class = tw_merge!("w-fit", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<div class=merged_class data-name="__Dialog">
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DialogTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DialogContext>();
|
||||||
|
let trigger_id = format!("trigger_{}", ctx.target_id);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Button
|
||||||
|
class=class
|
||||||
|
attr:id=trigger_id
|
||||||
|
attr:tabindex="0"
|
||||||
|
attr:data-dialog-trigger=ctx.target_id
|
||||||
|
variant=variant
|
||||||
|
size=size
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DialogContent(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] id: Option<String>,
|
||||||
|
#[prop(into, optional)] hide_close_button: Option<bool>,
|
||||||
|
#[prop(default = true)] close_on_backdrop_click: bool,
|
||||||
|
#[prop(default = "Dialog")] data_name_prefix: &'static str,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DialogContext>();
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"relative bg-background border rounded-2xl shadow-lg p-6 w-full max-w-[calc(100%-2rem)] max-h-[85vh] fixed top-[50%] left-[50%] translate-x-[-50%] translate-y-[-50%] z-100 transition-all duration-200 data-[state=closed]:opacity-0 data-[state=closed]:scale-95 data-[state=open]:opacity-100 data-[state=open]:scale-100",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let backdrop_data_name = format!("{}Backdrop", data_name_prefix);
|
||||||
|
let content_data_name = format!("{}Content", data_name_prefix);
|
||||||
|
|
||||||
|
let target_id_clone = ctx.target_id.clone();
|
||||||
|
let backdrop_id = format!("{}_backdrop", ctx.target_id);
|
||||||
|
let backdrop_id_for_script = backdrop_id.clone();
|
||||||
|
let backdrop_behavior = if close_on_backdrop_click { "auto" } else { "manual" };
|
||||||
|
|
||||||
|
// Use provided id or fallback to random target_id
|
||||||
|
let final_id = id.unwrap_or_else(|| ctx.target_id.clone());
|
||||||
|
let final_id_for_script = final_id.clone();
|
||||||
|
let trigger_id_for_script = ctx.target_id.clone();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<script src="/lock_scroll.js"></script>
|
||||||
|
|
||||||
|
<div
|
||||||
|
data-name=backdrop_data_name
|
||||||
|
id=backdrop_id
|
||||||
|
class="fixed inset-0 transition-opacity duration-200 pointer-events-none z-60 bg-black/50 data-[state=closed]:opacity-0 data-[state=open]:opacity-100"
|
||||||
|
data-state="closed"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div
|
||||||
|
data-name=content_data_name
|
||||||
|
class=merged_class
|
||||||
|
id=final_id
|
||||||
|
data-target="target__dialog"
|
||||||
|
data-state="closed"
|
||||||
|
data-backdrop=backdrop_behavior
|
||||||
|
style="pointer-events: none;"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class=format!(
|
||||||
|
"absolute top-4 right-4 p-1 rounded-sm focus:ring-2 focus:ring-offset-2 focus:outline-none [&_svg:not([class*='size-'])]:size-4 focus:ring-ring{}",
|
||||||
|
if hide_close_button.unwrap_or(false) { " hidden" } else { "" },
|
||||||
|
)
|
||||||
|
data-dialog-close=target_id_clone.clone()
|
||||||
|
aria-label="Close dialog"
|
||||||
|
>
|
||||||
|
<span class="hidden">"Close Dialog"</span>
|
||||||
|
<X />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupDialog = () => {{
|
||||||
|
const dialog = document.querySelector('#{}');
|
||||||
|
const backdrop = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-dialog-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!dialog || !backdrop || !trigger) {{
|
||||||
|
setTimeout(setupDialog, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (dialog.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
dialog.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
const openDialog = () => {{
|
||||||
|
if (window.ScrollLock) window.ScrollLock.lock();
|
||||||
|
dialog.setAttribute('data-state', 'open');
|
||||||
|
backdrop.setAttribute('data-state', 'open');
|
||||||
|
dialog.style.pointerEvents = 'auto';
|
||||||
|
backdrop.style.pointerEvents = 'auto';
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeDialog = () => {{
|
||||||
|
dialog.setAttribute('data-state', 'closed');
|
||||||
|
backdrop.setAttribute('data-state', 'closed');
|
||||||
|
dialog.style.pointerEvents = 'none';
|
||||||
|
backdrop.style.pointerEvents = 'none';
|
||||||
|
window.ScrollLock.unlock(200);
|
||||||
|
}};
|
||||||
|
|
||||||
|
trigger.addEventListener('click', openDialog);
|
||||||
|
dialog.querySelectorAll('[data-dialog-close]').forEach(btn => {{
|
||||||
|
btn.addEventListener('click', closeDialog);
|
||||||
|
}});
|
||||||
|
backdrop.addEventListener('click', () => {{
|
||||||
|
if (dialog.getAttribute('data-backdrop') === 'auto') {{
|
||||||
|
closeDialog();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && dialog.getAttribute('data-state') === 'open') {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeDialog();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
setupDialog();
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
final_id_for_script,
|
||||||
|
backdrop_id_for_script,
|
||||||
|
trigger_id_for_script,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DialogClose(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DialogContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Button
|
||||||
|
class=class
|
||||||
|
attr:data-dialog-close=ctx.target_id
|
||||||
|
attr:aria-label="Close dialog"
|
||||||
|
variant=variant
|
||||||
|
size=size
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DialogAction(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Default)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DialogContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Button
|
||||||
|
class=class
|
||||||
|
attr:data-dialog-close=ctx.target_id
|
||||||
|
attr:aria-label="Close dialog"
|
||||||
|
variant=variant
|
||||||
|
size=size
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
536
frontend/src/components/ui/dropdown_menu.rs
Normal file
536
frontend/src/components/ui/dropdown_menu.rs
Normal file
@@ -0,0 +1,536 @@
|
|||||||
|
use icons::{Check, ChevronRight};
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
pub use crate::components::ui::separator::Separator as DropdownMenuSeparator;
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {DropdownMenuLabel, span, "px-2 py-1.5 text-sm font-medium data-inset:pl-8", "mb-1"}
|
||||||
|
clx! {DropdownMenuGroup, ul, "group"}
|
||||||
|
clx! {DropdownMenuItem, li, "inline-flex gap-2 items-center w-full rounded-sm px-2 py-1.5 text-sm no-underline transition-colors duration-200 text-popover-foreground hover:bg-accent hover:text-accent-foreground [&_svg:not([class*='size-'])]:size-4"}
|
||||||
|
clx! {DropdownMenuSubContent, ul, "dropdown__menu_sub_content", "rounded-md border bg-card shadow-lg p-1 absolute z-[100] min-w-[160px] opacity-0 invisible translate-x-[-8px] transition-all duration-200 ease-out pointer-events-none"}
|
||||||
|
clx! {DropdownMenuLink, a, "w-full inline-flex gap-2 items-center"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* RADIO GROUP */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct DropdownMenuRadioContext<T: Clone + PartialEq + Send + Sync + 'static> {
|
||||||
|
value_signal: RwSignal<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A group of radio items where only one can be selected at a time.
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuRadioGroup<T>(
|
||||||
|
children: Children,
|
||||||
|
/// The signal holding the current selected value
|
||||||
|
value: RwSignal<T>,
|
||||||
|
) -> impl IntoView
|
||||||
|
where
|
||||||
|
T: Clone + PartialEq + Send + Sync + 'static,
|
||||||
|
{
|
||||||
|
let ctx = DropdownMenuRadioContext { value_signal: value };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<ul data-name="DropdownMenuRadioGroup" role="group" class="group">
|
||||||
|
{children()}
|
||||||
|
</ul>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A radio item that shows a checkmark when selected.
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuRadioItem<T>(
|
||||||
|
children: Children,
|
||||||
|
/// The value this item represents
|
||||||
|
value: T,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView
|
||||||
|
where
|
||||||
|
T: Clone + PartialEq + Send + Sync + 'static,
|
||||||
|
{
|
||||||
|
let ctx = expect_context::<DropdownMenuRadioContext<T>>();
|
||||||
|
|
||||||
|
let value_for_check = value.clone();
|
||||||
|
let value_for_click = value.clone();
|
||||||
|
let is_selected = move || ctx.value_signal.get() == value_for_check;
|
||||||
|
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"group inline-flex gap-2 items-center w-full rounded-sm pl-2 pr-2 py-1.5 text-sm cursor-pointer no-underline transition-colors duration-200 text-popover-foreground hover:bg-accent hover:text-accent-foreground [&_svg:not([class*='size-'])]:size-4",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<li
|
||||||
|
data-name="DropdownMenuRadioItem"
|
||||||
|
class=merged_class
|
||||||
|
role="menuitemradio"
|
||||||
|
aria-checked=move || is_selected().to_string()
|
||||||
|
data-dropdown-close="true"
|
||||||
|
on:click=move |_| {
|
||||||
|
ctx.value_signal.set(value_for_click.clone());
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
<Check class="ml-auto opacity-0 size-4 text-muted-foreground group-aria-checked:opacity-100" />
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An action item in a dropdown menu (no checkmark, just triggers an action).
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuAction(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] href: Option<String>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let _ctx = expect_context::<DropdownMenuContext>();
|
||||||
|
|
||||||
|
let class = tw_merge!(
|
||||||
|
"inline-flex gap-2 items-center w-full text-sm text-left transition-colors duration-200 focus:outline-none focus-visible:outline-none text-popover-foreground [&_svg:not([class*='size-'])]:size-4 hover:bg-accent hover:text-accent-foreground",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some(href) = href {
|
||||||
|
// Render as <a> tag when href is provided
|
||||||
|
view! {
|
||||||
|
<a data-name="DropdownMenuAction" class=class href=href data-dropdown-close="true">
|
||||||
|
{children()}
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{r#"
|
||||||
|
(function() {
|
||||||
|
const link = document.currentScript.previousElementSibling;
|
||||||
|
if (!link) return;
|
||||||
|
|
||||||
|
link.addEventListener('click', function() {
|
||||||
|
// Close dropdown on route change after navigation
|
||||||
|
let currentPath = window.location.pathname;
|
||||||
|
const checkRouteChange = () => {
|
||||||
|
if (window.location.pathname !== currentPath) {
|
||||||
|
currentPath = window.location.pathname;
|
||||||
|
|
||||||
|
// Find and close the dropdown
|
||||||
|
const dropdown = link.closest('[data-target="target__dropdown"]');
|
||||||
|
if (dropdown) {
|
||||||
|
dropdown.setAttribute('data-state', 'closed');
|
||||||
|
dropdown.style.pointerEvents = 'none';
|
||||||
|
|
||||||
|
// Unlock scroll
|
||||||
|
if (window.ScrollLock) {
|
||||||
|
window.ScrollLock.unlock(200);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
clearInterval(routeCheckInterval);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const routeCheckInterval = setInterval(checkRouteChange, 50);
|
||||||
|
|
||||||
|
// Clear interval after 2 seconds to prevent memory leaks
|
||||||
|
setTimeout(() => clearInterval(routeCheckInterval), 2000);
|
||||||
|
});
|
||||||
|
})();
|
||||||
|
"#}
|
||||||
|
</script>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
} else {
|
||||||
|
// Render as <button> tag when no href
|
||||||
|
view! {
|
||||||
|
<button type="button" data-name="DropdownMenuAction" class=class data-dropdown-close="true">
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default)]
|
||||||
|
pub enum DropdownMenuAlign {
|
||||||
|
#[default]
|
||||||
|
Start,
|
||||||
|
StartOuter,
|
||||||
|
End,
|
||||||
|
EndOuter,
|
||||||
|
Center,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct DropdownMenuContext {
|
||||||
|
target_id: String,
|
||||||
|
align: DropdownMenuAlign,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenu(
|
||||||
|
children: Children,
|
||||||
|
#[prop(default = DropdownMenuAlign::default())] align: DropdownMenuAlign,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let dropdown_target_id = use_random_id_for("dropdown");
|
||||||
|
|
||||||
|
let ctx = DropdownMenuContext { target_id: dropdown_target_id.clone(), align };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<style>
|
||||||
|
"
|
||||||
|
/* Submenu Styles */
|
||||||
|
.dropdown__menu_sub_content {
|
||||||
|
position: absolute;
|
||||||
|
inset-inline-start: calc(100% + 8px);
|
||||||
|
inset-block-start: -4px;
|
||||||
|
z-index: 100;
|
||||||
|
min-inline-size: 160px;
|
||||||
|
opacity: 0;
|
||||||
|
visibility: hidden;
|
||||||
|
transform: translateX(-8px);
|
||||||
|
transition: all 0.2s ease-out;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown__menu_sub_trigger:hover .dropdown__menu_sub_content {
|
||||||
|
opacity: 1;
|
||||||
|
visibility: visible;
|
||||||
|
transform: translateX(0);
|
||||||
|
pointer-events: auto;
|
||||||
|
}
|
||||||
|
"
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div data-name="DropdownMenu">{children()}</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuTrigger(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DropdownMenuContext>();
|
||||||
|
let button_class = tw_merge!(
|
||||||
|
"px-4 py-2 h-9 inline-flex justify-center items-center text-sm font-medium whitespace-nowrap rounded-md transition-colors w-fit focus:outline-none focus:ring-1 focus:ring-ring focus-visible:outline-hidden focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 [&_svg:not([class*='size-'])]:size-4 border bg-background border-input hover:bg-accent hover:text-accent-foreground",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class=button_class
|
||||||
|
data-name="DropdownMenuTrigger"
|
||||||
|
data-dropdown-trigger=ctx.target_id
|
||||||
|
tabindex="0"
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default)]
|
||||||
|
pub enum DropdownMenuPosition {
|
||||||
|
#[default]
|
||||||
|
Auto,
|
||||||
|
Top,
|
||||||
|
Bottom,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuContent(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = DropdownMenuPosition::default())] position: DropdownMenuPosition,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<DropdownMenuContext>();
|
||||||
|
|
||||||
|
let base_classes = "z-50 p-1 rounded-md border bg-card shadow-md h-fit fixed transition-all duration-200 data-[state=closed]:opacity-0 data-[state=closed]:scale-95 data-[state=open]:opacity-100 data-[state=open]:scale-100";
|
||||||
|
let width_class = match ctx.align {
|
||||||
|
DropdownMenuAlign::Center => "min-w-full",
|
||||||
|
_ => "w-[180px]",
|
||||||
|
};
|
||||||
|
|
||||||
|
let class = tw_merge!(width_class, base_classes, class);
|
||||||
|
|
||||||
|
let target_id_for_script = ctx.target_id.clone();
|
||||||
|
let align_for_script = match ctx.align {
|
||||||
|
DropdownMenuAlign::Start => "start",
|
||||||
|
DropdownMenuAlign::StartOuter => "start-outer",
|
||||||
|
DropdownMenuAlign::End => "end",
|
||||||
|
DropdownMenuAlign::EndOuter => "end-outer",
|
||||||
|
DropdownMenuAlign::Center => "center",
|
||||||
|
};
|
||||||
|
|
||||||
|
let position_for_script = match position {
|
||||||
|
DropdownMenuPosition::Auto => "auto",
|
||||||
|
DropdownMenuPosition::Top => "top",
|
||||||
|
DropdownMenuPosition::Bottom => "bottom",
|
||||||
|
};
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="DropdownMenuContent"
|
||||||
|
class=class
|
||||||
|
id=ctx.target_id
|
||||||
|
data-target="target__dropdown"
|
||||||
|
data-state="closed"
|
||||||
|
data-align=align_for_script
|
||||||
|
data-position=position_for_script
|
||||||
|
style="pointer-events: none;"
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupDropdown = () => {{
|
||||||
|
const dropdown = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-dropdown-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!dropdown || !trigger) {{
|
||||||
|
setTimeout(setupDropdown, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (dropdown.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
dropdown.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
let isOpen = false;
|
||||||
|
|
||||||
|
const updatePosition = () => {{
|
||||||
|
const triggerRect = trigger.getBoundingClientRect();
|
||||||
|
const dropdownRect = dropdown.getBoundingClientRect();
|
||||||
|
const viewportHeight = window.innerHeight;
|
||||||
|
const viewportWidth = window.innerWidth;
|
||||||
|
const spaceBelow = viewportHeight - triggerRect.bottom;
|
||||||
|
const spaceAbove = triggerRect.top;
|
||||||
|
|
||||||
|
const align = dropdown.getAttribute('data-align') || 'start';
|
||||||
|
const position = dropdown.getAttribute('data-position') || 'auto';
|
||||||
|
|
||||||
|
// Determine if we should position above
|
||||||
|
let shouldPositionAbove = false;
|
||||||
|
if (position === 'top') {{
|
||||||
|
shouldPositionAbove = true;
|
||||||
|
}} else if (position === 'bottom') {{
|
||||||
|
shouldPositionAbove = false;
|
||||||
|
}} else {{
|
||||||
|
// Auto: position above if there's space above AND not enough space below
|
||||||
|
shouldPositionAbove = spaceAbove >= dropdownRect.height && spaceBelow < dropdownRect.height;
|
||||||
|
}}
|
||||||
|
|
||||||
|
switch (align) {{
|
||||||
|
case 'start':
|
||||||
|
if (shouldPositionAbove) {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top - dropdownRect.height - 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'left bottom';
|
||||||
|
}} else {{
|
||||||
|
dropdown.style.top = `${{triggerRect.bottom + 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'left top';
|
||||||
|
}}
|
||||||
|
dropdown.style.left = `${{triggerRect.left}}px`;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'end':
|
||||||
|
if (shouldPositionAbove) {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top - dropdownRect.height - 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'right bottom';
|
||||||
|
}} else {{
|
||||||
|
dropdown.style.top = `${{triggerRect.bottom + 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'right top';
|
||||||
|
}}
|
||||||
|
dropdown.style.left = `${{triggerRect.right - dropdownRect.width}}px`;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'start-outer':
|
||||||
|
if (shouldPositionAbove) {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top - dropdownRect.height - 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'right bottom';
|
||||||
|
}} else {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'right top';
|
||||||
|
}}
|
||||||
|
dropdown.style.left = `${{triggerRect.left - dropdownRect.width - 16}}px`;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'end-outer':
|
||||||
|
if (shouldPositionAbove) {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top - dropdownRect.height - 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'left bottom';
|
||||||
|
}} else {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'left top';
|
||||||
|
}}
|
||||||
|
dropdown.style.left = `${{triggerRect.right + 8}}px`;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'center':
|
||||||
|
if (shouldPositionAbove) {{
|
||||||
|
dropdown.style.top = `${{triggerRect.top - dropdownRect.height - 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'center bottom';
|
||||||
|
}} else {{
|
||||||
|
dropdown.style.top = `${{triggerRect.bottom + 6}}px`;
|
||||||
|
dropdown.style.transformOrigin = 'center top';
|
||||||
|
}}
|
||||||
|
dropdown.style.left = `${{triggerRect.left}}px`;
|
||||||
|
dropdown.style.minWidth = `${{triggerRect.width}}px`;
|
||||||
|
break;
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
const openDropdown = () => {{
|
||||||
|
isOpen = true;
|
||||||
|
|
||||||
|
// Set state to open first to remove scale transform for accurate measurements
|
||||||
|
dropdown.setAttribute('data-state', 'open');
|
||||||
|
|
||||||
|
// Make dropdown invisible but rendered to measure true height
|
||||||
|
dropdown.style.visibility = 'hidden';
|
||||||
|
dropdown.style.pointerEvents = 'auto';
|
||||||
|
|
||||||
|
// Force reflow to ensure height is calculated
|
||||||
|
dropdown.offsetHeight;
|
||||||
|
|
||||||
|
// Calculate position with accurate height
|
||||||
|
updatePosition();
|
||||||
|
|
||||||
|
// Now make it visible
|
||||||
|
dropdown.style.visibility = 'visible';
|
||||||
|
|
||||||
|
// Lock all scrollable elements
|
||||||
|
window.ScrollLock.lock();
|
||||||
|
|
||||||
|
// Close on click outside
|
||||||
|
setTimeout(() => {{
|
||||||
|
document.addEventListener('click', handleClickOutside);
|
||||||
|
}}, 0);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeDropdown = () => {{
|
||||||
|
isOpen = false;
|
||||||
|
dropdown.setAttribute('data-state', 'closed');
|
||||||
|
dropdown.style.pointerEvents = 'none';
|
||||||
|
document.removeEventListener('click', handleClickOutside);
|
||||||
|
|
||||||
|
// Unlock scroll after animation (200ms delay)
|
||||||
|
window.ScrollLock.unlock(200);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const handleClickOutside = (e) => {{
|
||||||
|
if (!dropdown.contains(e.target) && !trigger.contains(e.target)) {{
|
||||||
|
closeDropdown();
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
// Toggle dropdown when trigger is clicked
|
||||||
|
trigger.addEventListener('click', (e) => {{
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
// Check if any other dropdown is open
|
||||||
|
const allDropdowns = document.querySelectorAll('[data-target=\"target__dropdown\"]');
|
||||||
|
let otherDropdownOpen = false;
|
||||||
|
allDropdowns.forEach(dd => {{
|
||||||
|
if (dd !== dropdown && dd.getAttribute('data-state') === 'open') {{
|
||||||
|
otherDropdownOpen = true;
|
||||||
|
dd.setAttribute('data-state', 'closed');
|
||||||
|
dd.style.pointerEvents = 'none';
|
||||||
|
// Unlock scroll
|
||||||
|
if (window.ScrollLock) {{
|
||||||
|
window.ScrollLock.unlock(200);
|
||||||
|
}}
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
// If another dropdown was open, just close it and don't open this one
|
||||||
|
if (otherDropdownOpen) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
// Normal toggle behavior
|
||||||
|
if (isOpen) {{
|
||||||
|
closeDropdown();
|
||||||
|
}} else {{
|
||||||
|
openDropdown();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
|
||||||
|
// Close when action is clicked
|
||||||
|
const actions = dropdown.querySelectorAll('[data-dropdown-close]');
|
||||||
|
actions.forEach(action => {{
|
||||||
|
action.addEventListener('click', () => {{
|
||||||
|
closeDropdown();
|
||||||
|
}});
|
||||||
|
}});
|
||||||
|
|
||||||
|
// Handle ESC key to close
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && isOpen) {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeDropdown();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
|
||||||
|
if (document.readyState === 'loading') {{
|
||||||
|
document.addEventListener('DOMContentLoaded', setupDropdown);
|
||||||
|
}} else {{
|
||||||
|
setupDropdown();
|
||||||
|
}}
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
target_id_for_script,
|
||||||
|
target_id_for_script,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuSub(children: Children) -> impl IntoView {
|
||||||
|
// TODO. Find a better way for dropdown__menu_sub_trigger.
|
||||||
|
clx! {DropdownMenuSubRoot, li, "dropdown__menu_sub_trigger", " relative inline-flex relative gap-2 items-center py-1.5 px-2 w-full text-sm no-underline rounded-sm transition-colors duration-200 cursor-pointer text-popover-foreground [&_svg:not([class*='size-'])]:size-4 hover:bg-accent hover:text-accent-foreground"}
|
||||||
|
|
||||||
|
view! { <DropdownMenuSubRoot>{children()}</DropdownMenuSubRoot> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuSubTrigger(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("flex items-center justify-between w-full", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<span attr:data-name="DropdownMenuSubTrigger" class=class>
|
||||||
|
<span class="flex gap-2 items-center">{children()}</span>
|
||||||
|
<ChevronRight class="opacity-70 size-4" />
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn DropdownMenuSubItem(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!(
|
||||||
|
"inline-flex gap-2 items-center w-full rounded-sm px-3 py-2 text-sm transition-all duration-150 ease text-popover-foreground hover:bg-accent hover:text-accent-foreground cursor-pointer hover:translate-x-[2px]",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<li data-name="DropdownMenuSubItem" class=class data-dropdown-close="true">
|
||||||
|
{children()}
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
}
|
||||||
35
frontend/src/components/ui/empty.rs
Normal file
35
frontend/src/components/ui/empty.rs
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::{clx, variants};
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {Empty, div, "flex flex-col items-center justify-center gap-4 rounded-lg border border-dashed p-8 text-center"}
|
||||||
|
clx! {EmptyHeader, div, "flex flex-col items-center gap-2"}
|
||||||
|
clx! {EmptyTitle, h3, "text-lg font-semibold leading-none"}
|
||||||
|
clx! {EmptyDescription, p, "text-muted-foreground text-sm"}
|
||||||
|
clx! {EmptyContent, div, "flex items-center justify-center gap-2"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
variants! {
|
||||||
|
EmptyMedia {
|
||||||
|
base: "flex shrink-0 items-center justify-center mb-2 [&_svg]:pointer-events-none [&_svg]:shrink-0",
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
Default: "bg-transparent",
|
||||||
|
Icon: "bg-muted text-foreground flex size-10 shrink-0 items-center justify-center rounded-lg [&_svg:not([class*='size-'])]:size-6",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
Default: "",
|
||||||
|
}
|
||||||
|
},
|
||||||
|
component: {
|
||||||
|
element: div
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
99
frontend/src/components/ui/input.rs
Normal file
99
frontend/src/components/ui/input.rs
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
use leptos::html;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use strum::AsRefStr;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[derive(Default, Clone, Copy, PartialEq, Eq, AsRefStr)]
|
||||||
|
#[strum(serialize_all = "lowercase")]
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub enum InputType {
|
||||||
|
#[default]
|
||||||
|
Text,
|
||||||
|
Email,
|
||||||
|
Password,
|
||||||
|
Number,
|
||||||
|
Tel,
|
||||||
|
Url,
|
||||||
|
Search,
|
||||||
|
Date,
|
||||||
|
Time,
|
||||||
|
#[strum(serialize = "datetime-local")]
|
||||||
|
DatetimeLocal,
|
||||||
|
Month,
|
||||||
|
Week,
|
||||||
|
Color,
|
||||||
|
File,
|
||||||
|
Hidden,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Input(
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
#[prop(default = InputType::default())] r#type: InputType,
|
||||||
|
#[prop(into, optional)] placeholder: Option<String>,
|
||||||
|
#[prop(into, optional)] name: Option<String>,
|
||||||
|
#[prop(into, optional)] id: Option<String>,
|
||||||
|
#[prop(into, optional)] title: Option<String>,
|
||||||
|
#[prop(optional)] disabled: bool,
|
||||||
|
#[prop(optional)] readonly: bool,
|
||||||
|
#[prop(optional)] required: bool,
|
||||||
|
#[prop(optional)] autofocus: bool,
|
||||||
|
#[prop(into, optional)] min: Option<String>,
|
||||||
|
#[prop(into, optional)] max: Option<String>,
|
||||||
|
#[prop(into, optional)] step: Option<String>,
|
||||||
|
#[prop(into, optional)] bind_value: Option<RwSignal<String>>,
|
||||||
|
#[prop(optional)] node_ref: NodeRef<html::Input>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 border-input flex h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
|
||||||
|
"focus-visible:border-ring focus-visible:ring-ring/50",
|
||||||
|
"focus-visible:ring-2",
|
||||||
|
"aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
||||||
|
"read-only:bg-muted",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let type_str = r#type.as_ref();
|
||||||
|
|
||||||
|
match bind_value {
|
||||||
|
Some(signal) => view! {
|
||||||
|
<input
|
||||||
|
data-name="Input"
|
||||||
|
type=type_str
|
||||||
|
class=merged_class
|
||||||
|
placeholder=placeholder
|
||||||
|
name=name
|
||||||
|
id=id
|
||||||
|
title=title
|
||||||
|
disabled=disabled
|
||||||
|
readonly=readonly
|
||||||
|
required=required
|
||||||
|
autofocus=autofocus
|
||||||
|
min=min
|
||||||
|
max=max
|
||||||
|
step=step
|
||||||
|
bind:value=signal
|
||||||
|
node_ref=node_ref
|
||||||
|
/>
|
||||||
|
}.into_any(),
|
||||||
|
None => view! {
|
||||||
|
<input
|
||||||
|
data-name="Input"
|
||||||
|
type=type_str
|
||||||
|
class=merged_class
|
||||||
|
placeholder=placeholder
|
||||||
|
name=name
|
||||||
|
id=id
|
||||||
|
title=title
|
||||||
|
disabled=disabled
|
||||||
|
readonly=readonly
|
||||||
|
required=required
|
||||||
|
autofocus=autofocus
|
||||||
|
min=min
|
||||||
|
max=max
|
||||||
|
step=step
|
||||||
|
node_ref=node_ref
|
||||||
|
/>
|
||||||
|
}.into_any(),
|
||||||
|
}
|
||||||
|
}
|
||||||
27
frontend/src/components/ui/mod.rs
Normal file
27
frontend/src/components/ui/mod.rs
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
pub mod accordion;
|
||||||
|
pub mod alert_dialog;
|
||||||
|
pub mod badge;
|
||||||
|
pub mod button;
|
||||||
|
pub mod button_action;
|
||||||
|
pub mod card;
|
||||||
|
pub mod checkbox;
|
||||||
|
pub mod context_menu;
|
||||||
|
pub mod data_table;
|
||||||
|
pub mod dialog;
|
||||||
|
pub mod dropdown_menu;
|
||||||
|
pub mod empty;
|
||||||
|
pub mod input;
|
||||||
|
pub mod multi_select;
|
||||||
|
pub mod select;
|
||||||
|
pub mod separator;
|
||||||
|
pub mod scroll_area;
|
||||||
|
pub mod sheet;
|
||||||
|
pub mod sidenav;
|
||||||
|
pub mod skeleton;
|
||||||
|
pub mod shimmer;
|
||||||
|
pub mod svg_icon;
|
||||||
|
pub mod switch;
|
||||||
|
pub mod table;
|
||||||
|
pub mod tabs;
|
||||||
|
pub mod theme_toggle;
|
||||||
|
pub mod toast;
|
||||||
294
frontend/src/components/ui/multi_select.rs
Normal file
294
frontend/src/components/ui/multi_select.rs
Normal file
@@ -0,0 +1,294 @@
|
|||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use icons::{Check, ChevronDown, ChevronUp};
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_can_scroll_vertical::use_can_scroll_vertical;
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
// * Reuse @select.rs
|
||||||
|
pub use crate::components::ui::select::{
|
||||||
|
SelectGroup as MultiSelectGroup, SelectItem as MultiSelectItem, SelectLabel as MultiSelectLabel,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default)]
|
||||||
|
pub enum MultiSelectAlign {
|
||||||
|
Start,
|
||||||
|
#[default]
|
||||||
|
Center,
|
||||||
|
End,
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn MultiSelectValue(#[prop(optional, into)] placeholder: String) -> impl IntoView {
|
||||||
|
let multi_select_ctx = expect_context::<MultiSelectContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<span data-name="MultiSelectValue" class="text-sm text-muted-foreground truncate">
|
||||||
|
{move || {
|
||||||
|
let values = multi_select_ctx.values_signal.get();
|
||||||
|
if values.is_empty() {
|
||||||
|
placeholder.clone()
|
||||||
|
} else {
|
||||||
|
let count = values.len();
|
||||||
|
if count == 1 { "1 selected".to_string() } else { format!("{} selected", count) }
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn MultiSelectOption(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] value: Option<String>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let multi_select_ctx = expect_context::<MultiSelectContext>();
|
||||||
|
|
||||||
|
let value_clone = value.clone();
|
||||||
|
let is_selected = Signal::derive(move || {
|
||||||
|
if let Some(ref val) = value_clone {
|
||||||
|
multi_select_ctx.values_signal.with(|values| values.contains(val))
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let class = tw_merge!(
|
||||||
|
"group inline-flex gap-2 items-center w-full text-sm text-left transition-colors duration-200 focus:outline-none focus-visible:outline-none text-popover-foreground [&_svg:not([class*='size-'])]:size-4 hover:bg-accent hover:text-accent-foreground disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-name="MultiSelectOption"
|
||||||
|
class=class
|
||||||
|
role="option"
|
||||||
|
aria-selected=move || is_selected.get().to_string()
|
||||||
|
on:click=move |ev: web_sys::MouseEvent| {
|
||||||
|
ev.prevent_default();
|
||||||
|
ev.stop_propagation();
|
||||||
|
if let Some(val) = value.clone() {
|
||||||
|
multi_select_ctx
|
||||||
|
.values_signal
|
||||||
|
.update(|values| {
|
||||||
|
if values.contains(&val) {
|
||||||
|
values.remove(&val);
|
||||||
|
} else {
|
||||||
|
values.insert(val);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
<Check class="ml-auto opacity-0 size-4 text-muted-foreground group-aria-selected:opacity-100" />
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct MultiSelectContext {
|
||||||
|
target_id: String,
|
||||||
|
values_signal: RwSignal<HashSet<String>>,
|
||||||
|
align: MultiSelectAlign,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn MultiSelect(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] values: Option<RwSignal<HashSet<String>>>,
|
||||||
|
#[prop(default = MultiSelectAlign::default())] align: MultiSelectAlign,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let multi_select_target_id = use_random_id_for("multi_select");
|
||||||
|
let values_signal = values.unwrap_or_else(|| RwSignal::new(HashSet::<String>::new()));
|
||||||
|
|
||||||
|
let multi_select_ctx = MultiSelectContext { target_id: multi_select_target_id.clone(), values_signal, align };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=multi_select_ctx>
|
||||||
|
<div data-name="MultiSelect" class="relative w-fit">
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn MultiSelectTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] id: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let multi_select_ctx = expect_context::<MultiSelectContext>();
|
||||||
|
|
||||||
|
let peer_class = if !id.is_empty() { format!("peer/{}", id) } else { String::new() };
|
||||||
|
|
||||||
|
let button_class = tw_merge!(
|
||||||
|
"w-full p-2 h-9 inline-flex items-center justify-between text-sm font-medium whitespace-nowrap rounded-md transition-colors focus:outline-none focus:ring-1 focus:ring-ring focus-visible:outline-hidden focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 [&_svg:not(:last-child)]:mr-2 [&_svg:not(:first-child)]:ml-2 [&_svg:not([class*='size-'])]:size-4 border bg-background border-input hover:bg-accent hover:text-accent-foreground",
|
||||||
|
&peer_class,
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let button_id = if !id.is_empty() { id } else { format!("trigger_{}", multi_select_ctx.target_id) };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-name="MultiSelectTrigger"
|
||||||
|
class=button_class
|
||||||
|
id=button_id
|
||||||
|
tabindex="0"
|
||||||
|
data-multi-select-trigger=multi_select_ctx.target_id
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
<ChevronDown class="text-muted-foreground" />
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn MultiSelectContent(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let multi_select_ctx = expect_context::<MultiSelectContext>();
|
||||||
|
|
||||||
|
let align_str = match multi_select_ctx.align {
|
||||||
|
MultiSelectAlign::Start => "start",
|
||||||
|
MultiSelectAlign::Center => "center",
|
||||||
|
MultiSelectAlign::End => "end",
|
||||||
|
};
|
||||||
|
|
||||||
|
let class = tw_merge!(
|
||||||
|
"w-[150px] overflow-auto z-50 p-1 rounded-md border bg-card shadow-md h-fit max-h-[300px] absolute top-[calc(100%+4px)] transition-all duration-200 data-[state=closed]:opacity-0 data-[state=closed]:scale-95 data-[state=open]:opacity-100 data-[state=open]:scale-100 data-[align=start]:left-0 data-[align=center]:left-1/2 data-[align=center]:-translate-x-1/2 data-[align=end]:right-0 [scrollbar-width:none] [&::-webkit-scrollbar]:hidden",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let target_id_for_script = multi_select_ctx.target_id.clone();
|
||||||
|
let target_id_for_script_2 = multi_select_ctx.target_id.clone();
|
||||||
|
|
||||||
|
// Scroll indicator signals
|
||||||
|
let (on_scroll, can_scroll_up_signal, can_scroll_down_signal) = use_can_scroll_vertical();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="MultiSelectContent"
|
||||||
|
class=class
|
||||||
|
id=multi_select_ctx.target_id
|
||||||
|
data-target="target__multi_select"
|
||||||
|
data-state="closed"
|
||||||
|
data-align=align_str
|
||||||
|
style="pointer-events: none;"
|
||||||
|
on:scroll=move |ev| on_scroll.run(ev)
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
data-scroll-up="true"
|
||||||
|
class=move || {
|
||||||
|
let is_up: bool = can_scroll_up_signal.get();
|
||||||
|
if is_up {
|
||||||
|
"sticky -top-1 z-10 flex items-center justify-center py-1 bg-card"
|
||||||
|
} else {
|
||||||
|
"hidden"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ChevronUp class="size-4 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
{children()}
|
||||||
|
<div
|
||||||
|
data-scroll-down="true"
|
||||||
|
class=move || {
|
||||||
|
let is_down: bool = can_scroll_down_signal.get();
|
||||||
|
if is_down {
|
||||||
|
"sticky -bottom-1 z-10 flex items-center justify-center py-1 bg-card"
|
||||||
|
} else {
|
||||||
|
"hidden"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ChevronDown class="size-4 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupMultiSelect = () => {{
|
||||||
|
const multiSelect = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-multi-select-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!multiSelect || !trigger) {{
|
||||||
|
setTimeout(setupMultiSelect, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (multiSelect.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
multiSelect.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
let isOpen = false;
|
||||||
|
|
||||||
|
const openMultiSelect = () => {{
|
||||||
|
isOpen = true;
|
||||||
|
if (window.ScrollLock) window.ScrollLock.lock();
|
||||||
|
multiSelect.setAttribute('data-state', 'open');
|
||||||
|
multiSelect.style.pointerEvents = 'auto';
|
||||||
|
const triggerRect = trigger.getBoundingClientRect();
|
||||||
|
multiSelect.style.minWidth = `${{triggerRect.width}}px`;
|
||||||
|
multiSelect.dispatchEvent(new Event('scroll'));
|
||||||
|
setTimeout(() => {{
|
||||||
|
document.addEventListener('click', handleClickOutside);
|
||||||
|
}}, 0);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeMultiSelect = () => {{
|
||||||
|
isOpen = false;
|
||||||
|
multiSelect.setAttribute('data-state', 'closed');
|
||||||
|
multiSelect.style.pointerEvents = 'none';
|
||||||
|
document.removeEventListener('click', handleClickOutside);
|
||||||
|
if (window.ScrollLock) window.ScrollLock.unlock(200);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const handleClickOutside = (e) => {{
|
||||||
|
if (!multiSelect.contains(e.target) && !trigger.contains(e.target)) {{
|
||||||
|
closeMultiSelect();
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
trigger.addEventListener('click', (e) => {{
|
||||||
|
e.stopPropagation();
|
||||||
|
if (isOpen) closeMultiSelect(); else openMultiSelect();
|
||||||
|
}});
|
||||||
|
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && isOpen) {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeMultiSelect();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
|
||||||
|
if (document.readyState === 'loading') {{
|
||||||
|
document.addEventListener('DOMContentLoaded', setupMultiSelect);
|
||||||
|
}} else {{
|
||||||
|
setupMultiSelect();
|
||||||
|
}}
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
target_id_for_script,
|
||||||
|
target_id_for_script_2,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}.into_any()
|
||||||
|
}
|
||||||
102
frontend/src/components/ui/scroll_area.rs
Normal file
102
frontend/src/components/ui/scroll_area.rs
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::void;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
// Removed unused fake components
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ COMPONENTS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ScrollArea(children: Children, #[prop(into, optional)] class: String) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!("relative overflow-hidden", class);
|
||||||
|
view! {
|
||||||
|
<div data-name="ScrollArea" class=merged_class>
|
||||||
|
<ScrollAreaViewport class="pr-3 pb-3 [&::-webkit-scrollbar]:w-1.5 [&::-webkit-scrollbar]:h-1.5 [&::-webkit-scrollbar-track]:bg-transparent [&::-webkit-scrollbar-thumb]:bg-border/60 [&::-webkit-scrollbar-thumb]:rounded-full hover:[&::-webkit-scrollbar-thumb]:bg-border/80">{children()}</ScrollAreaViewport>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ScrollAreaViewport(children: Children, #[prop(into, optional)] class: String) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"focus-visible:ring-ring/50 size-full rounded-[inherit] transition-[color,box-shadow] outline-none focus-visible:ring-[3px] focus-visible:outline-1 overflow-auto",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
view! {
|
||||||
|
<div data-name="ScrollAreaViewport" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* 🧬 ENUMS 🧬 */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
// Real scrollbars are now utilized in the viewport directly.
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* 🧬 STRUCT 🧬 */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SnapScrollArea(
|
||||||
|
#[prop(into, default = SnapAreaVariant::default())] variant: SnapAreaVariant,
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
children: Children,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let snap_item = SnapAreaClass { variant };
|
||||||
|
let merged_class = snap_item.with_class(class);
|
||||||
|
view! {
|
||||||
|
<div data-name="SnapScrollArea" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(TwClass, Default)]
|
||||||
|
#[tw(class = "")]
|
||||||
|
pub struct SnapAreaClass {
|
||||||
|
variant: SnapAreaVariant,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(TwVariant)]
|
||||||
|
pub enum SnapAreaVariant {
|
||||||
|
// * snap-x by default
|
||||||
|
#[tw(default, class = "overflow-x-auto snap-x")]
|
||||||
|
Center,
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* 🧬 STRUCT 🧬 */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SnapItem(
|
||||||
|
#[prop(into, default = SnapVariant::default())] variant: SnapVariant,
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
children: Children,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let snap_item = SnapItemClass { variant };
|
||||||
|
let merged_class = snap_item.with_class(class);
|
||||||
|
view! {
|
||||||
|
<div data-name="SnapItem" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(TwClass, Default)]
|
||||||
|
#[tw(class = "shrink-0")]
|
||||||
|
pub struct SnapItemClass {
|
||||||
|
variant: SnapVariant,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(TwVariant)]
|
||||||
|
pub enum SnapVariant {
|
||||||
|
// * snap-center by default
|
||||||
|
#[tw(default, class = "snap-center")]
|
||||||
|
Center,
|
||||||
|
}
|
||||||
311
frontend/src/components/ui/select.rs
Normal file
311
frontend/src/components/ui/select.rs
Normal file
@@ -0,0 +1,311 @@
|
|||||||
|
use icons::{Check, ChevronDown, ChevronUp};
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
use strum::{AsRefStr, Display};
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_can_scroll_vertical::use_can_scroll_vertical;
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Display, AsRefStr)]
|
||||||
|
pub enum SelectPosition {
|
||||||
|
#[default]
|
||||||
|
Below,
|
||||||
|
Above,
|
||||||
|
}
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {SelectLabel, span, "px-2 py-1.5 text-sm font-medium data-inset:pl-8", "mb-1"}
|
||||||
|
clx! {SelectItem, li, "inline-flex gap-2 items-center w-full rounded-sm px-2 py-1.5 text-sm no-underline transition-colors duration-200 text-popover-foreground hover:bg-accent hover:text-accent-foreground [&_svg:not([class*='size-'])]:size-4"}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SelectGroup(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = "Select options".into(), into)] aria_label: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!("group", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<ul data-name="SelectGroup" role="listbox" aria-label=aria_label class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</ul>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SelectValue(#[prop(optional, into)] placeholder: String) -> impl IntoView {
|
||||||
|
let select_ctx = expect_context::<SelectContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<span data-name="SelectValue" class="text-sm text-muted-foreground truncate">
|
||||||
|
{move || { select_ctx.value_signal.get().unwrap_or_else(|| placeholder.clone()) }}
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SelectOption(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = false.into(), into)] aria_selected: Signal<bool>,
|
||||||
|
#[prop(optional, into)] value: Option<String>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SelectContext>();
|
||||||
|
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"group inline-flex gap-2 items-center w-full rounded-sm px-2 py-1.5 text-sm cursor-pointer no-underline transition-colors duration-200 text-popover-foreground hover:bg-accent hover:text-accent-foreground [&_svg:not([class*='size-'])]:size-4",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let value_for_check = value.clone();
|
||||||
|
let is_selected = move || aria_selected.get() || ctx.value_signal.get() == value_for_check;
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<li
|
||||||
|
data-name="SelectOption"
|
||||||
|
class=merged_class
|
||||||
|
role="option"
|
||||||
|
tabindex="0"
|
||||||
|
aria-selected=move || is_selected().to_string()
|
||||||
|
data-select-option="true"
|
||||||
|
on:click=move |_| {
|
||||||
|
let val = value.clone();
|
||||||
|
ctx.value_signal.set(val.clone());
|
||||||
|
if let Some(on_change) = ctx.on_change {
|
||||||
|
on_change.run(val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
<Check class="ml-auto opacity-0 size-4 text-muted-foreground group-aria-selected:opacity-100" />
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct SelectContext {
|
||||||
|
target_id: String,
|
||||||
|
value_signal: RwSignal<Option<String>>,
|
||||||
|
on_change: Option<Callback<Option<String>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Select(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] default_value: Option<String>,
|
||||||
|
#[prop(optional)] on_change: Option<Callback<Option<String>>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let select_target_id = use_random_id_for("select");
|
||||||
|
let value_signal = RwSignal::new(default_value);
|
||||||
|
|
||||||
|
let ctx = SelectContext { target_id: select_target_id.clone(), value_signal, on_change };
|
||||||
|
|
||||||
|
let merged_class = tw_merge!("relative w-fit", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<div data-name="Select" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SelectTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(optional, into)] id: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SelectContext>();
|
||||||
|
|
||||||
|
let peer_class = if !id.is_empty() { format!("peer/{}", id) } else { String::new() };
|
||||||
|
|
||||||
|
let button_class = tw_merge!(
|
||||||
|
"w-full p-2 h-9 inline-flex items-center justify-between text-sm font-medium whitespace-nowrap rounded-md transition-colors focus:outline-none focus:ring-1 focus:ring-ring focus-visible:outline-hidden focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 [&_svg:not(:last-child)]:mr-2 [&_svg:not(:first-child)]:ml-2 [&_svg:not([class*='size-'])]:size-4 border bg-background border-input hover:bg-accent hover:text-accent-foreground",
|
||||||
|
&peer_class,
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let button_id = if !id.is_empty() { id } else { format!("trigger_{}", ctx.target_id) };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-name="SelectTrigger"
|
||||||
|
class=button_class
|
||||||
|
id=button_id
|
||||||
|
tabindex="0"
|
||||||
|
data-select-trigger=ctx.target_id
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
<ChevronDown class="text-muted-foreground" />
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SelectContent(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = SelectPosition::default())] position: SelectPosition,
|
||||||
|
#[prop(optional)] on_close: Option<Callback<()>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SelectContext>();
|
||||||
|
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"w-[150px] overflow-auto z-50 p-1 rounded-md border bg-card shadow-md h-fit max-h-[300px] absolute top-[calc(100%+4px)] left-0 data-[position=Above]:top-auto data-[position=Above]:bottom-[calc(100%+4px)] transition-all duration-200 data-[state=closed]:opacity-0 data-[state=closed]:scale-95 data-[state=open]:opacity-100 data-[state=open]:scale-100 data-[state=closed]:data-[position=Below]:origin-top data-[state=open]:data-[position=Below]:origin-top data-[state=closed]:data-[position=Above]:origin-bottom data-[state=open]:data-[position=Above]:origin-bottom [scrollbar-width:none] [&::-webkit-scrollbar]:hidden",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let target_id_for_script = ctx.target_id.clone();
|
||||||
|
let target_id_for_script_2 = ctx.target_id.clone();
|
||||||
|
|
||||||
|
// Scroll indicator signals
|
||||||
|
let (on_scroll, can_scroll_up_signal, can_scroll_down_signal) = use_can_scroll_vertical();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="SelectContent"
|
||||||
|
class=merged_class
|
||||||
|
on:selectclose=move |_: web_sys::CustomEvent| {
|
||||||
|
if let Some(cb) = on_close {
|
||||||
|
cb.run(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
id=ctx.target_id
|
||||||
|
data-target="target__select"
|
||||||
|
data-state="closed"
|
||||||
|
data-position=position.to_string()
|
||||||
|
style="pointer-events: none;"
|
||||||
|
on:scroll=move |ev| on_scroll.run(ev)
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
data-scroll-up="true"
|
||||||
|
class=move || {
|
||||||
|
let is_up: bool = can_scroll_up_signal.get();
|
||||||
|
if is_up {
|
||||||
|
"sticky -top-1 z-10 flex items-center justify-center py-1 bg-card"
|
||||||
|
} else {
|
||||||
|
"hidden"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ChevronUp class="size-4 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
{children()}
|
||||||
|
<div
|
||||||
|
data-scroll-down="true"
|
||||||
|
class=move || {
|
||||||
|
let is_down: bool = can_scroll_down_signal.get();
|
||||||
|
if is_down {
|
||||||
|
"sticky -bottom-1 z-10 flex items-center justify-center py-1 bg-card"
|
||||||
|
} else {
|
||||||
|
"hidden"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ChevronDown class="size-4 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupSelect = () => {{
|
||||||
|
const select = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-select-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!select || !trigger) {{
|
||||||
|
setTimeout(setupSelect, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (select.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
select.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
let isOpen = false;
|
||||||
|
|
||||||
|
const updatePosition = () => {{
|
||||||
|
const triggerRect = trigger.getBoundingClientRect();
|
||||||
|
const viewportHeight = window.innerHeight;
|
||||||
|
const spaceBelow = viewportHeight - triggerRect.bottom;
|
||||||
|
const spaceAbove = triggerRect.top;
|
||||||
|
|
||||||
|
if (spaceBelow < 200 && spaceAbove > spaceBelow) {{
|
||||||
|
select.setAttribute('data-position', 'Above');
|
||||||
|
}} else {{
|
||||||
|
select.setAttribute('data-position', 'Below');
|
||||||
|
}}
|
||||||
|
|
||||||
|
select.style.minWidth = `${{triggerRect.width}}px`;
|
||||||
|
}};
|
||||||
|
|
||||||
|
const openSelect = () => {{
|
||||||
|
isOpen = true;
|
||||||
|
if (window.ScrollLock) window.ScrollLock.lock();
|
||||||
|
updatePosition();
|
||||||
|
select.setAttribute('data-state', 'open');
|
||||||
|
select.style.pointerEvents = 'auto';
|
||||||
|
select.dispatchEvent(new Event('scroll'));
|
||||||
|
setTimeout(() => {{
|
||||||
|
document.addEventListener('click', handleClickOutside);
|
||||||
|
}}, 0);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeSelect = () => {{
|
||||||
|
isOpen = false;
|
||||||
|
select.setAttribute('data-state', 'closed');
|
||||||
|
select.style.pointerEvents = 'none';
|
||||||
|
document.removeEventListener('click', handleClickOutside);
|
||||||
|
select.dispatchEvent(new CustomEvent('selectclose', {{ bubbles: false }}));
|
||||||
|
if (window.ScrollLock) window.ScrollLock.unlock(200);
|
||||||
|
}};
|
||||||
|
|
||||||
|
const handleClickOutside = (e) => {{
|
||||||
|
if (!select.contains(e.target) && !trigger.contains(e.target)) {{
|
||||||
|
closeSelect();
|
||||||
|
}}
|
||||||
|
}};
|
||||||
|
|
||||||
|
trigger.addEventListener('click', (e) => {{
|
||||||
|
e.stopPropagation();
|
||||||
|
if (isOpen) closeSelect(); else openSelect();
|
||||||
|
}});
|
||||||
|
|
||||||
|
const options = select.querySelectorAll('[data-select-option]');
|
||||||
|
options.forEach(option => {{
|
||||||
|
option.addEventListener('click', () => closeSelect());
|
||||||
|
}});
|
||||||
|
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && isOpen) {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeSelect();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
|
||||||
|
if (document.readyState === 'loading') {{
|
||||||
|
document.addEventListener('DOMContentLoaded', setupSelect);
|
||||||
|
}} else {{
|
||||||
|
setupSelect();
|
||||||
|
}}
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
target_id_for_script,
|
||||||
|
target_id_for_script_2,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}.into_any()
|
||||||
|
}
|
||||||
35
frontend/src/components/ui/separator.rs
Normal file
35
frontend/src/components/ui/separator.rs
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Separator(
|
||||||
|
#[prop(into, optional)] orientation: Signal<SeparatorOrientation>,
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
// children: Children,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let merged_class = Memo::new(move |_| {
|
||||||
|
let orientation = orientation.get();
|
||||||
|
let separator = SeparatorClass { orientation };
|
||||||
|
separator.with_class(class.clone())
|
||||||
|
});
|
||||||
|
|
||||||
|
view! { <div class=merged_class role="separator" /> }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* 🧬 STRUCT 🧬 */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(TwClass, Default)]
|
||||||
|
#[tw(class = "shrink-0 bg-border")]
|
||||||
|
pub struct SeparatorClass {
|
||||||
|
orientation: SeparatorOrientation,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(TwVariant)]
|
||||||
|
pub enum SeparatorOrientation {
|
||||||
|
#[tw(default, class = "w-full h-[1px]")]
|
||||||
|
Default,
|
||||||
|
#[tw(class = "h-full w-[1px]")]
|
||||||
|
Vertical,
|
||||||
|
}
|
||||||
237
frontend/src/components/ui/sheet.rs
Normal file
237
frontend/src/components/ui/sheet.rs
Normal file
@@ -0,0 +1,237 @@
|
|||||||
|
use icons::X;
|
||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_ui::clx;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use super::button::ButtonSize;
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
use crate::components::ui::button::{Button, ButtonVariant};
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {SheetTitle, h2, "font-bold text-2xl"}
|
||||||
|
clx! {SheetDescription, p, "text-muted-foreground"}
|
||||||
|
clx! {SheetBody, div, "flex flex-col gap-4"}
|
||||||
|
clx! {SheetFooter, footer, "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end"}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ CONTEXT ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct SheetContext {
|
||||||
|
pub target_id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
pub type SheetVariant = ButtonVariant;
|
||||||
|
pub type SheetSize = ButtonSize;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Sheet(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let sheet_target_id = use_random_id_for("sheet");
|
||||||
|
let ctx = SheetContext { target_id: sheet_target_id };
|
||||||
|
|
||||||
|
let merged_class = tw_merge!("", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<div data-name="Sheet" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SheetTrigger(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SheetContext>();
|
||||||
|
let trigger_id = format!("trigger_{}", ctx.target_id);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Button class=class attr:id=trigger_id attr:data-sheet-trigger=ctx.target_id variant=variant size=size>
|
||||||
|
{children()}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SheetClose(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = ButtonVariant::Outline)] variant: ButtonVariant,
|
||||||
|
#[prop(default = ButtonSize::Default)] size: ButtonSize,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SheetContext>();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Button class=class attr:data-sheet-close=ctx.target_id attr:aria-label="Close sheet" variant=variant size=size>
|
||||||
|
{children()}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SheetContent(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
#[prop(default = SheetDirection::Right)] direction: SheetDirection,
|
||||||
|
#[prop(into, optional)] hide_close_button: Option<bool>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<SheetContext>();
|
||||||
|
|
||||||
|
let backdrop_id = format!("{}_backdrop", ctx.target_id);
|
||||||
|
let target_id_for_script = ctx.target_id.clone();
|
||||||
|
let backdrop_id_for_script = backdrop_id.clone();
|
||||||
|
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"fixed z-100 bg-card shadow-lg p-6 transition-transform duration-300 overflow-y-auto overscroll-y-contain",
|
||||||
|
direction.initial_position(),
|
||||||
|
direction.closed_class(),
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="SheetBackdrop"
|
||||||
|
id=backdrop_id
|
||||||
|
class="fixed inset-0 transition-opacity duration-200 pointer-events-none z-60 bg-black/50 data-[state=closed]:opacity-0 data-[state=open]:opacity-100"
|
||||||
|
data-state="closed"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div
|
||||||
|
data-name="SheetContent"
|
||||||
|
class=merged_class
|
||||||
|
id=ctx.target_id
|
||||||
|
data-direction=direction.to_string()
|
||||||
|
data-state="closed"
|
||||||
|
style="pointer-events: none;"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class=format!(
|
||||||
|
"absolute top-4 right-4 p-1 rounded-sm focus:ring-2 focus:ring-offset-2 focus:outline-none [&_svg:not([class*='size-'])]:size-4 focus:ring-ring{}",
|
||||||
|
if hide_close_button.unwrap_or(false) { " hidden" } else { "" },
|
||||||
|
)
|
||||||
|
data-sheet-close=ctx.target_id.clone()
|
||||||
|
aria-label="Close sheet"
|
||||||
|
>
|
||||||
|
<span class="hidden">"Close Sheet"</span>
|
||||||
|
<X />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{format!(
|
||||||
|
r#"
|
||||||
|
(function() {{
|
||||||
|
const setupSheet = () => {{
|
||||||
|
const sheet = document.querySelector('#{}');
|
||||||
|
const backdrop = document.querySelector('#{}');
|
||||||
|
const trigger = document.querySelector('[data-sheet-trigger="{}"]');
|
||||||
|
|
||||||
|
if (!sheet || !backdrop || !trigger) {{
|
||||||
|
setTimeout(setupSheet, 50);
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
|
||||||
|
if (sheet.hasAttribute('data-initialized')) {{
|
||||||
|
return;
|
||||||
|
}}
|
||||||
|
sheet.setAttribute('data-initialized', 'true');
|
||||||
|
|
||||||
|
const openSheet = () => {{
|
||||||
|
if (window.ScrollLock) window.ScrollLock.lock();
|
||||||
|
sheet.setAttribute('data-state', 'open');
|
||||||
|
backdrop.setAttribute('data-state', 'open');
|
||||||
|
sheet.style.pointerEvents = 'auto';
|
||||||
|
backdrop.style.pointerEvents = 'auto';
|
||||||
|
const direction = sheet.getAttribute('data-direction');
|
||||||
|
sheet.classList.remove('translate-x-full', '-translate-x-full', 'translate-y-full', '-translate-y-full');
|
||||||
|
sheet.classList.add('translate-x-0', 'translate-y-0');
|
||||||
|
}};
|
||||||
|
|
||||||
|
const closeSheet = () => {{
|
||||||
|
sheet.setAttribute('data-state', 'closed');
|
||||||
|
backdrop.setAttribute('data-state', 'closed');
|
||||||
|
sheet.style.pointerEvents = 'none';
|
||||||
|
backdrop.style.pointerEvents = 'none';
|
||||||
|
const direction = sheet.getAttribute('data-direction');
|
||||||
|
sheet.classList.remove('translate-x-0', 'translate-y-0');
|
||||||
|
if (direction === 'Right') sheet.classList.add('translate-x-full');
|
||||||
|
else if (direction === 'Left') sheet.classList.add('-translate-x-full');
|
||||||
|
else if (direction === 'Top') sheet.classList.add('-translate-y-full');
|
||||||
|
else if (direction === 'Bottom') sheet.classList.add('translate-y-full');
|
||||||
|
if (window.ScrollLock) window.ScrollLock.unlock(300);
|
||||||
|
}};
|
||||||
|
|
||||||
|
trigger.addEventListener('click', openSheet);
|
||||||
|
const closeButtons = sheet.querySelectorAll('[data-sheet-close]');
|
||||||
|
closeButtons.forEach(btn => btn.addEventListener('click', closeSheet));
|
||||||
|
backdrop.addEventListener('click', closeSheet);
|
||||||
|
document.addEventListener('keydown', (e) => {{
|
||||||
|
if (e.key === 'Escape' && sheet.getAttribute('data-state') === 'open') {{
|
||||||
|
e.preventDefault();
|
||||||
|
closeSheet();
|
||||||
|
}}
|
||||||
|
}});
|
||||||
|
}};
|
||||||
|
|
||||||
|
if (document.readyState === 'loading') {{
|
||||||
|
document.addEventListener('DOMContentLoaded', setupSheet);
|
||||||
|
}} else {{
|
||||||
|
setupSheet();
|
||||||
|
}}
|
||||||
|
}})();
|
||||||
|
"#,
|
||||||
|
target_id_for_script,
|
||||||
|
backdrop_id_for_script,
|
||||||
|
target_id_for_script,
|
||||||
|
)}
|
||||||
|
</script>
|
||||||
|
}.into_any()
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ ENUM ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, strum::AsRefStr, strum::Display)]
|
||||||
|
pub enum SheetDirection {
|
||||||
|
Right,
|
||||||
|
Left,
|
||||||
|
Top,
|
||||||
|
Bottom,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SheetDirection {
|
||||||
|
fn closed_class(self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
SheetDirection::Right => "translate-x-full",
|
||||||
|
SheetDirection::Left => "-translate-x-full",
|
||||||
|
SheetDirection::Top => "-translate-y-full",
|
||||||
|
SheetDirection::Bottom => "translate-y-full",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn initial_position(self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
SheetDirection::Right => "top-0 right-0 h-full w-[400px]",
|
||||||
|
SheetDirection::Left => "top-0 left-0 h-full w-[400px]",
|
||||||
|
SheetDirection::Top => "top-0 left-0 w-full h-[400px]",
|
||||||
|
SheetDirection::Bottom => "bottom-0 left-0 w-full h-[400px]",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
52
frontend/src/components/ui/shimmer.rs
Normal file
52
frontend/src/components/ui/shimmer.rs
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_random::use_random_id_for;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Shimmer(
|
||||||
|
/// Controls shimmer visibility (works with any bool signal)
|
||||||
|
#[prop(into)]
|
||||||
|
loading: Signal<bool>,
|
||||||
|
|
||||||
|
/// Color of the shimmer wave (default: "rgba(255,255,255,0.15)")
|
||||||
|
#[prop(into, optional)]
|
||||||
|
shimmer_color: Option<String>,
|
||||||
|
|
||||||
|
/// Background color of shimmer blocks (default: "rgba(255,255,255,0.08)")
|
||||||
|
#[prop(into, optional)]
|
||||||
|
background_color: Option<String>,
|
||||||
|
|
||||||
|
/// Animation duration in seconds (default: 1.5)
|
||||||
|
#[prop(optional)]
|
||||||
|
duration: Option<f64>,
|
||||||
|
|
||||||
|
/// Fallback border-radius for text elements in px (default: 4)
|
||||||
|
#[prop(optional)]
|
||||||
|
fallback_border_radius: Option<f64>,
|
||||||
|
|
||||||
|
/// Additional classes
|
||||||
|
#[prop(into, optional)]
|
||||||
|
class: String,
|
||||||
|
|
||||||
|
/// Children to wrap
|
||||||
|
children: Children,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let shimmer_id = use_random_id_for("Shimmer");
|
||||||
|
let merged_class = tw_merge!("relative", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
id=shimmer_id
|
||||||
|
class=merged_class
|
||||||
|
data-name="Shimmer"
|
||||||
|
data-shimmer-loading=move || loading.get().to_string()
|
||||||
|
data-shimmer-color=shimmer_color
|
||||||
|
data-shimmer-bg-color=background_color
|
||||||
|
data-shimmer-duration=duration.map(|d| d.to_string())
|
||||||
|
data-shimmer-fallback-radius=fallback_border_radius.map(|r| r.to_string())
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
232
frontend/src/components/ui/sidenav.rs
Normal file
232
frontend/src/components/ui/sidenav.rs
Normal file
@@ -0,0 +1,232 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use leptos_router::hooks::use_location;
|
||||||
|
use leptos_ui::{clx, variants, void};
|
||||||
|
|
||||||
|
mod components {
|
||||||
|
use super::*;
|
||||||
|
clx! {SidenavWrapper, div, "group/sidenav-wrapper has-data-[variant=Inset]:bg-sidenav flex h-full w-full"}
|
||||||
|
// clx! {SidenavInset, div, "bg-background relative flex w-full flex-1 flex-col md:peer-data-[variant=Inset]:m-2 md:peer-data-[variant=Inset]:ml-0 md:peer-data-[variant=Inset]:rounded-xl md:peer-data-[variant=Inset]:shadow-sm md:peer-data-[variant=Inset]:peer-data-[state=Collapsed]:ml-2"}
|
||||||
|
clx! {SidenavInset, div, "bg-background relative flex w-full flex-1 flex-col data-[variant=Inset]:rounded-lg data-[variant=Inset]:border data-[variant=Inset]:border-sidenav-border data-[variant=Inset]:shadow-sm data-[variant=Inset]:m-2"}
|
||||||
|
// * data-[], not group-data-[]
|
||||||
|
clx! {SidenavInner, div, "flex flex-col w-full h-full bg-sidenav data-[variant=Floating]:rounded-lg data-[variant=Floating]:border data-[variant=Floating]:border-sidenav-border data-[variant=Floating]:shadow-sm"}
|
||||||
|
clx! {SidenavHeader, div, "flex flex-col gap-2 p-2"}
|
||||||
|
clx! {SidenavMenu, ul, "flex flex-col gap-1 w-full min-w-0"}
|
||||||
|
clx! {SidenavMenuSub, ul, "border-sidenav-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-l px-2.5 py-0.5 group-data-[collapsible=Icon]:hidden"}
|
||||||
|
clx! {SidenavMenuItem, li, "relative group/menu-item"}
|
||||||
|
clx! {SidenavContent, div, "scrollbar__on_hover", "flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=Icon]:overflow-hidden"}
|
||||||
|
clx! {SidenavGroup, div, "flex relative flex-col p-2 w-full min-w-0"}
|
||||||
|
clx! {SidenavGroupContent, div, "w-full text-sm"}
|
||||||
|
clx! {SidenavGroupLabel, div, "text-sidenav-foreground/70 ring-sidenav-ring flex h-8 shrink-0 items-center rounded-md px-2 text-xs font-medium outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0 group-data-[collapsible=Icon]:-mt-8 group-data-[collapsible=Icon]:opacity-0"}
|
||||||
|
clx! {SidenavFooter, footer, "flex flex-col gap-2 p-2"}
|
||||||
|
// Button "More"
|
||||||
|
clx! {DropdownMenuTriggerEllipsis, button, "text-sidenav-foreground ring-sidenav-ring hover:bg-sidenav-accent hover:text-sidenav-accent-foreground peer-hover/menu-button:text-sidenav-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0 after:absolute after:-inset-2 md:after:hidden peer-data-[size=sm]/menu-button:top-1 peer-data-[size=default]/menu-button:top-1.5 peer-data-[size=lg]/menu-button:top-2.5 group-data-[collapsible=Icon]:hidden peer-data-[active=true]/menu-button:text-sidenav-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-[state=open]:opacity-100 md:opacity-0"}
|
||||||
|
|
||||||
|
void! {SidenavInput, input,
|
||||||
|
"file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 border-input flex h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
|
||||||
|
"focus-visible:border-ring focus-visible:ring-ring/50",
|
||||||
|
"focus-visible:ring-2", // TODO. Port tw_merge to Tailwind V4.
|
||||||
|
// "focus-visible:ring-[3px]", // TODO. Port tw_merge to Tailwind V4.
|
||||||
|
"aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
||||||
|
"read-only:bg-muted",
|
||||||
|
// Specific to Sidenav
|
||||||
|
"w-full h-8 shadow-none bg-background"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub use components::*;
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SidenavLink(
|
||||||
|
children: Children,
|
||||||
|
#[prop(into)] href: String,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left outline-hidden ring-sidenav-ring transition-[width,height,padding] focus-visible:ring-2 active:bg-sidenav-accent active:text-sidenav-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-data-[sidenav=menu-action]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 aria-[current=page]:bg-sidenav-accent aria-[current=page]:font-semibold aria-[current=page]:text-sidenav-accent-foreground data-[state=open]:hover:bg-sidenav-accent data-[state=open]:hover:text-sidenav-accent-foreground group-data-[collapsible=Icon]:size-8! group-data-[collapsible=Icon]:p-2! [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0 hover:bg-sidenav-accent hover:text-sidenav-accent-foreground h-8 text-sm",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
|
||||||
|
let location = use_location();
|
||||||
|
|
||||||
|
// Check if the link is active based on current path
|
||||||
|
let href_clone = href.clone();
|
||||||
|
let is_active = move || {
|
||||||
|
let path = location.pathname.get();
|
||||||
|
path == href_clone || path.starts_with(&format!("{}/", href_clone))
|
||||||
|
};
|
||||||
|
|
||||||
|
let aria_current = move || if is_active() { "page" } else { "false" };
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<a data-name="SidenavLink" class=merged_class href=href aria-current=aria_current>
|
||||||
|
{children()}
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
variants! {
|
||||||
|
SidenavMenuButton {
|
||||||
|
base: "peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left text-sm outline-hidden ring-sidenav-ring transition-[width,height,padding] hover:bg-sidenav-accent hover:text-sidenav-accent-foreground focus-visible:ring-2 active:bg-sidenav-accent active:text-sidenav-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-data-[sidenav=menu-action]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 aria-[current=page]:bg-sidenav-accent aria-[current=page]:font-medium aria-[current=page]:text-sidenav-accent-foreground data-[state=open]:hover:bg-sidenav-accent data-[state=open]:hover:text-sidenav-accent-foreground [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0 group-data-[collapsible=Icon]:size-8! group-data-[collapsible=Icon]:p-0! [&>svg]:stroke-2 aria-[current=page]:[&>svg]:stroke-[2.7]",
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
Default: "hover:bg-sidenav-accent hover:text-sidenav-accent-foreground", // Already in base
|
||||||
|
Outline: "bg-background shadow-[0_0_0_1px_hsl(var(--sidenav-border))] hover:bg-sidenav-accent hover:text-sidenav-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidenav-accent))]",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
Default: "h-8 text-sm",
|
||||||
|
Sm: "h-7 text-xs",
|
||||||
|
Lg: "h-12",
|
||||||
|
}
|
||||||
|
},
|
||||||
|
component: {
|
||||||
|
element: button,
|
||||||
|
support_href: true,
|
||||||
|
support_aria_current: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display, strum::IntoStaticStr)]
|
||||||
|
pub enum SidenavVariant {
|
||||||
|
#[default]
|
||||||
|
Sidenav,
|
||||||
|
Floating,
|
||||||
|
Inset,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display)]
|
||||||
|
pub enum SidenavSide {
|
||||||
|
#[default]
|
||||||
|
Left,
|
||||||
|
Right,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default, Clone, Copy, PartialEq, Eq, strum::Display)]
|
||||||
|
pub enum SidenavCollapsible {
|
||||||
|
#[default]
|
||||||
|
Offcanvas,
|
||||||
|
None,
|
||||||
|
Icon,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Sidenav(
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
#[prop(default = SidenavVariant::default())] variant: SidenavVariant,
|
||||||
|
#[prop(default = SidenavState::default())] data_state: SidenavState,
|
||||||
|
#[prop(default = SidenavSide::default())] data_side: SidenavSide,
|
||||||
|
#[prop(default = SidenavCollapsible::default())] data_collapsible: SidenavCollapsible,
|
||||||
|
children: Children,
|
||||||
|
) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
{if data_collapsible == SidenavCollapsible::None {
|
||||||
|
view! {
|
||||||
|
<aside
|
||||||
|
data-name="Sidenav"
|
||||||
|
class=tw_merge!(
|
||||||
|
"flex flex-col h-full bg-sidenav text-sidenav-foreground w-(--sidenav-width)", class.clone()
|
||||||
|
)
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</aside>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
} else {
|
||||||
|
view! {
|
||||||
|
<aside
|
||||||
|
data-name="Sidenav"
|
||||||
|
data-sidenav=data_state.to_string()
|
||||||
|
data-side=data_side.to_string()
|
||||||
|
class="hidden md:block group peer text-sidenav-foreground data-[state=Collapsed]:hidden"
|
||||||
|
>
|
||||||
|
// * SidenavGap: This is what handles the sidenav gap on desktop
|
||||||
|
<div
|
||||||
|
data-name="SidenavGap"
|
||||||
|
class=tw_merge!(
|
||||||
|
"relative w-(--sidenav-width) bg-transparent transition-[width] duration-200 ease-linear",
|
||||||
|
"group-data-[collapsible=Offcanvas]:w-0",
|
||||||
|
"group-data-[side=Right]:rotate-180",
|
||||||
|
match variant {
|
||||||
|
SidenavVariant::Sidenav => "group-data-[collapsible=Icon]:w-(--sidenav-width-icon)",
|
||||||
|
SidenavVariant::Floating | SidenavVariant::Inset =>
|
||||||
|
"group-data-[collapsible=Icon]:w-[calc(var(--sidenav-width-icon)+(--spacing(4)))]",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
data-name="SidenavContainer"
|
||||||
|
class=tw_merge!(
|
||||||
|
"fixed inset-y-0 z-10 hidden h-svh w-(--sidenav-width) transition-[left,right,width] duration-200 ease-linear md:flex",
|
||||||
|
class,
|
||||||
|
match data_side {
|
||||||
|
SidenavSide::Left => "left-0 group-data-[collapsible=Offcanvas]:left-[calc(var(--sidenav-width)*-1)]",
|
||||||
|
SidenavSide::Right => "right-0 group-data-[collapsible=Offcanvas]:right-[calc(var(--sidenav-width)*-1)]"
|
||||||
|
},
|
||||||
|
match variant {
|
||||||
|
SidenavVariant::Sidenav => "group-data-[collapsible=Icon]:w-(--sidenav-width-icon) group-data-[side=Left]:border-r group-data-[side=Right]:border-l",
|
||||||
|
SidenavVariant::Floating | SidenavVariant::Inset =>
|
||||||
|
"p-2 group-data-[collapsible=Icon]:w-[calc(var(--sidenav-width-icon)+(--spacing(4))+2px)]",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
>
|
||||||
|
// * Act as a Sidenav for the onclick trigger to work with nested Sidenavs.
|
||||||
|
<SidenavInner attr:data-sidenav="Sidenav" attr:data-variant=variant.to_string()>
|
||||||
|
{children()}
|
||||||
|
<SidenavToggleRail />
|
||||||
|
</SidenavInner>
|
||||||
|
</div>
|
||||||
|
</aside>
|
||||||
|
}
|
||||||
|
.into_any()
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ========================================================== */
|
||||||
|
/* ✨ FUNCTIONS ✨ */
|
||||||
|
/* ========================================================== */
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display)]
|
||||||
|
pub enum SidenavState {
|
||||||
|
#[default]
|
||||||
|
Expanded,
|
||||||
|
Collapsed,
|
||||||
|
}
|
||||||
|
|
||||||
|
const ONCLICK_TRIGGER: &str = "document.querySelector('[data-name=\"Sidenav\"]').setAttribute('data-state', document.querySelector('[data-name=\"Sidenav\"]').getAttribute('data-state') === 'Collapsed' ? 'Expanded' : 'Collapsed')";
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SidenavTrigger(children: Children) -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
// TODO. Use Button.
|
||||||
|
|
||||||
|
<button
|
||||||
|
onclick=ONCLICK_TRIGGER
|
||||||
|
data-name="SidenavTrigger"
|
||||||
|
class="inline-flex gap-2 justify-center items-center -ml-1 text-sm font-medium whitespace-nowrap rounded-md transition-all outline-none disabled:opacity-50 disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 aria-invalid:ring-destructive/20 aria-invalid:border-destructive size-7 dark:aria-invalid:ring-destructive/40 dark:hover:bg-accent/50 hover:bg-accent hover:text-accent-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]"
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
fn SidenavToggleRail() -> impl IntoView {
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
data-name="SidenavToggleRail"
|
||||||
|
aria-label="Toggle Sidenav"
|
||||||
|
tabindex="-1"
|
||||||
|
onclick=ONCLICK_TRIGGER
|
||||||
|
class="hidden absolute inset-y-0 z-20 w-4 transition-all ease-linear -translate-x-1/2 sm:flex group-data-[side=Left]:-right-4 group-data-[side=Right]:left-0 after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] in-data-[side=Left]:cursor-w-resize in-data-[side=Right]:cursor-e-resize [[data-side=Left][data-state=Collapsed]_&]:cursor-e-resize [[data-side=Right][data-state=Collapsed]_&]:cursor-w-resize group-data-[collapsible=Offcanvas]:translate-x-0 group-data-[collapsible=Offcanvas]:after:left-full [[data-side=Left][data-collapsible=Offcanvas]_&]:-right-2 [[data-side=Right][data-collapsible=Offcanvas]_&]:-left-0eft-2 hover:after:bg-sidenav-border hover:group-data-[collapsible=Offcanvas]:bg-sidenav"
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}
|
||||||
13
frontend/src/components/ui/skeleton.rs
Normal file
13
frontend/src/components/ui/skeleton.rs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Skeleton(
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let class = tw_merge!(
|
||||||
|
"animate-pulse rounded-md bg-muted",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
view! { <div class=class /> }
|
||||||
|
}
|
||||||
25
frontend/src/components/ui/svg_icon.rs
Normal file
25
frontend/src/components/ui/svg_icon.rs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SvgIcon(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let class = tw_merge!("size-4", class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
class=class
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</svg>
|
||||||
|
}
|
||||||
|
}
|
||||||
42
frontend/src/components/ui/switch.rs
Normal file
42
frontend/src/components/ui/switch.rs
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tailwind_fuse::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Switch(
|
||||||
|
#[prop(into)] checked: Signal<bool>,
|
||||||
|
#[prop(into, optional)] on_checked_change: Option<Callback<bool>>,
|
||||||
|
#[prop(into, optional)] class: String,
|
||||||
|
#[prop(into, optional)] disabled: Signal<bool>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let checked_val = move || checked.get();
|
||||||
|
let disabled_val = move || disabled.get();
|
||||||
|
|
||||||
|
let track_class = move || tw_merge!(
|
||||||
|
"inline-flex h-5 w-9 shrink-0 cursor-pointer items-center rounded-full border-2 border-transparent transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
if checked_val() { "bg-primary" } else { "bg-input" },
|
||||||
|
class.clone()
|
||||||
|
);
|
||||||
|
|
||||||
|
let thumb_class = move || tw_merge!(
|
||||||
|
"pointer-events-none block h-4 w-4 rounded-full bg-background shadow-lg ring-0 transition-transform",
|
||||||
|
if checked_val() { "translate-x-4" } else { "translate-x-0" }
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
role="switch"
|
||||||
|
aria-checked=move || checked_val().to_string()
|
||||||
|
disabled=disabled_val
|
||||||
|
class=track_class
|
||||||
|
on:click=move |e| {
|
||||||
|
e.prevent_default();
|
||||||
|
if let Some(cb) = on_checked_change {
|
||||||
|
cb.run(!checked_val());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<span class=thumb_class />
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
60
frontend/src/components/ui/table.rs
Normal file
60
frontend/src/components/ui/table.rs
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableWrapper(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("overflow-hidden rounded-md border w-full", class);
|
||||||
|
view! { <div class=class>{children()}</div> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Table(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("w-full text-sm border-collapse", class);
|
||||||
|
view! { <table class=class>{children()}</table> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableCaption(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("mt-4 text-sm text-muted-foreground", class);
|
||||||
|
view! { <caption class=class>{children()}</caption> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableHeader(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("[&_tr]:border-b bg-muted/50", class);
|
||||||
|
view! { <thead class=class>{children()}</thead> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableRow(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("border-b transition-colors data-[state=selected]:bg-muted hover:bg-muted/50", class);
|
||||||
|
view! { <tr class=class>{children()}</tr> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableHead(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!(
|
||||||
|
"h-10 px-4 text-left align-middle font-medium text-muted-foreground whitespace-nowrap",
|
||||||
|
"transition-all duration-100 active:scale-[0.98] cursor-pointer select-none hover:bg-muted/30 hover:text-foreground",
|
||||||
|
class
|
||||||
|
);
|
||||||
|
view! { <th class=class>{children()}</th> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableBody(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("[&_tr:last-child]:border-0", class);
|
||||||
|
view! { <tbody class=class>{children()}</tbody> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableCell(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("p-2 px-4 align-middle", class);
|
||||||
|
view! { <td class=class>{children()}</td> }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TableFooter(children: Children, #[prop(optional, into)] class: String) -> impl IntoView {
|
||||||
|
let class = tw_merge!("border-t bg-muted/50 font-medium [&>tr]:last:border-b-0", class);
|
||||||
|
view! { <tfoot class=class>{children()}</tfoot> }
|
||||||
|
}
|
||||||
108
frontend/src/components/ui/tabs.rs
Normal file
108
frontend/src/components/ui/tabs.rs
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
use leptos::context::Provider;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::tw_merge;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct TabsContext {
|
||||||
|
pub active_tab: RwSignal<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Tabs(
|
||||||
|
#[prop(into)] default_value: String,
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let active_tab = RwSignal::new(default_value);
|
||||||
|
let ctx = TabsContext { active_tab };
|
||||||
|
|
||||||
|
let merged_class = tw_merge!("w-full", &class);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<Provider value=ctx>
|
||||||
|
<div data-name="Tabs" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
</Provider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TabsList(
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let merged_class = tw_merge!(
|
||||||
|
"inline-flex h-9 items-center justify-center rounded-lg bg-muted p-1 text-muted-foreground",
|
||||||
|
&class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div data-name="TabsList" class=merged_class>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TabsTrigger(
|
||||||
|
#[prop(into)] value: String,
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<TabsContext>();
|
||||||
|
let v_clone = value.clone();
|
||||||
|
|
||||||
|
let is_active = Memo::new(move |_| ctx.active_tab.get() == v_clone);
|
||||||
|
|
||||||
|
let merged_class = move || tw_merge!(
|
||||||
|
"inline-flex items-center justify-center whitespace-nowrap rounded-md px-3 py-1 text-sm font-medium ring-offset-background transition-all focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 cursor-pointer select-none",
|
||||||
|
if is_active.get() {
|
||||||
|
"bg-background text-foreground shadow-sm"
|
||||||
|
} else {
|
||||||
|
"hover:bg-background/50 hover:text-foreground"
|
||||||
|
},
|
||||||
|
&class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<button
|
||||||
|
data-name="TabsTrigger"
|
||||||
|
type="button"
|
||||||
|
class=merged_class
|
||||||
|
data-state=move || if is_active.get() { "active" } else { "inactive" }
|
||||||
|
on:click=move |_| ctx.active_tab.set(value.clone())
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn TabsContent(
|
||||||
|
#[prop(into)] value: String,
|
||||||
|
children: Children,
|
||||||
|
#[prop(optional, into)] class: String,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let ctx = expect_context::<TabsContext>();
|
||||||
|
let v_clone = value.clone();
|
||||||
|
|
||||||
|
let is_active = Memo::new(move |_| ctx.active_tab.get() == v_clone);
|
||||||
|
|
||||||
|
let merged_class = move || tw_merge!(
|
||||||
|
"mt-2 ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2",
|
||||||
|
if !is_active.get() { "hidden" } else { "" },
|
||||||
|
&class
|
||||||
|
);
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
data-name="TabsContent"
|
||||||
|
class=merged_class
|
||||||
|
data-state=move || if is_active.get() { "active" } else { "inactive" }
|
||||||
|
tabindex=move || if is_active.get() { "0" } else { "-1" }
|
||||||
|
>
|
||||||
|
{children()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
76
frontend/src/components/ui/theme_toggle.rs
Normal file
76
frontend/src/components/ui/theme_toggle.rs
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
use crate::components::ui::svg_icon::SvgIcon;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
|
||||||
|
use crate::components::hooks::use_theme_mode::use_theme_mode;
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn ThemeToggle() -> impl IntoView {
|
||||||
|
let theme_mode = use_theme_mode();
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<style>
|
||||||
|
{"
|
||||||
|
.theme__toggle_transition {
|
||||||
|
-webkit-tap-highlight-color: transparent;
|
||||||
|
|
||||||
|
svg path {
|
||||||
|
transform-origin: center;
|
||||||
|
transition: all .6s ease;
|
||||||
|
transform: translate3d(0,0,0);
|
||||||
|
backface-visibility: hidden;
|
||||||
|
|
||||||
|
&.sun {
|
||||||
|
transform: scale(.4) rotate(60deg);
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.moon {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.switch {
|
||||||
|
svg path {
|
||||||
|
&.sun {
|
||||||
|
transform: scale(1) rotate(0);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.moon {
|
||||||
|
transform: scale(.4) rotate(-60deg);
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
aria-label="Toggle theme"
|
||||||
|
class=move || {
|
||||||
|
let base_class = "theme__toggle_transition";
|
||||||
|
if theme_mode.get() { format!("{base_class} switch") } else { base_class.to_string() }
|
||||||
|
}
|
||||||
|
on:click=move |_| theme_mode.toggle()
|
||||||
|
>
|
||||||
|
<SvgIcon class="size-4">
|
||||||
|
<path
|
||||||
|
d="M12 1.75V3.25M12 20.75V22.25M1.75 12H3.25M20.75 12H22.25M4.75216 4.75216L5.81282 5.81282M18.1872 18.1872L19.2478 19.2478M4.75216 19.2478L5.81282 18.1872M18.1872 5.81282L19.2478 4.75216M16.25 12C16.25 14.3472 14.3472 16.25 12 16.25C9.65279 16.25 7.75 14.3472 7.75 12C7.75 9.65279 9.65279 7.75 12 7.75C14.3472 7.75 16.25 9.65279 16.25 12Z"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="1.5"
|
||||||
|
stroke-linecap="round"
|
||||||
|
class="sun text-neutral-300"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M2.75 12C2.75 17.1086 6.89137 21.25 12 21.25C16.7154 21.25 20.6068 17.7216 21.1778 13.161C20.1198 13.8498 18.8566 14.25 17.5 14.25C13.7721 14.25 10.75 11.2279 10.75 7.5C10.75 5.66012 11.4861 3.99217 12.6799 2.77461C12.4554 2.7583 12.2287 2.75 12 2.75C6.89137 2.75 2.75 6.89137 2.75 12Z"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="1.5"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
class="moon text-neutral-700"
|
||||||
|
/>
|
||||||
|
</SvgIcon>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
}
|
||||||
251
frontend/src/components/ui/toast.rs
Normal file
251
frontend/src/components/ui/toast.rs
Normal file
@@ -0,0 +1,251 @@
|
|||||||
|
use leptos::prelude::*;
|
||||||
|
use tw_merge::*;
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display, Debug)]
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub enum ToastType {
|
||||||
|
#[default]
|
||||||
|
Default,
|
||||||
|
Success,
|
||||||
|
Error,
|
||||||
|
Warning,
|
||||||
|
Info,
|
||||||
|
Loading,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Default, strum::Display, Debug)]
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub enum SonnerPosition {
|
||||||
|
TopLeft,
|
||||||
|
TopCenter,
|
||||||
|
TopRight,
|
||||||
|
#[default]
|
||||||
|
BottomRight,
|
||||||
|
BottomCenter,
|
||||||
|
BottomLeft,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
|
pub struct ToastData {
|
||||||
|
pub id: u64,
|
||||||
|
pub title: String,
|
||||||
|
pub description: Option<String>,
|
||||||
|
pub variant: ToastType,
|
||||||
|
pub duration: u64, // ms
|
||||||
|
pub is_exiting: RwSignal<bool>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy)]
|
||||||
|
pub struct ToasterStore {
|
||||||
|
pub toasts: RwSignal<Vec<ToastData>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn SonnerTrigger(
|
||||||
|
toast: ToastData,
|
||||||
|
index: usize,
|
||||||
|
total: usize,
|
||||||
|
position: SonnerPosition,
|
||||||
|
is_expanded: Signal<bool>,
|
||||||
|
#[prop(optional)] on_dismiss: Option<Callback<()>>,
|
||||||
|
) -> impl IntoView {
|
||||||
|
let _ = is_expanded; // Silence unused warning while keeping prop name intact for builder
|
||||||
|
let variant_classes = match toast.variant {
|
||||||
|
ToastType::Default => "bg-background text-foreground border-border",
|
||||||
|
ToastType::Success => "bg-background text-foreground border-border [&_.icon]:text-green-500",
|
||||||
|
ToastType::Error => "bg-background text-foreground border-border [&_.icon]:text-destructive",
|
||||||
|
ToastType::Warning => "bg-background text-foreground border-border [&_.icon]:text-yellow-500",
|
||||||
|
ToastType::Info => "bg-background text-foreground border-border [&_.icon]:text-blue-500",
|
||||||
|
ToastType::Loading => "bg-background text-foreground border-border",
|
||||||
|
};
|
||||||
|
|
||||||
|
let bar_color = match toast.variant {
|
||||||
|
ToastType::Success => "bg-green-500",
|
||||||
|
ToastType::Error => "bg-destructive",
|
||||||
|
ToastType::Warning => "bg-yellow-500",
|
||||||
|
ToastType::Info => "bg-blue-500",
|
||||||
|
_ => "bg-primary",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Simplified Style (No manual translateY needed with Flexbox)
|
||||||
|
let style = move || {
|
||||||
|
format!(
|
||||||
|
"z-index: {}; opacity: 1; transition: all 0.3s ease;",
|
||||||
|
total - index
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
let animation_class = move || {
|
||||||
|
let pos = position.to_string();
|
||||||
|
let is_left = pos.contains("Left");
|
||||||
|
let is_exiting = toast.is_exiting.get();
|
||||||
|
|
||||||
|
match (is_left, is_exiting) {
|
||||||
|
(true, false) => "animate-sonner-in-left",
|
||||||
|
(true, true) => "animate-sonner-out-left",
|
||||||
|
(false, false) => "animate-sonner-in-right",
|
||||||
|
(false, true) => "animate-sonner-out-right",
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let icon = match toast.variant {
|
||||||
|
ToastType::Success => Some(view! { <span class="icon font-bold text-lg">"✓"</span> }.into_any()),
|
||||||
|
ToastType::Error => Some(view! { <span class="icon font-bold text-lg">"✕"</span> }.into_any()),
|
||||||
|
ToastType::Warning => Some(view! { <span class="icon font-bold text-lg">"⚠"</span> }.into_any()),
|
||||||
|
ToastType::Info => Some(view! { <span class="icon font-bold text-lg">"ℹ"</span> }.into_any()),
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<div
|
||||||
|
class=move || tw_merge!(
|
||||||
|
"relative transition-all duration-300 ease-in-out cursor-pointer pointer-events-auto",
|
||||||
|
"flex items-center gap-3 w-full max-w-[calc(100vw-2rem)] sm:max-w-[380px] p-4 rounded-lg border shadow-xl bg-card",
|
||||||
|
variant_classes,
|
||||||
|
animation_class()
|
||||||
|
)
|
||||||
|
style=style
|
||||||
|
on:click=move |_| {
|
||||||
|
if let Some(cb) = on_dismiss {
|
||||||
|
cb.run(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{icon}
|
||||||
|
<div class="flex flex-col gap-0.5 overflow-hidden flex-1">
|
||||||
|
<div class="text-sm font-bold truncate leading-tight">{toast.title}</div>
|
||||||
|
{move || toast.description.as_ref().map(|d| view! { <div class="text-xs opacity-80 truncate">{d.clone()}</div> })}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
// Progress Bar
|
||||||
|
<div
|
||||||
|
class=tw_merge!("absolute bottom-0 left-0 h-1 w-full opacity-30", bar_color)
|
||||||
|
style=format!(
|
||||||
|
"animation: sonner-progress {}ms linear forwards; transform-origin: left;",
|
||||||
|
toast.duration
|
||||||
|
)
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
|
}
|
||||||
|
|
||||||
|
thread_local! {
|
||||||
|
static TOASTS: std::cell::RefCell<Option<RwSignal<Vec<ToastData>>>> = std::cell::RefCell::new(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn provide_toaster() {
|
||||||
|
let toasts = RwSignal::new(Vec::<ToastData>::new());
|
||||||
|
TOASTS.with(|t| *t.borrow_mut() = Some(toasts));
|
||||||
|
provide_context(ToasterStore { toasts });
|
||||||
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Toaster(#[prop(default = SonnerPosition::default())] position: SonnerPosition) -> impl IntoView {
|
||||||
|
let store = use_context::<ToasterStore>().expect("Toaster context not found");
|
||||||
|
let toasts = store.toasts;
|
||||||
|
|
||||||
|
let is_bottom = position.to_string().contains("Bottom");
|
||||||
|
|
||||||
|
let container_class = match position {
|
||||||
|
SonnerPosition::TopLeft => "left-6 top-6 items-start",
|
||||||
|
SonnerPosition::TopRight => "right-6 top-6 items-end",
|
||||||
|
SonnerPosition::TopCenter => "left-1/2 -translate-x-1/2 top-6 items-center",
|
||||||
|
SonnerPosition::BottomCenter => "left-1/2 -translate-x-1/2 bottom-6 items-center",
|
||||||
|
SonnerPosition::BottomLeft => "left-6 bottom-6 items-start",
|
||||||
|
SonnerPosition::BottomRight => "right-6 bottom-6 items-end",
|
||||||
|
};
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<style>
|
||||||
|
"@keyframes sonner-progress { from { transform: scaleX(1); } to { transform: scaleX(0); } }
|
||||||
|
@keyframes sonner-in-right { from { transform: translateX(100%); opacity: 0; } to { transform: translateX(0); opacity: 1; } }
|
||||||
|
@keyframes sonner-out-right { from { transform: translateX(0); opacity: 1; } to { transform: translateX(100%); opacity: 0; } }
|
||||||
|
@keyframes sonner-in-left { from { transform: translateX(-100%); opacity: 0; } to { transform: translateX(0); opacity: 1; } }
|
||||||
|
@keyframes sonner-out-left { from { transform: translateX(0); opacity: 1; } to { transform: translateX(-100%); opacity: 0; } }
|
||||||
|
.animate-sonner-in-right { animation: sonner-in-right 0.3s ease-out forwards; }
|
||||||
|
.animate-sonner-out-right { animation: sonner-out-right 0.3s ease-in forwards; }
|
||||||
|
.animate-sonner-in-left { animation: sonner-in-left 0.3s ease-out forwards; }
|
||||||
|
.animate-sonner-out-left { animation: sonner-out-left 0.3s ease-in forwards; }"
|
||||||
|
</style>
|
||||||
|
<div
|
||||||
|
class=tw_merge!(
|
||||||
|
"fixed z-[100] flex gap-3 pointer-events-none w-full sm:w-[400px]",
|
||||||
|
"left-1/2 -translate-x-1/2 sm:left-auto sm:translate-x-0 px-4 sm:px-0", // Mobile centering fix
|
||||||
|
if is_bottom { "flex-col-reverse" } else { "flex-col" },
|
||||||
|
container_class,
|
||||||
|
"pb-[env(safe-area-inset-bottom)] pt-[env(safe-area-inset-top)]"
|
||||||
|
)
|
||||||
|
>
|
||||||
|
<For
|
||||||
|
each=move || {
|
||||||
|
let list = toasts.get();
|
||||||
|
list.into_iter().enumerate().collect::<Vec<_>>()
|
||||||
|
}
|
||||||
|
key=|(_, toast)| toast.id
|
||||||
|
children=move |(index, toast)| {
|
||||||
|
let id = toast.id;
|
||||||
|
let total = toasts.with(|t| t.len());
|
||||||
|
let is_exiting = toast.is_exiting;
|
||||||
|
|
||||||
|
view! {
|
||||||
|
<SonnerTrigger
|
||||||
|
toast=toast
|
||||||
|
index=index
|
||||||
|
total=total
|
||||||
|
position=position
|
||||||
|
is_expanded=Signal::derive(move || true)
|
||||||
|
on_dismiss=Callback::new(move |_| {
|
||||||
|
is_exiting.set(true);
|
||||||
|
leptos::task::spawn_local(async move {
|
||||||
|
gloo_timers::future::TimeoutFuture::new(300).await;
|
||||||
|
toasts.update(|vec| vec.retain(|t| t.id != id));
|
||||||
|
});
|
||||||
|
})
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
}.into_any()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn toast(title: impl Into<String>, variant: ToastType) {
|
||||||
|
let signal_opt = TOASTS.with(|t| *t.borrow());
|
||||||
|
|
||||||
|
if let Some(toasts) = signal_opt {
|
||||||
|
let id = js_sys::Math::random().to_bits();
|
||||||
|
let new_toast = ToastData {
|
||||||
|
id,
|
||||||
|
title: title.into(),
|
||||||
|
description: None,
|
||||||
|
variant,
|
||||||
|
duration: 4000,
|
||||||
|
is_exiting: RwSignal::new(false),
|
||||||
|
};
|
||||||
|
|
||||||
|
toasts.update(|t| {
|
||||||
|
t.push(new_toast.clone());
|
||||||
|
if t.len() > 5 {
|
||||||
|
t.remove(0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let duration = new_toast.duration;
|
||||||
|
let is_exiting = new_toast.is_exiting;
|
||||||
|
leptos::task::spawn_local(async move {
|
||||||
|
gloo_timers::future::TimeoutFuture::new(duration as u32).await;
|
||||||
|
is_exiting.set(true);
|
||||||
|
gloo_timers::future::TimeoutFuture::new(300).await;
|
||||||
|
toasts.update(|vec| vec.retain(|t| t.id != id));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn toast_success(title: impl Into<String>) { toast(title, ToastType::Success); }
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn toast_error(title: impl Into<String>) { toast(title, ToastType::Error); }
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn toast_warning(title: impl Into<String>) { toast(title, ToastType::Warning); }
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn toast_info(title: impl Into<String>) { toast(title, ToastType::Info); }
|
||||||
@@ -2,23 +2,29 @@ use futures::StreamExt;
|
|||||||
use gloo_net::eventsource::futures::EventSource;
|
use gloo_net::eventsource::futures::EventSource;
|
||||||
use leptos::prelude::*;
|
use leptos::prelude::*;
|
||||||
use leptos::task::spawn_local;
|
use leptos::task::spawn_local;
|
||||||
use shared::{AppEvent, GlobalStats, NotificationLevel, SystemNotification, Torrent};
|
use shared::{AppEvent, GlobalStats, NotificationLevel, Torrent};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use struct_patch::traits::Patch;
|
use struct_patch::traits::Patch;
|
||||||
use base64::{Engine as _, engine::general_purpose::STANDARD as BASE64};
|
use base64::{Engine as _, engine::general_purpose::{URL_SAFE_NO_PAD as BASE64_URL, STANDARD as BASE64}};
|
||||||
|
use wasm_bindgen::JsCast;
|
||||||
|
|
||||||
|
use crate::components::ui::toast::{ToastType, toast};
|
||||||
|
|
||||||
pub fn show_toast(level: NotificationLevel, message: impl Into<String>) {
|
pub fn show_toast(level: NotificationLevel, message: impl Into<String>) {
|
||||||
let msg = message.into();
|
let msg = message.into();
|
||||||
match level {
|
gloo_console::log!("TOAST CALL:", &msg, format!("{:?}", level));
|
||||||
NotificationLevel::Info => { leptos_shadcn_toast::toast::info(&msg).show(); },
|
log::info!("Displaying toast: [{:?}] {}", level, msg);
|
||||||
NotificationLevel::Success => { leptos_shadcn_toast::toast::success(&msg).show(); },
|
|
||||||
NotificationLevel::Warning => { leptos_shadcn_toast::toast::warning(&msg).show(); },
|
let variant = match level {
|
||||||
NotificationLevel::Error => { leptos_shadcn_toast::toast::error(&msg).show(); },
|
NotificationLevel::Success => ToastType::Success,
|
||||||
}
|
NotificationLevel::Error => ToastType::Error,
|
||||||
|
NotificationLevel::Warning => ToastType::Warning,
|
||||||
|
NotificationLevel::Info => ToastType::Info,
|
||||||
|
};
|
||||||
|
|
||||||
|
toast(msg, variant);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
pub fn toast_success(message: impl Into<String>) { show_toast(NotificationLevel::Success, message); }
|
pub fn toast_success(message: impl Into<String>) { show_toast(NotificationLevel::Success, message); }
|
||||||
pub fn toast_error(message: impl Into<String>) { show_toast(NotificationLevel::Error, message); }
|
pub fn toast_error(message: impl Into<String>) { show_toast(NotificationLevel::Error, message); }
|
||||||
|
|
||||||
@@ -47,6 +53,7 @@ pub struct TorrentStore {
|
|||||||
pub global_stats: RwSignal<GlobalStats>,
|
pub global_stats: RwSignal<GlobalStats>,
|
||||||
pub user: RwSignal<Option<String>>,
|
pub user: RwSignal<Option<String>>,
|
||||||
pub selected_torrent: RwSignal<Option<String>>,
|
pub selected_torrent: RwSignal<Option<String>>,
|
||||||
|
pub push_enabled: RwSignal<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn provide_torrent_store() {
|
pub fn provide_torrent_store() {
|
||||||
@@ -56,12 +63,20 @@ pub fn provide_torrent_store() {
|
|||||||
let global_stats = RwSignal::new(GlobalStats::default());
|
let global_stats = RwSignal::new(GlobalStats::default());
|
||||||
let user = RwSignal::new(Option::<String>::None);
|
let user = RwSignal::new(Option::<String>::None);
|
||||||
let selected_torrent = RwSignal::new(Option::<String>::None);
|
let selected_torrent = RwSignal::new(Option::<String>::None);
|
||||||
|
let push_enabled = RwSignal::new(false);
|
||||||
|
|
||||||
let show_browser_notification = crate::utils::notification::use_app_notification();
|
let show_browser_notification = crate::utils::notification::use_app_notification();
|
||||||
|
|
||||||
let store = TorrentStore { torrents, filter, search_query, global_stats, user, selected_torrent };
|
let store = TorrentStore { torrents, filter, search_query, global_stats, user, selected_torrent, push_enabled };
|
||||||
provide_context(store);
|
provide_context(store);
|
||||||
|
|
||||||
|
// Initial check for push status
|
||||||
|
spawn_local(async move {
|
||||||
|
if let Ok(enabled) = is_push_subscribed().await {
|
||||||
|
push_enabled.set(enabled);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
let global_stats_for_sse = global_stats;
|
let global_stats_for_sse = global_stats;
|
||||||
let torrents_for_sse = torrents;
|
let torrents_for_sse = torrents;
|
||||||
let show_browser_notification = show_browser_notification.clone();
|
let show_browser_notification = show_browser_notification.clone();
|
||||||
@@ -72,17 +87,12 @@ pub fn provide_torrent_store() {
|
|||||||
let mut disconnect_notified = false;
|
let mut disconnect_notified = false;
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
|
|
||||||
log::debug!("SSE: Creating EventSource...");
|
|
||||||
let es_result = EventSource::new("/api/events");
|
let es_result = EventSource::new("/api/events");
|
||||||
match es_result {
|
match es_result {
|
||||||
Ok(mut es) => {
|
Ok(mut es) => {
|
||||||
log::debug!("SSE: EventSource created, subscribing...");
|
|
||||||
if let Ok(mut stream) = es.subscribe("message") {
|
if let Ok(mut stream) = es.subscribe("message") {
|
||||||
log::debug!("SSE: Subscribed to message channel");
|
|
||||||
let mut got_first_message = false;
|
let mut got_first_message = false;
|
||||||
while let Some(Ok((_, msg))) = stream.next().await {
|
while let Some(Ok((_, msg))) = stream.next().await {
|
||||||
log::debug!("SSE: Received message");
|
|
||||||
if !got_first_message {
|
if !got_first_message {
|
||||||
got_first_message = true;
|
got_first_message = true;
|
||||||
backoff_ms = 1000;
|
backoff_ms = 1000;
|
||||||
@@ -94,47 +104,30 @@ pub fn provide_torrent_store() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let Some(data_str) = msg.data().as_string() {
|
if let Some(data_str) = msg.data().as_string() {
|
||||||
// Decode Base64
|
if let Ok(bytes) = BASE64.decode(&data_str) {
|
||||||
match BASE64.decode(&data_str) {
|
if let Ok(event) = rmp_serde::from_slice::<AppEvent>(&bytes) {
|
||||||
Ok(bytes) => {
|
match event {
|
||||||
// Deserialize MessagePack
|
AppEvent::FullList(list, _) => {
|
||||||
match rmp_serde::from_slice::<AppEvent>(&bytes) {
|
torrents_for_sse.update(|map| {
|
||||||
Ok(event) => {
|
let new_hashes: std::collections::HashSet<String> = list.iter().map(|t| t.hash.clone()).collect();
|
||||||
match event {
|
map.retain(|hash, _| new_hashes.contains(hash));
|
||||||
AppEvent::FullList(list, _) => {
|
for new_torrent in list { map.insert(new_torrent.hash.clone(), new_torrent); }
|
||||||
log::info!("SSE: Received FullList with {} torrents", list.len());
|
});
|
||||||
torrents_for_sse.update(|map| {
|
}
|
||||||
let new_hashes: std::collections::HashSet<String> = list.iter().map(|t| t.hash.clone()).collect();
|
AppEvent::Update(patch) => {
|
||||||
map.retain(|hash, _| new_hashes.contains(hash));
|
if let Some(hash) = patch.hash.clone() {
|
||||||
for new_torrent in list {
|
torrents_for_sse.update(|map| { if let Some(t) = map.get_mut(&hash) { t.apply(patch); } });
|
||||||
map.insert(new_torrent.hash.clone(), new_torrent);
|
}
|
||||||
}
|
}
|
||||||
});
|
AppEvent::Stats(stats) => { global_stats_for_sse.set(stats); }
|
||||||
log::debug!("SSE: torrents map now has {} entries", torrents_for_sse.with(|m| m.len()));
|
AppEvent::Notification(n) => {
|
||||||
}
|
show_toast(n.level.clone(), n.message.clone());
|
||||||
AppEvent::Update(patch) => {
|
if n.message.contains("tamamlandı") || n.level == shared::NotificationLevel::Error {
|
||||||
let hash_opt = patch.hash.clone();
|
show_browser_notification("VibeTorrent", &n.message);
|
||||||
if let Some(hash) = hash_opt {
|
|
||||||
torrents_for_sse.update(|map| {
|
|
||||||
if let Some(t) = map.get_mut(&hash) {
|
|
||||||
t.apply(patch);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
AppEvent::Stats(stats) => { global_stats_for_sse.set(stats); }
|
|
||||||
AppEvent::Notification(n) => {
|
|
||||||
show_toast(n.level.clone(), n.message.clone());
|
|
||||||
if n.message.contains("tamamlandı") || n.level == shared::NotificationLevel::Error {
|
|
||||||
show_browser_notification("VibeTorrent", &n.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(e) => log::error!("SSE: Failed to deserialize MessagePack: {}", e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(e) => log::error!("SSE: Failed to decode Base64: {}", e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -151,13 +144,106 @@ pub fn provide_torrent_store() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
log::debug!("SSE: Reconnecting in {}ms...", backoff_ms);
|
|
||||||
gloo_timers::future::TimeoutFuture::new(backoff_ms).await;
|
gloo_timers::future::TimeoutFuture::new(backoff_ms).await;
|
||||||
backoff_ms = std::cmp::min(backoff_ms * 2, 30000);
|
backoff_ms = std::cmp::min(backoff_ms * 2, 30000);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn subscribe_to_push_notifications() {
|
pub async fn is_push_subscribed() -> Result<bool, String> {
|
||||||
// ...
|
let window = web_sys::window().ok_or("no window")?;
|
||||||
|
let navigator = window.navigator();
|
||||||
|
let sw_container = navigator.service_worker();
|
||||||
|
|
||||||
|
let registration = wasm_bindgen_futures::JsFuture::from(sw_container.ready().map_err(|e| format!("{:?}", e))?)
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("{:?}", e))?
|
||||||
|
.dyn_into::<web_sys::ServiceWorkerRegistration>()
|
||||||
|
.map_err(|_| "not a registration")?;
|
||||||
|
|
||||||
|
let push_manager = registration.push_manager().map_err(|e| format!("{:?}", e))?;
|
||||||
|
let subscription = wasm_bindgen_futures::JsFuture::from(push_manager.get_subscription().map_err(|e| format!("{:?}", e))?)
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("{:?}", e))?;
|
||||||
|
|
||||||
|
Ok(!subscription.is_null())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn subscribe_to_push_notifications() {
|
||||||
|
let window = web_sys::window().expect("no window");
|
||||||
|
let navigator = window.navigator();
|
||||||
|
let sw_container = navigator.service_worker();
|
||||||
|
|
||||||
|
let registration = match wasm_bindgen_futures::JsFuture::from(sw_container.ready().expect("sw not ready")).await {
|
||||||
|
Ok(reg) => reg.dyn_into::<web_sys::ServiceWorkerRegistration>().expect("not a reg"),
|
||||||
|
Err(e) => { log::error!("SW Ready Error: {:?}", e); return; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// 1. Get Public Key from Backend
|
||||||
|
let public_key_res: Result<String, _> = shared::server_fns::push::get_public_key().await;
|
||||||
|
let public_key = match public_key_res {
|
||||||
|
Ok(key) => key,
|
||||||
|
Err(e) => { log::error!("Failed to get public key: {:?}", e); return; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// 2. Convert base64 key to Uint8Array
|
||||||
|
let decoded_key = BASE64_URL.decode(public_key.trim()).expect("invalid public key");
|
||||||
|
let key_array = js_sys::Uint8Array::from(&decoded_key[..]);
|
||||||
|
|
||||||
|
// 3. Prepare Options
|
||||||
|
let options = web_sys::PushSubscriptionOptionsInit::new();
|
||||||
|
options.set_user_visible_only(true);
|
||||||
|
options.set_application_server_key(&key_array.into());
|
||||||
|
|
||||||
|
// 4. Subscribe
|
||||||
|
let push_manager = registration.push_manager().expect("no push manager");
|
||||||
|
match wasm_bindgen_futures::JsFuture::from(push_manager.subscribe_with_options(&options).expect("subscribe failed")).await {
|
||||||
|
Ok(subscription) => {
|
||||||
|
let sub_js = subscription.clone();
|
||||||
|
|
||||||
|
// Use JS to extract JSON string representation
|
||||||
|
let json_str = js_sys::JSON::stringify(&sub_js).expect("stringify failed").as_string().expect("not a string");
|
||||||
|
let sub_obj: serde_json::Value = serde_json::from_str(&json_str).expect("serde from str failed");
|
||||||
|
|
||||||
|
let endpoint = sub_obj["endpoint"].as_str().expect("no endpoint").to_string();
|
||||||
|
let p256dh = sub_obj["keys"]["p256dh"].as_str().expect("no p256dh").to_string();
|
||||||
|
let auth = sub_obj["keys"]["auth"].as_str().expect("no auth").to_string();
|
||||||
|
|
||||||
|
// 5. Save to Backend
|
||||||
|
match shared::server_fns::push::subscribe_push(endpoint, p256dh, auth).await {
|
||||||
|
Ok(_) => {
|
||||||
|
log::info!("Push subscription saved successfully");
|
||||||
|
toast_success("Bildirimler aktif edildi");
|
||||||
|
}
|
||||||
|
Err(e) => log::error!("Failed to save subscription: {:?}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => log::error!("Subscription Error: {:?}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn unsubscribe_from_push_notifications() {
|
||||||
|
let window = web_sys::window().expect("no window");
|
||||||
|
let sw_container = window.navigator().service_worker();
|
||||||
|
|
||||||
|
let registration = wasm_bindgen_futures::JsFuture::from(sw_container.ready().expect("sw not ready")).await
|
||||||
|
.unwrap().dyn_into::<web_sys::ServiceWorkerRegistration>().unwrap();
|
||||||
|
|
||||||
|
let push_manager = registration.push_manager().unwrap();
|
||||||
|
if let Ok(sub_future) = push_manager.get_subscription() {
|
||||||
|
if let Ok(subscription) = wasm_bindgen_futures::JsFuture::from(sub_future).await {
|
||||||
|
if !subscription.is_null() {
|
||||||
|
let sub = subscription.dyn_into::<web_sys::PushSubscription>().unwrap();
|
||||||
|
let endpoint = sub.endpoint();
|
||||||
|
|
||||||
|
// 1. Unsubscribe in Browser
|
||||||
|
let _ = wasm_bindgen_futures::JsFuture::from(sub.unsubscribe().unwrap()).await;
|
||||||
|
|
||||||
|
// 2. Remove from Backend
|
||||||
|
let _ = shared::server_fns::push::unsubscribe_push(endpoint).await;
|
||||||
|
log::info!("Push subscription removed");
|
||||||
|
show_toast(NotificationLevel::Info, "Bildirimler kapatıldı");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
use web_sys::{Notification, NotificationOptions};
|
use web_sys::{Notification, NotificationOptions};
|
||||||
use leptos::prelude::*;
|
|
||||||
|
|
||||||
/// Request browser notification permission from user
|
/// Request browser notification permission from user
|
||||||
pub async fn request_notification_permission() -> bool {
|
pub async fn request_notification_permission() -> bool {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const CACHE_NAME = "vibetorrent-v2";
|
const CACHE_NAME = "vibetorrent-v3";
|
||||||
const ASSETS_TO_CACHE = [
|
const ASSETS_TO_CACHE = [
|
||||||
"/",
|
"/",
|
||||||
"/index.html",
|
"/index.html",
|
||||||
@@ -51,6 +51,11 @@ self.addEventListener("activate", (event) => {
|
|||||||
self.addEventListener("fetch", (event) => {
|
self.addEventListener("fetch", (event) => {
|
||||||
const url = new URL(event.request.url);
|
const url = new URL(event.request.url);
|
||||||
|
|
||||||
|
// Skip unsupported schemes (like chrome-extension://)
|
||||||
|
if (!url.protocol.startsWith("http")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Network-first strategy for API calls
|
// Network-first strategy for API calls
|
||||||
if (url.pathname.startsWith("/api/")) {
|
if (url.pathname.startsWith("/api/")) {
|
||||||
event.respondWith(
|
event.respondWith(
|
||||||
@@ -75,10 +80,12 @@ self.addEventListener("fetch", (event) => {
|
|||||||
fetch(event.request)
|
fetch(event.request)
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
// Cache the latest version of the HTML
|
// Cache the latest version of the HTML
|
||||||
const responseToCache = response.clone();
|
if (response && response.status === 200) {
|
||||||
caches.open(CACHE_NAME).then((cache) => {
|
const responseToCache = response.clone();
|
||||||
cache.put(event.request, responseToCache);
|
caches.open(CACHE_NAME).then((cache) => {
|
||||||
});
|
cache.put(event.request, responseToCache);
|
||||||
|
});
|
||||||
|
}
|
||||||
return response;
|
return response;
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
|
|||||||
@@ -26,4 +26,7 @@ module.exports = {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
plugins: [
|
||||||
|
require("tailwindcss-animate"),
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|||||||
2
frontend/ui_config.toml
Normal file
2
frontend/ui_config.toml
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
base_color = "neutral"
|
||||||
|
base_path_components = "src/components"
|
||||||
372
package-lock.json
generated
Normal file
372
package-lock.json
generated
Normal file
@@ -0,0 +1,372 @@
|
|||||||
|
{
|
||||||
|
"name": "vibetorrent-v3",
|
||||||
|
"lockfileVersion": 3,
|
||||||
|
"requires": true,
|
||||||
|
"packages": {
|
||||||
|
"": {
|
||||||
|
"dependencies": {
|
||||||
|
"@tailwindcss/cli": "^4.1.18",
|
||||||
|
"tailwindcss": "^4.1.18",
|
||||||
|
"tw-animate-css": "^1.4.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@jridgewell/gen-mapping": {
|
||||||
|
"version": "0.3.13",
|
||||||
|
"resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
|
||||||
|
"integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@jridgewell/sourcemap-codec": "^1.5.0",
|
||||||
|
"@jridgewell/trace-mapping": "^0.3.24"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@jridgewell/remapping": {
|
||||||
|
"version": "2.3.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
|
||||||
|
"integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@jridgewell/gen-mapping": "^0.3.5",
|
||||||
|
"@jridgewell/trace-mapping": "^0.3.24"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@jridgewell/resolve-uri": {
|
||||||
|
"version": "3.1.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
|
||||||
|
"integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@jridgewell/sourcemap-codec": {
|
||||||
|
"version": "1.5.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
|
||||||
|
"integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/@jridgewell/trace-mapping": {
|
||||||
|
"version": "0.3.31",
|
||||||
|
"resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
|
||||||
|
"integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@jridgewell/resolve-uri": "^3.1.0",
|
||||||
|
"@jridgewell/sourcemap-codec": "^1.4.14"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@parcel/watcher": {
|
||||||
|
"version": "2.5.6",
|
||||||
|
"resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-2.5.6.tgz",
|
||||||
|
"integrity": "sha512-tmmZ3lQxAe/k/+rNnXQRawJ4NjxO2hqiOLTHvWchtGZULp4RyFeh6aU4XdOYBFe2KE1oShQTv4AblOs2iOrNnQ==",
|
||||||
|
"hasInstallScript": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"detect-libc": "^2.0.3",
|
||||||
|
"is-glob": "^4.0.3",
|
||||||
|
"node-addon-api": "^7.0.0",
|
||||||
|
"picomatch": "^4.0.3"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10.0.0"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/parcel"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"@parcel/watcher-android-arm64": "2.5.6",
|
||||||
|
"@parcel/watcher-darwin-arm64": "2.5.6",
|
||||||
|
"@parcel/watcher-darwin-x64": "2.5.6",
|
||||||
|
"@parcel/watcher-freebsd-x64": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-arm-glibc": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-arm-musl": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-arm64-glibc": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-arm64-musl": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-x64-glibc": "2.5.6",
|
||||||
|
"@parcel/watcher-linux-x64-musl": "2.5.6",
|
||||||
|
"@parcel/watcher-win32-arm64": "2.5.6",
|
||||||
|
"@parcel/watcher-win32-ia32": "2.5.6",
|
||||||
|
"@parcel/watcher-win32-x64": "2.5.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@parcel/watcher-darwin-arm64": {
|
||||||
|
"version": "2.5.6",
|
||||||
|
"resolved": "https://registry.npmjs.org/@parcel/watcher-darwin-arm64/-/watcher-darwin-arm64-2.5.6.tgz",
|
||||||
|
"integrity": "sha512-Z2ZdrnwyXvvvdtRHLmM4knydIdU9adO3D4n/0cVipF3rRiwP+3/sfzpAwA/qKFL6i1ModaabkU7IbpeMBgiVEA==",
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10.0.0"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/parcel"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@tailwindcss/cli": {
|
||||||
|
"version": "4.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tailwindcss/cli/-/cli-4.1.18.tgz",
|
||||||
|
"integrity": "sha512-sMZ+lZbDyxwjD2E0L7oRUjJ01Ffjtme5OtjvvnC+cV4CEDcbqzbp25TCpxHj6kWLU9+DlqJOiNgSOgctC2aZmg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@parcel/watcher": "^2.5.1",
|
||||||
|
"@tailwindcss/node": "4.1.18",
|
||||||
|
"@tailwindcss/oxide": "4.1.18",
|
||||||
|
"enhanced-resolve": "^5.18.3",
|
||||||
|
"mri": "^1.2.0",
|
||||||
|
"picocolors": "^1.1.1",
|
||||||
|
"tailwindcss": "4.1.18"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"tailwindcss": "dist/index.mjs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@tailwindcss/node": {
|
||||||
|
"version": "4.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tailwindcss/node/-/node-4.1.18.tgz",
|
||||||
|
"integrity": "sha512-DoR7U1P7iYhw16qJ49fgXUlry1t4CpXeErJHnQ44JgTSKMaZUdf17cfn5mHchfJ4KRBZRFA/Coo+MUF5+gOaCQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@jridgewell/remapping": "^2.3.4",
|
||||||
|
"enhanced-resolve": "^5.18.3",
|
||||||
|
"jiti": "^2.6.1",
|
||||||
|
"lightningcss": "1.30.2",
|
||||||
|
"magic-string": "^0.30.21",
|
||||||
|
"source-map-js": "^1.2.1",
|
||||||
|
"tailwindcss": "4.1.18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@tailwindcss/oxide": {
|
||||||
|
"version": "4.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tailwindcss/oxide/-/oxide-4.1.18.tgz",
|
||||||
|
"integrity": "sha512-EgCR5tTS5bUSKQgzeMClT6iCY3ToqE1y+ZB0AKldj809QXk1Y+3jB0upOYZrn9aGIzPtUsP7sX4QQ4XtjBB95A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"@tailwindcss/oxide-android-arm64": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-darwin-arm64": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-darwin-x64": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-freebsd-x64": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-linux-arm64-gnu": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-linux-arm64-musl": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-linux-x64-gnu": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-linux-x64-musl": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-wasm32-wasi": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-win32-arm64-msvc": "4.1.18",
|
||||||
|
"@tailwindcss/oxide-win32-x64-msvc": "4.1.18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@tailwindcss/oxide-darwin-arm64": {
|
||||||
|
"version": "4.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.1.18.tgz",
|
||||||
|
"integrity": "sha512-Gc2q4Qhs660bhjyBSKgq6BYvwDz4G+BuyJ5H1xfhmDR3D8HnHCmT/BSkvSL0vQLy/nkMLY20PQ2OoYMO15Jd0A==",
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/detect-libc": {
|
||||||
|
"version": "2.1.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
|
||||||
|
"integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/enhanced-resolve": {
|
||||||
|
"version": "5.19.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.19.0.tgz",
|
||||||
|
"integrity": "sha512-phv3E1Xl4tQOShqSte26C7Fl84EwUdZsyOuSSk9qtAGyyQs2s3jJzComh+Abf4g187lUUAvH+H26omrqia2aGg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"graceful-fs": "^4.2.4",
|
||||||
|
"tapable": "^2.3.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10.13.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/graceful-fs": {
|
||||||
|
"version": "4.2.11",
|
||||||
|
"resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz",
|
||||||
|
"integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/is-extglob": {
|
||||||
|
"version": "2.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
|
||||||
|
"integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/is-glob": {
|
||||||
|
"version": "4.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
|
||||||
|
"integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"is-extglob": "^2.1.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/jiti": {
|
||||||
|
"version": "2.6.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/jiti/-/jiti-2.6.1.tgz",
|
||||||
|
"integrity": "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"bin": {
|
||||||
|
"jiti": "lib/jiti-cli.mjs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/lightningcss": {
|
||||||
|
"version": "1.30.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.30.2.tgz",
|
||||||
|
"integrity": "sha512-utfs7Pr5uJyyvDETitgsaqSyjCb2qNRAtuqUeWIAKztsOYdcACf2KtARYXg2pSvhkt+9NfoaNY7fxjl6nuMjIQ==",
|
||||||
|
"license": "MPL-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"detect-libc": "^2.0.3"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 12.0.0"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/parcel"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"lightningcss-android-arm64": "1.30.2",
|
||||||
|
"lightningcss-darwin-arm64": "1.30.2",
|
||||||
|
"lightningcss-darwin-x64": "1.30.2",
|
||||||
|
"lightningcss-freebsd-x64": "1.30.2",
|
||||||
|
"lightningcss-linux-arm-gnueabihf": "1.30.2",
|
||||||
|
"lightningcss-linux-arm64-gnu": "1.30.2",
|
||||||
|
"lightningcss-linux-arm64-musl": "1.30.2",
|
||||||
|
"lightningcss-linux-x64-gnu": "1.30.2",
|
||||||
|
"lightningcss-linux-x64-musl": "1.30.2",
|
||||||
|
"lightningcss-win32-arm64-msvc": "1.30.2",
|
||||||
|
"lightningcss-win32-x64-msvc": "1.30.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/lightningcss-darwin-arm64": {
|
||||||
|
"version": "1.30.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.30.2.tgz",
|
||||||
|
"integrity": "sha512-ylTcDJBN3Hp21TdhRT5zBOIi73P6/W0qwvlFEk22fkdXchtNTOU4Qc37SkzV+EKYxLouZ6M4LG9NfZ1qkhhBWA==",
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
],
|
||||||
|
"license": "MPL-2.0",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 12.0.0"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/parcel"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/magic-string": {
|
||||||
|
"version": "0.30.21",
|
||||||
|
"resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz",
|
||||||
|
"integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@jridgewell/sourcemap-codec": "^1.5.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/mri": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=4"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/node-addon-api": {
|
||||||
|
"version": "7.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-7.1.1.tgz",
|
||||||
|
"integrity": "sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/picocolors": {
|
||||||
|
"version": "1.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
||||||
|
"integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/picomatch": {
|
||||||
|
"version": "4.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
|
||||||
|
"integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/jonschlinkert"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/source-map-js": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
|
||||||
|
"integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
|
||||||
|
"license": "BSD-3-Clause",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tailwindcss": {
|
||||||
|
"version": "4.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.1.18.tgz",
|
||||||
|
"integrity": "sha512-4+Z+0yiYyEtUVCScyfHCxOYP06L5Ne+JiHhY2IjR2KWMIWhJOYZKLSGZaP5HkZ8+bY0cxfzwDE5uOmzFXyIwxw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/tapable": {
|
||||||
|
"version": "2.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz",
|
||||||
|
"integrity": "sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/webpack"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tw-animate-css": {
|
||||||
|
"version": "1.4.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/tw-animate-css/-/tw-animate-css-1.4.0.tgz",
|
||||||
|
"integrity": "sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/Wombosvideo"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
8
package.json
Normal file
8
package.json
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"type": "module",
|
||||||
|
"dependencies": {
|
||||||
|
"@tailwindcss/cli": "^4.1.18",
|
||||||
|
"tailwindcss": "^4.1.18",
|
||||||
|
"tw-animate-css": "^1.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -42,6 +42,11 @@ pub struct Torrent {
|
|||||||
pub error_message: String,
|
pub error_message: String,
|
||||||
pub added_date: i64,
|
pub added_date: i64,
|
||||||
pub label: Option<String>,
|
pub label: Option<String>,
|
||||||
|
pub ratio: f64,
|
||||||
|
pub uploaded: i64,
|
||||||
|
pub wasted: i64,
|
||||||
|
pub save_path: String,
|
||||||
|
pub free_disk_space: i64,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq, ToSchema)]
|
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq, ToSchema)]
|
||||||
@@ -121,6 +126,13 @@ pub struct TorrentTracker {
|
|||||||
pub url: String,
|
pub url: String,
|
||||||
pub status: String,
|
pub status: String,
|
||||||
pub message: String,
|
pub message: String,
|
||||||
|
pub is_enabled: bool,
|
||||||
|
pub group: i64,
|
||||||
|
pub seeders: i64,
|
||||||
|
pub peers: i64,
|
||||||
|
pub downloaded: i64,
|
||||||
|
pub last_updated: i64,
|
||||||
|
pub interval: i64,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize, Clone, ToSchema)]
|
#[derive(Debug, Serialize, Deserialize, Clone, ToSchema)]
|
||||||
|
|||||||
@@ -83,12 +83,19 @@ impl ScgiRequest {
|
|||||||
|
|
||||||
pub async fn send_request(socket_path: &str, request: ScgiRequest) -> Result<Bytes, ScgiError> {
|
pub async fn send_request(socket_path: &str, request: ScgiRequest) -> Result<Bytes, ScgiError> {
|
||||||
let perform_request = async {
|
let perform_request = async {
|
||||||
let mut stream = UnixStream::connect(socket_path).await?;
|
|
||||||
let data = request.encode();
|
let data = request.encode();
|
||||||
stream.write_all(&data).await?;
|
|
||||||
|
|
||||||
let mut response = Vec::new();
|
let mut response = Vec::new();
|
||||||
stream.read_to_end(&mut response).await?;
|
|
||||||
|
if socket_path.contains(':') {
|
||||||
|
let mut stream = tokio::net::TcpStream::connect(socket_path).await?;
|
||||||
|
stream.write_all(&data).await?;
|
||||||
|
stream.read_to_end(&mut response).await?;
|
||||||
|
} else {
|
||||||
|
let mut stream = tokio::net::UnixStream::connect(socket_path).await?;
|
||||||
|
stream.write_all(&data).await?;
|
||||||
|
stream.read_to_end(&mut response).await?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok::<Vec<u8>, std::io::Error>(response)
|
Ok::<Vec<u8>, std::io::Error>(response)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ pub struct SetupStatus {
|
|||||||
pub completed: bool,
|
pub completed: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(GetSetupStatus, "/api/server_fns/GetSetupStatus", input = MsgPack, output = MsgPack)]
|
#[server(GetSetupStatus, "/api/server_fns", input = MsgPack, output = MsgPack)]
|
||||||
pub async fn get_setup_status() -> Result<SetupStatus, ServerFnError> {
|
pub async fn get_setup_status() -> Result<SetupStatus, ServerFnError> {
|
||||||
use crate::DbContext;
|
use crate::DbContext;
|
||||||
|
|
||||||
@@ -33,7 +33,7 @@ pub async fn get_setup_status() -> Result<SetupStatus, ServerFnError> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(Setup, "/api/server_fns/Setup", input = MsgPack, output = MsgPack)]
|
#[server(Setup, "/api/server_fns", input = MsgPack, output = MsgPack)]
|
||||||
pub async fn setup(username: String, password: String) -> Result<(), ServerFnError> {
|
pub async fn setup(username: String, password: String) -> Result<(), ServerFnError> {
|
||||||
use crate::DbContext;
|
use crate::DbContext;
|
||||||
|
|
||||||
@@ -55,7 +55,7 @@ pub async fn setup(username: String, password: String) -> Result<(), ServerFnErr
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(Login, "/api/server_fns/Login", input = MsgPack, output = MsgPack)]
|
#[server(Login, "/api/server_fns", input = MsgPack, output = MsgPack)]
|
||||||
pub async fn login(username: String, password: String) -> Result<UserResponse, ServerFnError> {
|
pub async fn login(username: String, password: String) -> Result<UserResponse, ServerFnError> {
|
||||||
use crate::DbContext;
|
use crate::DbContext;
|
||||||
use leptos_axum::ResponseOptions;
|
use leptos_axum::ResponseOptions;
|
||||||
@@ -111,7 +111,7 @@ pub async fn login(username: String, password: String) -> Result<UserResponse, S
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(Logout, "/api/server_fns/Logout", input = MsgPack, output = MsgPack)]
|
#[server(Logout, "/api/server_fns", input = MsgPack, output = MsgPack)]
|
||||||
pub async fn logout() -> Result<(), ServerFnError> {
|
pub async fn logout() -> Result<(), ServerFnError> {
|
||||||
use leptos_axum::ResponseOptions;
|
use leptos_axum::ResponseOptions;
|
||||||
use cookie::{Cookie, SameSite};
|
use cookie::{Cookie, SameSite};
|
||||||
@@ -132,7 +132,7 @@ pub async fn logout() -> Result<(), ServerFnError> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(GetUser, "/api/server_fns/GetUser", input = MsgPack, output = MsgPack)]
|
#[server(GetUser, "/api/server_fns", input = MsgPack, output = MsgPack)]
|
||||||
pub async fn get_user() -> Result<Option<UserResponse>, ServerFnError> {
|
pub async fn get_user() -> Result<Option<UserResponse>, ServerFnError> {
|
||||||
use axum::http::HeaderMap;
|
use axum::http::HeaderMap;
|
||||||
use leptos_axum::extract;
|
use leptos_axum::extract;
|
||||||
|
|||||||
@@ -20,3 +20,13 @@ pub async fn subscribe_push(
|
|||||||
.await
|
.await
|
||||||
.map_err(|e| ServerFnError::new(format!("Failed to save subscription: {}", e)))
|
.map_err(|e| ServerFnError::new(format!("Failed to save subscription: {}", e)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[server(UnsubscribePush, "/api/server_fns")]
|
||||||
|
pub async fn unsubscribe_push(endpoint: String) -> Result<(), ServerFnError> {
|
||||||
|
let db_ctx = expect_context::<crate::DbContext>();
|
||||||
|
db_ctx
|
||||||
|
.db
|
||||||
|
.remove_push_subscription(&endpoint)
|
||||||
|
.await
|
||||||
|
.map_err(|e| ServerFnError::new(format!("Failed to remove subscription: {}", e)))
|
||||||
|
}
|
||||||
|
|||||||
@@ -193,8 +193,12 @@ pub async fn get_trackers(hash: String) -> Result<Vec<TorrentTracker>, ServerFnE
|
|||||||
RpcParam::from(hash.as_str()),
|
RpcParam::from(hash.as_str()),
|
||||||
RpcParam::from(""),
|
RpcParam::from(""),
|
||||||
RpcParam::from("t.url="),
|
RpcParam::from("t.url="),
|
||||||
RpcParam::from("t.activity_date_last="),
|
RpcParam::from("t.is_enabled="),
|
||||||
RpcParam::from("t.message="),
|
RpcParam::from("t.group="),
|
||||||
|
RpcParam::from("t.scrape_complete="),
|
||||||
|
RpcParam::from("t.scrape_incomplete="),
|
||||||
|
RpcParam::from("t.scrape_downloaded="),
|
||||||
|
RpcParam::from("t.normal_interval="),
|
||||||
];
|
];
|
||||||
|
|
||||||
let xml = client
|
let xml = client
|
||||||
@@ -205,14 +209,23 @@ pub async fn get_trackers(hash: String) -> Result<Vec<TorrentTracker>, ServerFnE
|
|||||||
let rows = parse_multicall_response(&xml)
|
let rows = parse_multicall_response(&xml)
|
||||||
.map_err(|e| ServerFnError::new(format!("Parse error: {}", e)))?;
|
.map_err(|e| ServerFnError::new(format!("Parse error: {}", e)))?;
|
||||||
|
|
||||||
Ok(rows
|
let result: Vec<TorrentTracker> = rows
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|row| TorrentTracker {
|
.map(|row| TorrentTracker {
|
||||||
url: row.get(0).cloned().unwrap_or_default(),
|
url: row.get(0).cloned().unwrap_or_default(),
|
||||||
status: "Unknown".to_string(),
|
is_enabled: row.get(1).and_then(|s| s.parse::<i64>().ok()).unwrap_or(0) != 0,
|
||||||
message: row.get(2).cloned().unwrap_or_default(),
|
group: row.get(2).and_then(|s| s.parse().ok()).unwrap_or(0),
|
||||||
|
seeders: row.get(3).and_then(|s| s.parse().ok()).unwrap_or(0),
|
||||||
|
peers: row.get(4).and_then(|s| s.parse().ok()).unwrap_or(0),
|
||||||
|
downloaded: row.get(5).and_then(|s| s.parse().ok()).unwrap_or(0),
|
||||||
|
interval: row.get(6).and_then(|s| s.parse().ok()).unwrap_or(0),
|
||||||
|
last_updated: 0,
|
||||||
|
status: "Unknown".to_string(), // Can derive from activity later, or keep unknown
|
||||||
|
message: "".to_string(),
|
||||||
})
|
})
|
||||||
.collect())
|
.collect();
|
||||||
|
|
||||||
|
Ok(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[server(SetFilePriority, "/api/server_fns")]
|
#[server(SetFilePriority, "/api/server_fns")]
|
||||||
@@ -225,6 +238,7 @@ pub async fn set_file_priority(
|
|||||||
let ctx = expect_context::<crate::ServerContext>();
|
let ctx = expect_context::<crate::ServerContext>();
|
||||||
let client = RtorrentClient::new(&ctx.scgi_socket_path);
|
let client = RtorrentClient::new(&ctx.scgi_socket_path);
|
||||||
|
|
||||||
|
// rTorrent f.set_priority takes: target = "HASH:fINDEX", value = priority
|
||||||
let target = format!("{}:f{}", hash, file_index);
|
let target = format!("{}:f{}", hash, file_index);
|
||||||
let params = vec![
|
let params = vec![
|
||||||
RpcParam::from(target.as_str()),
|
RpcParam::from(target.as_str()),
|
||||||
@@ -232,10 +246,11 @@ pub async fn set_file_priority(
|
|||||||
];
|
];
|
||||||
|
|
||||||
client
|
client
|
||||||
.call("f.set_priority", ¶ms)
|
.call("f.priority.set", ¶ms)
|
||||||
.await
|
.await
|
||||||
.map_err(|e| ServerFnError::new(format!("RPC error: {}", e)))?;
|
.map_err(|e| ServerFnError::new(format!("RPC error setting priority: {}", e)))?;
|
||||||
|
|
||||||
|
// Notify rTorrent to update its internal priority state
|
||||||
let _ = client
|
let _ = client
|
||||||
.call("d.update_priorities", &[RpcParam::from(hash.as_str())])
|
.call("d.update_priorities", &[RpcParam::from(hash.as_str())])
|
||||||
.await;
|
.await;
|
||||||
|
|||||||
26
test_rpc.rs
Normal file
26
test_rpc.rs
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
use shared::xmlrpc::{RtorrentClient, RpcParam, parse_multicall_response};
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() {
|
||||||
|
let client = RtorrentClient::new("/tmp/rtorrent.sock");
|
||||||
|
|
||||||
|
// Hardcode a known hash from the UI, e.g. "C3315ABFAD70C54505813D1303C1457900C5B795" (from first image)
|
||||||
|
let hash = "C3315ABFAD70C54505813D1303C1457900C5B795";
|
||||||
|
|
||||||
|
let params = vec![
|
||||||
|
RpcParam::from(hash),
|
||||||
|
RpcParam::from(""),
|
||||||
|
RpcParam::from("t.url="),
|
||||||
|
];
|
||||||
|
|
||||||
|
match client.call("t.multicall", ¶ms).await {
|
||||||
|
Ok(xml) => {
|
||||||
|
println!("Response XML:\n{}", xml);
|
||||||
|
match parse_multicall_response(&xml) {
|
||||||
|
Ok(rows) => println!("Rows ({})", rows.len()),
|
||||||
|
Err(e) => println!("Parse error: {:?}", e),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(e) => println!("Error: {:?}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
2
ui_config.toml
Normal file
2
ui_config.toml
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
base_color = "neutral"
|
||||||
|
base_path_components = "backend/src/components"
|
||||||
BIN
vibetorrent.db-shm
Normal file
BIN
vibetorrent.db-shm
Normal file
Binary file not shown.
BIN
vibetorrent.db-wal
Normal file
BIN
vibetorrent.db-wal
Normal file
Binary file not shown.
Reference in New Issue
Block a user