chore: ags configs removed
|
@ -1,130 +0,0 @@
|
|||
env:
|
||||
es2022: true
|
||||
extends:
|
||||
- "eslint:recommended"
|
||||
- "plugin:@typescript-eslint/recommended"
|
||||
parser: "@typescript-eslint/parser"
|
||||
parserOptions:
|
||||
ecmaVersion: 2022
|
||||
sourceType: "module"
|
||||
project: "./tsconfig.json"
|
||||
warnOnUnsupportedTypeScriptVersion: false
|
||||
root: true
|
||||
ignorePatterns:
|
||||
- types/
|
||||
plugins:
|
||||
- "@typescript-eslint"
|
||||
rules:
|
||||
"@typescript-eslint/ban-ts-comment":
|
||||
- "off"
|
||||
"@typescript-eslint/no-non-null-assertion":
|
||||
- "off"
|
||||
# "@typescript-eslint/no-explicit-any":
|
||||
# - "off"
|
||||
"@typescript-eslint/no-unused-vars":
|
||||
- error
|
||||
- varsIgnorePattern: (^unused|_$)
|
||||
argsIgnorePattern: ^(unused|_)
|
||||
"@typescript-eslint/no-empty-interface":
|
||||
- "off"
|
||||
|
||||
arrow-parens:
|
||||
- error
|
||||
- as-needed
|
||||
comma-dangle:
|
||||
- error
|
||||
- always-multiline
|
||||
comma-spacing:
|
||||
- error
|
||||
- before: false
|
||||
after: true
|
||||
comma-style:
|
||||
- error
|
||||
- last
|
||||
curly:
|
||||
- error
|
||||
- multi-or-nest
|
||||
- consistent
|
||||
dot-location:
|
||||
- error
|
||||
- property
|
||||
eol-last:
|
||||
- error
|
||||
eqeqeq:
|
||||
- error
|
||||
- always
|
||||
indent:
|
||||
- error
|
||||
- 4
|
||||
- SwitchCase: 1
|
||||
keyword-spacing:
|
||||
- error
|
||||
- before: true
|
||||
lines-between-class-members:
|
||||
- error
|
||||
- always
|
||||
- exceptAfterSingleLine: true
|
||||
padded-blocks:
|
||||
- error
|
||||
- never
|
||||
- allowSingleLineBlocks: false
|
||||
prefer-const:
|
||||
- error
|
||||
quotes:
|
||||
- error
|
||||
- double
|
||||
- avoidEscape: true
|
||||
semi:
|
||||
- error
|
||||
- never
|
||||
nonblock-statement-body-position:
|
||||
- error
|
||||
- below
|
||||
no-trailing-spaces:
|
||||
- error
|
||||
no-useless-escape:
|
||||
- off
|
||||
max-len:
|
||||
- error
|
||||
- code: 100
|
||||
func-call-spacing:
|
||||
- error
|
||||
array-bracket-spacing:
|
||||
- error
|
||||
space-before-function-paren:
|
||||
- error
|
||||
- anonymous: never
|
||||
named: never
|
||||
asyncArrow: ignore
|
||||
space-before-blocks:
|
||||
- error
|
||||
key-spacing:
|
||||
- error
|
||||
object-curly-spacing:
|
||||
- error
|
||||
- always
|
||||
globals:
|
||||
Widget: readonly
|
||||
Utils: readonly
|
||||
App: readonly
|
||||
Variable: readonly
|
||||
Service: readonly
|
||||
pkg: readonly
|
||||
ARGV: readonly
|
||||
Debugger: readonly
|
||||
GIRepositoryGType: readonly
|
||||
globalThis: readonly
|
||||
imports: readonly
|
||||
Intl: readonly
|
||||
log: readonly
|
||||
logError: readonly
|
||||
print: readonly
|
||||
printerr: readonly
|
||||
window: readonly
|
||||
TextEncoder: readonly
|
||||
TextDecoder: readonly
|
||||
console: readonly
|
||||
setTimeout: readonly
|
||||
setInterval: readonly
|
||||
clearTimeout: readonly
|
||||
clearInterval: readonly
|
6
hypr-configs/dotfiles/ags/.gitignore
vendored
|
@ -1,6 +0,0 @@
|
|||
node_modules
|
||||
types
|
||||
package-lock.json
|
||||
bun.lockb
|
||||
flake.lock
|
||||
.weather
|
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 8.96875 0 c -0.332031 0.0117188 -0.640625 0.1875 -0.816406 0.46875 l -5 8 c -0.105469 0.171875 -0.152344 0.355469 -0.152344 0.53125 v 1 h 3 v 5 c 0 1.003906 1.316406 1.378906 1.847656 0.53125 l 5 -8 c 0.105469 -0.171875 0.152344 -0.355469 0.152344 -0.53125 v -1 h -3 v -5 c 0 -0.5625 -0.464844 -1.015625 -1.03125 -1 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 476 B |
|
@ -1,5 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 14 3.175781 v 3.824219 c 0 2.179688 -1.820312 4 -4 4 h -3.585938 l -2 2 h 5.585938 l 3 3 v -3 c 1.644531 0 3 -1.355469 3 -3 v -4 c 0 -1.292969 -0.839844 -2.40625 -2 -2.824219 z m 0 0" fill-opacity="0.34902"/>
|
||||
<path d="m 3 0 c -1.644531 0 -3 1.355469 -3 3 v 4 c 0 1.644531 1.355469 3 3 3 v 3 l 3 -3 h 4 c 1.644531 0 3 -1.355469 3 -3 v -4 c 0 -1.644531 -1.355469 -3 -3 -3 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 534 B |
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 3.785156 2.03125 c -0.242187 0 -0.523437 0.066406 -0.804687 0.21875 c -1.039063 0.546875 -1.992188 2.335938 -2.511719 4.65625 c -0.4414062 1.972656 -0.605469 4.664062 -0.339844 5.75 c 0.226563 0.933594 0.625 1.34375 1.332032 1.34375 c 1.042968 -0.019531 2.359374 -1.183594 3.191406 -2.75 c 0.601562 -0.867188 2 -1.261719 3.347656 -1.21875 c 1.347656 -0.046875 2.746094 0.351562 3.347656 1.21875 c 0.832032 1.566406 2.148438 2.730469 3.191406 2.75 c 0.707032 0 1.105469 -0.410156 1.332032 -1.34375 c 0.265625 -1.085938 0.101562 -3.777344 -0.339844 -5.75 c -0.519531 -2.320312 -1.472656 -4.109375 -2.511719 -4.65625 c -0.566406 -0.304688 -1.039062 -0.296875 -1.453125 0 c -0.527344 0.375 -1.628906 0.78125 -3.566406 0.78125 c -1.9375 0.003906 -3.039062 -0.40625 -3.566406 -0.78125 c -0.207032 -0.148438 -0.40625 -0.21875 -0.648438 -0.21875 z m 0.246094 3 h 0.992188 v 1 h 0.992187 v 1 h -0.992187 v 1 h -0.992188 v -1 h -0.992188 v -1 h 0.992188 z m 7.441406 0 c 0.273438 0 0.496094 0.222656 0.496094 0.5 s -0.222656 0.5 -0.496094 0.5 c -0.273437 0 -0.496094 -0.222656 -0.496094 -0.5 s 0.222657 -0.5 0.496094 -0.5 z m -0.992187 1 c 0.273437 0 0.496093 0.222656 0.496093 0.5 s -0.222656 0.5 -0.496093 0.5 c -0.273438 0 -0.496094 -0.222656 -0.496094 -0.5 s 0.222656 -0.5 0.496094 -0.5 z m 1.984375 0 c 0.273437 0 0.496094 0.222656 0.496094 0.5 s -0.222657 0.5 -0.496094 0.5 c -0.273438 0 -0.496094 -0.222656 -0.496094 -0.5 s 0.222656 -0.5 0.496094 -0.5 z m -0.992188 1 c 0.273438 0 0.496094 0.222656 0.496094 0.5 s -0.222656 0.5 -0.496094 0.5 c -0.273437 0 -0.496094 -0.222656 -0.496094 -0.5 s 0.222657 -0.5 0.496094 -0.5 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 1.7 KiB |
|
@ -1,5 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 4.550781 1 c -1.9375 0 -3.5 1.5625 -3.5 3.5 s 1.5625 3.5 3.5 3.5 h 7 c 1.941407 0 3.5 -1.5625 3.5 -3.5 s -1.558593 -3.5 -3.5 -3.5 z m 7 1 c 1.386719 0 2.5 1.113281 2.5 2.5 c 0 1.382812 -1.113281 2.5 -2.5 2.5 c -1.382812 0 -2.5 -1.117188 -2.5 -2.5 c 0 -1.386719 1.117188 -2.5 2.5 -2.5 z m 0 0"/>
|
||||
<path d="m 4.550781 9 c -1.9375 0 -3.5 1.5625 -3.5 3.5 s 1.5625 3.5 3.5 3.5 h 7 c 1.941407 0 3.5 -1.5625 3.5 -3.5 s -1.558593 -3.5 -3.5 -3.5 z m 0 1 c 1.386719 0 2.5 1.113281 2.5 2.5 c 0 1.382812 -1.113281 2.5 -2.5 2.5 c -1.382812 0 -2.5 -1.117188 -2.5 -2.5 c 0 -1.386719 1.117188 -2.5 2.5 -2.5 z m 0 0" fill-opacity="0.34902"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 777 B |
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 0.917969 8.003906 c 0 3.914063 3.164062 7.078125 7.078125 7.078125 c 3.605468 -0.007812 6.617187 -2.703125 7.023437 -6.285156 c 0.042969 -0.378906 -0.136719 -0.75 -0.457031 -0.957031 c -0.324219 -0.203125 -0.738281 -0.207032 -1.0625 -0.003906 c -0.609375 0.375 -1.316406 0.578124 -2.03125 0.578124 c -2.140625 0 -3.882812 -1.742187 -3.882812 -3.882812 c 0 -0.714844 0.203124 -1.421875 0.578124 -2.03125 c 0.203126 -0.324219 0.199219 -0.738281 -0.003906 -1.0625 c -0.207031 -0.320312 -0.578125 -0.5 -0.957031 -0.457031 c -3.582031 0.40625 -6.277344 3.417969 -6.285156 7.023437 z m 4.667969 -3.472656 c 0 3.253906 2.628906 5.882812 5.886718 5.882812 c 1.085938 0 2.152344 -0.304687 3.078125 -0.878906 l -1.519531 -0.960937 c -0.289062 2.554687 -2.464844 4.503906 -5.035156 4.507812 c -2.796875 0 -5.078125 -2.28125 -5.078125 -5.078125 c 0.003906 -2.570312 1.953125 -4.746094 4.507812 -5.035156 l -0.960937 -1.519531 c -0.574219 0.925781 -0.875 1.992187 -0.878906 3.082031 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 1.1 KiB |
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 5 0 c -0.96875 0 -2 1.050781 -2 2 v 2.988281 c 0 0.429688 0.222656 0.675781 0.554688 1.007813 l 2.023437 2.003906 l -2.007813 1.992188 c -0.367187 0.363281 -0.570312 0.6875 -0.570312 1 v 3.007812 c 0 1.011719 0.988281 2 2 2 h 6 c 1.007812 0 2 -1.011719 2 -2.003906 v -3.003906 c 0 -0.3125 -0.222656 -0.628907 -0.570312 -0.976563 l -2.015626 -2.015625 l 1.988282 -1.988281 c 0.261718 -0.261719 0.585937 -0.6875 0.597656 -1.015625 v -2.996094 c 0 -1.003906 -1.007812 -2 -2 -2 z m 6 4 h -6 v -2 h 6 m -3.589844 7 h 1.175782 l 2.414062 2.414062 v 1.585938 h -6 v -1.613281 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 729 B |
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 8 0 c -0.554688 0 -1 0.445312 -1 1 v 1 c 0 0.554688 0.445312 1 1 1 s 1 -0.445312 1 -1 v -1 c 0 -0.554688 -0.445312 -1 -1 -1 z m -4.996094 2.003906 c -0.253906 0 -0.507812 0.097656 -0.707031 0.296875 c -0.390625 0.390625 -0.390625 1.019531 0 1.414063 l 0.707031 0.707031 c 0.394532 0.390625 1.023438 0.390625 1.414063 0 c 0.394531 -0.394531 0.394531 -1.023437 0 -1.414063 l -0.707031 -0.707031 c -0.195313 -0.199219 -0.449219 -0.296875 -0.707032 -0.296875 z m 9.988282 0 c -0.253907 0 -0.507813 0.097656 -0.707032 0.296875 l -0.707031 0.707031 c -0.390625 0.390626 -0.390625 1.019532 0 1.414063 c 0.394531 0.390625 1.023437 0.390625 1.414063 0 l 0.707031 -0.707031 c 0.394531 -0.394532 0.394531 -1.023438 0 -1.414063 c -0.195313 -0.199219 -0.449219 -0.296875 -0.707031 -0.296875 z m -4.992188 1.996094 c -2.210938 0 -4 1.789062 -4 4 s 1.789062 4 4 4 s 4 -1.789062 4 -4 s -1.789062 -4 -4 -4 z m 0 2 c 1.105469 0 2 0.894531 2 2 s -0.894531 2 -2 2 s -2 -0.894531 -2 -2 s 0.894531 -2 2 -2 z m -7 1 c -0.554688 0 -1 0.445312 -1 1 s 0.445312 1 1 1 h 1 c 0.554688 0 1 -0.445312 1 -1 s -0.445312 -1 -1 -1 z m 13 0 c -0.554688 0 -1 0.445312 -1 1 s 0.445312 1 1 1 h 1 c 0.554688 0 1 -0.445312 1 -1 s -0.445312 -1 -1 -1 z m -10.335938 4.289062 c -0.238281 0.007813 -0.472656 0.105469 -0.660156 0.292969 l -0.707031 0.707031 c -0.390625 0.390626 -0.390625 1.019532 0 1.414063 c 0.394531 0.390625 1.023437 0.390625 1.414063 0 l 0.707031 -0.707031 c 0.394531 -0.394532 0.394531 -1.023438 0 -1.414063 c -0.207031 -0.210937 -0.484375 -0.308593 -0.753907 -0.292969 z m 8.574219 0 c -0.238281 0.007813 -0.472656 0.105469 -0.660156 0.292969 c -0.390625 0.390625 -0.390625 1.019531 0 1.414063 l 0.707031 0.707031 c 0.394532 0.390625 1.023438 0.390625 1.414063 0 c 0.394531 -0.394531 0.394531 -1.023437 0 -1.414063 l -0.707031 -0.707031 c -0.207032 -0.210937 -0.484376 -0.308593 -0.753907 -0.292969 z m -4.292969 1.710938 c -0.527343 0.027344 -0.945312 0.464844 -0.945312 1 v 1 c 0 0.554688 0.445312 1 1 1 s 1 -0.445312 1 -1 v -1 c 0 -0.554688 -0.445312 -1 -1 -1 c -0.015625 0 -0.035156 0 -0.050781 0 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 2.2 KiB |
|
@ -1,6 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 11.5 1 c -1.921875 0 -3.5 1.578125 -3.5 3.5 s 1.578125 3.5 3.5 3.5 s 3.5 -1.578125 3.5 -3.5 s -1.578125 -3.5 -3.5 -3.5 z m 0 2 c 0.839844 0 1.5 0.660156 1.5 1.5 s -0.660156 1.5 -1.5 1.5 s -1.5 -0.660156 -1.5 -1.5 s 0.660156 -1.5 1.5 -1.5 z m 0 0"/>
|
||||
<path d="m 4.5 8 c -1.921875 0 -3.5 1.578125 -3.5 3.5 s 1.578125 3.5 3.5 3.5 c 1.386719 0 2.59375 -0.820312 3.15625 -2 h 5.84375 c 0.832031 0 1.5 -0.667969 1.5 -1.5 s -0.667969 -1.5 -1.5 -1.5 h -5.84375 c -0.5625 -1.179688 -1.769531 -2 -3.15625 -2 z m 0 2 c 0.839844 0 1.5 0.660156 1.5 1.5 s -0.660156 1.5 -1.5 1.5 s -1.5 -0.660156 -1.5 -1.5 s 0.660156 -1.5 1.5 -1.5 z m 0 0"/>
|
||||
<path d="m 2.5 3 c -0.832031 0 -1.5 0.667969 -1.5 1.5 s 0.667969 1.5 1.5 1.5 h 4.769531 c -0.175781 -0.480469 -0.265625 -0.988281 -0.269531 -1.5 c 0 -0.511719 0.09375 -1.019531 0.269531 -1.5 z m 0 0" fill-opacity="0.34902"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 1,009 B |
|
@ -1,155 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<svg
|
||||
width="496"
|
||||
height="496"
|
||||
version="1"
|
||||
id="svg6"
|
||||
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg"
|
||||
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||
xmlns:cc="http://creativecommons.org/ns#"
|
||||
xmlns:dc="http://purl.org/dc/elements/1.1/">
|
||||
<metadata
|
||||
id="metadata12">
|
||||
<rdf:RDF>
|
||||
<cc:Work
|
||||
rdf:about="">
|
||||
<dc:format>image/svg+xml</dc:format>
|
||||
<dc:type
|
||||
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
|
||||
</cc:Work>
|
||||
</rdf:RDF>
|
||||
</metadata>
|
||||
<defs
|
||||
id="defs10" />
|
||||
<g
|
||||
id="g946"
|
||||
transform="matrix(0.97173996,0,0,0.97173996,4.043873,36.112138)">
|
||||
<g
|
||||
id="layer7"
|
||||
style="display:none"
|
||||
transform="translate(-23.75651,-24.84972)">
|
||||
<rect
|
||||
transform="translate(-132.5822,958.04022)"
|
||||
style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:#ffffff;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:3;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate"
|
||||
id="rect5389"
|
||||
width="1543.4283"
|
||||
height="483.7439"
|
||||
x="132.5822"
|
||||
y="-957.77832" />
|
||||
</g>
|
||||
<g
|
||||
id="layer6"
|
||||
style="display:none"
|
||||
transform="translate(-156.33871,933.1905)">
|
||||
<rect
|
||||
y="-958.02759"
|
||||
x="132.65129"
|
||||
height="484.30399"
|
||||
width="550.41602"
|
||||
id="rect5379"
|
||||
style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:#5c201e;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:3;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" />
|
||||
<rect
|
||||
style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:#c24a46;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:3;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate"
|
||||
id="rect5372"
|
||||
width="501.94415"
|
||||
height="434.30405"
|
||||
x="156.12303"
|
||||
y="-933.02759" />
|
||||
<rect
|
||||
style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:#d98d8a;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:3;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate"
|
||||
id="rect5381"
|
||||
width="24.939611"
|
||||
height="24.939611"
|
||||
x="658.02826"
|
||||
y="-958.04022" />
|
||||
</g>
|
||||
<g
|
||||
id="layer3"
|
||||
style="display:inline;opacity:1"
|
||||
transform="translate(37.235605,912.8581)">
|
||||
<g
|
||||
id="g2072"
|
||||
transform="matrix(0.99894325,0,0,0.99894325,-36.551621,-913.90743)"
|
||||
style="fill:#cccccc;fill-opacity:1">
|
||||
<g
|
||||
style="display:none;fill:#cccccc;fill-opacity:1"
|
||||
transform="matrix(0.09048806,0,0,0.09048806,-14.15991,84.454917)"
|
||||
id="layer1-3">
|
||||
<rect
|
||||
y="-2102.4253"
|
||||
x="-1045.6049"
|
||||
height="7145.4614"
|
||||
width="7947.0356"
|
||||
id="rect995"
|
||||
style="opacity:1;fill:#cccccc;fill-opacity:1;stroke-width:10.3605" />
|
||||
</g>
|
||||
<g
|
||||
transform="translate(-156.48372,537.56136)"
|
||||
style="display:inline;opacity:1;fill:#cccccc;fill-opacity:1"
|
||||
id="layer3-6">
|
||||
<g
|
||||
style="fill:#cccccc;stroke-width:11.0512;fill-opacity:1"
|
||||
transform="matrix(0.09048806,0,0,0.09048806,142.32381,-453.10644)"
|
||||
id="g955">
|
||||
<g
|
||||
transform="matrix(11.047619,0,0,11.047619,-1572.2888,9377.7107)"
|
||||
id="g869"
|
||||
style="fill:#cccccc;fill-opacity:1">
|
||||
<g
|
||||
transform="rotate(-60,226.35754,-449.37199)"
|
||||
id="g932"
|
||||
style="fill:#cccccc;stroke-width:11.0512;fill-opacity:1">
|
||||
<path
|
||||
id="path3336-6-7"
|
||||
d="m 449.71876,-420.51322 c 40.73228,70.55837 81.46455,141.11675 122.19683,211.67512 -18.71902,0.1756 -37.43804,0.3512 -56.15706,0.5268 -10.87453,-18.9564 -21.74907,-37.9128 -32.6236,-56.8692 -10.95215,18.8551 -21.9043,37.7102 -32.85645,56.5653 -9.30079,-0.004 -18.60158,-0.007 -27.90237,-0.011 -4.76362,-8.22987 -9.52724,-16.45973 -14.29086,-24.6896 15.60349,-26.83003 31.20698,-53.66007 46.81047,-80.4901 -11.07649,-19.27523 -22.15297,-38.55047 -33.22946,-57.8257 9.35083,-16.29387 18.70167,-32.58775 28.0525,-48.88162 z"
|
||||
style="opacity:1;fill:#cccccc;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:33.1535;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
|
||||
</g>
|
||||
<path
|
||||
id="path4260-0-5"
|
||||
d="m 309.54892,-710.38827 c 40.73228,70.55837 81.46455,141.11675 122.19683,211.67512 -18.71902,0.1756 -37.43804,0.3512 -56.15706,0.5268 -10.87453,-18.9564 -21.74907,-37.9128 -32.6236,-56.8692 -10.95215,18.8551 -21.9043,37.7102 -32.85645,56.5653 -9.30079,-0.004 -18.60158,-0.007 -27.90237,-0.011 -4.76362,-8.22987 -9.52724,-16.45973 -14.29086,-24.6896 15.60349,-26.83003 31.20698,-53.66007 46.81047,-80.4901 -11.07649,-19.2752 -22.15297,-38.5504 -33.22946,-57.8256 9.35083,-16.29391 18.70167,-32.58781 28.0525,-48.88172 z"
|
||||
style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:#cccccc;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:33.1535;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" />
|
||||
<use
|
||||
x="0"
|
||||
y="0"
|
||||
xlink:href="#path3336-6-7"
|
||||
id="use3439-6-3"
|
||||
transform="rotate(60,728.23563,-692.24036)"
|
||||
width="100%"
|
||||
height="100%"
|
||||
style="fill:#cccccc;fill-opacity:1;stroke-width:11.0512" />
|
||||
<use
|
||||
x="0"
|
||||
y="0"
|
||||
xlink:href="#path3336-6-7"
|
||||
id="use3449-5-5"
|
||||
transform="rotate(180,477.5036,-570.81898)"
|
||||
width="100%"
|
||||
height="100%"
|
||||
style="fill:#cccccc;fill-opacity:1;stroke-width:11.0512" />
|
||||
<use
|
||||
style="display:inline;fill:#cccccc;fill-opacity:1;stroke-width:11.0512"
|
||||
x="0"
|
||||
y="0"
|
||||
xlink:href="#path4260-0-5"
|
||||
id="use4354-5-6"
|
||||
transform="rotate(120,407.33916,-716.08356)"
|
||||
width="100%"
|
||||
height="100%" />
|
||||
<use
|
||||
style="display:inline;fill:#cccccc;fill-opacity:1;stroke-width:11.0512"
|
||||
x="0"
|
||||
y="0"
|
||||
xlink:href="#path4260-0-5"
|
||||
id="use4362-2-2"
|
||||
transform="rotate(-120,407.28823,-715.86995)"
|
||||
width="100%"
|
||||
height="100%" />
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</svg>
|
Before Width: | Height: | Size: 8.6 KiB |
|
@ -1,321 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<svg
|
||||
height="16px"
|
||||
viewBox="0 0 16 16"
|
||||
width="16px"
|
||||
version="1.1"
|
||||
id="svg3533"
|
||||
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg">
|
||||
<defs
|
||||
id="defs3537" />
|
||||
<filter
|
||||
id="a"
|
||||
height="1"
|
||||
width="1"
|
||||
x="0"
|
||||
y="0">
|
||||
<feColorMatrix
|
||||
in="SourceGraphic"
|
||||
type="matrix"
|
||||
values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"
|
||||
id="feColorMatrix3414" />
|
||||
</filter>
|
||||
<mask
|
||||
id="b">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3419">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3417" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="c">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3422" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="d">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3427">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3425" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="e">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3430" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="f">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3435">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3433" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="g">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3438" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="h">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3443">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3441" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="i">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3446" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="j">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3451">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3449" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="k">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3454" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="l">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3459">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3457" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="m">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3462" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="n">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3467">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3465" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="o">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3470" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="p">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3475">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3473" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="q">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3478" />
|
||||
</clipPath>
|
||||
<mask
|
||||
id="r">
|
||||
<g
|
||||
filter="url(#a)"
|
||||
id="g3483">
|
||||
<image
|
||||
height="800"
|
||||
width="1024"
|
||||
xlink:href=""
|
||||
id="image3481" />
|
||||
</g>
|
||||
</mask>
|
||||
<clipPath
|
||||
id="s">
|
||||
<path
|
||||
d="m 0 0 h 1024 v 800 h -1024 z"
|
||||
id="path3486" />
|
||||
</clipPath>
|
||||
<path
|
||||
d="m 3 1 c -1.644531 0 -3 1.355469 -3 3 v 6 c 0 1.644531 1.355469 3 3 3 h 10 c 1.644531 0 3 -1.355469 3 -3 v -6 c 0 -0.570312 -0.167969 -1.101562 -0.449219 -1.558594 l -1.550781 1.554688 v 6.003906 c 0 0.570312 -0.429688 1 -1 1 h -10 c -0.570312 0 -1 -0.429688 -1 -1 v -6 c 0 -0.570312 0.429688 -1 1 -1 h 5.96875 l 2.007812 -2 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3489"
|
||||
style="fill:#000000" />
|
||||
<g
|
||||
clip-path="url(#c)"
|
||||
mask="url(#b)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3493"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 439.105469 225.78125 h 7.839843 c -0.890624 0.371094 -0.972656 1.847656 0 2.25 h -7.839843 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3491"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#e)"
|
||||
mask="url(#d)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3497"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 29.25 627.75 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3495"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#g)"
|
||||
mask="url(#f)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3501"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 30 627 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3499"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#i)"
|
||||
mask="url(#h)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3505"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 30.75 629.25 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3503"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#k)"
|
||||
mask="url(#j)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3509"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 29.25 629.25 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3507"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#m)"
|
||||
mask="url(#l)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3513"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 30 630 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3511"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#o)"
|
||||
mask="url(#n)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3517"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 31.5 630 h 0.75 v 0.75 h -0.75 z m 0 0"
|
||||
fill="#2e3436"
|
||||
fill-rule="evenodd"
|
||||
id="path3515"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<g
|
||||
clip-path="url(#q)"
|
||||
mask="url(#p)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3521"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 119.253906 648.75 v 5.25 h 5.25 v -5.25 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3519"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<path
|
||||
d="m 11 7 c 0 1.65625 -1.339844 3.007812 -3 3 h -3 v -3 c 0 -1.660156 1.34375 -3 3 -3 c 1.660156 0 3 1.339844 3 3 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3523"
|
||||
style="fill:#000000" />
|
||||
<path
|
||||
d="m 13.398438 0 l -3.46875 3.457031 c 0.683593 0.355469 1.234374 0.910157 1.589843 1.589844 l 0.171875 -0.171875 l 0.007813 0.007812 l 4.300781 -4.300781 v -0.582031 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3525"
|
||||
style="fill:#000000" />
|
||||
<g
|
||||
clip-path="url(#s)"
|
||||
mask="url(#r)"
|
||||
transform="matrix(1 0 0 1 -40 -620)"
|
||||
id="g3529"
|
||||
style="fill:#000000">
|
||||
<path
|
||||
d="m 181.503906 635.25 h 2.25 v 9 h -2.25 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3527"
|
||||
style="fill:#000000" />
|
||||
</g>
|
||||
<path
|
||||
d="m 5 14 c -1.105469 0 -2 0.894531 -2 2 h 10 c 0 -1.105469 -0.894531 -2 -2 -2 z m 0 0"
|
||||
fill="#2e3436"
|
||||
id="path3531"
|
||||
style="fill:#000000" />
|
||||
</svg>
|
Before Width: | Height: | Size: 46 KiB |
|
@ -1,17 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 5 5 h 6 v 6 h -6 z m 0 0"/>
|
||||
<path d="m 13 5 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 13 7 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 13 9 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 0 6 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 0 8 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 0 10 h 3 v 1 h -3 z m 0 0"/>
|
||||
<path d="m 5 0 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 7 0 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 9 0 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 10 13 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 8 13 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 6 13 h 1 v 3 h -1 z m 0 0"/>
|
||||
<path d="m 5 2 c -1.644531 0 -3 1.355469 -3 3 v 6 c 0 1.644531 1.355469 3 3 3 h 6 c 1.644531 0 3 -1.355469 3 -3 v -6 c 0 -1.644531 -1.355469 -3 -3 -3 z m 0 2 h 6 c 0.570312 0 1 0.429688 1 1 v 6 c 0 0.570312 -0.429688 1 -1 1 h -6 c -0.570312 0 -1 -0.429688 -1 -1 v -6 c 0 -0.570312 0.429688 -1 1 -1 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 1,014 B |
|
@ -1,5 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 2.199219 0 c -1.207031 0 -2.199219 1.007812 -2.199219 2.207031 v 10.585938 c 0 1.199219 0.992188 2.207031 2.199219 2.207031 h 11.601562 c 1.207031 0 2.199219 -1.007812 2.199219 -2.207031 v -10.585938 c 0 -1.199219 -0.992188 -2.207031 -2.199219 -2.207031 z m 0 2 h 11.601562 c 0.121094 0 0.199219 0.070312 0.199219 0.207031 v 10.585938 c 0 0.136719 -0.078125 0.207031 -0.199219 0.207031 h -11.601562 c -0.121094 0 -0.199219 -0.070312 -0.199219 -0.207031 v -10.585938 c 0 -0.136719 0.078125 -0.207031 0.199219 -0.207031 z m 0 0"/>
|
||||
<path d="m 4.515625 5.898438 c -0.164063 -0.003907 -0.324219 0.0625 -0.441406 0.175781 c -0.230469 0.234375 -0.230469 0.617187 0 0.851562 l 1.578125 1.574219 l -1.578125 1.574219 c -0.230469 0.234375 -0.230469 0.617187 0 0.851562 c 0.234375 0.230469 0.617187 0.230469 0.851562 0 l 2 -2 c 0.230469 -0.234375 0.230469 -0.617187 0 -0.851562 l -2 -2 c -0.109375 -0.105469 -0.257812 -0.167969 -0.410156 -0.175781 z m 3.484375 4.101562 v 1 h 3 v -1 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 1.1 KiB |
|
@ -1,4 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<svg height="16px" viewBox="0 0 16 16" width="16px" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="m 2 0 c -1.214844 0 -2 0.828125 -2 2 v 12 c 0 1 1 2 2 2 h 11.984375 c 1 0 2 -1 2 -2 v -12 c 0 -1.238281 -0.828125 -2 -2 -2 z m 0 2 h 2 v 2 h -2 z m 3 0 h 2 v 2 h -2 z m 3 0 h 2 v 2 h -2 z m -6 4 h 11.984375 v 8 h -11.984375 z m 0 0"/>
|
||||
</svg>
|
Before Width: | Height: | Size: 382 B |
|
@ -1,46 +0,0 @@
|
|||
import GLib from "gi://GLib"
|
||||
|
||||
const main = "/tmp/asztal/main.js"
|
||||
const entry = `${App.configDir}/main.ts`
|
||||
const bundler = GLib.getenv("AGS_BUNDLER") || "bun"
|
||||
|
||||
const v = {
|
||||
ags: pkg.version?.split(".").map(Number) || [],
|
||||
expect: [1, 8, 1],
|
||||
}
|
||||
|
||||
try {
|
||||
switch (bundler) {
|
||||
case "bun": await Utils.execAsync([
|
||||
"bun", "build", entry,
|
||||
"--outfile", main,
|
||||
"--external", "resource://*",
|
||||
"--external", "gi://*",
|
||||
"--external", "file://*",
|
||||
]); break
|
||||
|
||||
case "esbuild": await Utils.execAsync([
|
||||
"esbuild", "--bundle", entry,
|
||||
"--format=esm",
|
||||
`--outfile=${main}`,
|
||||
"--external:resource://*",
|
||||
"--external:gi://*",
|
||||
"--external:file://*",
|
||||
]); break
|
||||
|
||||
default:
|
||||
throw `"${bundler}" is not a valid bundler`
|
||||
}
|
||||
|
||||
if (v.ags[1] < v.expect[1] || v.ags[2] < v.expect[2]) {
|
||||
print(`my config needs at least v${v.expect.join(".")}, yours is v${v.ags.join(".")}`)
|
||||
App.quit()
|
||||
}
|
||||
|
||||
await import(`file://${main}`)
|
||||
} catch (error) {
|
||||
console.error(error)
|
||||
App.quit()
|
||||
}
|
||||
|
||||
export { }
|
|
@ -1,103 +0,0 @@
|
|||
{
|
||||
inputs,
|
||||
writeShellScript,
|
||||
system,
|
||||
stdenv,
|
||||
cage,
|
||||
swww,
|
||||
esbuild,
|
||||
dart-sass,
|
||||
fd,
|
||||
fzf,
|
||||
brightnessctl,
|
||||
accountsservice,
|
||||
slurp,
|
||||
wf-recorder,
|
||||
wl-clipboard,
|
||||
wayshot,
|
||||
swappy,
|
||||
hyprpicker,
|
||||
pavucontrol,
|
||||
networkmanager,
|
||||
gtk3,
|
||||
which,
|
||||
}: let
|
||||
name = "asztal";
|
||||
|
||||
ags = inputs.ags.packages.${system}.default.override {
|
||||
extraPackages = [accountsservice];
|
||||
};
|
||||
|
||||
dependencies = [
|
||||
which
|
||||
dart-sass
|
||||
fd
|
||||
fzf
|
||||
brightnessctl
|
||||
swww
|
||||
inputs.matugen.packages.${system}.default
|
||||
slurp
|
||||
wf-recorder
|
||||
wl-clipboard
|
||||
wayshot
|
||||
swappy
|
||||
hyprpicker
|
||||
pavucontrol
|
||||
networkmanager
|
||||
gtk3
|
||||
];
|
||||
|
||||
addBins = list: builtins.concatStringsSep ":" (builtins.map (p: "${p}/bin") list);
|
||||
|
||||
greeter = writeShellScript "greeter" ''
|
||||
export PATH=$PATH:${addBins dependencies}
|
||||
${cage}/bin/cage -ds -m last ${ags}/bin/ags -- -c ${config}/greeter.js
|
||||
'';
|
||||
|
||||
desktop = writeShellScript name ''
|
||||
export PATH=$PATH:${addBins dependencies}
|
||||
${ags}/bin/ags -b ${name} -c ${config}/config.js $@
|
||||
'';
|
||||
|
||||
config = stdenv.mkDerivation {
|
||||
inherit name;
|
||||
src = ./.;
|
||||
|
||||
buildPhase = ''
|
||||
${esbuild}/bin/esbuild \
|
||||
--bundle ./main.ts \
|
||||
--outfile=main.js \
|
||||
--format=esm \
|
||||
--external:resource://\* \
|
||||
--external:gi://\* \
|
||||
|
||||
${esbuild}/bin/esbuild \
|
||||
--bundle ./greeter/greeter.ts \
|
||||
--outfile=greeter.js \
|
||||
--format=esm \
|
||||
--external:resource://\* \
|
||||
--external:gi://\* \
|
||||
'';
|
||||
|
||||
installPhase = ''
|
||||
mkdir -p $out
|
||||
cp -r assets $out
|
||||
cp -r style $out
|
||||
cp -r greeter $out
|
||||
cp -r widget $out
|
||||
cp -f main.js $out/config.js
|
||||
cp -f greeter.js $out/greeter.js
|
||||
'';
|
||||
};
|
||||
in
|
||||
stdenv.mkDerivation {
|
||||
inherit name;
|
||||
src = config;
|
||||
|
||||
installPhase = ''
|
||||
mkdir -p $out/bin
|
||||
cp -r . $out
|
||||
cp ${desktop} $out/bin/${name}
|
||||
cp ${greeter} $out/bin/greeter
|
||||
'';
|
||||
}
|
|
@ -1,18 +0,0 @@
|
|||
const main = "/tmp/ags/greeter.js"
|
||||
const entry = `${App.configDir}/greeter/greeter.ts`
|
||||
|
||||
try {
|
||||
await Utils.execAsync([
|
||||
"bun", "build", entry,
|
||||
"--outfile", main,
|
||||
"--external", "resource://*",
|
||||
"--external", "gi://*",
|
||||
"--external", "file://*",
|
||||
])
|
||||
await import(`file://${main}`)
|
||||
} catch (error) {
|
||||
console.error(error)
|
||||
App.quit()
|
||||
}
|
||||
|
||||
export { }
|
|
@ -1,114 +0,0 @@
|
|||
import GLib from "gi://GLib?version=2.0"
|
||||
import icons from "lib/icons"
|
||||
import { bash } from "lib/utils"
|
||||
|
||||
const userName = await bash("find /home -maxdepth 1 -printf '%f\n' | tail -n 1")
|
||||
const iconFile = `/var/lib/AccountsService/icons/${userName}`
|
||||
|
||||
// FIXME: AccountsService crashes?
|
||||
// import AccountsService from "gi://AccountsService?version=1.0"
|
||||
// const { iconFile, realName, userName } = AccountsService.UserManager
|
||||
// .get_default().list_users()[0]
|
||||
|
||||
const loggingin = Variable(false)
|
||||
|
||||
const CMD = GLib.getenv("ASZTAL_DM_CMD")
|
||||
|| "Hyprland"
|
||||
|
||||
const ENV = GLib.getenv("ASZTAL_DM_ENV")
|
||||
|| "WLR_NO_HARDWARE_CURSORS=1 _JAVA_AWT_WM_NONREPARENTING=1"
|
||||
|
||||
async function login(pw: string) {
|
||||
loggingin.value = true
|
||||
const greetd = await Service.import("greetd")
|
||||
return greetd.login(userName, pw, CMD, ENV.split(/\s+/))
|
||||
.catch(res => {
|
||||
loggingin.value = false
|
||||
response.label = res?.description || JSON.stringify(res)
|
||||
password.text = ""
|
||||
revealer.reveal_child = true
|
||||
})
|
||||
}
|
||||
|
||||
const avatar = Widget.Box({
|
||||
class_name: "avatar",
|
||||
hpack: "center",
|
||||
css: `background-image: url('${iconFile}')`,
|
||||
})
|
||||
|
||||
const password = Widget.Entry({
|
||||
placeholder_text: "Password",
|
||||
hexpand: true,
|
||||
visibility: false,
|
||||
on_accept: ({ text }) => { login(text || "") },
|
||||
})
|
||||
|
||||
const response = Widget.Label({
|
||||
class_name: "response",
|
||||
wrap: true,
|
||||
max_width_chars: 35,
|
||||
hpack: "center",
|
||||
hexpand: true,
|
||||
xalign: .5,
|
||||
})
|
||||
|
||||
const revealer = Widget.Revealer({
|
||||
transition: "slide_down",
|
||||
child: response,
|
||||
})
|
||||
|
||||
export default Widget.Box({
|
||||
class_name: "auth",
|
||||
attribute: { password },
|
||||
vertical: true,
|
||||
children: [
|
||||
Widget.Overlay({
|
||||
child: Widget.Box(
|
||||
{
|
||||
css: "min-width: 200px; min-height: 200px;",
|
||||
vertical: true,
|
||||
},
|
||||
Widget.Box({
|
||||
class_name: "wallpaper",
|
||||
css: `background-image: url('${WALLPAPER}')`,
|
||||
}),
|
||||
Widget.Box({
|
||||
class_name: "wallpaper-contrast",
|
||||
vexpand: true,
|
||||
}),
|
||||
),
|
||||
overlay: Widget.Box(
|
||||
{
|
||||
vpack: "end",
|
||||
vertical: true,
|
||||
},
|
||||
avatar,
|
||||
Widget.Box({
|
||||
hpack: "center",
|
||||
children: [
|
||||
Widget.Icon(icons.ui.avatar),
|
||||
Widget.Label(userName),
|
||||
],
|
||||
}),
|
||||
Widget.Box(
|
||||
{
|
||||
class_name: "password",
|
||||
},
|
||||
Widget.Spinner({
|
||||
visible: loggingin.bind(),
|
||||
active: true,
|
||||
}),
|
||||
Widget.Icon({
|
||||
visible: loggingin.bind().as(b => !b),
|
||||
icon: icons.ui.lock,
|
||||
}),
|
||||
password,
|
||||
),
|
||||
),
|
||||
}),
|
||||
Widget.Box(
|
||||
{ class_name: "response-box" },
|
||||
revealer,
|
||||
),
|
||||
],
|
||||
})
|
|
@ -1,37 +0,0 @@
|
|||
import "./session"
|
||||
import "style/style"
|
||||
import GLib from "gi://GLib?version=2.0"
|
||||
import RegularWindow from "widget/RegularWindow"
|
||||
import statusbar from "./statusbar"
|
||||
import auth from "./auth"
|
||||
|
||||
const win = RegularWindow({
|
||||
name: "greeter",
|
||||
setup: self => {
|
||||
self.set_default_size(500, 500)
|
||||
self.show_all()
|
||||
auth.attribute.password.grab_focus()
|
||||
},
|
||||
child: Widget.Overlay({
|
||||
child: Widget.Box({ expand: true }),
|
||||
overlays: [
|
||||
Widget.Box({
|
||||
vpack: "start",
|
||||
hpack: "fill",
|
||||
hexpand: true,
|
||||
child: statusbar,
|
||||
}),
|
||||
Widget.Box({
|
||||
vpack: "center",
|
||||
hpack: "center",
|
||||
child: auth,
|
||||
}),
|
||||
],
|
||||
}),
|
||||
})
|
||||
|
||||
App.config({
|
||||
icons: "./assets",
|
||||
windows: [win],
|
||||
cursorTheme: GLib.getenv("XCURSOR_THEME")!,
|
||||
})
|
|
@ -1,23 +0,0 @@
|
|||
import GLib from "gi://GLib?version=2.0"
|
||||
import { bash } from "lib/utils"
|
||||
|
||||
// import AccountsService from "gi://AccountsService?version=1.0"
|
||||
// const { userName } = AccountsService.UserManager.get_default().list_users()[0]
|
||||
|
||||
const userName = await bash("find /home -maxdepth 1 -printf '%f\n' | tail -n 1")
|
||||
|
||||
declare global {
|
||||
const WALLPAPER: string
|
||||
}
|
||||
|
||||
Object.assign(globalThis, {
|
||||
TMP: `${GLib.get_tmp_dir()}/greeter`,
|
||||
OPTIONS: "/var/cache/greeter/options.json",
|
||||
WALLPAPER: "/var/cache/greeter/background",
|
||||
// TMP: "/tmp/ags",
|
||||
// OPTIONS: Utils.CACHE_DIR + "/options.json",
|
||||
// WALLPAPER: Utils.HOME + "/.config/background",
|
||||
USER: userName,
|
||||
})
|
||||
|
||||
Utils.ensureDirectory(TMP)
|
|
@ -1,46 +0,0 @@
|
|||
import { clock } from "lib/variables"
|
||||
import options from "options"
|
||||
import icons from "lib/icons"
|
||||
import BatteryBar from "widget/bar/buttons/BatteryBar"
|
||||
import PanelButton from "widget/bar/PanelButton"
|
||||
|
||||
const { scheme } = options.theme
|
||||
const { monochrome } = options.bar.powermenu
|
||||
const { format } = options.bar.date
|
||||
|
||||
const poweroff = PanelButton({
|
||||
class_name: "powermenu",
|
||||
child: Widget.Icon(icons.powermenu.shutdown),
|
||||
on_clicked: () => Utils.exec("shutdown now"),
|
||||
setup: self => self.hook(monochrome, () => {
|
||||
self.toggleClassName("colored", !monochrome.value)
|
||||
self.toggleClassName("box")
|
||||
}),
|
||||
})
|
||||
|
||||
const date = PanelButton({
|
||||
class_name: "date",
|
||||
child: Widget.Label({
|
||||
label: clock.bind().as(c => c.format(`${format}`)!),
|
||||
}),
|
||||
})
|
||||
|
||||
const darkmode = PanelButton({
|
||||
class_name: "darkmode",
|
||||
child: Widget.Icon({ icon: scheme.bind().as(s => icons.color[s]) }),
|
||||
on_clicked: () => scheme.value = scheme.value === "dark" ? "light" : "dark",
|
||||
})
|
||||
|
||||
export default Widget.CenterBox({
|
||||
class_name: "bar",
|
||||
hexpand: true,
|
||||
center_widget: date,
|
||||
end_widget: Widget.Box({
|
||||
hpack: "end",
|
||||
children: [
|
||||
darkmode,
|
||||
BatteryBar(),
|
||||
poweroff,
|
||||
],
|
||||
}),
|
||||
})
|
|
@ -1,16 +0,0 @@
|
|||
import icons from "./icons"
|
||||
|
||||
export default async function init() {
|
||||
const bat = await Service.import("battery")
|
||||
bat.connect("notify::percent", ({ percent, charging }) => {
|
||||
const low = 30
|
||||
if (percent !== low || percent !== low / 2 || !charging)
|
||||
return
|
||||
|
||||
Utils.notify({
|
||||
summary: `${percent}% Battery Percentage`,
|
||||
iconName: icons.battery.warning,
|
||||
urgency: "critical",
|
||||
})
|
||||
})
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
import Gio from "gi://Gio"
|
||||
import options from "options"
|
||||
|
||||
const settings = new Gio.Settings({
|
||||
schema: "org.gnome.desktop.interface",
|
||||
})
|
||||
|
||||
function gtk() {
|
||||
const scheme = options.theme.scheme.value
|
||||
settings.set_string("color-scheme", `prefer-${scheme}`)
|
||||
}
|
||||
|
||||
export default function init() {
|
||||
options.theme.scheme.connect("changed", gtk)
|
||||
gtk()
|
||||
}
|
|
@ -1,82 +0,0 @@
|
|||
import options from "options"
|
||||
const { messageAsync } = await Service.import("hyprland")
|
||||
|
||||
const {
|
||||
hyprland,
|
||||
theme: {
|
||||
spacing,
|
||||
radius,
|
||||
border: { width },
|
||||
blur,
|
||||
shadows,
|
||||
dark: {
|
||||
primary: { bg: darkActive },
|
||||
},
|
||||
light: {
|
||||
primary: { bg: lightActive },
|
||||
},
|
||||
scheme,
|
||||
},
|
||||
} = options
|
||||
|
||||
const deps = [
|
||||
"hyprland",
|
||||
spacing.id,
|
||||
radius.id,
|
||||
blur.id,
|
||||
width.id,
|
||||
shadows.id,
|
||||
darkActive.id,
|
||||
lightActive.id,
|
||||
scheme.id,
|
||||
]
|
||||
|
||||
function primary() {
|
||||
return scheme.value === "dark"
|
||||
? darkActive.value
|
||||
: lightActive.value
|
||||
}
|
||||
|
||||
function rgba(color: string) {
|
||||
return `rgba(${color}ff)`.replace("#", "")
|
||||
}
|
||||
|
||||
function sendBatch(batch: string[]) {
|
||||
const cmd = batch
|
||||
.filter(x => !!x)
|
||||
.map(x => `keyword ${x}`)
|
||||
.join("; ")
|
||||
|
||||
return messageAsync(`[[BATCH]]/${cmd}`)
|
||||
}
|
||||
|
||||
async function setupHyprland() {
|
||||
const wm_gaps = Math.floor(hyprland.gaps.value * spacing.value)
|
||||
|
||||
sendBatch([
|
||||
`general:border_size ${width}`,
|
||||
`general:gaps_out ${wm_gaps}`,
|
||||
`general:gaps_in ${Math.floor(wm_gaps / 2)}`,
|
||||
`general:col.active_border ${rgba(primary())}`,
|
||||
`general:col.inactive_border ${rgba(hyprland.inactiveBorder.value)}`,
|
||||
`decoration:rounding ${radius}`,
|
||||
`decoration:drop_shadow ${shadows.value ? "yes" : "no"}`,
|
||||
`dwindle:no_gaps_when_only ${hyprland.gapsWhenOnly.value ? 0 : 1}`,
|
||||
`master:no_gaps_when_only ${hyprland.gapsWhenOnly.value ? 0 : 1}`,
|
||||
])
|
||||
|
||||
await sendBatch(App.windows.map(({ name }) => `layerrule unset, ${name}`))
|
||||
|
||||
if (blur.value > 0) {
|
||||
sendBatch(App.windows.flatMap(({ name }) => [
|
||||
`layerrule unset, ${name}`,
|
||||
`layerrule blur, ${name}`,
|
||||
`layerrule ignorealpha ${/* based on shadow color */.29}, ${name}`,
|
||||
]))
|
||||
}
|
||||
}
|
||||
|
||||
export default function init() {
|
||||
options.handler(deps, setupHyprland)
|
||||
setupHyprland()
|
||||
}
|
|
@ -1,145 +0,0 @@
|
|||
export const substitutes = {
|
||||
"transmission-gtk": "transmission",
|
||||
"blueberry.py": "blueberry",
|
||||
"Caprine": "facebook-messenger",
|
||||
"com.raggesilver.BlackBox-symbolic": "terminal-symbolic",
|
||||
"org.wezfurlong.wezterm-symbolic": "terminal-symbolic",
|
||||
"audio-headset-bluetooth": "audio-headphones-symbolic",
|
||||
"audio-card-analog-usb": "audio-speakers-symbolic",
|
||||
"audio-card-analog-pci": "audio-card-symbolic",
|
||||
"preferences-system": "emblem-system-symbolic",
|
||||
"com.github.Aylur.ags-symbolic": "controls-symbolic",
|
||||
"com.github.Aylur.ags": "controls-symbolic",
|
||||
}
|
||||
|
||||
export default {
|
||||
missing: "image-missing-symbolic",
|
||||
nix: {
|
||||
nix: "nix-snowflake-symbolic",
|
||||
},
|
||||
app: {
|
||||
terminal: "terminal-symbolic",
|
||||
},
|
||||
fallback: {
|
||||
executable: "application-x-executable",
|
||||
notification: "dialog-information-symbolic",
|
||||
video: "video-x-generic-symbolic",
|
||||
audio: "audio-x-generic-symbolic",
|
||||
},
|
||||
ui: {
|
||||
close: "window-close-symbolic",
|
||||
colorpicker: "color-select-symbolic",
|
||||
info: "info-symbolic",
|
||||
link: "external-link-symbolic",
|
||||
lock: "system-lock-screen-symbolic",
|
||||
menu: "open-menu-symbolic",
|
||||
refresh: "view-refresh-symbolic",
|
||||
search: "system-search-symbolic",
|
||||
settings: "emblem-system-symbolic",
|
||||
themes: "preferences-desktop-theme-symbolic",
|
||||
tick: "object-select-symbolic",
|
||||
time: "hourglass-symbolic",
|
||||
toolbars: "toolbars-symbolic",
|
||||
warning: "dialog-warning-symbolic",
|
||||
avatar: "avatar-default-symbolic",
|
||||
arrow: {
|
||||
right: "pan-end-symbolic",
|
||||
left: "pan-start-symbolic",
|
||||
down: "pan-down-symbolic",
|
||||
up: "pan-up-symbolic",
|
||||
},
|
||||
},
|
||||
audio: {
|
||||
mic: {
|
||||
muted: "microphone-disabled-symbolic",
|
||||
low: "microphone-sensitivity-low-symbolic",
|
||||
medium: "microphone-sensitivity-medium-symbolic",
|
||||
high: "microphone-sensitivity-high-symbolic",
|
||||
},
|
||||
volume: {
|
||||
muted: "audio-volume-muted-symbolic",
|
||||
low: "audio-volume-low-symbolic",
|
||||
medium: "audio-volume-medium-symbolic",
|
||||
high: "audio-volume-high-symbolic",
|
||||
overamplified: "audio-volume-overamplified-symbolic",
|
||||
},
|
||||
type: {
|
||||
headset: "audio-headphones-symbolic",
|
||||
speaker: "audio-speakers-symbolic",
|
||||
card: "audio-card-symbolic",
|
||||
},
|
||||
mixer: "mixer-symbolic",
|
||||
},
|
||||
powerprofile: {
|
||||
balanced: "power-profile-balanced-symbolic",
|
||||
"power-saver": "power-profile-power-saver-symbolic",
|
||||
performance: "power-profile-performance-symbolic",
|
||||
},
|
||||
asusctl: {
|
||||
profile: {
|
||||
Balanced: "power-profile-balanced-symbolic",
|
||||
Quiet: "power-profile-power-saver-symbolic",
|
||||
Performance: "power-profile-performance-symbolic",
|
||||
},
|
||||
mode: {
|
||||
Integrated: "processor-symbolic",
|
||||
Hybrid: "controller-symbolic",
|
||||
},
|
||||
},
|
||||
battery: {
|
||||
charging: "battery-flash-symbolic",
|
||||
warning: "battery-empty-symbolic",
|
||||
},
|
||||
bluetooth: {
|
||||
enabled: "bluetooth-active-symbolic",
|
||||
disabled: "bluetooth-disabled-symbolic",
|
||||
},
|
||||
brightness: {
|
||||
indicator: "display-brightness-symbolic",
|
||||
keyboard: "keyboard-brightness-symbolic",
|
||||
screen: "display-brightness-symbolic",
|
||||
},
|
||||
powermenu: {
|
||||
sleep: "weather-clear-night-symbolic",
|
||||
reboot: "system-reboot-symbolic",
|
||||
logout: "system-log-out-symbolic",
|
||||
shutdown: "system-shutdown-symbolic",
|
||||
},
|
||||
recorder: {
|
||||
recording: "media-record-symbolic",
|
||||
},
|
||||
notifications: {
|
||||
noisy: "org.gnome.Settings-notifications-symbolic",
|
||||
silent: "notifications-disabled-symbolic",
|
||||
message: "chat-bubbles-symbolic",
|
||||
},
|
||||
trash: {
|
||||
full: "user-trash-full-symbolic",
|
||||
empty: "user-trash-symbolic",
|
||||
},
|
||||
mpris: {
|
||||
shuffle: {
|
||||
enabled: "media-playlist-shuffle-symbolic",
|
||||
disabled: "media-playlist-consecutive-symbolic",
|
||||
},
|
||||
loop: {
|
||||
none: "media-playlist-repeat-symbolic",
|
||||
track: "media-playlist-repeat-song-symbolic",
|
||||
playlist: "media-playlist-repeat-symbolic",
|
||||
},
|
||||
playing: "media-playback-pause-symbolic",
|
||||
paused: "media-playback-start-symbolic",
|
||||
stopped: "media-playback-start-symbolic",
|
||||
prev: "media-skip-backward-symbolic",
|
||||
next: "media-skip-forward-symbolic",
|
||||
},
|
||||
system: {
|
||||
cpu: "org.gnome.SystemMonitor-symbolic",
|
||||
ram: "drive-harddisk-solidstate-symbolic",
|
||||
temp: "temperature-symbolic",
|
||||
},
|
||||
color: {
|
||||
dark: "dark-mode-symbolic",
|
||||
light: "light-mode-symbolic",
|
||||
},
|
||||
}
|
|
@ -1,19 +0,0 @@
|
|||
import matugen from "./matugen"
|
||||
import hyprland from "./hyprland"
|
||||
import tmux from "./tmux"
|
||||
import gtk from "./gtk"
|
||||
import lowBattery from "./battery"
|
||||
import notifications from "./notifications"
|
||||
|
||||
export default function init() {
|
||||
try {
|
||||
gtk()
|
||||
tmux()
|
||||
matugen()
|
||||
lowBattery()
|
||||
notifications()
|
||||
hyprland()
|
||||
} catch (error) {
|
||||
logError(error)
|
||||
}
|
||||
}
|
|
@ -1,113 +0,0 @@
|
|||
import wallpaper from "service/wallpaper"
|
||||
import options from "options"
|
||||
import { sh, dependencies } from "./utils"
|
||||
|
||||
export default function init() {
|
||||
wallpaper.connect("changed", () => matugen())
|
||||
options.autotheme.connect("changed", () => matugen())
|
||||
}
|
||||
|
||||
function animate(...setters: Array<() => void>) {
|
||||
const delay = options.transition.value / 2
|
||||
setters.forEach((fn, i) => Utils.timeout(delay * i, fn))
|
||||
}
|
||||
|
||||
export async function matugen(
|
||||
type: "image" | "color" = "image",
|
||||
arg = wallpaper.wallpaper,
|
||||
) {
|
||||
if (!options.autotheme.value || !dependencies("matugen"))
|
||||
return
|
||||
|
||||
const colors = await sh(`matugen --dry-run -j hex ${type} ${arg}`)
|
||||
const c = JSON.parse(colors).colors as { light: Colors, dark: Colors }
|
||||
const { dark, light } = options.theme
|
||||
|
||||
animate(
|
||||
() => {
|
||||
dark.widget.value = c.dark.on_surface
|
||||
light.widget.value = c.light.on_surface
|
||||
},
|
||||
() => {
|
||||
dark.border.value = c.dark.outline
|
||||
light.border.value = c.light.outline
|
||||
},
|
||||
() => {
|
||||
dark.bg.value = c.dark.surface
|
||||
light.bg.value = c.light.surface
|
||||
},
|
||||
() => {
|
||||
dark.fg.value = c.dark.on_surface
|
||||
light.fg.value = c.light.on_surface
|
||||
},
|
||||
() => {
|
||||
dark.primary.bg.value = c.dark.primary
|
||||
light.primary.bg.value = c.light.primary
|
||||
options.bar.battery.charging.value = options.theme.scheme.value === "dark"
|
||||
? c.dark.primary : c.light.primary
|
||||
},
|
||||
() => {
|
||||
dark.primary.fg.value = c.dark.on_primary
|
||||
light.primary.fg.value = c.light.on_primary
|
||||
},
|
||||
() => {
|
||||
dark.error.bg.value = c.dark.error
|
||||
light.error.bg.value = c.light.error
|
||||
},
|
||||
() => {
|
||||
dark.error.fg.value = c.dark.on_error
|
||||
light.error.fg.value = c.light.on_error
|
||||
},
|
||||
)
|
||||
}
|
||||
|
||||
type Colors = {
|
||||
background: string
|
||||
error: string
|
||||
error_container: string
|
||||
inverse_on_surface: string
|
||||
inverse_primary: string
|
||||
inverse_surface: string
|
||||
on_background: string
|
||||
on_error: string
|
||||
on_error_container: string
|
||||
on_primary: string
|
||||
on_primary_container: string
|
||||
on_primary_fixed: string
|
||||
on_primary_fixed_variant: string
|
||||
on_secondary: string
|
||||
on_secondary_container: string
|
||||
on_secondary_fixed: string
|
||||
on_secondary_fixed_variant: string
|
||||
on_surface: string
|
||||
on_surface_variant: string
|
||||
on_tertiary: string
|
||||
on_tertiary_container: string
|
||||
on_tertiary_fixed: string
|
||||
on_tertiary_fixed_variant: string
|
||||
outline: string
|
||||
outline_variant: string
|
||||
primary: string
|
||||
primary_container: string
|
||||
primary_fixed: string
|
||||
primary_fixed_dim: string
|
||||
scrim: string
|
||||
secondary: string
|
||||
secondary_container: string
|
||||
secondary_fixed: string
|
||||
secondary_fixed_dim: string
|
||||
shadow: string
|
||||
surface: string
|
||||
surface_bright: string
|
||||
surface_container: string
|
||||
surface_container_high: string
|
||||
surface_container_highest: string
|
||||
surface_container_low: string
|
||||
surface_container_lowest: string
|
||||
surface_dim: string
|
||||
surface_variant: string
|
||||
tertiary: string
|
||||
tertiary_container: string
|
||||
tertiary_fixed: string
|
||||
tertiary_fixed_dim: string
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
import options from "options"
|
||||
const notifs = await Service.import("notifications")
|
||||
|
||||
// TODO: consider adding this to upstream
|
||||
|
||||
const { blacklist } = options.notifications
|
||||
|
||||
export default function init() {
|
||||
const notify = notifs.constructor.prototype.Notify.bind(notifs)
|
||||
notifs.constructor.prototype.Notify = function(appName: string, ...rest: unknown[]) {
|
||||
if (blacklist.value.includes(appName))
|
||||
return Number.MAX_SAFE_INTEGER
|
||||
|
||||
return notify(appName, ...rest)
|
||||
}
|
||||
}
|
|
@ -1,115 +0,0 @@
|
|||
import { Variable } from "resource:///com/github/Aylur/ags/variable.js"
|
||||
|
||||
type OptProps = {
|
||||
persistent?: boolean
|
||||
}
|
||||
|
||||
export class Opt<T = unknown> extends Variable<T> {
|
||||
static { Service.register(this) }
|
||||
|
||||
constructor(initial: T, { persistent = false }: OptProps = {}) {
|
||||
super(initial)
|
||||
this.initial = initial
|
||||
this.persistent = persistent
|
||||
}
|
||||
|
||||
initial: T
|
||||
id = ""
|
||||
persistent: boolean
|
||||
toString() { return `${this.value}` }
|
||||
toJSON() { return `opt:${this.value}` }
|
||||
|
||||
getValue = (): T => {
|
||||
return super.getValue()
|
||||
}
|
||||
|
||||
init(cacheFile: string) {
|
||||
const cacheV = JSON.parse(Utils.readFile(cacheFile) || "{}")[this.id]
|
||||
if (cacheV !== undefined)
|
||||
this.value = cacheV
|
||||
|
||||
this.connect("changed", () => {
|
||||
const cache = JSON.parse(Utils.readFile(cacheFile) || "{}")
|
||||
cache[this.id] = this.value
|
||||
Utils.writeFileSync(JSON.stringify(cache, null, 2), cacheFile)
|
||||
})
|
||||
}
|
||||
|
||||
reset() {
|
||||
if (this.persistent)
|
||||
return
|
||||
|
||||
if (JSON.stringify(this.value) !== JSON.stringify(this.initial)) {
|
||||
this.value = this.initial
|
||||
return this.id
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export const opt = <T>(initial: T, opts?: OptProps) => new Opt(initial, opts)
|
||||
|
||||
function getOptions(object: object, path = ""): Opt[] {
|
||||
return Object.keys(object).flatMap(key => {
|
||||
const obj: Opt = object[key]
|
||||
const id = path ? path + "." + key : key
|
||||
|
||||
if (obj instanceof Variable) {
|
||||
obj.id = id
|
||||
return obj
|
||||
}
|
||||
|
||||
if (typeof obj === "object")
|
||||
return getOptions(obj, id)
|
||||
|
||||
return []
|
||||
})
|
||||
}
|
||||
|
||||
export function mkOptions<T extends object>(cacheFile: string, object: T) {
|
||||
for (const opt of getOptions(object))
|
||||
opt.init(cacheFile)
|
||||
|
||||
Utils.ensureDirectory(cacheFile.split("/").slice(0, -1).join("/"))
|
||||
|
||||
const configFile = `${TMP}/config.json`
|
||||
const values = getOptions(object).reduce((obj, { id, value }) => ({ [id]: value, ...obj }), {})
|
||||
Utils.writeFileSync(JSON.stringify(values, null, 2), configFile)
|
||||
Utils.monitorFile(configFile, () => {
|
||||
const cache = JSON.parse(Utils.readFile(configFile) || "{}")
|
||||
for (const opt of getOptions(object)) {
|
||||
if (JSON.stringify(cache[opt.id]) !== JSON.stringify(opt.value))
|
||||
opt.value = cache[opt.id]
|
||||
}
|
||||
})
|
||||
|
||||
function sleep(ms = 0) {
|
||||
return new Promise(r => setTimeout(r, ms))
|
||||
}
|
||||
|
||||
async function reset(
|
||||
[opt, ...list] = getOptions(object),
|
||||
id = opt?.reset(),
|
||||
): Promise<Array<string>> {
|
||||
if (!opt)
|
||||
return sleep().then(() => [])
|
||||
|
||||
return id
|
||||
? [id, ...(await sleep(50).then(() => reset(list)))]
|
||||
: await sleep().then(() => reset(list))
|
||||
}
|
||||
|
||||
return Object.assign(object, {
|
||||
configFile,
|
||||
array: () => getOptions(object),
|
||||
async reset() {
|
||||
return (await reset()).join("\n")
|
||||
},
|
||||
handler(deps: string[], callback: () => void) {
|
||||
for (const opt of getOptions(object)) {
|
||||
if (deps.some(i => opt.id.startsWith(i)))
|
||||
opt.connect("changed", callback)
|
||||
}
|
||||
},
|
||||
})
|
||||
}
|
||||
|
|
@ -1,16 +0,0 @@
|
|||
import GLib from "gi://GLib?version=2.0"
|
||||
|
||||
declare global {
|
||||
const OPTIONS: string
|
||||
const TMP: string
|
||||
const USER: string
|
||||
}
|
||||
|
||||
Object.assign(globalThis, {
|
||||
OPTIONS: `${GLib.get_user_cache_dir()}/ags/options.json`,
|
||||
TMP: `${GLib.get_tmp_dir()}/asztal`,
|
||||
USER: GLib.get_user_name(),
|
||||
})
|
||||
|
||||
Utils.ensureDirectory(TMP)
|
||||
App.addIcons(`${App.configDir}/assets`)
|
|
@ -1,14 +0,0 @@
|
|||
import options from "options"
|
||||
import { sh } from "./utils"
|
||||
|
||||
export async function tmux() {
|
||||
const { scheme, dark, light } = options.theme
|
||||
const hex = scheme.value === "dark" ? dark.primary.bg.value : light.primary.bg.value
|
||||
if (await sh("which tmux").catch(() => false))
|
||||
sh(`tmux set @main_accent "${hex}"`)
|
||||
}
|
||||
|
||||
export default function init() {
|
||||
options.theme.dark.primary.bg.connect("changed", tmux)
|
||||
options.theme.light.primary.bg.connect("changed", tmux)
|
||||
}
|
|
@ -1,113 +0,0 @@
|
|||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||
import { type Application } from "types/service/applications"
|
||||
import icons, { substitutes } from "./icons"
|
||||
import Gtk from "gi://Gtk?version=3.0"
|
||||
import Gdk from "gi://Gdk"
|
||||
import GLib from "gi://GLib?version=2.0"
|
||||
|
||||
export type Binding<T> = import("types/service").Binding<any, any, T>
|
||||
|
||||
/**
|
||||
* @returns substitute icon || name || fallback icon
|
||||
*/
|
||||
export function icon(name: string | null, fallback = icons.missing) {
|
||||
if (!name)
|
||||
return fallback || ""
|
||||
|
||||
if (GLib.file_test(name, GLib.FileTest.EXISTS))
|
||||
return name
|
||||
|
||||
const icon = (substitutes[name] || name)
|
||||
if (Utils.lookUpIcon(icon))
|
||||
return icon
|
||||
|
||||
print(`no icon substitute "${icon}" for "${name}", fallback: "${fallback}"`)
|
||||
return fallback
|
||||
}
|
||||
|
||||
/**
|
||||
* @returns execAsync(["bash", "-c", cmd])
|
||||
*/
|
||||
export async function bash(strings: TemplateStringsArray | string, ...values: unknown[]) {
|
||||
const cmd = typeof strings === "string" ? strings : strings
|
||||
.flatMap((str, i) => str + `${values[i] ?? ""}`)
|
||||
.join("")
|
||||
|
||||
return Utils.execAsync(["bash", "-c", cmd]).catch(err => {
|
||||
console.error(cmd, err)
|
||||
return ""
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* @returns execAsync(cmd)
|
||||
*/
|
||||
export async function sh(cmd: string | string[]) {
|
||||
return Utils.execAsync(cmd).catch(err => {
|
||||
console.error(typeof cmd === "string" ? cmd : cmd.join(" "), err)
|
||||
return ""
|
||||
})
|
||||
}
|
||||
|
||||
export function forMonitors(widget: (monitor: number) => Gtk.Window) {
|
||||
const n = Gdk.Display.get_default()?.get_n_monitors() || 1
|
||||
return range(n, 0).flatMap(widget)
|
||||
}
|
||||
|
||||
/**
|
||||
* @returns [start...length]
|
||||
*/
|
||||
export function range(length: number, start = 1) {
|
||||
return Array.from({ length }, (_, i) => i + start)
|
||||
}
|
||||
|
||||
/**
|
||||
* @returns true if all of the `bins` are found
|
||||
*/
|
||||
export function dependencies(...bins: string[]) {
|
||||
const missing = bins.filter(bin => Utils.exec({
|
||||
cmd: `which ${bin}`,
|
||||
out: () => false,
|
||||
err: () => true,
|
||||
}))
|
||||
|
||||
if (missing.length > 0) {
|
||||
console.warn(Error(`missing dependencies: ${missing.join(", ")}`))
|
||||
Utils.notify(`missing dependencies: ${missing.join(", ")}`)
|
||||
}
|
||||
|
||||
return missing.length === 0
|
||||
}
|
||||
|
||||
/**
|
||||
* run app detached
|
||||
*/
|
||||
export function launchApp(app: Application) {
|
||||
const exe = app.executable
|
||||
.split(/\s+/)
|
||||
.filter(str => !str.startsWith("%") && !str.startsWith("@"))
|
||||
.join(" ")
|
||||
|
||||
bash(`${exe} &`)
|
||||
app.frequency += 1
|
||||
}
|
||||
|
||||
/**
|
||||
* to use with drag and drop
|
||||
*/
|
||||
export function createSurfaceFromWidget(widget: Gtk.Widget) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
const cairo = imports.gi.cairo as any
|
||||
const alloc = widget.get_allocation()
|
||||
const surface = new cairo.ImageSurface(
|
||||
cairo.Format.ARGB32,
|
||||
alloc.width,
|
||||
alloc.height,
|
||||
)
|
||||
const cr = new cairo.Context(surface)
|
||||
cr.setSourceRGBA(255, 255, 255, 0)
|
||||
cr.rectangle(0, 0, alloc.width, alloc.height)
|
||||
cr.fill()
|
||||
widget.draw(cr)
|
||||
return surface
|
||||
}
|
|
@ -1,42 +0,0 @@
|
|||
import GLib from "gi://GLib"
|
||||
// import options from "options"
|
||||
//
|
||||
// const intval = options.system.fetchInterval.value
|
||||
// const tempPath = options.system.temperature.value
|
||||
|
||||
export const clock = Variable(GLib.DateTime.new_now_local(), {
|
||||
poll: [1000, () => GLib.DateTime.new_now_local()],
|
||||
})
|
||||
|
||||
export const uptime = Variable(0, {
|
||||
poll: [60_000, "cat /proc/uptime", line =>
|
||||
Number.parseInt(line.split(".")[0]) / 60,
|
||||
],
|
||||
})
|
||||
|
||||
export const distro = {
|
||||
id: GLib.get_os_info("ID"),
|
||||
logo: GLib.get_os_info("LOGO"),
|
||||
}
|
||||
|
||||
// const divide = ([total, free]: string[]) => Number.parseInt(free) / Number.parseInt(total)
|
||||
//
|
||||
// export const cpu = Variable(0, {
|
||||
// poll: [intval, "top -b -n 1", out => divide(["100", out.split("\n")
|
||||
// .find(line => line.includes("Cpu(s)"))
|
||||
// ?.split(/\s+/)[1]
|
||||
// .replace(",", ".") || "0"])],
|
||||
// })
|
||||
//
|
||||
// export const ram = Variable(0, {
|
||||
// poll: [intval, "free", out => divide(out.split("\n")
|
||||
// .find(line => line.includes("Mem:"))
|
||||
// ?.split(/\s+/)
|
||||
// .splice(1, 2) || ["1", "1"])],
|
||||
// })
|
||||
//
|
||||
// export const temperature = Variable(0, {
|
||||
// poll: [intval, `cat ${tempPath}`, n => {
|
||||
// return Number.parseInt(n) / 100_000
|
||||
// }],
|
||||
// })
|
|
@ -1,41 +0,0 @@
|
|||
import "lib/session"
|
||||
import "style/style"
|
||||
import init from "lib/init"
|
||||
import options from "options"
|
||||
import Bar from "widget/bar/Bar"
|
||||
import Launcher from "widget/launcher/Launcher"
|
||||
import NotificationPopups from "widget/notifications/NotificationPopups"
|
||||
import OSD from "widget/osd/OSD"
|
||||
import Overview from "widget/overview/Overview"
|
||||
import PowerMenu from "widget/powermenu/PowerMenu"
|
||||
import ScreenCorners from "widget/bar/ScreenCorners"
|
||||
import SettingsDialog from "widget/settings/SettingsDialog"
|
||||
import Verification from "widget/powermenu/Verification"
|
||||
import { forMonitors } from "lib/utils"
|
||||
import { setupQuickSettings } from "widget/quicksettings/QuickSettings"
|
||||
import { setupDateMenu } from "widget/datemenu/DateMenu"
|
||||
|
||||
App.config({
|
||||
onConfigParsed: () => {
|
||||
setupQuickSettings()
|
||||
setupDateMenu()
|
||||
init()
|
||||
},
|
||||
closeWindowDelay: {
|
||||
"launcher": options.transition.value,
|
||||
"overview": options.transition.value,
|
||||
"quicksettings": options.transition.value,
|
||||
"datemenu": options.transition.value,
|
||||
},
|
||||
windows: () => [
|
||||
...forMonitors(Bar),
|
||||
...forMonitors(NotificationPopups),
|
||||
...forMonitors(ScreenCorners),
|
||||
...forMonitors(OSD),
|
||||
Launcher(),
|
||||
Overview(),
|
||||
PowerMenu(),
|
||||
SettingsDialog(),
|
||||
Verification(),
|
||||
],
|
||||
})
|
|
@ -1,243 +0,0 @@
|
|||
import { opt, mkOptions } from "lib/option"
|
||||
import { distro } from "lib/variables"
|
||||
import { icon } from "lib/utils"
|
||||
import icons from "lib/icons"
|
||||
|
||||
const options = mkOptions(OPTIONS, {
|
||||
autotheme: opt(false),
|
||||
|
||||
wallpaper: {
|
||||
resolution: opt<import("service/wallpaper").Resolution>(1920),
|
||||
market: opt<import("service/wallpaper").Market>("random"),
|
||||
},
|
||||
|
||||
theme: {
|
||||
dark: {
|
||||
primary: {
|
||||
bg: opt("#51a4e7"),
|
||||
fg: opt("#141414"),
|
||||
},
|
||||
error: {
|
||||
bg: opt("#e55f86"),
|
||||
fg: opt("#141414"),
|
||||
},
|
||||
bg: opt("#171717"),
|
||||
fg: opt("#eeeeee"),
|
||||
widget: opt("#eeeeee"),
|
||||
border: opt("#eeeeee"),
|
||||
},
|
||||
light: {
|
||||
primary: {
|
||||
bg: opt("#426ede"),
|
||||
fg: opt("#eeeeee"),
|
||||
},
|
||||
error: {
|
||||
bg: opt("#b13558"),
|
||||
fg: opt("#eeeeee"),
|
||||
},
|
||||
bg: opt("#fffffa"),
|
||||
fg: opt("#080808"),
|
||||
widget: opt("#080808"),
|
||||
border: opt("#080808"),
|
||||
},
|
||||
|
||||
blur: opt(0),
|
||||
scheme: opt<"dark" | "light">("dark"),
|
||||
widget: { opacity: opt(94) },
|
||||
border: {
|
||||
width: opt(1),
|
||||
opacity: opt(96),
|
||||
},
|
||||
|
||||
shadows: opt(true),
|
||||
padding: opt(7),
|
||||
spacing: opt(12),
|
||||
radius: opt(11),
|
||||
},
|
||||
|
||||
transition: opt(200),
|
||||
|
||||
font: {
|
||||
size: opt(13),
|
||||
name: opt("Ubuntu Nerd Font"),
|
||||
},
|
||||
|
||||
bar: {
|
||||
flatButtons: opt(true),
|
||||
position: opt<"top" | "bottom">("top"),
|
||||
corners: opt(50),
|
||||
transparent: opt(false),
|
||||
layout: {
|
||||
start: opt<Array<import("widget/bar/Bar").BarWidget>>([
|
||||
"launcher",
|
||||
"workspaces",
|
||||
"taskbar",
|
||||
"expander",
|
||||
"messages",
|
||||
]),
|
||||
center: opt<Array<import("widget/bar/Bar").BarWidget>>([
|
||||
"date",
|
||||
]),
|
||||
end: opt<Array<import("widget/bar/Bar").BarWidget>>([
|
||||
"media",
|
||||
"expander",
|
||||
"systray",
|
||||
"colorpicker",
|
||||
"screenrecord",
|
||||
"system",
|
||||
"battery",
|
||||
"powermenu",
|
||||
]),
|
||||
},
|
||||
launcher: {
|
||||
icon: {
|
||||
colored: opt(true),
|
||||
icon: opt(icon(distro.logo, icons.ui.search)),
|
||||
},
|
||||
label: {
|
||||
colored: opt(false),
|
||||
label: opt(" Applications"),
|
||||
},
|
||||
action: opt(() => App.toggleWindow("launcher")),
|
||||
},
|
||||
date: {
|
||||
format: opt("%H:%M - %A %e."),
|
||||
action: opt(() => App.toggleWindow("datemenu")),
|
||||
},
|
||||
battery: {
|
||||
bar: opt<"hidden" | "regular" | "whole">("regular"),
|
||||
charging: opt("#00D787"),
|
||||
percentage: opt(true),
|
||||
blocks: opt(7),
|
||||
width: opt(50),
|
||||
low: opt(30),
|
||||
},
|
||||
workspaces: {
|
||||
workspaces: opt(7),
|
||||
},
|
||||
taskbar: {
|
||||
iconSize: opt(0),
|
||||
monochrome: opt(true),
|
||||
exclusive: opt(false),
|
||||
},
|
||||
messages: {
|
||||
action: opt(() => App.toggleWindow("datemenu")),
|
||||
},
|
||||
systray: {
|
||||
ignore: opt([
|
||||
"KDE Connect Indicator",
|
||||
"spotify-client",
|
||||
]),
|
||||
},
|
||||
media: {
|
||||
monochrome: opt(true),
|
||||
preferred: opt("spotify"),
|
||||
direction: opt<"left" | "right">("right"),
|
||||
format: opt("{artists} - {title}"),
|
||||
length: opt(40),
|
||||
},
|
||||
powermenu: {
|
||||
monochrome: opt(false),
|
||||
action: opt(() => App.toggleWindow("powermenu")),
|
||||
},
|
||||
},
|
||||
|
||||
launcher: {
|
||||
width: opt(0),
|
||||
margin: opt(80),
|
||||
nix: {
|
||||
pkgs: opt("nixpkgs/nixos-unstable"),
|
||||
max: opt(8),
|
||||
},
|
||||
sh: {
|
||||
max: opt(16),
|
||||
},
|
||||
apps: {
|
||||
iconSize: opt(62),
|
||||
max: opt(6),
|
||||
favorites: opt([
|
||||
[
|
||||
"firefox",
|
||||
"wezterm",
|
||||
"org.gnome.Nautilus",
|
||||
"org.gnome.Calendar",
|
||||
"spotify",
|
||||
],
|
||||
]),
|
||||
},
|
||||
},
|
||||
|
||||
overview: {
|
||||
scale: opt(9),
|
||||
workspaces: opt(7),
|
||||
monochromeIcon: opt(true),
|
||||
},
|
||||
|
||||
powermenu: {
|
||||
sleep: opt("systemctl suspend"),
|
||||
reboot: opt("systemctl reboot"),
|
||||
logout: opt("pkill Hyprland"),
|
||||
shutdown: opt("shutdown now"),
|
||||
layout: opt<"line" | "box">("line"),
|
||||
labels: opt(true),
|
||||
},
|
||||
|
||||
quicksettings: {
|
||||
avatar: {
|
||||
image: opt(`/var/lib/AccountsService/icons/${Utils.USER}`),
|
||||
size: opt(70),
|
||||
},
|
||||
width: opt(380),
|
||||
position: opt<"left" | "center" | "right">("right"),
|
||||
networkSettings: opt("gtk-launch gnome-control-center"),
|
||||
media: {
|
||||
monochromeIcon: opt(true),
|
||||
coverSize: opt(100),
|
||||
},
|
||||
},
|
||||
|
||||
datemenu: {
|
||||
position: opt<"left" | "center" | "right">("center"),
|
||||
weather: {
|
||||
interval: opt(60_000),
|
||||
unit: opt<"metric" | "imperial" | "standard">("metric"),
|
||||
key: opt<string>(
|
||||
JSON.parse(Utils.readFile(`${App.configDir}/.weather`) || "{}")?.key || "",
|
||||
),
|
||||
cities: opt<Array<number>>(
|
||||
JSON.parse(Utils.readFile(`${App.configDir}/.weather`) || "{}")?.cities || [],
|
||||
),
|
||||
},
|
||||
},
|
||||
|
||||
osd: {
|
||||
progress: {
|
||||
vertical: opt(true),
|
||||
pack: {
|
||||
h: opt<"start" | "center" | "end">("end"),
|
||||
v: opt<"start" | "center" | "end">("center"),
|
||||
},
|
||||
},
|
||||
microphone: {
|
||||
pack: {
|
||||
h: opt<"start" | "center" | "end">("center"),
|
||||
v: opt<"start" | "center" | "end">("end"),
|
||||
},
|
||||
},
|
||||
},
|
||||
|
||||
notifications: {
|
||||
position: opt<Array<"top" | "bottom" | "left" | "right">>(["top", "right"]),
|
||||
blacklist: opt(["Spotify"]),
|
||||
width: opt(440),
|
||||
},
|
||||
|
||||
hyprland: {
|
||||
gaps: opt(2.4),
|
||||
inactiveBorder: opt("#282828"),
|
||||
gapsWhenOnly: opt(false),
|
||||
},
|
||||
})
|
||||
|
||||
globalThis["options"] = options
|
||||
export default options
|
|
@ -1,19 +0,0 @@
|
|||
{
|
||||
"name": "ags-dotfiles",
|
||||
"author": "Aylur",
|
||||
"kofi": "https://ko-fi.com/aylur",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/Aylur/dotfiles.git"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@girs/accountsservice-1.0": "^1.0.0-3.2.7",
|
||||
"@typescript-eslint/eslint-plugin": "^6.20.0",
|
||||
"eslint": "^8.56.0",
|
||||
"eslint-config-standard-with-typescript": "^43.0.1",
|
||||
"eslint-plugin-import": "^2.29.1",
|
||||
"eslint-plugin-n": "^16.6.2",
|
||||
"eslint-plugin-promise": "^6.1.1",
|
||||
"typescript": "^5.3.3"
|
||||
}
|
||||
}
|
|
@ -1,55 +0,0 @@
|
|||
import { sh } from "lib/utils"
|
||||
|
||||
type Profile = "Performance" | "Balanced" | "Quiet"
|
||||
type Mode = "Hybrid" | "Integrated"
|
||||
|
||||
class Asusctl extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"profile": ["string", "r"],
|
||||
"mode": ["string", "r"],
|
||||
})
|
||||
}
|
||||
|
||||
get available() {
|
||||
return Utils.exec("which asusctl", () => true, () => false)
|
||||
}
|
||||
|
||||
#profile: Profile = "Balanced"
|
||||
#mode: Mode = "Hybrid"
|
||||
|
||||
async nextProfile() {
|
||||
await sh("asusctl profile -n")
|
||||
const profile = await sh("asusctl profile -p")
|
||||
const p = profile.split(" ")[3] as Profile
|
||||
this.#profile = p
|
||||
this.changed("profile")
|
||||
}
|
||||
|
||||
async setProfile(prof: Profile) {
|
||||
await sh(`asusctl profile --profile-set ${prof}`)
|
||||
this.#profile = prof
|
||||
this.changed("profile")
|
||||
}
|
||||
|
||||
async nextMode() {
|
||||
await sh(`supergfxctl -m ${this.#mode === "Hybrid" ? "Integrated" : "Hybrid"}`)
|
||||
this.#mode = await sh("supergfxctl -g") as Mode
|
||||
this.changed("profile")
|
||||
}
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
|
||||
if (this.available) {
|
||||
sh("asusctl profile -p").then(p => this.#profile = p.split(" ")[3] as Profile)
|
||||
sh("supergfxctl -g").then(m => this.#mode = m as Mode)
|
||||
}
|
||||
}
|
||||
|
||||
get profiles(): Profile[] { return ["Performance", "Balanced", "Quiet"] }
|
||||
get profile() { return this.#profile }
|
||||
get mode() { return this.#mode }
|
||||
}
|
||||
|
||||
export default new Asusctl
|
|
@ -1,69 +0,0 @@
|
|||
import { bash, dependencies, sh } from "lib/utils"
|
||||
|
||||
if (!dependencies("brightnessctl"))
|
||||
App.quit()
|
||||
|
||||
const get = (args: string) => Number(Utils.exec(`brightnessctl ${args}`))
|
||||
const screen = await bash`ls -w1 /sys/class/backlight | head -1`
|
||||
const kbd = await bash`ls -w1 /sys/class/leds | head -1`
|
||||
|
||||
class Brightness extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"screen": ["float", "rw"],
|
||||
"kbd": ["int", "rw"],
|
||||
})
|
||||
}
|
||||
|
||||
#kbdMax = get(`--device ${kbd} max`)
|
||||
#kbd = get(`--device ${kbd} get`)
|
||||
#screenMax = get("max")
|
||||
#screen = get("get") / (get("max") || 1)
|
||||
|
||||
get kbd() { return this.#kbd }
|
||||
get screen() { return this.#screen }
|
||||
|
||||
set kbd(value) {
|
||||
if (value < 0 || value > this.#kbdMax)
|
||||
return
|
||||
|
||||
sh(`brightnessctl -d ${kbd} s ${value} -q`).then(() => {
|
||||
this.#kbd = value
|
||||
this.changed("kbd")
|
||||
})
|
||||
}
|
||||
|
||||
set screen(percent) {
|
||||
if (percent < 0)
|
||||
percent = 0
|
||||
|
||||
if (percent > 1)
|
||||
percent = 1
|
||||
|
||||
sh(`brightnessctl set ${Math.floor(percent * 100)}% -q`).then(() => {
|
||||
this.#screen = percent
|
||||
this.changed("screen")
|
||||
})
|
||||
}
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
|
||||
const screenPath = `/sys/class/backlight/${screen}/brightness`
|
||||
const kbdPath = `/sys/class/leds/${kbd}/brightness`
|
||||
|
||||
Utils.monitorFile(screenPath, async f => {
|
||||
const v = await Utils.readFileAsync(f)
|
||||
this.#screen = Number(v) / this.#screenMax
|
||||
this.changed("screen")
|
||||
})
|
||||
|
||||
Utils.monitorFile(kbdPath, async f => {
|
||||
const v = await Utils.readFileAsync(f)
|
||||
this.#kbd = Number(v) / this.#kbdMax
|
||||
this.changed("kbd")
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
export default new Brightness
|
|
@ -1,56 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import { bash, dependencies } from "lib/utils"
|
||||
|
||||
const COLORS_CACHE = Utils.CACHE_DIR + "/colorpicker.json"
|
||||
const MAX_NUM_COLORS = 10
|
||||
|
||||
class ColorPicker extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"colors": ["jsobject"],
|
||||
})
|
||||
}
|
||||
|
||||
#notifID = 0
|
||||
#colors = JSON.parse(Utils.readFile(COLORS_CACHE) || "[]") as string[]
|
||||
|
||||
get colors() { return [...this.#colors] }
|
||||
set colors(colors) {
|
||||
this.#colors = colors
|
||||
this.changed("colors")
|
||||
}
|
||||
|
||||
// TODO: doesn't work?
|
||||
async wlCopy(color: string) {
|
||||
if (dependencies("wl-copy"))
|
||||
bash(`wl-copy ${color}`)
|
||||
}
|
||||
|
||||
readonly pick = async () => {
|
||||
if (!dependencies("hyprpicker"))
|
||||
return
|
||||
|
||||
const color = await bash("hyprpicker -a -r")
|
||||
if (!color)
|
||||
return
|
||||
|
||||
this.wlCopy(color)
|
||||
const list = this.colors
|
||||
if (!list.includes(color)) {
|
||||
list.push(color)
|
||||
if (list.length > MAX_NUM_COLORS)
|
||||
list.shift()
|
||||
|
||||
this.colors = list
|
||||
Utils.writeFile(JSON.stringify(list, null, 2), COLORS_CACHE)
|
||||
}
|
||||
|
||||
this.#notifID = await Utils.notify({
|
||||
id: this.#notifID,
|
||||
iconName: icons.ui.colorpicker,
|
||||
summary: color,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
export default new ColorPicker
|
|
@ -1,110 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import { bash, dependencies } from "lib/utils"
|
||||
import options from "options"
|
||||
|
||||
const CACHE = `${Utils.CACHE_DIR}/nixpkgs`
|
||||
const PREFIX = "legacyPackages.x86_64-linux."
|
||||
const MAX = options.launcher.nix.max
|
||||
const nixpkgs = options.launcher.nix.pkgs
|
||||
|
||||
export type Nixpkg = {
|
||||
name: string
|
||||
description: string
|
||||
pname: string
|
||||
version: string
|
||||
}
|
||||
|
||||
class Nix extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"available": ["boolean", "r"],
|
||||
"ready": ["boolean", "rw"],
|
||||
})
|
||||
}
|
||||
|
||||
#db: { [name: string]: Nixpkg } = {}
|
||||
#ready = true
|
||||
|
||||
private set ready(r: boolean) {
|
||||
this.#ready = r
|
||||
this.changed("ready")
|
||||
}
|
||||
|
||||
get db() { return this.#db }
|
||||
get ready() { return this.#ready }
|
||||
get available() {
|
||||
return Utils.exec("which nix", () => true, () => false)
|
||||
}
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
if (!this.available)
|
||||
return this
|
||||
|
||||
this.#updateList()
|
||||
nixpkgs.connect("changed", this.#updateList)
|
||||
}
|
||||
|
||||
query = async (filter: string) => {
|
||||
if (!dependencies("fzf", "nix") || !this.#ready)
|
||||
return [] as string[]
|
||||
|
||||
return bash(`cat ${CACHE} | fzf -f ${filter} -e | head -n ${MAX} `)
|
||||
.then(str => str.split("\n").filter(i => i))
|
||||
}
|
||||
|
||||
nix(cmd: string, bin: string, args: string) {
|
||||
return Utils.execAsync(`nix ${cmd} ${nixpkgs}#${bin} --impure ${args}`)
|
||||
}
|
||||
|
||||
run = async (input: string) => {
|
||||
if (!dependencies("nix"))
|
||||
return
|
||||
|
||||
try {
|
||||
const [bin, ...args] = input.trim().split(/\s+/)
|
||||
|
||||
this.ready = false
|
||||
await this.nix("shell", bin, "--command sh -c 'exit'")
|
||||
this.ready = true
|
||||
|
||||
this.nix("run", bin, ["--", ...args].join(" "))
|
||||
} catch (err) {
|
||||
if (typeof err === "string")
|
||||
Utils.notify("NixRun Error", err, icons.nix.nix)
|
||||
else
|
||||
logError(err)
|
||||
} finally {
|
||||
this.ready = true
|
||||
}
|
||||
}
|
||||
|
||||
#updateList = async () => {
|
||||
if (!dependencies("nix"))
|
||||
return
|
||||
|
||||
this.ready = false
|
||||
this.#db = {}
|
||||
|
||||
const search = await bash(`nix search ${nixpkgs} --json`)
|
||||
if (!search) {
|
||||
this.ready = true
|
||||
return
|
||||
}
|
||||
|
||||
const json = Object.entries(JSON.parse(search) as {
|
||||
[name: string]: Nixpkg
|
||||
})
|
||||
|
||||
for (const [pkg, info] of json) {
|
||||
const name = pkg.replace(PREFIX, "")
|
||||
this.#db[name] = { ...info, name }
|
||||
}
|
||||
|
||||
const list = Object.keys(this.#db).join("\n")
|
||||
await Utils.writeFile(list, CACHE)
|
||||
this.ready = true
|
||||
}
|
||||
}
|
||||
|
||||
export default new Nix
|
|
@ -1,47 +0,0 @@
|
|||
import options from "options"
|
||||
|
||||
const { sleep, reboot, logout, shutdown } = options.powermenu
|
||||
|
||||
export type Action = "sleep" | "reboot" | "logout" | "shutdown"
|
||||
|
||||
class PowerMenu extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"title": ["string"],
|
||||
"cmd": ["string"],
|
||||
})
|
||||
}
|
||||
|
||||
#title = ""
|
||||
#cmd = ""
|
||||
|
||||
get title() { return this.#title }
|
||||
|
||||
action(action: Action) {
|
||||
[this.#cmd, this.#title] = {
|
||||
sleep: [sleep.value, "Sleep"],
|
||||
reboot: [reboot.value, "Reboot"],
|
||||
logout: [logout.value, "Log Out"],
|
||||
shutdown: [shutdown.value, "Shutdown"],
|
||||
}[action]
|
||||
|
||||
this.notify("cmd")
|
||||
this.notify("title")
|
||||
this.emit("changed")
|
||||
App.closeWindow("powermenu")
|
||||
App.openWindow("verification")
|
||||
}
|
||||
|
||||
readonly shutdown = () => {
|
||||
this.action("shutdown")
|
||||
}
|
||||
|
||||
readonly exec = () => {
|
||||
App.closeWindow("verification")
|
||||
Utils.exec(this.#cmd)
|
||||
}
|
||||
}
|
||||
|
||||
const powermenu = new PowerMenu
|
||||
Object.assign(globalThis, { powermenu })
|
||||
export default powermenu
|
|
@ -1,102 +0,0 @@
|
|||
import GLib from "gi://GLib"
|
||||
import icons from "lib/icons"
|
||||
import { dependencies, sh, bash } from "lib/utils"
|
||||
|
||||
const now = () => GLib.DateTime.new_now_local().format("%Y-%m-%d_%H-%M-%S")
|
||||
|
||||
class Recorder extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"timer": ["int"],
|
||||
"recording": ["boolean"],
|
||||
})
|
||||
}
|
||||
|
||||
#recordings = Utils.HOME + "/Videos/Screencasting"
|
||||
#screenshots = Utils.HOME + "/Pictures/Screenshots"
|
||||
#file = ""
|
||||
#interval = 0
|
||||
|
||||
recording = false
|
||||
timer = 0
|
||||
|
||||
async start() {
|
||||
if (!dependencies("slurp", "wf-recorder"))
|
||||
return
|
||||
|
||||
if (this.recording)
|
||||
return
|
||||
|
||||
Utils.ensureDirectory(this.#recordings)
|
||||
this.#file = `${this.#recordings}/${now()}.mp4`
|
||||
sh(`wf-recorder -g "${await sh("slurp")}" -f ${this.#file} --pixel-format yuv420p`)
|
||||
|
||||
this.recording = true
|
||||
this.changed("recording")
|
||||
|
||||
this.timer = 0
|
||||
this.#interval = Utils.interval(1000, () => {
|
||||
this.changed("timer")
|
||||
this.timer++
|
||||
})
|
||||
}
|
||||
|
||||
async stop() {
|
||||
if (!this.recording)
|
||||
return
|
||||
|
||||
await bash("killall -INT wf-recorder")
|
||||
this.recording = false
|
||||
this.changed("recording")
|
||||
GLib.source_remove(this.#interval)
|
||||
|
||||
Utils.notify({
|
||||
iconName: icons.fallback.video,
|
||||
summary: "Screenrecord",
|
||||
body: this.#file,
|
||||
actions: {
|
||||
"Show in Files": () => sh(`xdg-open ${this.#recordings}`),
|
||||
"View": () => sh(`xdg-open ${this.#file}`),
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
async screenshot(full = false) {
|
||||
if (!dependencies("slurp", "wayshot"))
|
||||
return
|
||||
|
||||
const file = `${this.#screenshots}/${now()}.png`
|
||||
Utils.ensureDirectory(this.#screenshots)
|
||||
|
||||
if (full) {
|
||||
await sh(`wayshot -f ${file}`)
|
||||
}
|
||||
else {
|
||||
const size = await sh("slurp")
|
||||
if (!size)
|
||||
return
|
||||
|
||||
await sh(`wayshot -f ${file} -s "${size}"`)
|
||||
}
|
||||
|
||||
bash(`wl-copy < ${file}`)
|
||||
|
||||
Utils.notify({
|
||||
image: file,
|
||||
summary: "Screenshot",
|
||||
body: file,
|
||||
actions: {
|
||||
"Show in Files": () => sh(`xdg-open ${this.#screenshots}`),
|
||||
"View": () => sh(`xdg-open ${file}`),
|
||||
"Edit": () => {
|
||||
if (dependencies("swappy"))
|
||||
sh(`swappy -f ${file}`)
|
||||
},
|
||||
},
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
const recorder = new Recorder
|
||||
Object.assign(globalThis, { recorder })
|
||||
export default recorder
|
|
@ -1,48 +0,0 @@
|
|||
import GLib from "gi://GLib?version=2.0"
|
||||
import { bash, dependencies } from "lib/utils"
|
||||
import icons from "lib/icons"
|
||||
import options from "options"
|
||||
|
||||
const MAX = options.launcher.sh.max
|
||||
const BINS = `${Utils.CACHE_DIR}/binaries`
|
||||
|
||||
async function ls(path: string) {
|
||||
return Utils.execAsync(`ls ${path}`).catch(() => "")
|
||||
}
|
||||
|
||||
async function reload() {
|
||||
const bins = await Promise.all(GLib.getenv("PATH")!
|
||||
.split(":")
|
||||
.map(ls))
|
||||
|
||||
Utils.writeFile(bins.join("\n"), BINS)
|
||||
}
|
||||
|
||||
async function query(filter: string) {
|
||||
if (!dependencies("fzf"))
|
||||
return [] as string[]
|
||||
|
||||
return bash(`cat ${BINS} | fzf -f ${filter} | head -n ${MAX}`)
|
||||
.then(str => Array.from(new Set(str.split("\n").filter(i => i)).values()))
|
||||
.catch(err => { print(err); return [] })
|
||||
}
|
||||
|
||||
function run(args: string) {
|
||||
Utils.execAsync(args)
|
||||
.then(out => {
|
||||
print(`:sh ${args.trim()}:`)
|
||||
print(out)
|
||||
})
|
||||
.catch(err => {
|
||||
Utils.notify("ShRun Error", err, icons.app.terminal)
|
||||
})
|
||||
}
|
||||
|
||||
class Sh extends Service {
|
||||
static { Service.register(this) }
|
||||
constructor() { super(); reload() }
|
||||
query = query
|
||||
run = run
|
||||
}
|
||||
|
||||
export default new Sh
|
|
@ -1,99 +0,0 @@
|
|||
import options from "options"
|
||||
import { dependencies, sh } from "lib/utils"
|
||||
|
||||
export type Resolution = 1920 | 1366 | 3840
|
||||
export type Market =
|
||||
| "random"
|
||||
| "en-US"
|
||||
| "ja-JP"
|
||||
| "en-AU"
|
||||
| "en-GB"
|
||||
| "de-DE"
|
||||
| "en-NZ"
|
||||
| "en-CA"
|
||||
|
||||
const WP = `${Utils.HOME}/.config/background`
|
||||
const Cache = `${Utils.HOME}/Pictures/Wallpapers/Bing`
|
||||
|
||||
class Wallpaper extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"wallpaper": ["string"],
|
||||
})
|
||||
}
|
||||
|
||||
#blockMonitor = false
|
||||
|
||||
#wallpaper() {
|
||||
if (!dependencies("swww"))
|
||||
return
|
||||
|
||||
sh("hyprctl cursorpos").then(pos => {
|
||||
sh([
|
||||
"swww", "img",
|
||||
"--invert-y",
|
||||
"--transition-type", "grow",
|
||||
"--transition-pos", pos.replace(" ", ""),
|
||||
WP,
|
||||
]).then(() => {
|
||||
this.changed("wallpaper")
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
async #setWallpaper(path: string) {
|
||||
this.#blockMonitor = true
|
||||
|
||||
await sh(`cp ${path} ${WP}`)
|
||||
this.#wallpaper()
|
||||
|
||||
this.#blockMonitor = false
|
||||
}
|
||||
|
||||
async #fetchBing() {
|
||||
const res = await Utils.fetch("https://bing.biturl.top/", {
|
||||
params: {
|
||||
resolution: options.wallpaper.resolution.value,
|
||||
format: "json",
|
||||
image_format: "jpg",
|
||||
index: "random",
|
||||
mkt: options.wallpaper.market.value,
|
||||
},
|
||||
}).then(res => res.text())
|
||||
|
||||
if (!res.startsWith("{"))
|
||||
return console.warn("bing api", res)
|
||||
|
||||
const { url } = JSON.parse(res)
|
||||
const file = `${Cache}/${url.replace("https://www.bing.com/th?id=", "")}`
|
||||
|
||||
if (dependencies("curl")) {
|
||||
Utils.ensureDirectory(Cache)
|
||||
await sh(`curl "${url}" --output ${file}`)
|
||||
this.#setWallpaper(file)
|
||||
}
|
||||
}
|
||||
|
||||
readonly random = () => { this.#fetchBing() }
|
||||
readonly set = (path: string) => { this.#setWallpaper(path) }
|
||||
get wallpaper() { return WP }
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
|
||||
if (!dependencies("swww"))
|
||||
return this
|
||||
|
||||
// gtk portal
|
||||
Utils.monitorFile(WP, () => {
|
||||
if (!this.#blockMonitor)
|
||||
this.#wallpaper()
|
||||
})
|
||||
|
||||
Utils.execAsync("swww-daemon")
|
||||
.then(this.#wallpaper)
|
||||
.catch(() => null)
|
||||
}
|
||||
}
|
||||
|
||||
export default new Wallpaper
|
|
@ -1,59 +0,0 @@
|
|||
import options from "options"
|
||||
|
||||
const { interval, key, cities, unit } = options.datemenu.weather
|
||||
|
||||
class Weather extends Service {
|
||||
static {
|
||||
Service.register(this, {}, {
|
||||
"forecasts": ["jsobject"],
|
||||
})
|
||||
}
|
||||
|
||||
#forecasts: Forecast[] = []
|
||||
get forecasts() { return this.#forecasts }
|
||||
|
||||
async #fetch(placeid: number) {
|
||||
const url = "https://api.openweathermap.org/data/2.5/forecast"
|
||||
const res = await Utils.fetch(url, {
|
||||
params: {
|
||||
id: placeid,
|
||||
appid: key.value,
|
||||
untis: unit.value,
|
||||
},
|
||||
})
|
||||
return await res.json()
|
||||
}
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
if (!key.value)
|
||||
return this
|
||||
|
||||
Utils.interval(interval.value, () => {
|
||||
Promise.all(cities.value.map(this.#fetch)).then(forecasts => {
|
||||
this.#forecasts = forecasts as Forecast[]
|
||||
this.changed("forecasts")
|
||||
})
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
export default new Weather
|
||||
|
||||
type Forecast = {
|
||||
city: {
|
||||
name: string,
|
||||
}
|
||||
list: Array<{
|
||||
dt: number
|
||||
main: {
|
||||
temp: number
|
||||
feels_like: number
|
||||
},
|
||||
weather: Array<{
|
||||
main: string,
|
||||
description: string,
|
||||
icon: string,
|
||||
}>
|
||||
}>
|
||||
}
|
|
@ -1,67 +0,0 @@
|
|||
@import './mixins/button.scss';
|
||||
|
||||
* {
|
||||
font-size: $font-size;
|
||||
font-family: $font-name;
|
||||
}
|
||||
|
||||
separator {
|
||||
&.horizontal {
|
||||
min-height: $border-width;
|
||||
}
|
||||
|
||||
&.vertical {
|
||||
min-width: $border-width;
|
||||
}
|
||||
}
|
||||
|
||||
window.popup {
|
||||
>* {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
menu {
|
||||
border-radius: $popover-radius;
|
||||
background-color: $bg;
|
||||
padding: $popover-padding;
|
||||
border: $border-width solid $popover-border-color;
|
||||
|
||||
separator {
|
||||
background-color: $border-color;
|
||||
}
|
||||
|
||||
menuitem {
|
||||
@include button;
|
||||
padding: $spacing * .5;
|
||||
margin: ($spacing * .5) 0;
|
||||
|
||||
&:first-child {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
tooltip {
|
||||
* {
|
||||
all: unset;
|
||||
}
|
||||
|
||||
background-color: transparent;
|
||||
border: none;
|
||||
|
||||
>*>* {
|
||||
background-color: $bg;
|
||||
border-radius: $radius;
|
||||
border: $border-width solid $popover-border-color;
|
||||
color: $fg;
|
||||
padding: 8px;
|
||||
margin: 4px;
|
||||
box-shadow: 0 0 3px 0 $shadow-color;
|
||||
}
|
||||
}
|
|
@ -1,48 +0,0 @@
|
|||
@import './button';
|
||||
|
||||
@mixin accs-button($flat: false, $reactive: true) {
|
||||
@include unset;
|
||||
color: $fg;
|
||||
|
||||
>* {
|
||||
border-radius: $radius;
|
||||
transition: $transition;
|
||||
|
||||
@if $flat {
|
||||
background-color: transparent;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
@else {
|
||||
background-color: $widget-bg;
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@if $reactive {
|
||||
|
||||
&:focus>*,
|
||||
&.focused>* {
|
||||
@include button-focus;
|
||||
}
|
||||
|
||||
&:hover>* {
|
||||
@include button-hover;
|
||||
}
|
||||
|
||||
&:active,
|
||||
&.active,
|
||||
&.on,
|
||||
&:checked {
|
||||
>* {
|
||||
@include button-active;
|
||||
}
|
||||
|
||||
&:hover>* {
|
||||
box-shadow: inset 0 0 0 $border-width $border-color,
|
||||
inset 0 0 0 99px $hover-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,70 +0,0 @@
|
|||
@mixin button-focus() {
|
||||
box-shadow: inset 0 0 0 $border-width $primary-bg;
|
||||
background-color: $hover-bg;
|
||||
color: $hover-fg;
|
||||
}
|
||||
|
||||
@mixin button-hover() {
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
background-color: $hover-bg;
|
||||
color: $hover-fg;
|
||||
}
|
||||
|
||||
@mixin button-active() {
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
background-image: $active-gradient;
|
||||
background-color: $primary-bg;
|
||||
color: $primary-fg;
|
||||
}
|
||||
|
||||
@mixin button-disabled() {
|
||||
box-shadow: none;
|
||||
background-color: transparent;
|
||||
color: transparentize($fg, 0.7);
|
||||
}
|
||||
|
||||
@mixin button($flat: false, $reactive: true, $radius: $radius, $focusable: true) {
|
||||
all: unset;
|
||||
transition: $transition;
|
||||
border-radius: $radius;
|
||||
color: $fg;
|
||||
|
||||
@if $flat {
|
||||
background-color: transparent;
|
||||
background-image: none;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
@else {
|
||||
background-color: $widget-bg;
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
}
|
||||
|
||||
@if $reactive {
|
||||
@if $focusable {
|
||||
&:focus {
|
||||
@include button-focus;
|
||||
}
|
||||
}
|
||||
|
||||
&:hover {
|
||||
@include button-hover;
|
||||
}
|
||||
|
||||
&:active,
|
||||
&.on,
|
||||
&.active,
|
||||
&:checked {
|
||||
@include button-active;
|
||||
|
||||
&:hover {
|
||||
box-shadow: inset 0 0 0 $border-width $border-color,
|
||||
inset 0 0 0 99px $hover-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
&:disabled {
|
||||
@include button-disabled;
|
||||
}
|
||||
}
|
|
@ -1,12 +0,0 @@
|
|||
@mixin floating-widget {
|
||||
@if $shadows {
|
||||
box-shadow: 0 0 5px 0 $shadow-color;
|
||||
}
|
||||
|
||||
margin: max($spacing, 8px);
|
||||
border: $border-width solid $popover-border-color;
|
||||
background-color: $bg;
|
||||
color: $fg;
|
||||
border-radius: $popover-radius;
|
||||
padding: $popover-padding;
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
@mixin hidden {
|
||||
background-color: transparent;
|
||||
background-image: none;
|
||||
border-color: transparent;
|
||||
box-shadow: none;
|
||||
-gtk-icon-transform: scale(0);
|
||||
|
||||
* {
|
||||
background-color: transparent;
|
||||
background-image: none;
|
||||
border-color: transparent;
|
||||
box-shadow: none;
|
||||
-gtk-icon-transform: scale(0);
|
||||
}
|
||||
}
|
|
@ -1,42 +0,0 @@
|
|||
@mixin media() {
|
||||
@include widget;
|
||||
padding: $padding;
|
||||
|
||||
.cover {
|
||||
@if $shadows {
|
||||
box-shadow: 2px 2px 2px 0 $shadow-color;
|
||||
}
|
||||
|
||||
background-size: cover;
|
||||
background-position: center;
|
||||
border-radius: $radius*0.8;
|
||||
margin-right: $spacing;
|
||||
}
|
||||
|
||||
button {
|
||||
@include button($flat: true);
|
||||
padding: $padding * .5;
|
||||
|
||||
&.play-pause {
|
||||
margin: 0 ($spacing * .5);
|
||||
}
|
||||
|
||||
image {
|
||||
font-size: 1.2em;
|
||||
}
|
||||
}
|
||||
|
||||
.artist {
|
||||
color: transparentize($fg, .2);
|
||||
font-size: .9em;
|
||||
}
|
||||
|
||||
scale {
|
||||
@include slider($width: .5em, $slider: false, $gradient: linear-gradient($fg, $fg));
|
||||
margin-bottom: $padding * .5;
|
||||
|
||||
trough {
|
||||
border: none;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,42 +0,0 @@
|
|||
@mixin scrollable($top: false, $bottom: false) {
|
||||
|
||||
@if $top and $shadows {
|
||||
undershoot.top {
|
||||
background: linear-gradient(to bottom, $shadow-color, transparent, transparent, transparent, transparent, transparent);
|
||||
}
|
||||
}
|
||||
|
||||
@if $bottom and $shadows {
|
||||
undershoot.bottom {
|
||||
background: linear-gradient(to top, $shadow-color, transparent, transparent, transparent, transparent, transparent);
|
||||
}
|
||||
}
|
||||
|
||||
scrollbar,
|
||||
scrollbar * {
|
||||
all: unset;
|
||||
}
|
||||
|
||||
scrollbar.vertical {
|
||||
transition: $transition;
|
||||
background-color: transparentize($bg, 0.7);
|
||||
|
||||
&:hover {
|
||||
background-color: transparentize($bg, 0.3);
|
||||
|
||||
slider {
|
||||
background-color: transparentize($fg, 0.3);
|
||||
min-width: .6em;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
scrollbar.vertical slider {
|
||||
background-color: transparentize($fg, 0.5);
|
||||
border-radius: $radius;
|
||||
min-width: .4em;
|
||||
min-height: 2em;
|
||||
transition: $transition;
|
||||
}
|
||||
}
|
|
@ -1,74 +0,0 @@
|
|||
@import './unset';
|
||||
|
||||
@mixin slider($width: 0.7em, $slider-width: .5em, $gradient: $active-gradient, $slider: true, $focusable: true, $radius: $radius) {
|
||||
@include unset($rec: true);
|
||||
|
||||
trough {
|
||||
transition: $transition;
|
||||
border-radius: $radius;
|
||||
border: $border;
|
||||
background-color: $widget-bg;
|
||||
min-height: $width;
|
||||
min-width: $width;
|
||||
|
||||
highlight,
|
||||
progress {
|
||||
border-radius: max($radius - $border-width, 0);
|
||||
background-image: $gradient;
|
||||
min-height: $width;
|
||||
min-width: $width;
|
||||
}
|
||||
}
|
||||
|
||||
slider {
|
||||
box-shadow: none;
|
||||
background-color: transparent;
|
||||
border: $border-width solid transparent;
|
||||
transition: $transition;
|
||||
border-radius: $radius;
|
||||
min-height: $width;
|
||||
min-width: $width;
|
||||
margin: -$slider-width;
|
||||
}
|
||||
|
||||
&:hover {
|
||||
trough {
|
||||
background-color: $hover-bg;
|
||||
}
|
||||
|
||||
slider {
|
||||
@if $slider {
|
||||
background-color: $fg;
|
||||
border-color: $border-color;
|
||||
|
||||
@if $shadows {
|
||||
box-shadow: 0 0 3px 0 $shadow-color;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
&:disabled {
|
||||
|
||||
highlight,
|
||||
progress {
|
||||
background-color: transparentize($fg, 0.4);
|
||||
background-image: none;
|
||||
}
|
||||
}
|
||||
|
||||
@if $focusable {
|
||||
trough:focus {
|
||||
background-color: $hover-bg;
|
||||
box-shadow: inset 0 0 0 $border-width $primary-bg;
|
||||
|
||||
slider {
|
||||
@if $slider {
|
||||
background-color: $fg;
|
||||
box-shadow: inset 0 0 0 $border-width $primary-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
|
@ -1,53 +0,0 @@
|
|||
@mixin spacing($multiplier: 1, $spacing: $spacing, $rec: false) {
|
||||
&.horizontal>* {
|
||||
margin: 0 calc($spacing * $multiplier / 2);
|
||||
|
||||
&:first-child {
|
||||
margin-left: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-right: 0;
|
||||
}
|
||||
}
|
||||
|
||||
&.vertical>* {
|
||||
margin: calc($spacing * $multiplier / 2) 0;
|
||||
|
||||
&:first-child {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
|
||||
@if $rec {
|
||||
box {
|
||||
&.horizontal>* {
|
||||
margin: 0 $spacing * $multiplier / 2;
|
||||
|
||||
&:first-child {
|
||||
margin-left: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-right: 0;
|
||||
}
|
||||
}
|
||||
|
||||
&.vertical>* {
|
||||
margin: $spacing * $multiplier / 2 0;
|
||||
|
||||
&:first-child {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
@import './button';
|
||||
|
||||
@mixin switch {
|
||||
@include button;
|
||||
|
||||
slider {
|
||||
background-color: $primary-fg;
|
||||
border-radius: $radius;
|
||||
min-width: 24px;
|
||||
min-height: 24px;
|
||||
}
|
||||
|
||||
image {
|
||||
color: transparent;
|
||||
}
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
@mixin unset($rec: false) {
|
||||
all: unset;
|
||||
|
||||
@if $rec {
|
||||
* {
|
||||
all: unset
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,7 +0,0 @@
|
|||
@mixin widget {
|
||||
transition: $transition;
|
||||
border-radius: $radius;
|
||||
color: $fg;
|
||||
background-color: $widget-bg;
|
||||
border: $border;
|
||||
}
|
|
@ -1,115 +0,0 @@
|
|||
/* eslint-disable max-len */
|
||||
import { type Opt } from "lib/option"
|
||||
import { bash, dependencies } from "lib/utils"
|
||||
import options from "options"
|
||||
|
||||
const deps = [
|
||||
"font",
|
||||
"theme",
|
||||
"bar.corners",
|
||||
"bar.flatButtons",
|
||||
"bar.position",
|
||||
"bar.battery.charging",
|
||||
"bar.battery.blocks",
|
||||
]
|
||||
|
||||
const {
|
||||
dark,
|
||||
light,
|
||||
blur,
|
||||
scheme,
|
||||
padding,
|
||||
spacing,
|
||||
radius,
|
||||
shadows,
|
||||
widget,
|
||||
border,
|
||||
} = options.theme
|
||||
|
||||
const popoverPaddingMultiplier = 1.6
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
const t = (dark: Opt<any> | string, light: Opt<any> | string) => scheme.value === "dark"
|
||||
? `${dark}` : `${light}`
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
const $ = (name: string, value: string | Opt<any>) => `$${name}: ${value};`
|
||||
|
||||
const variables = () => [
|
||||
$("bg", blur.value ? `transparentize(${t(dark.bg, light.bg)}, ${blur.value / 100})` : t(dark.bg, light.bg)),
|
||||
$("fg", t(dark.fg, light.fg)),
|
||||
|
||||
$("primary-bg", t(dark.primary.bg, light.primary.bg)),
|
||||
$("primary-fg", t(dark.primary.fg, light.primary.fg)),
|
||||
|
||||
$("error-bg", t(dark.error.bg, light.error.bg)),
|
||||
$("error-fg", t(dark.error.fg, light.error.fg)),
|
||||
|
||||
$("scheme", scheme),
|
||||
$("padding", `${padding}pt`),
|
||||
$("spacing", `${spacing}pt`),
|
||||
$("radius", `${radius}px`),
|
||||
$("transition", `${options.transition}ms`),
|
||||
|
||||
$("shadows", `${shadows}`),
|
||||
|
||||
$("widget-bg", `transparentize(${t(dark.widget, light.widget)}, ${widget.opacity.value / 100})`),
|
||||
|
||||
$("hover-bg", `transparentize(${t(dark.widget, light.widget)}, ${(widget.opacity.value * .9) / 100})`),
|
||||
$("hover-fg", `lighten(${t(dark.fg, light.fg)}, 8%)`),
|
||||
|
||||
$("border-width", `${border.width}px`),
|
||||
$("border-color", `transparentize(${t(dark.border, light.border)}, ${border.opacity.value / 100})`),
|
||||
$("border", "$border-width solid $border-color"),
|
||||
|
||||
$("active-gradient", `linear-gradient(to right, ${t(dark.primary.bg, light.primary.bg)}, darken(${t(dark.primary.bg, light.primary.bg)}, 4%))`),
|
||||
$("shadow-color", t("rgba(0,0,0,.6)", "rgba(0,0,0,.4)")),
|
||||
$("text-shadow", t("2pt 2pt 2pt $shadow-color", "none")),
|
||||
$("box-shadow", t("2pt 2pt 2pt 0 $shadow-color, inset 0 0 0 $border-width $border-color", "none")),
|
||||
|
||||
$("popover-border-color", `transparentize(${t(dark.border, light.border)}, ${Math.max(((border.opacity.value - 1) / 100), 0)})`),
|
||||
$("popover-padding", `$padding * ${popoverPaddingMultiplier}`),
|
||||
$("popover-radius", radius.value === 0 ? "0" : "$radius + $popover-padding"),
|
||||
|
||||
$("font-size", `${options.font.size}pt`),
|
||||
$("font-name", options.font.name),
|
||||
|
||||
// etc
|
||||
$("charging-bg", options.bar.battery.charging),
|
||||
$("bar-battery-blocks", options.bar.battery.blocks),
|
||||
$("bar-position", options.bar.position),
|
||||
$("hyprland-gaps-multiplier", options.hyprland.gaps),
|
||||
$("screen-corner-multiplier", `${options.bar.corners.value * 0.01}`),
|
||||
]
|
||||
|
||||
async function resetCss() {
|
||||
if (!dependencies("sass", "fd"))
|
||||
return
|
||||
|
||||
try {
|
||||
const vars = `${TMP}/variables.scss`
|
||||
const scss = `${TMP}/main.scss`
|
||||
const css = `${TMP}/main.css`
|
||||
|
||||
const fd = await bash(`fd ".scss" ${App.configDir}`)
|
||||
const files = fd.split(/\s+/)
|
||||
const imports = [vars, ...files].map(f => `@import '${f}';`)
|
||||
|
||||
await Utils.writeFile(variables().join("\n"), vars)
|
||||
await Utils.writeFile(imports.join("\n"), scss)
|
||||
|
||||
await bash`sass ${scss} ${css}`
|
||||
|
||||
App.applyCss(css, true)
|
||||
} catch (error) {
|
||||
error instanceof Error
|
||||
? logError(error)
|
||||
: console.error(error)
|
||||
}
|
||||
}
|
||||
|
||||
Utils.monitorFile(`${App.configDir}/style`, resetCss)
|
||||
|
||||
options.handler(deps, resetCss)
|
||||
|
||||
await resetCss()
|
|
@ -1,265 +0,0 @@
|
|||
@use 'sass:color';
|
||||
|
||||
$bar-spacing: $spacing * .3;
|
||||
$button-radius: $radius;
|
||||
|
||||
@mixin panel-button($flat: true, $reactive: true) {
|
||||
@include accs-button($flat, $reactive);
|
||||
|
||||
>* {
|
||||
border-radius: $button-radius;
|
||||
margin: $bar-spacing;
|
||||
}
|
||||
|
||||
label,
|
||||
image {
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
>* {
|
||||
padding: $padding * 0.4 $padding * 0.8;
|
||||
}
|
||||
}
|
||||
|
||||
.bar {
|
||||
transition: $transition;
|
||||
background-color: $bg;
|
||||
|
||||
.panel-button {
|
||||
@include panel-button;
|
||||
|
||||
&:not(.flat) {
|
||||
|
||||
@include accs-button($flat: false);
|
||||
}
|
||||
}
|
||||
|
||||
.launcher {
|
||||
.colored {
|
||||
color: transparentize($primary-bg, 0.2);
|
||||
}
|
||||
|
||||
&:hover .colored {
|
||||
color: $primary-bg;
|
||||
}
|
||||
|
||||
&:active .colored,
|
||||
&.active .colored {
|
||||
color: $primary-fg;
|
||||
}
|
||||
}
|
||||
|
||||
.workspaces {
|
||||
label {
|
||||
font-size: 0;
|
||||
min-width: 5pt;
|
||||
min-height: 5pt;
|
||||
border-radius: $radius*.6;
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
margin: 0 $padding * .5;
|
||||
transition: $transition* .5;
|
||||
background-color: transparentize($fg, .8);
|
||||
|
||||
&.occupied {
|
||||
background-color: transparentize($fg, .2);
|
||||
min-width: 7pt;
|
||||
min-height: 7pt;
|
||||
}
|
||||
|
||||
&.active {
|
||||
// background-color: $primary-bg;
|
||||
background-image: $active-gradient;
|
||||
min-width: 20pt;
|
||||
min-height: 12pt;
|
||||
}
|
||||
}
|
||||
|
||||
&.active,
|
||||
&:active {
|
||||
label {
|
||||
background-color: transparentize($primary-fg, .3);
|
||||
|
||||
&.occupied {
|
||||
background-color: transparentize($primary-fg, .15);
|
||||
}
|
||||
|
||||
&.active {
|
||||
background-color: $primary-fg;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.media label {
|
||||
margin: 0 ($spacing * .5)
|
||||
}
|
||||
|
||||
.taskbar .indicator.active {
|
||||
background-color: $primary-bg;
|
||||
border-radius: $radius;
|
||||
min-height: 4pt;
|
||||
min-width: 6pt;
|
||||
margin: 2pt;
|
||||
}
|
||||
|
||||
.powermenu.colored,
|
||||
.recorder {
|
||||
image {
|
||||
color: transparentize($error-bg, 0.3);
|
||||
}
|
||||
|
||||
&:hover image {
|
||||
color: transparentize($error-bg, 0.15);
|
||||
}
|
||||
|
||||
&:active image {
|
||||
color: $primary-fg;
|
||||
}
|
||||
}
|
||||
|
||||
.quicksettings>box>box {
|
||||
@include spacing($spacing: if($bar-spacing==0, $padding / 2, $bar-spacing));
|
||||
}
|
||||
|
||||
.quicksettings:not(.active):not(:active) {
|
||||
.bluetooth {
|
||||
color: $primary-bg;
|
||||
|
||||
label {
|
||||
font-size: $font-size * .7;
|
||||
color: $fg;
|
||||
text-shadow: $text-shadow;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.battery-bar {
|
||||
>* {
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
&.bar-hidden>box {
|
||||
padding: 0 $spacing * .5;
|
||||
|
||||
image {
|
||||
margin: 0;
|
||||
}
|
||||
}
|
||||
|
||||
levelbar * {
|
||||
all: unset;
|
||||
transition: $transition;
|
||||
}
|
||||
|
||||
.whole {
|
||||
@if $shadows {
|
||||
image {
|
||||
-gtk-icon-shadow: $text-shadow;
|
||||
}
|
||||
|
||||
label {
|
||||
text-shadow: $text-shadow;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.regular image {
|
||||
margin-left: $spacing * .5;
|
||||
}
|
||||
|
||||
trough {
|
||||
@include widget;
|
||||
min-height: 12pt;
|
||||
min-width: 12pt;
|
||||
}
|
||||
|
||||
.regular trough {
|
||||
margin-right: $spacing * .5;
|
||||
}
|
||||
|
||||
block {
|
||||
margin: 0;
|
||||
|
||||
&:last-child {
|
||||
border-radius: 0 $button-radius $button-radius 0;
|
||||
}
|
||||
|
||||
&:first-child {
|
||||
border-radius: $button-radius 0 0 $button-radius;
|
||||
}
|
||||
}
|
||||
|
||||
.vertical {
|
||||
block {
|
||||
&:last-child {
|
||||
border-radius: 0 0 $button-radius $button-radius;
|
||||
}
|
||||
|
||||
&:first-child {
|
||||
border-radius: $button-radius $button-radius 0 0;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@for $i from 1 through $bar-battery-blocks {
|
||||
block:nth-child(#{$i}).filled {
|
||||
background-color: color.mix($bg, $primary-bg, $i*3)
|
||||
}
|
||||
|
||||
&.low block:nth-child(#{$i}).filled {
|
||||
background-color: color.mix($bg, $error-bg, $i*3)
|
||||
}
|
||||
|
||||
&.charging block:nth-child(#{$i}).filled {
|
||||
background-color: color.mix($bg, $charging-bg, $i*3)
|
||||
}
|
||||
|
||||
&:active .regular block:nth-child(#{$i}).filled {
|
||||
background-color: color.mix($bg, $primary-fg, $i*3)
|
||||
}
|
||||
}
|
||||
|
||||
&.low image {
|
||||
color: $error-bg
|
||||
}
|
||||
|
||||
&.charging image {
|
||||
color: $charging-bg
|
||||
}
|
||||
|
||||
&:active image {
|
||||
color: $primary-fg
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.bar.transparent {
|
||||
background-color: transparent;
|
||||
|
||||
.panel-button {
|
||||
&:hover>* {
|
||||
box-shadow: 1px 1px 3px 0 $shadow-color, inset 0 0 0 $border-width $border-color;
|
||||
background-color: $bg;
|
||||
}
|
||||
|
||||
&:not(:hover):not(.active) {
|
||||
|
||||
label,
|
||||
image {
|
||||
text-shadow: $text-shadow;
|
||||
-gtk-icon-shadow: $text-shadow;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.workspaces label {
|
||||
box-shadow: inset 0 0 0 $border-width $border-color,
|
||||
1px 1px 3px 0 $shadow-color;
|
||||
}
|
||||
|
||||
.battery-bar trough {
|
||||
box-shadow: 1px 1px 3px 0 $shadow-color;
|
||||
|
||||
}
|
||||
}
|
|
@ -1,110 +0,0 @@
|
|||
@import "./notifications.scss";
|
||||
|
||||
@mixin calendar {
|
||||
@include widget;
|
||||
padding: $padding*2 $padding*2 0;
|
||||
|
||||
calendar {
|
||||
all: unset;
|
||||
|
||||
&.button {
|
||||
@include button($flat: true);
|
||||
}
|
||||
|
||||
&:selected {
|
||||
box-shadow: inset 0 -8px 0 0 transparentize($primary-bg, 0.5),
|
||||
inset 0 0 0 1px $primary-bg;
|
||||
border-radius: $radius*0.6;
|
||||
}
|
||||
|
||||
&.header {
|
||||
background-color: transparent;
|
||||
border: none;
|
||||
color: transparentize($fg, 0.5);
|
||||
}
|
||||
|
||||
&.highlight {
|
||||
background-color: transparent;
|
||||
color: transparentize($primary-bg, 0.5);
|
||||
}
|
||||
|
||||
&:indeterminate {
|
||||
color: transparentize($fg, 0.9);
|
||||
}
|
||||
|
||||
font-size: 1.1em;
|
||||
padding: .2em;
|
||||
}
|
||||
}
|
||||
|
||||
window#datemenu .datemenu {
|
||||
@include floating-widget;
|
||||
|
||||
.notifications {
|
||||
.header {
|
||||
margin-bottom: $spacing;
|
||||
margin-right: $spacing;
|
||||
|
||||
>label {
|
||||
margin-left: $radius * .5;
|
||||
}
|
||||
|
||||
button {
|
||||
@include button;
|
||||
padding: $padding*.7 $padding;
|
||||
}
|
||||
}
|
||||
|
||||
.notification-scrollable {
|
||||
@include scrollable($top: true, $bottom: true);
|
||||
}
|
||||
|
||||
.notification-list {
|
||||
margin-right: $spacing;
|
||||
}
|
||||
|
||||
.notification {
|
||||
@include notification;
|
||||
@include widget;
|
||||
padding: $padding;
|
||||
margin-bottom: $spacing;
|
||||
}
|
||||
|
||||
.placeholder {
|
||||
image {
|
||||
font-size: 7em;
|
||||
}
|
||||
|
||||
label {
|
||||
font-size: 1.2em;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
separator {
|
||||
background-color: $popover-border-color;
|
||||
border-radius: $radius;
|
||||
margin-right: $spacing;
|
||||
}
|
||||
|
||||
.datemenu {
|
||||
@include spacing;
|
||||
}
|
||||
|
||||
.clock-box {
|
||||
padding: $padding;
|
||||
|
||||
.clock {
|
||||
font-size: 5em;
|
||||
}
|
||||
|
||||
.uptime {
|
||||
color: transparentize($fg, 0.2);
|
||||
}
|
||||
}
|
||||
|
||||
.calendar {
|
||||
@include calendar;
|
||||
}
|
||||
}
|
|
@ -1,64 +0,0 @@
|
|||
@import "../mixins/floating-widget.scss";
|
||||
@import "../mixins/widget.scss";
|
||||
@import "../mixins/spacing.scss";
|
||||
@import "../mixins/unset.scss";
|
||||
@import "../mixins/a11y-button.scss";
|
||||
@import "./bar.scss";
|
||||
|
||||
window#greeter {
|
||||
background-color: lighten($bg, 6%);
|
||||
color: $fg;
|
||||
|
||||
.bar {
|
||||
background-color: transparent;
|
||||
|
||||
.date {
|
||||
@include unset($rec: true);
|
||||
@include panel-button($flat: true, $reactive: false);
|
||||
}
|
||||
}
|
||||
|
||||
.auth {
|
||||
@include floating_widget;
|
||||
border-radius: $radius;
|
||||
min-width: 400px;
|
||||
padding: 0;
|
||||
|
||||
.wallpaper {
|
||||
min-height: 220px;
|
||||
background-size: cover;
|
||||
border-top-left-radius: $radius;
|
||||
border-top-right-radius: $radius;
|
||||
}
|
||||
|
||||
.wallpaper-contrast {
|
||||
min-height: 100px;
|
||||
}
|
||||
|
||||
.avatar {
|
||||
border-radius: 99px;
|
||||
min-width: 140px;
|
||||
min-height: 140px;
|
||||
background-size: cover;
|
||||
box-shadow: 3px 3px 6px 0 $shadow-color;
|
||||
margin-bottom: $spacing;
|
||||
}
|
||||
|
||||
|
||||
.password {
|
||||
entry {
|
||||
@include button;
|
||||
padding: $padding*.7 $padding;
|
||||
margin-left: $spacing*.5;
|
||||
}
|
||||
|
||||
margin: 0 $padding*4;
|
||||
margin-top: $spacing;
|
||||
}
|
||||
|
||||
.response-box {
|
||||
color: $error-bg;
|
||||
margin: $spacing 0;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,143 +0,0 @@
|
|||
@use "sass:math";
|
||||
@use "sass:color";
|
||||
|
||||
window#launcher .launcher {
|
||||
@include floating_widget;
|
||||
|
||||
.quicklaunch {
|
||||
@include spacing;
|
||||
|
||||
button {
|
||||
@include button($flat: true);
|
||||
padding: $padding;
|
||||
}
|
||||
}
|
||||
|
||||
entry {
|
||||
@include button;
|
||||
padding: $padding;
|
||||
margin: $spacing;
|
||||
|
||||
selection {
|
||||
color: color.mix($fg, $bg, 50%);
|
||||
background-color: transparent;
|
||||
}
|
||||
|
||||
label,
|
||||
image {
|
||||
color: $fg;
|
||||
}
|
||||
}
|
||||
|
||||
image.spinner {
|
||||
color: $primary-bg;
|
||||
margin-right: $spacing;
|
||||
}
|
||||
|
||||
separator {
|
||||
margin: 4pt 0;
|
||||
background-color: $popover-border-color;
|
||||
}
|
||||
|
||||
button.app-item {
|
||||
@include button($flat: true, $reactive: false);
|
||||
|
||||
>box {
|
||||
@include spacing(0.5);
|
||||
}
|
||||
|
||||
transition: $transition;
|
||||
padding: $padding;
|
||||
|
||||
label {
|
||||
transition: $transition;
|
||||
|
||||
&.title {
|
||||
color: $fg;
|
||||
}
|
||||
|
||||
&.description {
|
||||
color: transparentize($fg, 0.3);
|
||||
}
|
||||
}
|
||||
|
||||
image {
|
||||
transition: $transition;
|
||||
}
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
.title {
|
||||
color: $primary-bg;
|
||||
}
|
||||
|
||||
.description {
|
||||
color: transparentize($primary-bg, .4);
|
||||
}
|
||||
|
||||
image {
|
||||
-gtk-icon-shadow: 2px 2px $primary-bg;
|
||||
}
|
||||
}
|
||||
|
||||
&:active {
|
||||
background-color: transparentize($primary-bg, 0.5);
|
||||
border-radius: $radius;
|
||||
box-shadow: inset 0 0 0 $border-width $border-color;
|
||||
|
||||
.title {
|
||||
color: $fg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
button.help,
|
||||
button.nix-item {
|
||||
@include button($flat: true, $reactive: false);
|
||||
padding: 0 ($padding * .5);
|
||||
|
||||
label {
|
||||
transition: $transition;
|
||||
color: $fg;
|
||||
}
|
||||
|
||||
.name {
|
||||
font-size: 1.2em;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.description {
|
||||
color: transparentize($fg, .3)
|
||||
}
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
label {
|
||||
text-shadow: $text-shadow;
|
||||
}
|
||||
|
||||
.name,
|
||||
.version {
|
||||
color: $primary-bg;
|
||||
}
|
||||
|
||||
.description {
|
||||
color: transparentize($primary-bg, .3)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
button.sh-item {
|
||||
@include button($flat: true, $reactive: false);
|
||||
padding: 0 ($padding * .5);
|
||||
|
||||
transition: $transition;
|
||||
color: $fg;
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
color: $primary-bg;
|
||||
text-shadow: $text-shadow;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,79 +0,0 @@
|
|||
@mixin notification() {
|
||||
&.critical {
|
||||
box-shadow: inset 0 0 .5em 0 $error-bg;
|
||||
}
|
||||
|
||||
&:hover button.close-button {
|
||||
@include button-hover;
|
||||
background-color: transparentize($error-bg, .5);
|
||||
}
|
||||
|
||||
.content {
|
||||
.title {
|
||||
margin-right: $spacing;
|
||||
color: $fg;
|
||||
font-size: 1.1em;
|
||||
}
|
||||
|
||||
.time {
|
||||
color: transparentize($fg, .2);
|
||||
}
|
||||
|
||||
.description {
|
||||
font-size: .9em;
|
||||
color: transparentize($fg, .2);
|
||||
}
|
||||
|
||||
.icon {
|
||||
border-radius: $radius*0.8;
|
||||
margin-right: $spacing;
|
||||
|
||||
&.img {
|
||||
border: $border;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
box.actions {
|
||||
@include spacing(0.5);
|
||||
margin-top: $spacing;
|
||||
|
||||
button {
|
||||
@include button;
|
||||
border-radius: $radius*0.8;
|
||||
font-size: 1.2em;
|
||||
padding: $padding * 0.7;
|
||||
}
|
||||
}
|
||||
|
||||
button.close-button {
|
||||
@include button($flat: true);
|
||||
margin-left: $spacing / 2;
|
||||
border-radius: $radius*0.8;
|
||||
min-width: 1.2em;
|
||||
min-height: 1.2em;
|
||||
|
||||
&:hover {
|
||||
background-color: transparentize($error-bg, .2);
|
||||
}
|
||||
|
||||
&:active {
|
||||
background-image: none;
|
||||
background-color: $error-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
window.notifications {
|
||||
@include unset;
|
||||
|
||||
.notification {
|
||||
@include notification;
|
||||
@include floating-widget;
|
||||
border-radius: $radius;
|
||||
|
||||
.description {
|
||||
min-width: 350px;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,26 +0,0 @@
|
|||
window.indicator {
|
||||
.progress {
|
||||
@include floating-widget;
|
||||
padding: $padding * .5;
|
||||
border-radius: if($radius >0, calc($radius + $padding*.5), 0);
|
||||
@debug $radius;
|
||||
|
||||
.fill {
|
||||
border-radius: $radius;
|
||||
background-color: $primary-bg;
|
||||
color: $primary-fg;
|
||||
|
||||
image {
|
||||
-gtk-icon-transform: scale(0.7);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.microphone {
|
||||
@include floating-widget;
|
||||
margin: $spacing * 2;
|
||||
padding: $popover-padding * 2;
|
||||
font-size: 58px;
|
||||
color: transparentize($fg, .1)
|
||||
}
|
||||
}
|
|
@ -1,34 +0,0 @@
|
|||
window#overview .overview {
|
||||
@include floating-widget;
|
||||
@include spacing;
|
||||
|
||||
.workspace {
|
||||
&.active>widget {
|
||||
border-color: $primary-bg
|
||||
}
|
||||
|
||||
>widget {
|
||||
@include widget;
|
||||
border-radius: if($radius ==0, 0, $radius + $padding);
|
||||
|
||||
&:hover {
|
||||
background-color: $hover-bg;
|
||||
}
|
||||
|
||||
&:drop(active) {
|
||||
border-color: $primary-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.client {
|
||||
@include button;
|
||||
border-radius: $radius;
|
||||
margin: $padding;
|
||||
|
||||
&.hidden {
|
||||
@include hidden;
|
||||
transition: 0;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,110 +0,0 @@
|
|||
window#powermenu,
|
||||
window#verification {
|
||||
// the fraction has to be more than hyprland ignorealpha
|
||||
background-color: rgba(0, 0, 0, .4);
|
||||
}
|
||||
|
||||
window#verification .verification {
|
||||
@include floating-widget;
|
||||
padding: $popover-padding * 1.5;
|
||||
min-width: 300px;
|
||||
min-height: 100px;
|
||||
|
||||
.text-box {
|
||||
margin-bottom: $spacing;
|
||||
|
||||
.title {
|
||||
font-size: 1.6em;
|
||||
}
|
||||
|
||||
.desc {
|
||||
color: transparentize($fg, 0.1);
|
||||
font-size: 1.1em;
|
||||
}
|
||||
}
|
||||
|
||||
.buttons {
|
||||
@include spacing;
|
||||
margin-top: $padding;
|
||||
|
||||
button {
|
||||
@include button;
|
||||
font-size: 1.5em;
|
||||
padding: $padding;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
window#powermenu .powermenu {
|
||||
@include floating-widget;
|
||||
|
||||
&.line {
|
||||
padding: $popover-padding * 1.5;
|
||||
|
||||
button {
|
||||
padding: $popover-padding;
|
||||
}
|
||||
|
||||
label {
|
||||
margin-bottom: $spacing * -.5;
|
||||
}
|
||||
}
|
||||
|
||||
&.box {
|
||||
padding: $popover-padding * 2;
|
||||
|
||||
button {
|
||||
padding: $popover-padding * 1.5;
|
||||
}
|
||||
|
||||
label {
|
||||
margin-bottom: $spacing * -1;
|
||||
}
|
||||
}
|
||||
|
||||
button {
|
||||
@include unset;
|
||||
|
||||
image {
|
||||
@include button;
|
||||
border-radius: $radius + ($popover-padding * 1.4);
|
||||
min-width: 1.7em;
|
||||
min-height: 1.7em;
|
||||
font-size: 4em;
|
||||
}
|
||||
|
||||
label,
|
||||
image {
|
||||
color: transparentize($fg, 0.1);
|
||||
}
|
||||
|
||||
label {
|
||||
margin-top: $spacing * .3;
|
||||
}
|
||||
|
||||
&:hover {
|
||||
image {
|
||||
@include button-hover;
|
||||
}
|
||||
|
||||
label {
|
||||
color: $fg;
|
||||
}
|
||||
}
|
||||
|
||||
&:focus image {
|
||||
@include button-focus;
|
||||
}
|
||||
|
||||
&:active image {
|
||||
@include button-active;
|
||||
}
|
||||
|
||||
&:focus,
|
||||
&:active {
|
||||
label {
|
||||
color: $primary-bg;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,177 +0,0 @@
|
|||
window#quicksettings .quicksettings {
|
||||
@include floating-widget;
|
||||
@include spacing;
|
||||
|
||||
padding: $popover-padding * 1.4;
|
||||
|
||||
.avatar {
|
||||
@include widget;
|
||||
border-radius: $radius * 3;
|
||||
}
|
||||
|
||||
.header {
|
||||
@include spacing(.5);
|
||||
color: transparentize($fg, .15);
|
||||
|
||||
button {
|
||||
@include button;
|
||||
padding: $padding;
|
||||
|
||||
image {
|
||||
font-size: 1.4em;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.sliders-box {
|
||||
@include widget;
|
||||
padding: $padding;
|
||||
|
||||
button {
|
||||
@include button($flat: true);
|
||||
padding: $padding * .5;
|
||||
}
|
||||
|
||||
.volume button.arrow:last-child {
|
||||
margin-left: $spacing * .4;
|
||||
}
|
||||
|
||||
.volume,
|
||||
.brightness {
|
||||
padding: $padding * .5;
|
||||
}
|
||||
|
||||
scale {
|
||||
@include slider;
|
||||
margin: 0 ($spacing * .5);
|
||||
|
||||
&.muted highlight {
|
||||
background-image: none;
|
||||
background-color: transparentize($fg, $amount: .2);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.row {
|
||||
@include spacing;
|
||||
}
|
||||
|
||||
.menu {
|
||||
@include unset;
|
||||
@include widget;
|
||||
padding: $padding;
|
||||
margin-top: $spacing;
|
||||
|
||||
.icon {
|
||||
margin: 0 ($spacing * .5);
|
||||
margin-left: $spacing * .2;
|
||||
}
|
||||
|
||||
.title {
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
separator {
|
||||
margin: ($radius * .5);
|
||||
background-color: $border-color;
|
||||
}
|
||||
|
||||
button {
|
||||
@include button($flat: true);
|
||||
padding: ($padding * .5);
|
||||
|
||||
image:first-child {
|
||||
margin-right: $spacing * .5;
|
||||
}
|
||||
}
|
||||
|
||||
.bluetooth-devices {
|
||||
@include spacing(.5);
|
||||
}
|
||||
|
||||
switch {
|
||||
@include switch;
|
||||
}
|
||||
}
|
||||
|
||||
.sliders-box .menu {
|
||||
margin: ($spacing * .5) 0;
|
||||
|
||||
&.app-mixer {
|
||||
.mixer-item {
|
||||
padding: $padding * .5;
|
||||
padding-left: 0;
|
||||
padding-right: $padding * 2;
|
||||
|
||||
scale {
|
||||
@include slider($width: .5em);
|
||||
}
|
||||
|
||||
image {
|
||||
font-size: 1.2em;
|
||||
margin: 0 $padding;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.toggle-button {
|
||||
@include button;
|
||||
font-weight: bold;
|
||||
|
||||
image {
|
||||
font-size: 1.3em;
|
||||
}
|
||||
|
||||
label {
|
||||
margin-left: $spacing * .3;
|
||||
}
|
||||
|
||||
button {
|
||||
@include button($flat: true);
|
||||
|
||||
&:first-child {
|
||||
padding: $padding * 1.2;
|
||||
border-top-right-radius: 0;
|
||||
border-bottom-right-radius: 0;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
padding: $padding * .5;
|
||||
border-top-left-radius: 0;
|
||||
border-bottom-left-radius: 0;
|
||||
}
|
||||
}
|
||||
|
||||
&.active {
|
||||
background-color: $primary-bg;
|
||||
|
||||
label,
|
||||
image {
|
||||
color: $primary-fg;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.simple-toggle {
|
||||
@include button;
|
||||
font-weight: bold;
|
||||
padding: $padding * 1.2;
|
||||
|
||||
label {
|
||||
margin-left: $spacing * .3;
|
||||
}
|
||||
|
||||
image {
|
||||
font-size: 1.3em;
|
||||
}
|
||||
}
|
||||
|
||||
.media {
|
||||
@include spacing;
|
||||
|
||||
.player {
|
||||
@include media;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,52 +0,0 @@
|
|||
$_shadow-size: $padding;
|
||||
$_radius: $radius * $hyprland-gaps-multiplier * $screen-corner-multiplier;
|
||||
$_margin: 99px;
|
||||
|
||||
window.screen-corner:not(.hidden) {
|
||||
transition: $transition;
|
||||
|
||||
box.shadow {
|
||||
margin-right: $_margin * -1;
|
||||
margin-left: $_margin * -1;
|
||||
|
||||
@if $shadows {
|
||||
box-shadow: inset 0 0 $_shadow-size 0 $shadow-color;
|
||||
}
|
||||
|
||||
@if $bar-position =="top" {
|
||||
margin-bottom: $_margin * -1;
|
||||
}
|
||||
|
||||
@if $bar-position =="bottom" {
|
||||
margin-top: $_margin * -1;
|
||||
}
|
||||
}
|
||||
|
||||
box.border {
|
||||
@if $bar-position =="top" {
|
||||
border-top: $border-width solid $bg;
|
||||
}
|
||||
|
||||
@if $bar-position =="bottom" {
|
||||
border-bottom: $border-width solid $bg;
|
||||
}
|
||||
|
||||
margin-right: $_margin;
|
||||
margin-left: $_margin;
|
||||
}
|
||||
|
||||
box.corner {
|
||||
box-shadow: 0 0 0 $border-width $border-color;
|
||||
}
|
||||
|
||||
&.corners {
|
||||
box.border {
|
||||
border-radius: if($radius>0, $_radius, 0);
|
||||
box-shadow: 0 0 0 $_radius $bg;
|
||||
}
|
||||
|
||||
box.corner {
|
||||
border-radius: if($radius>0, $_radius, 0);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,144 +0,0 @@
|
|||
window.settings-dialog {
|
||||
background-color: $bg;
|
||||
color: $fg;
|
||||
|
||||
.header {
|
||||
.pager {
|
||||
@include spacing(.5);
|
||||
}
|
||||
|
||||
padding: $padding;
|
||||
|
||||
button {
|
||||
@include button;
|
||||
font-weight: bold;
|
||||
padding: $padding*.5 $padding;
|
||||
|
||||
box {
|
||||
@include spacing($spacing: .3em);
|
||||
}
|
||||
}
|
||||
|
||||
button.close {
|
||||
padding: $padding * .5;
|
||||
}
|
||||
|
||||
button.reset {
|
||||
@include button($flat: true);
|
||||
padding: $padding*.5;
|
||||
}
|
||||
}
|
||||
|
||||
.page {
|
||||
@include scrollable($top: true);
|
||||
|
||||
.page-content {
|
||||
padding: $padding*2;
|
||||
padding-top: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.group {
|
||||
.group-title {
|
||||
color: $primary-bg;
|
||||
margin-bottom: $spacing*.5;
|
||||
}
|
||||
|
||||
.group-reset {
|
||||
@include button($flat: true);
|
||||
margin: $spacing * .5;
|
||||
padding: $padding * .5;
|
||||
|
||||
&:disabled {
|
||||
color: transparent;
|
||||
}
|
||||
}
|
||||
|
||||
&:not(:first-child) {
|
||||
margin-top: $spacing;
|
||||
}
|
||||
}
|
||||
|
||||
.row {
|
||||
background-color: $widget-bg;
|
||||
padding: $padding;
|
||||
border: $border;
|
||||
border-top: none;
|
||||
|
||||
&:first-child {
|
||||
border-radius: $radius $radius 0 0;
|
||||
border: $border;
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
border-radius: 0 0 $radius $radius;
|
||||
}
|
||||
|
||||
&:first-child:last-child {
|
||||
border-radius: $radius;
|
||||
border: $border;
|
||||
}
|
||||
|
||||
button.reset {
|
||||
margin-left: $spacing;
|
||||
}
|
||||
|
||||
label.id,
|
||||
label.note {
|
||||
color: transparentize($fg, .4)
|
||||
}
|
||||
|
||||
entry,
|
||||
button {
|
||||
@include button;
|
||||
padding: $padding;
|
||||
}
|
||||
|
||||
switch {
|
||||
@include switch;
|
||||
}
|
||||
|
||||
spinbutton {
|
||||
@include unset;
|
||||
|
||||
entry {
|
||||
border-radius: $radius 0 0 $radius;
|
||||
}
|
||||
|
||||
button {
|
||||
border-radius: 0;
|
||||
}
|
||||
|
||||
button:last-child {
|
||||
border-radius: 0 $radius $radius 0;
|
||||
}
|
||||
}
|
||||
|
||||
.enum-setter {
|
||||
label {
|
||||
background-color: $widget-bg;
|
||||
border: $border;
|
||||
padding: 0 $padding;
|
||||
border-radius: $radius 0 0 $radius;
|
||||
}
|
||||
|
||||
button {
|
||||
border-radius: 0;
|
||||
}
|
||||
|
||||
button:last-child {
|
||||
border-radius: 0 $radius $radius 0;
|
||||
}
|
||||
}
|
||||
|
||||
&.wallpaper {
|
||||
button {
|
||||
margin-top: $spacing * .5;
|
||||
}
|
||||
|
||||
.preview {
|
||||
border-radius: $radius;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,19 +0,0 @@
|
|||
{
|
||||
"compilerOptions": {
|
||||
"target": "ES2022",
|
||||
"module": "ES2022",
|
||||
"lib": [
|
||||
"ES2022"
|
||||
],
|
||||
"allowJs": true,
|
||||
"checkJs": true,
|
||||
"strict": true,
|
||||
"noImplicitAny": false,
|
||||
"baseUrl": ".",
|
||||
"typeRoots": [
|
||||
"./types",
|
||||
"./node_modules/@girs"
|
||||
],
|
||||
"skipLibCheck": true
|
||||
}
|
||||
}
|
|
@ -1,156 +0,0 @@
|
|||
import { type WindowProps } from "types/widgets/window"
|
||||
import { type RevealerProps } from "types/widgets/revealer"
|
||||
import { type EventBoxProps } from "types/widgets/eventbox"
|
||||
import type Gtk from "gi://Gtk?version=3.0"
|
||||
import options from "options"
|
||||
|
||||
type Transition = RevealerProps["transition"]
|
||||
type Child = WindowProps["child"]
|
||||
|
||||
type PopupWindowProps = Omit<WindowProps, "name"> & {
|
||||
name: string
|
||||
layout?: keyof ReturnType<typeof Layout>
|
||||
transition?: Transition,
|
||||
}
|
||||
|
||||
export const Padding = (name: string, {
|
||||
css = "",
|
||||
hexpand = true,
|
||||
vexpand = true,
|
||||
}: EventBoxProps = {}) => Widget.EventBox({
|
||||
hexpand,
|
||||
vexpand,
|
||||
can_focus: false,
|
||||
child: Widget.Box({ css }),
|
||||
setup: w => w.on("button-press-event", () => App.toggleWindow(name)),
|
||||
})
|
||||
|
||||
const PopupRevealer = (
|
||||
name: string,
|
||||
child: Child,
|
||||
transition: Transition = "slide_down",
|
||||
) => Widget.Box(
|
||||
{ css: "padding: 1px;" },
|
||||
Widget.Revealer({
|
||||
transition,
|
||||
child: Widget.Box({
|
||||
class_name: "window-content",
|
||||
child,
|
||||
}),
|
||||
transitionDuration: options.transition.bind(),
|
||||
setup: self => self.hook(App, (_, wname, visible) => {
|
||||
if (wname === name)
|
||||
self.reveal_child = visible
|
||||
}),
|
||||
}),
|
||||
)
|
||||
|
||||
const Layout = (name: string, child: Child, transition?: Transition) => ({
|
||||
"center": () => Widget.CenterBox({},
|
||||
Padding(name),
|
||||
Widget.CenterBox(
|
||||
{ vertical: true },
|
||||
Padding(name),
|
||||
PopupRevealer(name, child, transition),
|
||||
Padding(name),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"top": () => Widget.CenterBox({},
|
||||
Padding(name),
|
||||
Widget.Box(
|
||||
{ vertical: true },
|
||||
PopupRevealer(name, child, transition),
|
||||
Padding(name),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"top-right": () => Widget.Box({},
|
||||
Padding(name),
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
PopupRevealer(name, child, transition),
|
||||
Padding(name),
|
||||
),
|
||||
),
|
||||
"top-center": () => Widget.Box({},
|
||||
Padding(name),
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
PopupRevealer(name, child, transition),
|
||||
Padding(name),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"top-left": () => Widget.Box({},
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
PopupRevealer(name, child, transition),
|
||||
Padding(name),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"bottom-left": () => Widget.Box({},
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
Padding(name),
|
||||
PopupRevealer(name, child, transition),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"bottom-center": () => Widget.Box({},
|
||||
Padding(name),
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
Padding(name),
|
||||
PopupRevealer(name, child, transition),
|
||||
),
|
||||
Padding(name),
|
||||
),
|
||||
"bottom-right": () => Widget.Box({},
|
||||
Padding(name),
|
||||
Widget.Box(
|
||||
{
|
||||
hexpand: false,
|
||||
vertical: true,
|
||||
},
|
||||
Padding(name),
|
||||
PopupRevealer(name, child, transition),
|
||||
),
|
||||
),
|
||||
})
|
||||
|
||||
export default ({
|
||||
name,
|
||||
child,
|
||||
layout = "center",
|
||||
transition,
|
||||
exclusivity = "ignore",
|
||||
...props
|
||||
}: PopupWindowProps) => Widget.Window<Gtk.Widget>({
|
||||
name,
|
||||
class_names: [name, "popup-window"],
|
||||
setup: w => w.keybind("Escape", () => App.closeWindow(name)),
|
||||
visible: false,
|
||||
keymode: "on-demand",
|
||||
exclusivity,
|
||||
layer: "top",
|
||||
anchor: ["top", "bottom", "right", "left"],
|
||||
child: Layout(name, child, transition)[layout](),
|
||||
...props,
|
||||
})
|
|
@ -1,3 +0,0 @@
|
|||
import Gtk from "gi://Gtk?version=3.0"
|
||||
|
||||
export default Widget.subclass<typeof Gtk.Window, Gtk.Window.ConstructorProperties>(Gtk.Window)
|
|
@ -1,60 +0,0 @@
|
|||
import BatteryBar from "./buttons/BatteryBar"
|
||||
import ColorPicker from "./buttons/ColorPicker"
|
||||
import Date from "./buttons/Date"
|
||||
import Launcher from "./buttons/Launcher"
|
||||
import Media from "./buttons/Media"
|
||||
import PowerMenu from "./buttons/PowerMenu"
|
||||
import SysTray from "./buttons/SysTray"
|
||||
import SystemIndicators from "./buttons/SystemIndicators"
|
||||
import Taskbar from "./buttons/Taskbar"
|
||||
import Workspaces from "./buttons/Workspaces"
|
||||
import ScreenRecord from "./buttons/ScreenRecord"
|
||||
import Messages from "./buttons/Messages"
|
||||
import options from "options"
|
||||
|
||||
const { start, center, end } = options.bar.layout
|
||||
const { transparent, position } = options.bar
|
||||
|
||||
export type BarWidget = keyof typeof widget
|
||||
|
||||
const widget = {
|
||||
battery: BatteryBar,
|
||||
colorpicker: ColorPicker,
|
||||
date: Date,
|
||||
launcher: Launcher,
|
||||
media: Media,
|
||||
powermenu: PowerMenu,
|
||||
systray: SysTray,
|
||||
system: SystemIndicators,
|
||||
taskbar: Taskbar,
|
||||
workspaces: Workspaces,
|
||||
screenrecord: ScreenRecord,
|
||||
messages: Messages,
|
||||
expander: () => Widget.Box({ expand: true }),
|
||||
}
|
||||
|
||||
export default (monitor: number) => Widget.Window({
|
||||
monitor,
|
||||
class_name: "bar",
|
||||
name: `bar${monitor}`,
|
||||
exclusivity: "exclusive",
|
||||
anchor: position.bind().as(pos => [pos, "right", "left"]),
|
||||
child: Widget.CenterBox({
|
||||
css: "min-width: 2px; min-height: 2px;",
|
||||
startWidget: Widget.Box({
|
||||
hexpand: true,
|
||||
children: start.bind().as(s => s.map(w => widget[w]())),
|
||||
}),
|
||||
centerWidget: Widget.Box({
|
||||
hpack: "center",
|
||||
children: center.bind().as(c => c.map(w => widget[w]())),
|
||||
}),
|
||||
endWidget: Widget.Box({
|
||||
hexpand: true,
|
||||
children: end.bind().as(e => e.map(w => widget[w]())),
|
||||
}),
|
||||
}),
|
||||
setup: self => self.hook(transparent, () => {
|
||||
self.toggleClassName("transparent", transparent.value)
|
||||
}),
|
||||
})
|
|
@ -1,46 +0,0 @@
|
|||
import options from "options"
|
||||
import { ButtonProps } from "types/widgets/button"
|
||||
|
||||
type PanelButtonProps = ButtonProps & {
|
||||
window?: string,
|
||||
flat?: boolean
|
||||
}
|
||||
|
||||
export default ({
|
||||
window = "",
|
||||
flat,
|
||||
child,
|
||||
setup,
|
||||
...rest
|
||||
}: PanelButtonProps) => Widget.Button({
|
||||
child: Widget.Box({ child }),
|
||||
setup: self => {
|
||||
let open = false
|
||||
|
||||
self.toggleClassName("panel-button")
|
||||
self.toggleClassName(window)
|
||||
|
||||
self.hook(options.bar.flatButtons, () => {
|
||||
self.toggleClassName("flat", flat ?? options.bar.flatButtons.value)
|
||||
})
|
||||
|
||||
self.hook(App, (_, win, visible) => {
|
||||
if (win !== window)
|
||||
return
|
||||
|
||||
if (open && !visible) {
|
||||
open = false
|
||||
self.toggleClassName("active", false)
|
||||
}
|
||||
|
||||
if (visible) {
|
||||
open = true
|
||||
self.toggleClassName("active")
|
||||
}
|
||||
})
|
||||
|
||||
if (setup)
|
||||
setup(self)
|
||||
},
|
||||
...rest,
|
||||
})
|
|
@ -1,29 +0,0 @@
|
|||
import options from "options"
|
||||
|
||||
const { corners, transparent } = options.bar
|
||||
|
||||
export default (monitor: number) => Widget.Window({
|
||||
monitor,
|
||||
name: `corner${monitor}`,
|
||||
class_name: "screen-corner",
|
||||
anchor: ["top", "bottom", "right", "left"],
|
||||
click_through: true,
|
||||
child: Widget.Box({
|
||||
class_name: "shadow",
|
||||
child: Widget.Box({
|
||||
class_name: "border",
|
||||
expand: true,
|
||||
child: Widget.Box({
|
||||
class_name: "corner",
|
||||
expand: true,
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
setup: self => self
|
||||
.hook(corners, () => {
|
||||
self.toggleClassName("corners", corners.value > 0)
|
||||
})
|
||||
.hook(transparent, () => {
|
||||
self.toggleClassName("hidden", transparent.value)
|
||||
}),
|
||||
})
|
|
@ -1,94 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import options from "options"
|
||||
import PanelButton from "../PanelButton"
|
||||
|
||||
const battery = await Service.import("battery")
|
||||
const { bar, percentage, blocks, width, low } = options.bar.battery
|
||||
|
||||
const Indicator = () => Widget.Icon({
|
||||
setup: self => self.hook(battery, () => {
|
||||
self.icon = battery.charging || battery.charged
|
||||
? icons.battery.charging
|
||||
: battery.icon_name
|
||||
}),
|
||||
})
|
||||
|
||||
const PercentLabel = () => Widget.Revealer({
|
||||
transition: "slide_right",
|
||||
click_through: true,
|
||||
reveal_child: percentage.bind(),
|
||||
child: Widget.Label({
|
||||
label: battery.bind("percent").as(p => `${p}%`),
|
||||
}),
|
||||
})
|
||||
|
||||
const LevelBar = () => {
|
||||
const level = Widget.LevelBar({
|
||||
bar_mode: "discrete",
|
||||
max_value: blocks.bind(),
|
||||
visible: bar.bind().as(b => b !== "hidden"),
|
||||
value: battery.bind("percent").as(p => (p / 100) * blocks.value),
|
||||
})
|
||||
const update = () => {
|
||||
level.value = (battery.percent / 100) * blocks.value
|
||||
level.css = `block { min-width: ${width.value / blocks.value}pt; }`
|
||||
}
|
||||
return level
|
||||
.hook(width, update)
|
||||
.hook(blocks, update)
|
||||
.hook(bar, () => {
|
||||
level.vpack = bar.value === "whole" ? "fill" : "center"
|
||||
level.hpack = bar.value === "whole" ? "fill" : "center"
|
||||
})
|
||||
}
|
||||
|
||||
const WholeButton = () => Widget.Overlay({
|
||||
vexpand: true,
|
||||
child: LevelBar(),
|
||||
class_name: "whole",
|
||||
pass_through: true,
|
||||
overlay: Widget.Box({
|
||||
hpack: "center",
|
||||
children: [
|
||||
Widget.Icon({
|
||||
icon: icons.battery.charging,
|
||||
visible: Utils.merge([
|
||||
battery.bind("charging"),
|
||||
battery.bind("charged"),
|
||||
], (ing, ed) => ing || ed),
|
||||
}),
|
||||
Widget.Box({
|
||||
hpack: "center",
|
||||
vpack: "center",
|
||||
child: PercentLabel(),
|
||||
}),
|
||||
],
|
||||
}),
|
||||
})
|
||||
|
||||
const Regular = () => Widget.Box({
|
||||
class_name: "regular",
|
||||
children: [
|
||||
Indicator(),
|
||||
PercentLabel(),
|
||||
LevelBar(),
|
||||
],
|
||||
})
|
||||
|
||||
export default () => PanelButton({
|
||||
class_name: "battery-bar",
|
||||
hexpand: false,
|
||||
on_clicked: () => { percentage.value = !percentage.value },
|
||||
visible: battery.bind("available"),
|
||||
child: Widget.Box({
|
||||
expand: true,
|
||||
visible: battery.bind("available"),
|
||||
child: bar.bind().as(b => b === "whole" ? WholeButton() : Regular()),
|
||||
}),
|
||||
setup: self => self
|
||||
.hook(bar, w => w.toggleClassName("bar-hidden", bar.value === "hidden"))
|
||||
.hook(battery, w => {
|
||||
w.toggleClassName("charging", battery.charging || battery.charged)
|
||||
w.toggleClassName("low", battery.percent < low.value)
|
||||
}),
|
||||
})
|
|
@ -1,37 +0,0 @@
|
|||
import PanelButton from "../PanelButton"
|
||||
import colorpicker from "service/colorpicker"
|
||||
import Gdk from "gi://Gdk"
|
||||
|
||||
const css = (color: string) => `
|
||||
* {
|
||||
background-color: ${color};
|
||||
color: transparent;
|
||||
}
|
||||
*:hover {
|
||||
color: white;
|
||||
text-shadow: 2px 2px 3px rgba(0,0,0,.8);
|
||||
}`
|
||||
|
||||
export default () => {
|
||||
const menu = Widget.Menu({
|
||||
class_name: "colorpicker",
|
||||
children: colorpicker.bind("colors").as(c => c.map(color => Widget.MenuItem({
|
||||
child: Widget.Label(color),
|
||||
css: css(color),
|
||||
on_activate: () => colorpicker.wlCopy(color),
|
||||
}))),
|
||||
})
|
||||
|
||||
return PanelButton({
|
||||
class_name: "color-picker",
|
||||
child: Widget.Icon("color-select-symbolic"),
|
||||
tooltip_text: colorpicker.bind("colors").as(v => `${v.length} colors`),
|
||||
on_clicked: colorpicker.pick,
|
||||
on_secondary_click: self => {
|
||||
if (colorpicker.colors.length === 0)
|
||||
return
|
||||
|
||||
menu.popup_at_widget(self, Gdk.Gravity.SOUTH, Gdk.Gravity.NORTH, null)
|
||||
},
|
||||
})
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
import { clock } from "lib/variables"
|
||||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
|
||||
const { format, action } = options.bar.date
|
||||
const time = Utils.derive([clock, format], (c, f) => c.format(f) || "")
|
||||
|
||||
export default () => PanelButton({
|
||||
window: "datemenu",
|
||||
on_clicked: action.bind(),
|
||||
child: Widget.Label({
|
||||
justification: "center",
|
||||
label: time.bind(),
|
||||
}),
|
||||
})
|
|
@ -1,49 +0,0 @@
|
|||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
import nix from "service/nix"
|
||||
|
||||
const { icon, label, action } = options.bar.launcher
|
||||
|
||||
function Spinner() {
|
||||
const child = Widget.Icon({
|
||||
icon: icon.icon.bind(),
|
||||
class_name: Utils.merge([
|
||||
icon.colored.bind(),
|
||||
nix.bind("ready"),
|
||||
], (c, r) => `${c ? "colored" : ""} ${r ? "" : "spinning"}`),
|
||||
css: `
|
||||
@keyframes spin {
|
||||
to { -gtk-icon-transform: rotate(1turn); }
|
||||
}
|
||||
|
||||
image.spinning {
|
||||
animation-name: spin;
|
||||
animation-duration: 1s;
|
||||
animation-timing-function: linear;
|
||||
animation-iteration-count: infinite;
|
||||
}
|
||||
`,
|
||||
})
|
||||
|
||||
return Widget.Revealer({
|
||||
transition: "slide_left",
|
||||
child,
|
||||
reveal_child: Utils.merge([
|
||||
icon.icon.bind(),
|
||||
nix.bind("ready"),
|
||||
], (i, r) => Boolean(i || r)),
|
||||
})
|
||||
}
|
||||
|
||||
export default () => PanelButton({
|
||||
window: "launcher",
|
||||
on_clicked: action.bind(),
|
||||
child: Widget.Box([
|
||||
Spinner(),
|
||||
Widget.Label({
|
||||
class_name: label.colored.bind().as(c => c ? "colored" : ""),
|
||||
visible: label.label.bind().as(v => !!v),
|
||||
label: label.label.bind(),
|
||||
}),
|
||||
]),
|
||||
})
|
|
@ -1,92 +0,0 @@
|
|||
import { type MprisPlayer } from "types/service/mpris"
|
||||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
import icons from "lib/icons"
|
||||
import { icon } from "lib/utils"
|
||||
|
||||
const mpris = await Service.import("mpris")
|
||||
const { length, direction, preferred, monochrome, format } = options.bar.media
|
||||
|
||||
const getPlayer = (name = preferred.value) =>
|
||||
mpris.getPlayer(name) || mpris.players[0] || null
|
||||
|
||||
const Content = (player: MprisPlayer) => {
|
||||
const revealer = Widget.Revealer({
|
||||
click_through: true,
|
||||
visible: length.bind().as(l => l > 0),
|
||||
transition: direction.bind().as(d => `slide_${d}` as const),
|
||||
setup: self => {
|
||||
let current = ""
|
||||
self.hook(player, () => {
|
||||
if (current === player.track_title)
|
||||
return
|
||||
|
||||
current = player.track_title
|
||||
self.reveal_child = true
|
||||
Utils.timeout(3000, () => {
|
||||
!self.is_destroyed && (self.reveal_child = false)
|
||||
})
|
||||
})
|
||||
},
|
||||
child: Widget.Label({
|
||||
truncate: "end",
|
||||
max_width_chars: length.bind().as(n => n > 0 ? n : -1),
|
||||
label: Utils.merge([
|
||||
player.bind("track_title"),
|
||||
player.bind("track_artists"),
|
||||
format.bind(),
|
||||
], () => `${format}`
|
||||
.replace("{title}", player.track_title)
|
||||
.replace("{artists}", player.track_artists.join(", "))
|
||||
.replace("{artist}", player.track_artists[0] || "")
|
||||
.replace("{album}", player.track_album)
|
||||
.replace("{name}", player.name)
|
||||
.replace("{identity}", player.identity),
|
||||
),
|
||||
}),
|
||||
})
|
||||
|
||||
const playericon = Widget.Icon({
|
||||
icon: Utils.merge([player.bind("entry"), monochrome.bind()], (entry => {
|
||||
const name = `${entry}${monochrome.value ? "-symbolic" : ""}`
|
||||
return icon(name, icons.fallback.audio)
|
||||
})),
|
||||
})
|
||||
|
||||
return Widget.Box({
|
||||
attribute: { revealer },
|
||||
children: direction.bind().as(d => d === "right"
|
||||
? [playericon, revealer] : [revealer, playericon]),
|
||||
})
|
||||
}
|
||||
|
||||
export default () => {
|
||||
let player = getPlayer()
|
||||
|
||||
const btn = PanelButton({
|
||||
class_name: "media",
|
||||
child: Widget.Icon(icons.fallback.audio),
|
||||
})
|
||||
|
||||
const update = () => {
|
||||
player = getPlayer()
|
||||
btn.visible = !!player
|
||||
|
||||
if (!player)
|
||||
return
|
||||
|
||||
const content = Content(player)
|
||||
const { revealer } = content.attribute
|
||||
btn.child = content
|
||||
btn.on_primary_click = () => { player.playPause() }
|
||||
btn.on_secondary_click = () => { player.playPause() }
|
||||
btn.on_scroll_up = () => { player.next() }
|
||||
btn.on_scroll_down = () => { player.previous() }
|
||||
btn.on_hover = () => { revealer.reveal_child = true }
|
||||
btn.on_hover_lost = () => { revealer.reveal_child = false }
|
||||
}
|
||||
|
||||
return btn
|
||||
.hook(preferred, update)
|
||||
.hook(mpris, update, "notify::players")
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
|
||||
const n = await Service.import("notifications")
|
||||
const notifs = n.bind("notifications")
|
||||
const action = options.bar.messages.action.bind()
|
||||
|
||||
export default () => PanelButton({
|
||||
class_name: "messages",
|
||||
on_clicked: action,
|
||||
visible: notifs.as(n => n.length > 0),
|
||||
child: Widget.Box([
|
||||
Widget.Icon(icons.notifications.message),
|
||||
]),
|
||||
})
|
|
@ -1,15 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
|
||||
const { monochrome, action } = options.bar.powermenu
|
||||
|
||||
export default () => PanelButton({
|
||||
window: "powermenu",
|
||||
on_clicked: action.bind(),
|
||||
child: Widget.Icon(icons.powermenu.shutdown),
|
||||
setup: self => self.hook(monochrome, () => {
|
||||
self.toggleClassName("colored", !monochrome.value)
|
||||
self.toggleClassName("box")
|
||||
}),
|
||||
})
|
|
@ -1,21 +0,0 @@
|
|||
import PanelButton from "../PanelButton"
|
||||
import screenrecord from "service/screenrecord"
|
||||
import icons from "lib/icons"
|
||||
|
||||
export default () => PanelButton({
|
||||
class_name: "recorder",
|
||||
on_clicked: () => screenrecord.stop(),
|
||||
visible: screenrecord.bind("recording"),
|
||||
child: Widget.Box({
|
||||
children: [
|
||||
Widget.Icon(icons.recorder.recording),
|
||||
Widget.Label({
|
||||
label: screenrecord.bind("timer").as(time => {
|
||||
const sec = time % 60
|
||||
const min = Math.floor(time / 60)
|
||||
return `${min}:${sec < 10 ? "0" + sec : sec}`
|
||||
}),
|
||||
}),
|
||||
],
|
||||
}),
|
||||
})
|
|
@ -1,39 +0,0 @@
|
|||
import { type TrayItem } from "types/service/systemtray"
|
||||
import PanelButton from "../PanelButton"
|
||||
import Gdk from "gi://Gdk"
|
||||
import options from "options"
|
||||
|
||||
const systemtray = await Service.import("systemtray")
|
||||
const { ignore } = options.bar.systray
|
||||
|
||||
const SysTrayItem = (item: TrayItem) => PanelButton({
|
||||
class_name: "tray-item",
|
||||
child: Widget.Icon({ icon: item.bind("icon") }),
|
||||
tooltip_markup: item.bind("tooltip_markup"),
|
||||
setup: self => {
|
||||
const { menu } = item
|
||||
if (!menu)
|
||||
return
|
||||
|
||||
const id = menu.connect("popped-up", () => {
|
||||
self.toggleClassName("active")
|
||||
menu.connect("notify::visible", () => {
|
||||
self.toggleClassName("active", menu.visible)
|
||||
})
|
||||
menu.disconnect(id!)
|
||||
})
|
||||
|
||||
self.connect("destroy", () => menu.disconnect(id))
|
||||
},
|
||||
|
||||
on_primary_click: btn => item.menu?.popup_at_widget(
|
||||
btn, Gdk.Gravity.SOUTH, Gdk.Gravity.NORTH, null),
|
||||
|
||||
on_secondary_click: btn => item.menu?.popup_at_widget(
|
||||
btn, Gdk.Gravity.SOUTH, Gdk.Gravity.NORTH, null),
|
||||
})
|
||||
|
||||
export default () => Widget.Box()
|
||||
.bind("children", systemtray, "items", i => i
|
||||
.filter(({ id }) => !ignore.value.includes(id))
|
||||
.map(SysTrayItem))
|
|
@ -1,98 +0,0 @@
|
|||
import PanelButton from "../PanelButton"
|
||||
import icons from "lib/icons"
|
||||
import asusctl from "service/asusctl"
|
||||
|
||||
const notifications = await Service.import("notifications")
|
||||
const bluetooth = await Service.import("bluetooth")
|
||||
const audio = await Service.import("audio")
|
||||
const network = await Service.import("network")
|
||||
const powerprof = await Service.import("powerprofiles")
|
||||
|
||||
const ProfileIndicator = () => {
|
||||
const visible = asusctl.available
|
||||
? asusctl.bind("profile").as(p => p !== "Balanced")
|
||||
: powerprof.bind("active_profile").as(p => p !== "balanced")
|
||||
|
||||
const icon = asusctl.available
|
||||
? asusctl.bind("profile").as(p => icons.asusctl.profile[p])
|
||||
: powerprof.bind("active_profile").as(p => icons.powerprofile[p])
|
||||
|
||||
return Widget.Icon({ visible, icon })
|
||||
}
|
||||
|
||||
const ModeIndicator = () => {
|
||||
if (!asusctl.available) {
|
||||
return Widget.Icon({
|
||||
setup(self) {
|
||||
Utils.idle(() => self.visible = false)
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
return Widget.Icon({
|
||||
visible: asusctl.bind("mode").as(m => m !== "Hybrid"),
|
||||
icon: asusctl.bind("mode").as(m => icons.asusctl.mode[m]),
|
||||
})
|
||||
}
|
||||
|
||||
const MicrophoneIndicator = () => Widget.Icon()
|
||||
.hook(audio, self => self.visible =
|
||||
audio.recorders.length > 0
|
||||
|| audio.microphone.is_muted
|
||||
|| false)
|
||||
.hook(audio.microphone, self => {
|
||||
const vol = audio.microphone.is_muted ? 0 : audio.microphone.volume
|
||||
const { muted, low, medium, high } = icons.audio.mic
|
||||
const cons = [[67, high], [34, medium], [1, low], [0, muted]] as const
|
||||
self.icon = cons.find(([n]) => n <= vol * 100)?.[1] || ""
|
||||
})
|
||||
|
||||
const DNDIndicator = () => Widget.Icon({
|
||||
visible: notifications.bind("dnd"),
|
||||
icon: icons.notifications.silent,
|
||||
})
|
||||
|
||||
const BluetoothIndicator = () => Widget.Overlay({
|
||||
class_name: "bluetooth",
|
||||
passThrough: true,
|
||||
visible: bluetooth.bind("enabled"),
|
||||
child: Widget.Icon({
|
||||
icon: icons.bluetooth.enabled,
|
||||
}),
|
||||
overlay: Widget.Label({
|
||||
hpack: "end",
|
||||
vpack: "start",
|
||||
label: bluetooth.bind("connected_devices").as(c => `${c.length}`),
|
||||
visible: bluetooth.bind("connected_devices").as(c => c.length > 0),
|
||||
}),
|
||||
})
|
||||
|
||||
const NetworkIndicator = () => Widget.Icon().hook(network, self => {
|
||||
const icon = network[network.primary || "wifi"]?.icon_name
|
||||
self.icon = icon || ""
|
||||
self.visible = !!icon
|
||||
})
|
||||
|
||||
const AudioIndicator = () => Widget.Icon()
|
||||
.hook(audio.speaker, self => {
|
||||
const vol = audio.speaker.is_muted ? 0 : audio.speaker.volume
|
||||
const { muted, low, medium, high, overamplified } = icons.audio.volume
|
||||
const cons = [[101, overamplified], [67, high], [34, medium], [1, low], [0, muted]] as const
|
||||
self.icon = cons.find(([n]) => n <= vol * 100)?.[1] || ""
|
||||
})
|
||||
|
||||
export default () => PanelButton({
|
||||
window: "quicksettings",
|
||||
on_clicked: () => App.toggleWindow("quicksettings"),
|
||||
on_scroll_up: () => audio.speaker.volume += 0.02,
|
||||
on_scroll_down: () => audio.speaker.volume -= 0.02,
|
||||
child: Widget.Box([
|
||||
ProfileIndicator(),
|
||||
ModeIndicator(),
|
||||
DNDIndicator(),
|
||||
BluetoothIndicator(),
|
||||
NetworkIndicator(),
|
||||
AudioIndicator(),
|
||||
MicrophoneIndicator(),
|
||||
]),
|
||||
})
|
|
@ -1,90 +0,0 @@
|
|||
import { launchApp, icon } from "lib/utils"
|
||||
import icons from "lib/icons"
|
||||
import options from "options"
|
||||
import PanelButton from "../PanelButton"
|
||||
|
||||
const hyprland = await Service.import("hyprland")
|
||||
const apps = await Service.import("applications")
|
||||
const { monochrome, exclusive, iconSize } = options.bar.taskbar
|
||||
const { position } = options.bar
|
||||
|
||||
const focus = (address: string) => hyprland.messageAsync(
|
||||
`dispatch focuswindow address:${address}`)
|
||||
|
||||
const DummyItem = (address: string) => Widget.Box({
|
||||
attribute: { address },
|
||||
visible: false,
|
||||
})
|
||||
|
||||
const AppItem = (address: string) => {
|
||||
const client = hyprland.getClient(address)
|
||||
if (!client || client.class === "")
|
||||
return DummyItem(address)
|
||||
|
||||
const app = apps.list.find(app => app.match(client.class))
|
||||
|
||||
const btn = PanelButton({
|
||||
class_name: "panel-button",
|
||||
tooltip_text: Utils.watch(client.title, hyprland, () =>
|
||||
hyprland.getClient(address)?.title || "",
|
||||
),
|
||||
on_primary_click: () => focus(address),
|
||||
on_middle_click: () => app && launchApp(app),
|
||||
child: Widget.Icon({
|
||||
size: iconSize.bind(),
|
||||
icon: monochrome.bind().as(m => icon(
|
||||
(app?.icon_name || client.class) + (m ? "-symbolic" : ""),
|
||||
icons.fallback.executable + (m ? "-symbolic" : ""),
|
||||
)),
|
||||
}),
|
||||
})
|
||||
|
||||
return Widget.Box(
|
||||
{
|
||||
attribute: { address },
|
||||
visible: Utils.watch(true, [exclusive, hyprland], () => {
|
||||
return exclusive.value
|
||||
? hyprland.active.workspace.id === client.workspace.id
|
||||
: true
|
||||
}),
|
||||
},
|
||||
Widget.Overlay({
|
||||
child: btn,
|
||||
pass_through: true,
|
||||
overlay: Widget.Box({
|
||||
className: "indicator",
|
||||
hpack: "center",
|
||||
vpack: position.bind().as(p => p === "top" ? "start" : "end"),
|
||||
setup: w => w.hook(hyprland, () => {
|
||||
w.toggleClassName("active", hyprland.active.client.address === address)
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
function sortItems<T extends { attribute: { address: string } }>(arr: T[]) {
|
||||
return arr.sort(({ attribute: a }, { attribute: b }) => {
|
||||
const aclient = hyprland.getClient(a.address)!
|
||||
const bclient = hyprland.getClient(b.address)!
|
||||
return aclient.workspace.id - bclient.workspace.id
|
||||
})
|
||||
}
|
||||
|
||||
export default () => Widget.Box({
|
||||
class_name: "taskbar",
|
||||
children: sortItems(hyprland.clients.map(c => AppItem(c.address))),
|
||||
setup: w => w
|
||||
.hook(hyprland, (w, address?: string) => {
|
||||
if (typeof address === "string")
|
||||
w.children = w.children.filter(ch => ch.attribute.address !== address)
|
||||
}, "client-removed")
|
||||
.hook(hyprland, (w, address?: string) => {
|
||||
if (typeof address === "string")
|
||||
w.children = sortItems([...w.children, AppItem(address)])
|
||||
}, "client-added")
|
||||
.hook(hyprland, (w, event?: string) => {
|
||||
if (event === "movewindow")
|
||||
w.children = sortItems(w.children)
|
||||
}, "event"),
|
||||
})
|
|
@ -1,38 +0,0 @@
|
|||
import PanelButton from "../PanelButton"
|
||||
import options from "options"
|
||||
import { sh, range } from "lib/utils"
|
||||
|
||||
const hyprland = await Service.import("hyprland")
|
||||
const { workspaces } = options.bar.workspaces
|
||||
|
||||
const dispatch = (arg: string | number) => {
|
||||
sh(`hyprctl dispatch workspace ${arg}`)
|
||||
}
|
||||
|
||||
const Workspaces = (ws: number) => Widget.Box({
|
||||
children: range(ws || 20).map(i => Widget.Label({
|
||||
attribute: i,
|
||||
vpack: "center",
|
||||
label: `${i}`,
|
||||
setup: self => self.hook(hyprland, () => {
|
||||
self.toggleClassName("active", hyprland.active.workspace.id === i)
|
||||
self.toggleClassName("occupied", (hyprland.getWorkspace(i)?.windows || 0) > 0)
|
||||
}),
|
||||
})),
|
||||
setup: box => {
|
||||
if (ws === 0) {
|
||||
box.hook(hyprland.active.workspace, () => box.children.map(btn => {
|
||||
btn.visible = hyprland.workspaces.some(ws => ws.id === btn.attribute)
|
||||
}))
|
||||
}
|
||||
},
|
||||
})
|
||||
|
||||
export default () => PanelButton({
|
||||
window: "overview",
|
||||
class_name: "workspaces",
|
||||
on_scroll_up: () => dispatch("m+1"),
|
||||
on_scroll_down: () => dispatch("m-1"),
|
||||
on_clicked: () => App.toggleWindow("overview"),
|
||||
child: workspaces.bind().as(Workspaces),
|
||||
})
|
|
@ -1,37 +0,0 @@
|
|||
import { clock, uptime } from "lib/variables"
|
||||
|
||||
function up(up: number) {
|
||||
const h = Math.floor(up / 60)
|
||||
const m = Math.floor(up % 60)
|
||||
return `uptime: ${h}:${m < 10 ? "0" + m : m}`
|
||||
}
|
||||
|
||||
export default () => Widget.Box({
|
||||
vertical: true,
|
||||
class_name: "date-column vertical",
|
||||
children: [
|
||||
Widget.Box({
|
||||
class_name: "clock-box",
|
||||
vertical: true,
|
||||
children: [
|
||||
Widget.Label({
|
||||
class_name: "clock",
|
||||
label: clock.bind().as(t => t.format("%H:%M")!),
|
||||
}),
|
||||
Widget.Label({
|
||||
class_name: "uptime",
|
||||
label: uptime.bind().as(up),
|
||||
}),
|
||||
],
|
||||
}),
|
||||
Widget.Box({
|
||||
class_name: "calendar",
|
||||
children: [
|
||||
Widget.Calendar({
|
||||
hexpand: true,
|
||||
hpack: "center",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
],
|
||||
})
|
|
@ -1,36 +0,0 @@
|
|||
import PopupWindow from "widget/PopupWindow"
|
||||
import NotificationColumn from "./NotificationColumn"
|
||||
import DateColumn from "./DateColumn"
|
||||
import options from "options"
|
||||
|
||||
const { bar, datemenu } = options
|
||||
const pos = bar.position.bind()
|
||||
const layout = Utils.derive([bar.position, datemenu.position], (bar, qs) =>
|
||||
`${bar}-${qs}` as const,
|
||||
)
|
||||
|
||||
const Settings = () => Widget.Box({
|
||||
class_name: "datemenu horizontal",
|
||||
vexpand: false,
|
||||
children: [
|
||||
NotificationColumn(),
|
||||
Widget.Separator({ orientation: 1 }),
|
||||
DateColumn(),
|
||||
],
|
||||
})
|
||||
|
||||
const DateMenu = () => PopupWindow({
|
||||
name: "datemenu",
|
||||
exclusivity: "exclusive",
|
||||
transition: pos.as(pos => pos === "top" ? "slide_down" : "slide_up"),
|
||||
layout: layout.value,
|
||||
child: Settings(),
|
||||
})
|
||||
|
||||
export function setupDateMenu() {
|
||||
App.addWindow(DateMenu())
|
||||
layout.connect("changed", () => {
|
||||
App.removeWindow("datemenu")
|
||||
App.addWindow(DateMenu())
|
||||
})
|
||||
}
|
|
@ -1,113 +0,0 @@
|
|||
import { type Notification as Notif } from "types/service/notifications"
|
||||
import Notification from "widget/notifications/Notification"
|
||||
import options from "options"
|
||||
import icons from "lib/icons"
|
||||
|
||||
const notifications = await Service.import("notifications")
|
||||
const notifs = notifications.bind("notifications")
|
||||
|
||||
const Animated = (n: Notif) => Widget.Revealer({
|
||||
transition_duration: options.transition.value,
|
||||
transition: "slide_down",
|
||||
child: Notification(n),
|
||||
setup: self => Utils.timeout(options.transition.value, () => {
|
||||
if (!self.is_destroyed)
|
||||
self.reveal_child = true
|
||||
}),
|
||||
})
|
||||
|
||||
const ClearButton = () => Widget.Button({
|
||||
on_clicked: notifications.clear,
|
||||
sensitive: notifs.as(n => n.length > 0),
|
||||
child: Widget.Box({
|
||||
children: [
|
||||
Widget.Label("Clear "),
|
||||
Widget.Icon({
|
||||
icon: notifs.as(n => icons.trash[n.length > 0 ? "full" : "empty"]),
|
||||
}),
|
||||
],
|
||||
}),
|
||||
})
|
||||
|
||||
const Header = () => Widget.Box({
|
||||
class_name: "header",
|
||||
children: [
|
||||
Widget.Label({ label: "Notifications", hexpand: true, xalign: 0 }),
|
||||
ClearButton(),
|
||||
],
|
||||
})
|
||||
|
||||
const NotificationList = () => {
|
||||
const map: Map<number, ReturnType<typeof Animated>> = new Map
|
||||
const box = Widget.Box({
|
||||
vertical: true,
|
||||
children: notifications.notifications.map(n => {
|
||||
const w = Animated(n)
|
||||
map.set(n.id, w)
|
||||
return w
|
||||
}),
|
||||
visible: notifs.as(n => n.length > 0),
|
||||
})
|
||||
|
||||
function remove(_: unknown, id: number) {
|
||||
const n = map.get(id)
|
||||
if (n) {
|
||||
n.reveal_child = false
|
||||
Utils.timeout(options.transition.value, () => {
|
||||
n.destroy()
|
||||
map.delete(id)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
return box
|
||||
.hook(notifications, remove, "closed")
|
||||
.hook(notifications, (_, id: number) => {
|
||||
if (id !== undefined) {
|
||||
if (map.has(id))
|
||||
remove(null, id)
|
||||
|
||||
const n = notifications.getNotification(id)!
|
||||
|
||||
const w = Animated(n)
|
||||
map.set(id, w)
|
||||
box.children = [w, ...box.children]
|
||||
}
|
||||
}, "notified")
|
||||
}
|
||||
|
||||
const Placeholder = () => Widget.Box({
|
||||
class_name: "placeholder",
|
||||
vertical: true,
|
||||
vpack: "center",
|
||||
hpack: "center",
|
||||
vexpand: true,
|
||||
hexpand: true,
|
||||
visible: notifs.as(n => n.length === 0),
|
||||
children: [
|
||||
Widget.Icon(icons.notifications.silent),
|
||||
Widget.Label("Your inbox is empty"),
|
||||
],
|
||||
})
|
||||
|
||||
export default () => Widget.Box({
|
||||
class_name: "notifications",
|
||||
css: options.notifications.width.bind().as(w => `min-width: ${w}px`),
|
||||
vertical: true,
|
||||
children: [
|
||||
Header(),
|
||||
Widget.Scrollable({
|
||||
vexpand: true,
|
||||
hscroll: "never",
|
||||
class_name: "notification-scrollable",
|
||||
child: Widget.Box({
|
||||
class_name: "notification-list vertical",
|
||||
vertical: true,
|
||||
children: [
|
||||
NotificationList(),
|
||||
Placeholder(),
|
||||
],
|
||||
}),
|
||||
}),
|
||||
],
|
||||
})
|
|
@ -1,40 +0,0 @@
|
|||
import options from "options"
|
||||
import { matugen } from "lib/matugen"
|
||||
const mpris = await Service.import("mpris")
|
||||
|
||||
const pref = () => options.bar.media.preferred.value
|
||||
|
||||
export default (monitor: number) => Widget.Window({
|
||||
monitor,
|
||||
layer: "bottom",
|
||||
name: `desktop${monitor}`,
|
||||
class_name: "desktop",
|
||||
anchor: ["top", "bottom", "left", "right"],
|
||||
child: Widget.Box({
|
||||
expand: true,
|
||||
css: options.theme.dark.primary.bg.bind().as(c => `
|
||||
transition: 500ms;
|
||||
background-color: ${c}`),
|
||||
child: Widget.Box({
|
||||
class_name: "wallpaper",
|
||||
expand: true,
|
||||
vpack: "center",
|
||||
hpack: "center",
|
||||
setup: self => self
|
||||
.hook(mpris, () => {
|
||||
const img = mpris.getPlayer(pref())!.cover_path
|
||||
matugen("image", img)
|
||||
Utils.timeout(500, () => self.css = `
|
||||
background-image: url('${img}');
|
||||
background-size: contain;
|
||||
background-repeat: no-repeat;
|
||||
transition: 200ms;
|
||||
min-width: 700px;
|
||||
min-height: 700px;
|
||||
border-radius: 30px;
|
||||
box-shadow: 25px 25px 30px 0 rgba(0,0,0,0.5);`,
|
||||
)
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
})
|
|
@ -1,130 +0,0 @@
|
|||
import { type Application } from "types/service/applications"
|
||||
import { launchApp, icon } from "lib/utils"
|
||||
import options from "options"
|
||||
import icons from "lib/icons"
|
||||
|
||||
const apps = await Service.import("applications")
|
||||
const { query } = apps
|
||||
const { iconSize } = options.launcher.apps
|
||||
|
||||
const QuickAppButton = (app: Application) => Widget.Button({
|
||||
hexpand: true,
|
||||
tooltip_text: app.name,
|
||||
on_clicked: () => {
|
||||
App.closeWindow("launcher")
|
||||
launchApp(app)
|
||||
},
|
||||
child: Widget.Icon({
|
||||
size: iconSize.bind(),
|
||||
icon: icon(app.icon_name, icons.fallback.executable),
|
||||
}),
|
||||
})
|
||||
|
||||
const AppItem = (app: Application) => {
|
||||
const title = Widget.Label({
|
||||
class_name: "title",
|
||||
label: app.name,
|
||||
hexpand: true,
|
||||
xalign: 0,
|
||||
vpack: "center",
|
||||
truncate: "end",
|
||||
})
|
||||
|
||||
const description = Widget.Label({
|
||||
class_name: "description",
|
||||
label: app.description || "",
|
||||
hexpand: true,
|
||||
wrap: true,
|
||||
max_width_chars: 30,
|
||||
xalign: 0,
|
||||
justification: "left",
|
||||
vpack: "center",
|
||||
})
|
||||
|
||||
const appicon = Widget.Icon({
|
||||
icon: icon(app.icon_name, icons.fallback.executable),
|
||||
size: iconSize.bind(),
|
||||
})
|
||||
|
||||
const textBox = Widget.Box({
|
||||
vertical: true,
|
||||
vpack: "center",
|
||||
children: app.description ? [title, description] : [title],
|
||||
})
|
||||
|
||||
return Widget.Button({
|
||||
class_name: "app-item",
|
||||
attribute: { app },
|
||||
child: Widget.Box({
|
||||
children: [appicon, textBox],
|
||||
}),
|
||||
on_clicked: () => {
|
||||
App.closeWindow("launcher")
|
||||
launchApp(app)
|
||||
},
|
||||
})
|
||||
}
|
||||
export function Favorites() {
|
||||
const favs = options.launcher.apps.favorites.bind()
|
||||
return Widget.Revealer({
|
||||
visible: favs.as(f => f.length > 0),
|
||||
child: Widget.Box({
|
||||
vertical: true,
|
||||
children: favs.as(favs => favs.flatMap(fs => [
|
||||
Widget.Separator(),
|
||||
Widget.Box({
|
||||
class_name: "quicklaunch horizontal",
|
||||
children: fs
|
||||
.map(f => query(f)?.[0])
|
||||
.filter(f => f)
|
||||
.map(QuickAppButton),
|
||||
}),
|
||||
])),
|
||||
}),
|
||||
})
|
||||
}
|
||||
|
||||
export function Launcher() {
|
||||
const applist = Variable(query(""))
|
||||
const max = options.launcher.apps.max
|
||||
let first = applist.value[0]
|
||||
|
||||
function SeparatedAppItem(app: Application) {
|
||||
return Widget.Revealer(
|
||||
{ attribute: { app } },
|
||||
Widget.Box(
|
||||
{ vertical: true },
|
||||
Widget.Separator(),
|
||||
AppItem(app),
|
||||
),
|
||||
)
|
||||
}
|
||||
|
||||
const list = Widget.Box({
|
||||
vertical: true,
|
||||
children: applist.bind().as(list => list.map(SeparatedAppItem)),
|
||||
setup: self => self
|
||||
.hook(apps, () => applist.value = query(""), "notify::frequents"),
|
||||
})
|
||||
|
||||
return Object.assign(list, {
|
||||
filter(text: string | null) {
|
||||
first = query(text || "")[0]
|
||||
list.children.reduce((i, item) => {
|
||||
if (!text || i >= max.value) {
|
||||
item.reveal_child = false
|
||||
return i
|
||||
}
|
||||
if (item.attribute.app.match(text)) {
|
||||
item.reveal_child = true
|
||||
return ++i
|
||||
}
|
||||
item.reveal_child = false
|
||||
return i
|
||||
}, 0)
|
||||
},
|
||||
launchFirst() {
|
||||
launchApp(first)
|
||||
},
|
||||
})
|
||||
}
|
|
@ -1,139 +0,0 @@
|
|||
import { type Binding } from "lib/utils"
|
||||
import PopupWindow, { Padding } from "widget/PopupWindow"
|
||||
import icons from "lib/icons"
|
||||
import options from "options"
|
||||
import nix from "service/nix"
|
||||
import * as AppLauncher from "./AppLauncher"
|
||||
import * as NixRun from "./NixRun"
|
||||
import * as ShRun from "./ShRun"
|
||||
|
||||
const { width, margin } = options.launcher
|
||||
const isnix = nix.available
|
||||
|
||||
function Launcher() {
|
||||
const favs = AppLauncher.Favorites()
|
||||
const applauncher = AppLauncher.Launcher()
|
||||
const sh = ShRun.ShRun()
|
||||
const shicon = ShRun.Icon()
|
||||
const nix = NixRun.NixRun()
|
||||
const nixload = NixRun.Spinner()
|
||||
|
||||
function HelpButton(cmd: string, desc: string | Binding<string>) {
|
||||
return Widget.Box(
|
||||
{ vertical: true },
|
||||
Widget.Separator(),
|
||||
Widget.Button(
|
||||
{
|
||||
class_name: "help",
|
||||
on_clicked: () => {
|
||||
entry.grab_focus()
|
||||
entry.text = `:${cmd} `
|
||||
entry.set_position(-1)
|
||||
},
|
||||
},
|
||||
Widget.Box([
|
||||
Widget.Label({
|
||||
class_name: "name",
|
||||
label: `:${cmd}`,
|
||||
}),
|
||||
Widget.Label({
|
||||
hexpand: true,
|
||||
hpack: "end",
|
||||
class_name: "description",
|
||||
label: desc,
|
||||
}),
|
||||
]),
|
||||
),
|
||||
)
|
||||
}
|
||||
|
||||
const help = Widget.Revealer({
|
||||
child: Widget.Box(
|
||||
{ vertical: true },
|
||||
HelpButton("sh", "run a binary"),
|
||||
isnix ? HelpButton("nx", options.launcher.nix.pkgs.bind().as(pkg =>
|
||||
`run a nix package from ${pkg}`,
|
||||
)) : Widget.Box(),
|
||||
),
|
||||
})
|
||||
|
||||
const entry = Widget.Entry({
|
||||
hexpand: true,
|
||||
primary_icon_name: icons.ui.search,
|
||||
on_accept: ({ text }) => {
|
||||
if (text?.startsWith(":nx"))
|
||||
nix.run(text.substring(3))
|
||||
else if (text?.startsWith(":sh"))
|
||||
sh.run(text.substring(3))
|
||||
else
|
||||
applauncher.launchFirst()
|
||||
|
||||
App.toggleWindow("launcher")
|
||||
entry.text = ""
|
||||
},
|
||||
on_change: ({ text }) => {
|
||||
text ||= ""
|
||||
favs.reveal_child = text === ""
|
||||
help.reveal_child = text.split(" ").length === 1 && text?.startsWith(":")
|
||||
|
||||
if (text?.startsWith(":nx"))
|
||||
nix.filter(text.substring(3))
|
||||
else
|
||||
nix.filter("")
|
||||
|
||||
if (text?.startsWith(":sh"))
|
||||
sh.filter(text.substring(3))
|
||||
else
|
||||
sh.filter("")
|
||||
|
||||
if (!text?.startsWith(":"))
|
||||
applauncher.filter(text)
|
||||
},
|
||||
})
|
||||
|
||||
function focus() {
|
||||
entry.text = "Search"
|
||||
entry.set_position(-1)
|
||||
entry.select_region(0, -1)
|
||||
entry.grab_focus()
|
||||
favs.reveal_child = true
|
||||
}
|
||||
|
||||
const layout = Widget.Box({
|
||||
css: width.bind().as(v => `min-width: ${v}pt;`),
|
||||
class_name: "launcher",
|
||||
vertical: true,
|
||||
vpack: "start",
|
||||
setup: self => self.hook(App, (_, win, visible) => {
|
||||
if (win !== "launcher")
|
||||
return
|
||||
|
||||
entry.text = ""
|
||||
if (visible)
|
||||
focus()
|
||||
}),
|
||||
children: [
|
||||
Widget.Box([entry, nixload, shicon]),
|
||||
favs,
|
||||
help,
|
||||
applauncher,
|
||||
nix,
|
||||
sh,
|
||||
],
|
||||
})
|
||||
|
||||
return Widget.Box(
|
||||
{ vertical: true, css: "padding: 1px" },
|
||||
Padding("launcher", {
|
||||
css: margin.bind().as(v => `min-height: ${v}pt;`),
|
||||
vexpand: false,
|
||||
}),
|
||||
layout,
|
||||
)
|
||||
}
|
||||
|
||||
export default () => PopupWindow({
|
||||
name: "launcher",
|
||||
layout: "top",
|
||||
child: Launcher(),
|
||||
})
|
|
@ -1,118 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import nix, { type Nixpkg } from "service/nix"
|
||||
|
||||
const iconVisible = Variable(false)
|
||||
|
||||
function Item(pkg: Nixpkg) {
|
||||
const name = Widget.Label({
|
||||
class_name: "name",
|
||||
label: pkg.name.split(".").at(-1),
|
||||
})
|
||||
|
||||
const subpkg = pkg.name.includes(".") ? Widget.Label({
|
||||
class_name: "description",
|
||||
hpack: "end",
|
||||
hexpand: true,
|
||||
label: ` ${pkg.name.split(".").slice(0, -1).join(".")}`,
|
||||
}) : null
|
||||
|
||||
const version = Widget.Label({
|
||||
class_name: "version",
|
||||
label: pkg.version,
|
||||
hexpand: true,
|
||||
hpack: "end",
|
||||
})
|
||||
|
||||
const description = pkg.description ? Widget.Label({
|
||||
class_name: "description",
|
||||
label: pkg.description,
|
||||
justification: "left",
|
||||
wrap: true,
|
||||
hpack: "start",
|
||||
max_width_chars: 40,
|
||||
}) : null
|
||||
|
||||
return Widget.Box(
|
||||
{
|
||||
attribute: { name: pkg.name },
|
||||
vertical: true,
|
||||
},
|
||||
Widget.Separator(),
|
||||
Widget.Button(
|
||||
{
|
||||
class_name: "nix-item",
|
||||
on_clicked: () => {
|
||||
nix.run(pkg.name)
|
||||
App.closeWindow("launcher")
|
||||
},
|
||||
},
|
||||
Widget.Box(
|
||||
{ vertical: true },
|
||||
Widget.Box([name, version]),
|
||||
Widget.Box([
|
||||
description as ReturnType<typeof Widget.Label>,
|
||||
subpkg as ReturnType<typeof Widget.Label>,
|
||||
]),
|
||||
),
|
||||
),
|
||||
)
|
||||
}
|
||||
|
||||
export function Spinner() {
|
||||
const icon = Widget.Icon({
|
||||
icon: icons.nix.nix,
|
||||
class_name: "spinner",
|
||||
css: `
|
||||
@keyframes spin {
|
||||
to { -gtk-icon-transform: rotate(1turn); }
|
||||
}
|
||||
|
||||
image.spinning {
|
||||
animation-name: spin;
|
||||
animation-duration: 1s;
|
||||
animation-timing-function: linear;
|
||||
animation-iteration-count: infinite;
|
||||
}
|
||||
`,
|
||||
setup: self => self.hook(nix, () => {
|
||||
self.toggleClassName("spinning", !nix.ready)
|
||||
}),
|
||||
})
|
||||
|
||||
return Widget.Revealer({
|
||||
transition: "slide_left",
|
||||
child: icon,
|
||||
reveal_child: Utils.merge([
|
||||
nix.bind("ready"),
|
||||
iconVisible.bind(),
|
||||
], (ready, show) => !ready || show),
|
||||
})
|
||||
}
|
||||
|
||||
export function NixRun() {
|
||||
const list = Widget.Box<ReturnType<typeof Item>>({
|
||||
vertical: true,
|
||||
})
|
||||
|
||||
const revealer = Widget.Revealer({
|
||||
child: list,
|
||||
})
|
||||
|
||||
async function filter(term: string) {
|
||||
iconVisible.value = Boolean(term)
|
||||
|
||||
if (!term)
|
||||
revealer.reveal_child = false
|
||||
|
||||
if (term.trim()) {
|
||||
const found = await nix.query(term)
|
||||
list.children = found.map(k => Item(nix.db[k]))
|
||||
revealer.reveal_child = true
|
||||
}
|
||||
}
|
||||
|
||||
return Object.assign(revealer, {
|
||||
filter,
|
||||
run: nix.run,
|
||||
})
|
||||
}
|
|
@ -1,66 +0,0 @@
|
|||
import icons from "lib/icons"
|
||||
import sh from "service/sh"
|
||||
|
||||
const iconVisible = Variable(false)
|
||||
|
||||
function Item(bin: string) {
|
||||
return Widget.Box(
|
||||
{
|
||||
attribute: { bin },
|
||||
vertical: true,
|
||||
},
|
||||
Widget.Separator(),
|
||||
Widget.Button({
|
||||
child: Widget.Label({
|
||||
label: bin,
|
||||
hpack: "start",
|
||||
}),
|
||||
class_name: "sh-item",
|
||||
on_clicked: () => {
|
||||
Utils.execAsync(bin)
|
||||
App.closeWindow("launcher")
|
||||
},
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
export function Icon() {
|
||||
const icon = Widget.Icon({
|
||||
icon: icons.app.terminal,
|
||||
class_name: "spinner",
|
||||
})
|
||||
|
||||
return Widget.Revealer({
|
||||
transition: "slide_left",
|
||||
child: icon,
|
||||
reveal_child: iconVisible.bind(),
|
||||
})
|
||||
}
|
||||
|
||||
export function ShRun() {
|
||||
const list = Widget.Box<ReturnType<typeof Item>>({
|
||||
vertical: true,
|
||||
})
|
||||
|
||||
const revealer = Widget.Revealer({
|
||||
child: list,
|
||||
})
|
||||
|
||||
async function filter(term: string) {
|
||||
iconVisible.value = Boolean(term)
|
||||
|
||||
if (!term)
|
||||
revealer.reveal_child = false
|
||||
|
||||
if (term.trim()) {
|
||||
const found = await sh.query(term)
|
||||
list.children = found.map(Item)
|
||||
revealer.reveal_child = true
|
||||
}
|
||||
}
|
||||
|
||||
return Object.assign(revealer, {
|
||||
filter,
|
||||
run: sh.run,
|
||||
})
|
||||
}
|
|
@ -1,138 +0,0 @@
|
|||
import { type Notification } from "types/service/notifications"
|
||||
import GLib from "gi://GLib"
|
||||
import icons from "lib/icons"
|
||||
|
||||
const time = (time: number, format = "%H:%M") => GLib.DateTime
|
||||
.new_from_unix_local(time)
|
||||
.format(format)
|
||||
|
||||
const NotificationIcon = ({ app_entry, app_icon, image }: Notification) => {
|
||||
if (image) {
|
||||
return Widget.Box({
|
||||
vpack: "start",
|
||||
hexpand: false,
|
||||
class_name: "icon img",
|
||||
css: `
|
||||
background-image: url("${image}");
|
||||
background-size: cover;
|
||||
background-repeat: no-repeat;
|
||||
background-position: center;
|
||||
min-width: 78px;
|
||||
min-height: 78px;
|
||||
`,
|
||||
})
|
||||
}
|
||||
|
||||
let icon = icons.fallback.notification
|
||||
if (Utils.lookUpIcon(app_icon))
|
||||
icon = app_icon
|
||||
|
||||
if (Utils.lookUpIcon(app_entry || ""))
|
||||
icon = app_entry || ""
|
||||
|
||||
return Widget.Box({
|
||||
vpack: "start",
|
||||
hexpand: false,
|
||||
class_name: "icon",
|
||||
css: `
|
||||
min-width: 78px;
|
||||
min-height: 78px;
|
||||
`,
|
||||
child: Widget.Icon({
|
||||
icon,
|
||||
size: 58,
|
||||
hpack: "center", hexpand: true,
|
||||
vpack: "center", vexpand: true,
|
||||
}),
|
||||
})
|
||||
}
|
||||
|
||||
export default (notification: Notification) => {
|
||||
const content = Widget.Box({
|
||||
class_name: "content",
|
||||
children: [
|
||||
NotificationIcon(notification),
|
||||
Widget.Box({
|
||||
hexpand: true,
|
||||
vertical: true,
|
||||
children: [
|
||||
Widget.Box({
|
||||
children: [
|
||||
Widget.Label({
|
||||
class_name: "title",
|
||||
xalign: 0,
|
||||
justification: "left",
|
||||
hexpand: true,
|
||||
max_width_chars: 24,
|
||||
truncate: "end",
|
||||
wrap: true,
|
||||
label: notification.summary.trim(),
|
||||
use_markup: true,
|
||||
}),
|
||||
Widget.Label({
|
||||
class_name: "time",
|
||||
vpack: "start",
|
||||
label: time(notification.time),
|
||||
}),
|
||||
Widget.Button({
|
||||
class_name: "close-button",
|
||||
vpack: "start",
|
||||
child: Widget.Icon("window-close-symbolic"),
|
||||
on_clicked: notification.close,
|
||||
}),
|
||||
],
|
||||
}),
|
||||
Widget.Label({
|
||||
class_name: "description",
|
||||
hexpand: true,
|
||||
use_markup: true,
|
||||
xalign: 0,
|
||||
justification: "left",
|
||||
label: notification.body.trim(),
|
||||
max_width_chars: 24,
|
||||
wrap: true,
|
||||
}),
|
||||
],
|
||||
}),
|
||||
],
|
||||
})
|
||||
|
||||
const actionsbox = notification.actions.length > 0 ? Widget.Revealer({
|
||||
transition: "slide_down",
|
||||
child: Widget.EventBox({
|
||||
child: Widget.Box({
|
||||
class_name: "actions horizontal",
|
||||
children: notification.actions.map(action => Widget.Button({
|
||||
class_name: "action-button",
|
||||
on_clicked: () => notification.invoke(action.id),
|
||||
hexpand: true,
|
||||
child: Widget.Label(action.label),
|
||||
})),
|
||||
}),
|
||||
}),
|
||||
}) : null
|
||||
|
||||
const eventbox = Widget.EventBox({
|
||||
vexpand: false,
|
||||
on_primary_click: notification.dismiss,
|
||||
on_hover() {
|
||||
if (actionsbox)
|
||||
actionsbox.reveal_child = true
|
||||
},
|
||||
on_hover_lost() {
|
||||
if (actionsbox)
|
||||
actionsbox.reveal_child = true
|
||||
|
||||
notification.dismiss()
|
||||
},
|
||||
child: Widget.Box({
|
||||
vertical: true,
|
||||
children: actionsbox ? [content, actionsbox] : [content],
|
||||
}),
|
||||
})
|
||||
|
||||
return Widget.Box({
|
||||
class_name: `notification ${notification.urgency}`,
|
||||
child: eventbox,
|
||||
})
|
||||
}
|
|
@ -1,90 +0,0 @@
|
|||
import Notification from "./Notification"
|
||||
import options from "options"
|
||||
|
||||
const notifications = await Service.import("notifications")
|
||||
const { transition } = options
|
||||
const { position } = options.notifications
|
||||
const { timeout, idle } = Utils
|
||||
|
||||
function Animated(id: number) {
|
||||
const n = notifications.getNotification(id)!
|
||||
const widget = Notification(n)
|
||||
|
||||
const inner = Widget.Revealer({
|
||||
transition: "slide_left",
|
||||
transition_duration: transition.value,
|
||||
child: widget,
|
||||
})
|
||||
|
||||
const outer = Widget.Revealer({
|
||||
transition: "slide_down",
|
||||
transition_duration: transition.value,
|
||||
child: inner,
|
||||
})
|
||||
|
||||
const box = Widget.Box({
|
||||
hpack: "end",
|
||||
child: outer,
|
||||
})
|
||||
|
||||
idle(() => {
|
||||
outer.reveal_child = true
|
||||
timeout(transition.value, () => {
|
||||
inner.reveal_child = true
|
||||
})
|
||||
})
|
||||
|
||||
return Object.assign(box, {
|
||||
dismiss() {
|
||||
inner.reveal_child = false
|
||||
timeout(transition.value, () => {
|
||||
outer.reveal_child = false
|
||||
timeout(transition.value, () => {
|
||||
box.destroy()
|
||||
})
|
||||
})
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
function PopupList() {
|
||||
const map: Map<number, ReturnType<typeof Animated>> = new Map
|
||||
const box = Widget.Box({
|
||||
hpack: "end",
|
||||
vertical: true,
|
||||
css: options.notifications.width.bind().as(w => `min-width: ${w}px;`),
|
||||
})
|
||||
|
||||
function remove(_: unknown, id: number) {
|
||||
map.get(id)?.dismiss()
|
||||
map.delete(id)
|
||||
}
|
||||
|
||||
return box
|
||||
.hook(notifications, (_, id: number) => {
|
||||
if (id !== undefined) {
|
||||
if (map.has(id))
|
||||
remove(null, id)
|
||||
|
||||
if (notifications.dnd)
|
||||
return
|
||||
|
||||
const w = Animated(id)
|
||||
map.set(id, w)
|
||||
box.children = [w, ...box.children]
|
||||
}
|
||||
}, "notified")
|
||||
.hook(notifications, remove, "dismissed")
|
||||
.hook(notifications, remove, "closed")
|
||||
}
|
||||
|
||||
export default (monitor: number) => Widget.Window({
|
||||
monitor,
|
||||
name: `notifications${monitor}`,
|
||||
anchor: position.bind(),
|
||||
class_name: "notifications",
|
||||
child: Widget.Box({
|
||||
css: "padding: 2px;",
|
||||
child: PopupList(),
|
||||
}),
|
||||
})
|
|
@ -1,111 +0,0 @@
|
|||
import { icon } from "lib/utils"
|
||||
import icons from "lib/icons"
|
||||
import Progress from "./Progress"
|
||||
import brightness from "service/brightness"
|
||||
import options from "options"
|
||||
|
||||
const audio = await Service.import("audio")
|
||||
const { progress, microphone } = options.osd
|
||||
|
||||
const DELAY = 2500
|
||||
|
||||
function OnScreenProgress(vertical: boolean) {
|
||||
const indicator = Widget.Icon({
|
||||
size: 42,
|
||||
vpack: "start",
|
||||
})
|
||||
const progress = Progress({
|
||||
vertical,
|
||||
width: vertical ? 42 : 300,
|
||||
height: vertical ? 300 : 42,
|
||||
child: indicator,
|
||||
})
|
||||
|
||||
const revealer = Widget.Revealer({
|
||||
transition: "slide_left",
|
||||
child: progress,
|
||||
})
|
||||
|
||||
let count = 0
|
||||
function show(value: number, icon: string) {
|
||||
revealer.reveal_child = true
|
||||
indicator.icon = icon
|
||||
progress.setValue(value)
|
||||
count++
|
||||
Utils.timeout(DELAY, () => {
|
||||
count--
|
||||
|
||||
if (count === 0)
|
||||
revealer.reveal_child = false
|
||||
})
|
||||
}
|
||||
|
||||
return revealer
|
||||
.hook(brightness, () => show(
|
||||
brightness.screen,
|
||||
icons.brightness.screen,
|
||||
), "notify::screen")
|
||||
.hook(brightness, () => show(
|
||||
brightness.kbd,
|
||||
icons.brightness.keyboard,
|
||||
), "notify::kbd")
|
||||
.hook(audio.speaker, () => show(
|
||||
audio.speaker.volume,
|
||||
icon(audio.speaker.icon_name || "", icons.audio.type.speaker),
|
||||
), "notify::volume")
|
||||
}
|
||||
|
||||
function MicrophoneMute() {
|
||||
const icon = Widget.Icon({
|
||||
class_name: "microphone",
|
||||
})
|
||||
|
||||
const revealer = Widget.Revealer({
|
||||
transition: "slide_up",
|
||||
child: icon,
|
||||
})
|
||||
|
||||
let count = 0
|
||||
let mute = audio.microphone.stream?.is_muted ?? false
|
||||
|
||||
return revealer.hook(audio.microphone, () => Utils.idle(() => {
|
||||
if (mute !== audio.microphone.stream?.is_muted) {
|
||||
mute = audio.microphone.stream!.is_muted
|
||||
icon.icon = icons.audio.mic[mute ? "muted" : "high"]
|
||||
revealer.reveal_child = true
|
||||
count++
|
||||
|
||||
Utils.timeout(DELAY, () => {
|
||||
count--
|
||||
if (count === 0)
|
||||
revealer.reveal_child = false
|
||||
})
|
||||
}
|
||||
}))
|
||||
}
|
||||
|
||||
export default (monitor: number) => Widget.Window({
|
||||
monitor,
|
||||
name: `indicator${monitor}`,
|
||||
class_name: "indicator",
|
||||
layer: "overlay",
|
||||
click_through: true,
|
||||
anchor: ["right", "left", "top", "bottom"],
|
||||
child: Widget.Box({
|
||||
css: "padding: 2px;",
|
||||
expand: true,
|
||||
child: Widget.Overlay(
|
||||
{ child: Widget.Box({ expand: true }) },
|
||||
Widget.Box({
|
||||
hpack: progress.pack.h.bind(),
|
||||
vpack: progress.pack.v.bind(),
|
||||
child: progress.vertical.bind().as(OnScreenProgress),
|
||||
}),
|
||||
Widget.Box({
|
||||
hpack: microphone.pack.h.bind(),
|
||||
vpack: microphone.pack.v.bind(),
|
||||
child: MicrophoneMute(),
|
||||
}),
|
||||
),
|
||||
}),
|
||||
})
|
|
@ -1,74 +0,0 @@
|
|||
import type Gtk from "gi://Gtk?version=3.0"
|
||||
import GLib from "gi://GLib?version=2.0"
|
||||
import { range } from "lib/utils"
|
||||
import options from "options"
|
||||
|
||||
type ProgressProps = {
|
||||
height?: number
|
||||
width?: number
|
||||
vertical?: boolean
|
||||
child: Gtk.Widget
|
||||
}
|
||||
|
||||
export default ({
|
||||
height = 18,
|
||||
width = 180,
|
||||
vertical = false,
|
||||
child,
|
||||
}: ProgressProps) => {
|
||||
const fill = Widget.Box({
|
||||
class_name: "fill",
|
||||
hexpand: vertical,
|
||||
vexpand: !vertical,
|
||||
hpack: vertical ? "fill" : "start",
|
||||
vpack: vertical ? "end" : "fill",
|
||||
child,
|
||||
})
|
||||
|
||||
const container = Widget.Box({
|
||||
class_name: "progress",
|
||||
child: fill,
|
||||
css: `
|
||||
min-width: ${width}px;
|
||||
min-height: ${height}px;
|
||||
`,
|
||||
})
|
||||
|
||||
let fill_size = 0
|
||||
let animations: number[] = []
|
||||
|
||||
return Object.assign(container, {
|
||||
setValue(value: number) {
|
||||
if (value < 0)
|
||||
return
|
||||
|
||||
if (animations.length > 0) {
|
||||
for (const id of animations)
|
||||
GLib.source_remove(id)
|
||||
|
||||
animations = []
|
||||
}
|
||||
|
||||
const axis = vertical ? "height" : "width"
|
||||
const axisv = vertical ? height : width
|
||||
const min = vertical ? width : height
|
||||
const preferred = (axisv - min) * value + min
|
||||
|
||||
if (!fill_size) {
|
||||
fill_size = preferred
|
||||
fill.css = `min-${axis}: ${preferred}px;`
|
||||
return
|
||||
}
|
||||
|
||||
const frames = options.transition.value / 10
|
||||
const goal = preferred - fill_size
|
||||
const step = goal / frames
|
||||
|
||||
animations = range(frames, 0).map(i => Utils.timeout(5 * i, () => {
|
||||
fill_size += step
|
||||
fill.css = `min-${axis}: ${fill_size}px`
|
||||
animations.shift()
|
||||
}))
|
||||
},
|
||||
})
|
||||
}
|