mirror of
https://github.com/anotherhadi/nixy.git
synced 2026-05-20 13:22:34 +02:00
Compare commits
142 Commits
v4.0.0
...
922a412cda
| Author | SHA1 | Date | |
|---|---|---|---|
| 922a412cda | |||
| cedf87698c | |||
| 52561cf335 | |||
| 9429c9f93b | |||
| 5e0dd70cfa | |||
| 9e24c44c53 | |||
| 241db5ce7a | |||
| 00faa213bf | |||
| 0aac097e77 | |||
| 5aaeba1512 | |||
| 4f3cf99b99 | |||
| f7a2aa6256 | |||
| ac3c0886b7 | |||
| 17f976d4e7 | |||
| daaf0d2b27 | |||
| 5f0d3bff5f | |||
| 36ad6aeb57 | |||
| 2326857f65 | |||
| 28b7923e47 | |||
| 29bdd6468a | |||
| 08db0a42c4 | |||
| 66cc2a3958 | |||
| 80afef207a | |||
| d18d67d86e | |||
| 2e10d6feff | |||
| dea17ebf3c | |||
| ae024e308d | |||
| 302f0ceec9 | |||
| 1cddeef739 | |||
| d7b72277f7 | |||
| 5d9ab362a3 | |||
| a6b2f66ac8 | |||
| e2a60b15dc | |||
| d029d6f205 | |||
| d9b4102bed | |||
| a744d85f48 | |||
| 5d50819d49 | |||
| 087f71d2d5 | |||
| 3f36665767 | |||
| 6cb4d1eb96 | |||
| 983c83b981 | |||
| c482e6e075 | |||
| 0a01e8f3e8 | |||
| 641411ad51 | |||
| 86fd60fd3d | |||
| 9a41aff92d | |||
| e4a2a7abd0 | |||
| b4d8e87abe | |||
| 5de0ce50d6 | |||
| aba6dafffa | |||
| 290471fcc2 | |||
| ac90e05258 | |||
| 562f2507c0 | |||
| a393b97680 | |||
| 59901b0081 | |||
| 6ecf475301 | |||
| 75483ea97a | |||
| 4cd0eccde5 | |||
| d32c4a317f | |||
| 5ecc516a2f | |||
| de489d7d9c | |||
| 19bfd1201b | |||
| 20f7bca099 | |||
| 72ed018351 | |||
| b904fd4fde | |||
| b1cf5fb3a1 | |||
| fee44db81c | |||
| 214095c02a | |||
| 29fc893847 | |||
| 3689215480 | |||
| 14168a65ea | |||
| 43ae447b34 | |||
| 07fa66f1b6 | |||
| 168423cf16 | |||
| bde39be205 | |||
| 18137f444a | |||
| e835f2466b | |||
| d2a9fff0d1 | |||
| 081e05c702 | |||
| 4a4b997ef3 | |||
| 37abf49364 | |||
| 36f5ef3792 | |||
| 3d4099c97b | |||
| 89413d06a8 | |||
| 76e3368f3e | |||
| 0f40fccb5e | |||
| cefebdbd0a | |||
| 05cfe684c2 | |||
| f11bd21e00 | |||
| 40477c6562 | |||
| d1fec1128d | |||
| 677a279eda | |||
| 1489261fa9 | |||
| 29d19d5502 | |||
| 2f69c6e1d4 | |||
| 9ad71c5775 | |||
| b5c64e96c4 | |||
| af787952ad | |||
| 4ebf579743 | |||
| 78aa49547d | |||
| d21fc81a19 | |||
| b0684b2e56 | |||
| 49a9f3f169 | |||
| 8caa02139c | |||
| 0c0a487488 | |||
| 6f463fc7f9 | |||
| 3e82b1cd60 | |||
| f9d171cef3 | |||
| 89c61650bc | |||
| c94559d29f | |||
| e91047a3e1 | |||
| 1d75175be8 | |||
| ea18558cb3 | |||
| 72c13294e6 | |||
| a771f685b2 | |||
| 27ad0783bc | |||
| d61e378405 | |||
| baa1ce858d | |||
| f32ea27565 | |||
| b657e5a891 | |||
| 48f3f3170b | |||
| 87778bd3fa | |||
| 5736c8ccf2 | |||
| 31031447ce | |||
| 4fb4058dca | |||
| 33b3ef66f2 | |||
| 52ee08e13b | |||
| 8c4b9340b6 | |||
| a1d57253f0 | |||
| 6e21914113 | |||
| 7acddfafa1 | |||
| 9fb4072893 | |||
| fd644671af | |||
| bfb7c2f4e0 | |||
| 22b5aa9797 | |||
| 4ac3945174 | |||
| d3cdc25481 | |||
| 6a0b3418a7 | |||
| f2c2540cd4 | |||
| 223f785c3a | |||
| 81ab17fed7 | |||
| 8f16767240 |
@@ -1,5 +1,5 @@
|
|||||||
<div align="center">
|
<div align="center">
|
||||||
<img alt="nixy logo" src="https://raw.githubusercontent.com/anotherhadi/nixy/main/.github/assets/logo.png" width="120px" />
|
<img alt="nixy logo" src="./.github/assets/logo.png" width="120px" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
@@ -122,3 +122,11 @@ sudo nixos-rebuild switch --flake ~/.config/nixos#yourhostname
|
|||||||
|
|
||||||
- [CONTRIBUTING](docs/CONTRIBUTING.md): How to contribute
|
- [CONTRIBUTING](docs/CONTRIBUTING.md): How to contribute
|
||||||
- [LICENSE](LICENSE): MIT License
|
- [LICENSE](LICENSE): MIT License
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
<a href="https://github.com/anotherhadi/nixy">github</a> |
|
||||||
|
<a href="https://gitlab.com/anotherhadi_mirror/nixy">gitlab (mirror)</a> |
|
||||||
|
<a href="https://git.hadi.icu/anotherhadi/nixy">gitea (mirror)</a>
|
||||||
|
</div
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 412 KiB After Width: | Height: | Size: 412 KiB |
@@ -1,2 +1,3 @@
|
|||||||
.sops.yaml
|
.sops.yaml
|
||||||
|
.claude/
|
||||||
old/
|
old/
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[//]: # (This file is autogenerated)
|
[//]: # (This file is autogenerated)
|
||||||
<div align="center">
|
<div align="center">
|
||||||
<img alt="nixy logo" src="https://raw.githubusercontent.com/anotherhadi/nixy/main/.github/assets/logo.png" width="120px" />
|
<img alt="nixy logo" src="./.github/assets/logo.png" width="120px" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
@@ -132,3 +132,11 @@ sudo nixos-rebuild switch --flake ~/.config/nixos#yourhostname
|
|||||||
|
|
||||||
- [CONTRIBUTING](docs/CONTRIBUTING.md): How to contribute
|
- [CONTRIBUTING](docs/CONTRIBUTING.md): How to contribute
|
||||||
- [LICENSE](LICENSE): MIT License
|
- [LICENSE](LICENSE): MIT License
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
<a href="https://github.com/anotherhadi/nixy">github</a> |
|
||||||
|
<a href="https://gitlab.com/anotherhadi_mirror/nixy">gitlab (mirror)</a> |
|
||||||
|
<a href="https://git.hadi.icu/anotherhadi/nixy">gitea (mirror)</a>
|
||||||
|
</div
|
||||||
|
|||||||
+12
-36
@@ -1,53 +1,29 @@
|
|||||||
# SERVER
|
# SERVER
|
||||||
|
|
||||||
> Update in comming. Early 2026
|
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
This document describes the architecture and setup of the self-hosted **NixOS server**, which is securely accessible via **Tailscale**. The server is designed for private, secure, and easily manageable self-hosting of various services.
|
This document describes the architecture and setup of the self-hosted **NixOS server**, which is securely accessible via **a Cloudflare Tunnel**.
|
||||||
|
The server is designed for private, secure, and easily manageable self-hosting of various services.
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
## **Why This Setup?**
|
## **Why This Setup?**
|
||||||
|
|
||||||
- **Private & Secure**: Services are only accessible through Tailscale, preventing exposure to the public internet.
|
- **Private & Secure**: Services are only accessible through Cloudflare's access control, preventing exposure to the "public internet".
|
||||||
- **Domain-based Access**: A custom domain (`example.org`) maps to the server's Tailscale IP, making service access simple and consistent.
|
- **Domain-based Access**: A custom domain maps to the server's tunnel, making service access simple and consistent.
|
||||||
- **Automatic SSL Certificates**: Using DNS-01 challenges, valid SSL certificates are generated even though the services are not publicly exposed.
|
- **Modular & Declarative**: Everything is managed through NixOS modules (except for access control), ensuring reproducibility and easy configuration.
|
||||||
- **Modular & Declarative**: Everything is managed through NixOS modules, ensuring reproducibility and easy configuration.
|
|
||||||
|
|
||||||
## **Self-Hosted Services**
|
## **Self-Hosted Services**
|
||||||
|
|
||||||
The server hosts several key applications:
|
The server hosts several key applications:
|
||||||
|
|
||||||
### **Core Infrastructure**
|
- **NGINX**: Reverse proxy for routing traffic to services via my domain name.
|
||||||
|
|
||||||
- **NGINX**: Reverse proxy for routing traffic to services via `example.org`.
|
|
||||||
|
|
||||||
### **Networking & Security**
|
|
||||||
|
|
||||||
- **AdGuard Home**: A self-hosted DNS ad blocker for network-wide ad and tracker filtering.
|
- **AdGuard Home**: A self-hosted DNS ad blocker for network-wide ad and tracker filtering.
|
||||||
|
|
||||||
### **Monitoring & Storage**
|
|
||||||
|
|
||||||
- **Glance**: An awesome dashboard! (See the screenshot above)
|
- **Glance**: An awesome dashboard! (See the screenshot above)
|
||||||
|
|
||||||
### **Media & Content Management**
|
|
||||||
|
|
||||||
- **Arr Stack (Radarr, Sonarr, etc.)**: Automated media management tools for handling movies and TV shows. (legaly ofc)
|
- **Arr Stack (Radarr, Sonarr, etc.)**: Automated media management tools for handling movies and TV shows. (legaly ofc)
|
||||||
|
- **Mealie**: A self-hosted recipe manager and meal planner with a clean user interface.
|
||||||
## **How It Works**
|
- **Stirling-PDF**: A powerful, locally hosted web application for editing, merging, and converting PDF files.
|
||||||
|
- **CyberChef**: The "Cyber Swiss Army Knife" for data analysis, decoding, and encryption tasks.
|
||||||
1. **Domain Configuration**
|
- **Mazanoke**: A utility service for image processing, specialized in format conversion and downgrading/optimization.
|
||||||
- `example.org` is pointed to the Tailscale IP of the server. (cloudflare A record, not proxied)
|
- **SSH**: Secure remote access configuration for server management. (via browser too)
|
||||||
- This allows for easy access without exposing services to the internet.
|
- **Security related stuff**: Cloudflared, Fail2Ban, Firewall
|
||||||
|
|
||||||
2. **SSL Certificate Generation**
|
|
||||||
- Certificates are obtained using a **DNS-01 challenge**, verifying domain ownership without requiring public access.
|
|
||||||
|
|
||||||
3. **NGINX Reverse Proxy**
|
|
||||||
- Routes incoming requests from `*.example.org` to the correct internal service.
|
|
||||||
- Ensures SSL termination and secure connections.
|
|
||||||
|
|
||||||
4. **Access Control**
|
|
||||||
- Only devices within the Tailscale network can reach the services.
|
|
||||||
- Firewall rules restrict access further based on necessity.
|
|
||||||
|
|||||||
+3
-3
@@ -12,6 +12,6 @@ Wallpapers are loaded from the [hadi's awesome-wallpapers](https://github.com/an
|
|||||||
|
|
||||||
### Rose-pine
|
### Rose-pine
|
||||||
|
|
||||||

|

|
||||||

|

|
||||||

|

|
||||||
|
|||||||
Generated
+772
-300
File diff suppressed because it is too large
Load Diff
@@ -7,13 +7,18 @@
|
|||||||
|
|
||||||
inputs = {
|
inputs = {
|
||||||
nixpkgs.url = "github:nixos/nixpkgs/nixos-unstable";
|
nixpkgs.url = "github:nixos/nixpkgs/nixos-unstable";
|
||||||
nixpkgs-stable.url = "github:nixos/nixpkgs/nixos-25.05";
|
nixpkgs-stable.url = "github:nixos/nixpkgs/nixos-25.11";
|
||||||
nixos-hardware.url = "github:NixOS/nixos-hardware/master";
|
nixos-hardware.url = "github:NixOS/nixos-hardware/master";
|
||||||
hyprland.url = "git+https://github.com/hyprwm/Hyprland?submodules=1";
|
hyprland.url = "git+https://github.com/hyprwm/Hyprland?submodules=1";
|
||||||
stylix.url = "github:danth/stylix";
|
stylix.url = "github:danth/stylix";
|
||||||
nixcord.url = "github:kaylorben/nixcord";
|
|
||||||
sops-nix.url = "github:Mic92/sops-nix";
|
sops-nix.url = "github:Mic92/sops-nix";
|
||||||
nvf.url = "github:notashelf/nvf";
|
nvf.url = "github:notashelf/nvf";
|
||||||
|
bun2nix.url = "github:nix-community/bun2nix";
|
||||||
|
|
||||||
|
nix-index-database = {
|
||||||
|
url = "github:nix-community/nix-index-database";
|
||||||
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
|
};
|
||||||
home-manager = {
|
home-manager = {
|
||||||
url = "github:nix-community/home-manager";
|
url = "github:nix-community/home-manager";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
@@ -31,13 +36,18 @@
|
|||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
};
|
};
|
||||||
# Server
|
# Server
|
||||||
eleakxir.url = "github:anotherhadi/eleakxir";
|
# FIXME: Deleted repo for now
|
||||||
|
# eleakxir.url = "github:anotherhadi/eleakxir";
|
||||||
nixarr.url = "github:rasmus-kirk/nixarr";
|
nixarr.url = "github:rasmus-kirk/nixarr";
|
||||||
|
default-creds.url = "github:anotherhadi/default-creds";
|
||||||
|
blog.url = "github:anotherhadi/blog";
|
||||||
|
awesome-wallpapers.url = "github:anotherhadi/awesome-wallpapers";
|
||||||
|
iknowyou.url = "github:anotherhadi/iknowyou";
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = inputs @ {nixpkgs, ...}: {
|
outputs = inputs @ {nixpkgs, ...}: {
|
||||||
nixosConfigurations = {
|
nixosConfigurations = {
|
||||||
nixy =
|
h-laptop =
|
||||||
# CHANGEME: This should match the 'hostname' in your variables.nix file
|
# CHANGEME: This should match the 'hostname' in your variables.nix file
|
||||||
nixpkgs.lib.nixosSystem {
|
nixpkgs.lib.nixosSystem {
|
||||||
modules = [
|
modules = [
|
||||||
@@ -50,9 +60,11 @@
|
|||||||
inputs.nixos-hardware.nixosModules.omen-16-n0005ne # CHANGEME: check https://github.com/NixOS/nixos-hardware
|
inputs.nixos-hardware.nixosModules.omen-16-n0005ne # CHANGEME: check https://github.com/NixOS/nixos-hardware
|
||||||
inputs.home-manager.nixosModules.home-manager
|
inputs.home-manager.nixosModules.home-manager
|
||||||
inputs.stylix.nixosModules.stylix
|
inputs.stylix.nixosModules.stylix
|
||||||
|
inputs.nix-index-database.nixosModules.default
|
||||||
./hosts/laptop/configuration.nix # CHANGEME: change the path to match your host folder
|
./hosts/laptop/configuration.nix # CHANGEME: change the path to match your host folder
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
# Jack is my server
|
# Jack is my server
|
||||||
jack = nixpkgs.lib.nixosSystem {
|
jack = nixpkgs.lib.nixosSystem {
|
||||||
modules = [
|
modules = [
|
||||||
@@ -61,7 +73,9 @@
|
|||||||
inputs.stylix.nixosModules.stylix
|
inputs.stylix.nixosModules.stylix
|
||||||
inputs.sops-nix.nixosModules.sops
|
inputs.sops-nix.nixosModules.sops
|
||||||
inputs.nixarr.nixosModules.default
|
inputs.nixarr.nixosModules.default
|
||||||
inputs.eleakxir.nixosModules.eleakxir
|
# inputs.eleakxir.nixosModules.eleakxir
|
||||||
|
inputs.nix-index-database.nixosModules.default
|
||||||
|
inputs.default-creds.nixosModules.default
|
||||||
./hosts/server/configuration.nix
|
./hosts/server/configuration.nix
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
{
|
||||||
|
programs.brave = {
|
||||||
|
enable = true;
|
||||||
|
commandLineArgs = [
|
||||||
|
# Wayland Native
|
||||||
|
"--enable-features=UseOzonePlatform"
|
||||||
|
"--ozone-platform=wayland"
|
||||||
|
|
||||||
|
# Hardware Acceleration (NVIDIA optimized)
|
||||||
|
"--enable-accelerated-video-decode"
|
||||||
|
"--enable-gpu-rasterization"
|
||||||
|
"--enable-zero-copy"
|
||||||
|
"--ignore-gpu-blocklist"
|
||||||
|
|
||||||
|
# Performance
|
||||||
|
"--enable-features=VaapiVideoDecoder"
|
||||||
|
"--enable-features=VaapiVideoEncoder"
|
||||||
|
"--enable-features=CanvasOopRasterization"
|
||||||
|
"--disable-features=UseChromeOSDirectVideoDecoder"
|
||||||
|
|
||||||
|
# Privacy & Security
|
||||||
|
"--disable-features=MediaRouter" # Disable Chromecast
|
||||||
|
"--disable-features=OptimizationHints" # No Google suggestions
|
||||||
|
"--disable-features=AutofillSavePaymentMethods"
|
||||||
|
"--disable-background-networking" # No telemetry
|
||||||
|
"--disable-sync" # Manual sync control
|
||||||
|
|
||||||
|
# Wayland-specific fixes
|
||||||
|
"--disable-features=WaylandWpColorManagerV1" # Color management fix
|
||||||
|
|
||||||
|
# UI/UX
|
||||||
|
"--force-dark-mode" # Match Stylix theme
|
||||||
|
"--enable-features=WebUIDarkMode"
|
||||||
|
"--no-default-browser-check"
|
||||||
|
];
|
||||||
|
extensions = let
|
||||||
|
ids = [
|
||||||
|
];
|
||||||
|
in
|
||||||
|
map (id: {inherit id;}) ids;
|
||||||
|
};
|
||||||
|
|
||||||
|
xdg.configFile."BraveSoftware/Brave-Browser/Policies/managed/policy.json".text = builtins.toJSON {
|
||||||
|
BraveShieldsAdControl = 2;
|
||||||
|
BraveShieldsTrackersBlocked = 1;
|
||||||
|
BraveShieldsHttpsEverywhere = 1;
|
||||||
|
BraveRewardsDisabled = 1;
|
||||||
|
BraveWalletDisabled = 1;
|
||||||
|
BraveVPNDisabled = 1;
|
||||||
|
BraveAIChatEnabled = 0;
|
||||||
|
PasswordManagerEnabled = 0;
|
||||||
|
BravePlaylistEnabled = 0;
|
||||||
|
BraveWebDiscoveryEnabled = 0;
|
||||||
|
BraveStatsPingEnabled = 0;
|
||||||
|
DnsOverHttpsMode = "automatic";
|
||||||
|
BraveDarkMode = 1;
|
||||||
|
};
|
||||||
|
|
||||||
|
xdg.desktopEntries = {
|
||||||
|
brave-incognito = {
|
||||||
|
name = "Brave (Private window)";
|
||||||
|
genericName = "Navigateur Web";
|
||||||
|
exec = "brave --incognito";
|
||||||
|
icon = "brave-browser";
|
||||||
|
terminal = false;
|
||||||
|
categories = ["Network" "WebBrowser"];
|
||||||
|
mimeType = ["text/html" "text/xml"];
|
||||||
|
};
|
||||||
|
brave-tor = {
|
||||||
|
name = "Brave (Private window w/Tor)";
|
||||||
|
genericName = "Navigateur Web";
|
||||||
|
exec = "brave --tor";
|
||||||
|
icon = "brave-browser";
|
||||||
|
terminal = false;
|
||||||
|
categories = ["Network" "WebBrowser"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
# Discord is a popular chat application.
|
|
||||||
{inputs, ...}: {
|
|
||||||
imports = [inputs.nixcord.homeModules.nixcord];
|
|
||||||
|
|
||||||
programs.nixcord = {
|
|
||||||
enable = true;
|
|
||||||
config = {frameless = true;};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
# Import all fetch scripts
|
|
||||||
{imports = [./neofetch ./nerdfetch];}
|
|
||||||
@@ -1,148 +0,0 @@
|
|||||||
# Legacy
|
|
||||||
{pkgs, ...}: {
|
|
||||||
home.packages = with pkgs; [neofetch];
|
|
||||||
|
|
||||||
xdg.configFile."neofetch/ascii.txt".text = ''
|
|
||||||
''${c6}
|
|
||||||
⡏⠉⠉⠉⠉⠉⠉⠋⠉⠉⠉⠉⠉⠉⠋⠉⠉⠉⠉⠉⠉⠉⠉⠉⠉⠙⠉⠉⠉⢹
|
|
||||||
⡇⢸⣿⡟⠛⢿⣷⠀⢸⣿⡟⠛⢿⣷⡄⢸⣿⡇⠀⢸⣿⡇⢸⣿⡇⠀⢸⣿⡇⢸
|
|
||||||
⡇⢸⣿⣧⣤⣾⠿⠀⢸⣿⣇⣀⣸⡿⠃⢸⣿⡇⠀⢸⣿⡇⢸⣿⣇⣀⣸⣿⡇⢸
|
|
||||||
⡇⢸⣿⡏⠉⢹⣿⡆⢸⣿⡟⠛⢻⣷⡄⢸⣿⡇⠀⢸⣿⡇⢸⣿⡏⠉⢹⣿⡇⢸
|
|
||||||
⡇⢸⣿⣧⣤⣼⡿⠃⢸⣿⡇⠀⢸⣿⡇⠸⣿⣧⣤⣼⡿⠁⢸⣿⡇⠀⢸⣿⡇⢸
|
|
||||||
⣇⣀⣀⣀⣀⣀⣀⣄⣀⣀⣀⣀⣀⣀⣀⣠⣀⡈⠉⣁⣀⣄⣀⣀⣀⣠⣀⣀⣀⣸
|
|
||||||
⣇⣿⠘⣿⣿⣿⡿⡿⣟⣟⢟⢟⢝⠵⡝⣿⡿⢂⣼⣿⣷⣌⠩⡫⡻⣝⠹⢿⣿⣷
|
|
||||||
⡆⣿⣆⠱⣝⡵⣝⢅⠙⣿⢕⢕⢕⢕⢝⣥⢒⠅⣿⣿⣿⡿⣳⣌⠪⡪⣡⢑⢝⣇
|
|
||||||
⡆⣿⣿⣦⠹⣳⣳⣕⢅⠈⢗⢕⢕⢕⢕⢕⢈⢆⠟⠋⠉⠁⠉⠉⠁⠈⠼⢐⢕⢽
|
|
||||||
⡗⢰⣶⣶⣦⣝⢝⢕⢕⠅⡆⢕⢕⢕⢕⢕⣴⠏⣠⡶⠛⡉⡉⡛⢶⣦⡀⠐⣕⢕
|
|
||||||
⡝⡄⢻⢟⣿⣿⣷⣕⣕⣅⣿⣔⣕⣵⣵⣿⣿⢠⣿⢠⣮⡈⣌⠨⠅⠹⣷⡀⢱⢕
|
|
||||||
⡝⡵⠟⠈⢀⣀⣀⡀⠉⢿⣿⣿⣿⣿⣿⣿⣿⣼⣿⢈⡋⠴⢿⡟⣡⡇⣿⡇⡀⢕
|
|
||||||
⡝⠁⣠⣾⠟⡉⡉⡉⠻⣦⣻⣿⣿⣿⣿⣿⣿⣿⣿⣧⠸⣿⣦⣥⣿⡇⡿⣰⢗⢄
|
|
||||||
⠁⢰⣿⡏⣴⣌⠈⣌⠡⠈⢻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣬⣉⣉⣁⣄⢖⢕⢕⢕
|
|
||||||
⡀⢻⣿⡇⢙⠁⠴⢿⡟⣡⡆⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣵⣵⣿
|
|
||||||
⡻⣄⣻⣿⣌⠘⢿⣷⣥⣿⠇⣿⣿⣿⣿⣿⣿⠛⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿
|
|
||||||
⣷⢄⠻⣿⣟⠿⠦⠍⠉⣡⣾⣿⣿⣿⣿⣿⣿⢸⣿⣦⠙⣿⣿⣿⣿⣿⣿⣿⣿⠟
|
|
||||||
⡕⡑⣑⣈⣻⢗⢟⢞⢝⣻⣿⣿⣿⣿⣿⣿⣿⠸⣿⠿⠃⣿⣿⣿⣿⣿⣿⡿⠁⣠
|
|
||||||
⡝⡵⡈⢟⢕⢕⢕⢕⣵⣿⣿⣿⣿⣿⣿⣿⣿⣿⣶⣶⣿⣿⣿⣿⣿⠿⠋⣀⣈⠙
|
|
||||||
⡝⡵⡕⡀⠑⠳⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⠛⢉⡠⡲⡫⡪⡪⡣
|
|
||||||
'';
|
|
||||||
|
|
||||||
xdg.configFile."neofetch/config.conf".text = ''
|
|
||||||
# From: https://github.com/Chick2D/neofetch-themes/
|
|
||||||
# Made by https://github.com/Dan1jel
|
|
||||||
|
|
||||||
print_info() {
|
|
||||||
prin "\n"
|
|
||||||
info "\n \n OS" distro
|
|
||||||
info "\n \n Host" model
|
|
||||||
info "\n \n Kernel" kernel
|
|
||||||
info "\n \n Uptime" uptime
|
|
||||||
info "\n \n Packages" packages
|
|
||||||
info "\n \n Shell" shell
|
|
||||||
info "\n \n Resolution" resolution
|
|
||||||
info "\n \n DE" de
|
|
||||||
info "\n \n WM" wm
|
|
||||||
info "\n \n WM Theme" wm_theme
|
|
||||||
info "\n \n Theme" theme
|
|
||||||
info "\n \n Icons" icons
|
|
||||||
info "\n \n Terminal" term
|
|
||||||
info "\n \n Terminal Font" term_font
|
|
||||||
info "\n \n CPU" cpu
|
|
||||||
info "\n \n GPU" gpu
|
|
||||||
info "\n \n Memory" memory
|
|
||||||
info "\n \n Disk" disk
|
|
||||||
info "\n \n Battery" battery
|
|
||||||
info "\n \n Font" font
|
|
||||||
info "\n \n Song" song
|
|
||||||
[[ "$player" ]] && prin "\n \n Music Player" "$player"
|
|
||||||
info "\n \n Local IP" local_ip
|
|
||||||
prin "\n"
|
|
||||||
|
|
||||||
# prin "\n \n ''${cl0}──''${cl1}────''${cl2}────''${cl3}────''${cl4}────''${cl5}────''${cl6}────''${cl7}──"
|
|
||||||
prin "\n \n \n \n \n \n ''${cl0}⬤ \n \n ''${cl1}⬤ \n \n ''${cl2}⬤ \n \n ''${cl3}⬤ \n \n ''${cl4}⬤ \n \n ''${cl5}⬤ \n \n ''${cl6}⬤ \n \n ''${cl7}⬤"
|
|
||||||
}
|
|
||||||
|
|
||||||
reset="\033[0m"
|
|
||||||
gray="\033[1;90m"
|
|
||||||
red="\033[1;31m"
|
|
||||||
green="\033[1;32m"
|
|
||||||
yellow="\033[1;33m"
|
|
||||||
blue="\033[1;34m"
|
|
||||||
magenta="\033[1;35m"
|
|
||||||
cyan="\033[1;36m"
|
|
||||||
white="\033[1;37m"
|
|
||||||
|
|
||||||
cl0="''${gray}"
|
|
||||||
cl1="''${red}"
|
|
||||||
cl2="''${green}"
|
|
||||||
cl3="''${yellow}"
|
|
||||||
cl4="''${blue}"
|
|
||||||
cl5="''${magenta}"
|
|
||||||
cl6="''${cyan}"
|
|
||||||
cl7="''${white}"
|
|
||||||
|
|
||||||
title_fqdn="off"
|
|
||||||
kernel_shorthand="on"
|
|
||||||
distro_shorthand="off"
|
|
||||||
os_arch="off"
|
|
||||||
uptime_shorthand="on"
|
|
||||||
memory_percent="off"
|
|
||||||
memory_unit="mib"
|
|
||||||
package_managers="on"
|
|
||||||
shell_path="off"
|
|
||||||
shell_version="on"
|
|
||||||
speed_type="bios_limit"
|
|
||||||
speed_shorthand="off"
|
|
||||||
cpu_brand="on"
|
|
||||||
cpu_speed="on"
|
|
||||||
cpu_cores="logical"
|
|
||||||
cpu_temp="off"
|
|
||||||
gpu_brand="on"
|
|
||||||
gpu_type="all"
|
|
||||||
refresh_rate="off"
|
|
||||||
gtk_shorthand="off"
|
|
||||||
gtk2="on"
|
|
||||||
gtk3="on"
|
|
||||||
de_version="on"
|
|
||||||
disk_show=('/storage/emulated' '/' '/server')
|
|
||||||
disk_subtitle="none"
|
|
||||||
disk_percent="on"
|
|
||||||
music_player="auto"
|
|
||||||
song_format="%artist% - %album% - %title%"
|
|
||||||
song_shorthand="off"
|
|
||||||
mpc_args=()
|
|
||||||
bold="on"
|
|
||||||
underline_enabled="on"
|
|
||||||
underline_char="-"
|
|
||||||
separator=":"
|
|
||||||
block_range=(0 15)
|
|
||||||
color_blocks="on"
|
|
||||||
block_width=3
|
|
||||||
block_height=1
|
|
||||||
col_offset="auto"
|
|
||||||
bar_char_elapsed="-"
|
|
||||||
bar_char_total="="
|
|
||||||
bar_border="on"
|
|
||||||
bar_length=15
|
|
||||||
bar_color_elapsed="distro"
|
|
||||||
bar_color_total="distro"
|
|
||||||
cpu_display="off"
|
|
||||||
memory_display="off"
|
|
||||||
battery_display="off"
|
|
||||||
disk_display="off"
|
|
||||||
image_backend="ascii"
|
|
||||||
image_source="$HOME/.config/neofetch/ascii.txt"
|
|
||||||
ascii_distro="auto"
|
|
||||||
ascii_colors=(distro)
|
|
||||||
ascii_bold="on"
|
|
||||||
image_loop="off"
|
|
||||||
thumbnail_dir="''${XDG_CACHE_HOME:-''${HOME}/.cache}/thumbnails/neofetch"
|
|
||||||
crop_mode="normal"
|
|
||||||
crop_offset="center"
|
|
||||||
image_size="auto"
|
|
||||||
gap=2
|
|
||||||
yoffset=0
|
|
||||||
xoffset=0
|
|
||||||
background_color=
|
|
||||||
stdout="off"
|
|
||||||
'';
|
|
||||||
}
|
|
||||||
@@ -1,367 +0,0 @@
|
|||||||
# Nerdfetch, a simple system info script written in bash
|
|
||||||
# Source: https://github.com/ThatOneCalculator/NerdFetch
|
|
||||||
{pkgs, ...}: let
|
|
||||||
nerdfetch = pkgs.writeShellScriptBin "nerdfetch" ''
|
|
||||||
ostype="$(uname)"
|
|
||||||
|
|
||||||
version=8.1.1
|
|
||||||
font=nerd
|
|
||||||
distrotype=none
|
|
||||||
osi=""
|
|
||||||
ki=""
|
|
||||||
ri=""
|
|
||||||
pi=""
|
|
||||||
ui=""
|
|
||||||
ci=""
|
|
||||||
|
|
||||||
case $1 in
|
|
||||||
"-p")
|
|
||||||
font=phosphor
|
|
||||||
osi=""
|
|
||||||
ki=""
|
|
||||||
ri=""
|
|
||||||
pi=""
|
|
||||||
ui=""
|
|
||||||
ci=""
|
|
||||||
;;
|
|
||||||
"-c")
|
|
||||||
font=cozette
|
|
||||||
ki="♥"
|
|
||||||
ri=""
|
|
||||||
pi=""
|
|
||||||
ui=""
|
|
||||||
ci=""
|
|
||||||
;;
|
|
||||||
"-e")
|
|
||||||
font=emoji
|
|
||||||
osi="🐧"
|
|
||||||
ki="💓"
|
|
||||||
ri="🐐"
|
|
||||||
pi="📦"
|
|
||||||
ui="☕"
|
|
||||||
ci="🎨"
|
|
||||||
;;
|
|
||||||
"-v")
|
|
||||||
echo "NerdFetch $version"
|
|
||||||
exit
|
|
||||||
;;
|
|
||||||
"-h")
|
|
||||||
echo "Flags:
|
|
||||||
-c: Cozette font
|
|
||||||
-p: Phosphor font
|
|
||||||
-e: Emoji font
|
|
||||||
-v: Version"
|
|
||||||
exit
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
if command -v getprop 1>/dev/null; then
|
|
||||||
distrotype=android
|
|
||||||
fi
|
|
||||||
kernel="$(echo $(uname -r) | cut -d'-' -f1-1)"
|
|
||||||
case $ostype in
|
|
||||||
*"Linux"*)
|
|
||||||
if [ $distrotype = android ]; then
|
|
||||||
host="$(hostname)"
|
|
||||||
USER="$(whoami)"
|
|
||||||
os="Android $(getprop ro.build.version.release)"
|
|
||||||
case $font in
|
|
||||||
phosphor) osi="" ;;
|
|
||||||
emoji) osi="🤖" ;;
|
|
||||||
*) osi="" ;;
|
|
||||||
esac
|
|
||||||
else
|
|
||||||
host="$(cat /proc/sys/kernel/hostname)"
|
|
||||||
. /etc/os-release
|
|
||||||
if [ -f /bedrock/etc/bedrock-release ]; then
|
|
||||||
os="$(brl version)"
|
|
||||||
else
|
|
||||||
os="''${PRETTY_NAME}"
|
|
||||||
if [ $font = nerd ]; then
|
|
||||||
case $(echo $ID | sed 's/ .*//') in
|
|
||||||
debian) osi="" ;;
|
|
||||||
arch) osi="" ;;
|
|
||||||
endeavouros) osi="" ;;
|
|
||||||
fedora) osi="" ;;
|
|
||||||
gentoo) osi="" ;;
|
|
||||||
rhel) osi="" ;;
|
|
||||||
slackware) osi="" ;;
|
|
||||||
void) osi="" ;;
|
|
||||||
alpine) osi="" ;;
|
|
||||||
nixos) osi="" ;;
|
|
||||||
artix) osi="" ;;
|
|
||||||
exherbo) osi="" ;;
|
|
||||||
mageia) osi="" ;;
|
|
||||||
manjaro) osi="" ;;
|
|
||||||
opensuse) osi="" ;;
|
|
||||||
solus) osi="" ;;
|
|
||||||
ubuntu) osi="" ;;
|
|
||||||
mint) osi="" ;;
|
|
||||||
trisquel) osi="" ;;
|
|
||||||
puppy) osi="" ;;
|
|
||||||
coreos) osi="" ;;
|
|
||||||
mx) osi="" ;;
|
|
||||||
vanilla) osi="" ;;
|
|
||||||
pop_os) osi="" ;;
|
|
||||||
raspbian) osi="" ;;
|
|
||||||
deepin) osi="" ;;
|
|
||||||
almalinux) osi="" ;;
|
|
||||||
garuda) osi="" ;;
|
|
||||||
centos) osi="" ;;
|
|
||||||
rocky) osi="" ;;
|
|
||||||
esac
|
|
||||||
elif [ $font = cozette ]; then
|
|
||||||
case $(echo $ID | sed 's/ .*//') in
|
|
||||||
debian) osi="" ;;
|
|
||||||
arch) osi="" ;;
|
|
||||||
fedora) osi="" ;;
|
|
||||||
gentoo) osi="" ;;
|
|
||||||
slackware) osi="" ;;
|
|
||||||
void) osi="" ;;
|
|
||||||
alpine) osi="" ;;
|
|
||||||
nixos) osi="" ;;
|
|
||||||
mageia) osi="" ;;
|
|
||||||
manjaro) osi="" ;;
|
|
||||||
opensuse) osi="" ;;
|
|
||||||
ubuntu) osi="" ;;
|
|
||||||
mint) osi="" ;;
|
|
||||||
coreos) osi="" ;;
|
|
||||||
centos) osi="" ;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
shell=$(basename "$SHELL")
|
|
||||||
;;
|
|
||||||
*"Darwin"*)
|
|
||||||
host="$(hostname -f | sed -e 's/^[^.]*\.//')"
|
|
||||||
mac_product="$(/usr/libexec/PlistBuddy -c "Print:ProductName" /System/Library/CoreServices/SystemVersion.plist)"
|
|
||||||
mac_version="$(/usr/libexec/PlistBuddy -c "Print:ProductVersion" /System/Library/CoreServices/SystemVersion.plist)"
|
|
||||||
os="''${mac_product} ''${mac_version}"
|
|
||||||
case $font in
|
|
||||||
nerd) osi="" ;;
|
|
||||||
phosphor) osi="" ;;
|
|
||||||
cozette) osi="" ;;
|
|
||||||
emoji) osi="🍎" ;;
|
|
||||||
esac
|
|
||||||
;;
|
|
||||||
*"FreeBSD"*)
|
|
||||||
host="$(hostname)"
|
|
||||||
distrotype=bsd
|
|
||||||
os="FreeBSD $(freebsd-version | sed 's/-.*//')"
|
|
||||||
case $font in
|
|
||||||
nerd) osi="" ;;
|
|
||||||
phosphor) osi="" ;;
|
|
||||||
cozette) osi="" ;;
|
|
||||||
emoji) osi="😈" ;;
|
|
||||||
esac
|
|
||||||
;;
|
|
||||||
*"OpenBSD"*)
|
|
||||||
host="$(hostname)"
|
|
||||||
distrotype=bsd
|
|
||||||
os="OpenBSD $(uname -r)"
|
|
||||||
case $font in
|
|
||||||
nerd) osi="" ;;
|
|
||||||
phosphor) osi="" ;;
|
|
||||||
cozette) osi="⌘" ;;
|
|
||||||
emoji) osi="🐡" ;;
|
|
||||||
esac
|
|
||||||
;;
|
|
||||||
*"NetBSD"*)
|
|
||||||
host="$(hostname)"
|
|
||||||
distrotype=netbsd
|
|
||||||
os="NetBSD $(uname -r)"
|
|
||||||
case $font in
|
|
||||||
nerd) osi="" ;;
|
|
||||||
phosphor) osi="" ;;
|
|
||||||
cozette) osi="" ;;
|
|
||||||
emoji) osi="🚩" ;;
|
|
||||||
esac
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
os="Unix-like"
|
|
||||||
host="host"
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
## PACKAGE MANAGER AND PACKAGES DETECTION
|
|
||||||
|
|
||||||
MANAGER=$(which nix-env pkg flatpak yum zypper dnf rpm dpkg-query brew port pacman xbps-query emerge cave apk kiss pmm /usr/sbin/slackpkg bulge birb yay paru pacstall cpm pmm eopkg getprop 2>/dev/null)
|
|
||||||
manager=$(basename "$MANAGER")
|
|
||||||
if [ $distrotype = netbsd ]; then
|
|
||||||
manager="pkg_info-netbsd"
|
|
||||||
fi
|
|
||||||
case $manager in
|
|
||||||
cpm) packages="$(cpm C)" ;;
|
|
||||||
flatpak) packages="$(flatpak list --app | wc -l)" ;;
|
|
||||||
brew) packages="$(printf '%s\n' "$(brew --cellar)/"* | wc -l)" ;;
|
|
||||||
port) packages="$(port installed | wc -l)" ;;
|
|
||||||
dpkg-query) packages="$(dpkg-query -f '${"binary:Package"}\n' -W | wc -l)" ;;
|
|
||||||
rpm) packages="$(rpm -qa --last | wc -l)" ;;
|
|
||||||
yum) packages="$(yum list installed | wc -l)" ;;
|
|
||||||
dnf) packages="$(dnf list installed | wc -l)" ;;
|
|
||||||
zypper) packages="$(zypper se | wc -l)" ;;
|
|
||||||
pacman) packages="$(pacman -Q | wc -l)" ;;
|
|
||||||
yay) packages="$(yay -Q | wc -l)" ;;
|
|
||||||
paru) packages="$(paru -Q | wc -l)" ;;
|
|
||||||
pacstall) packages="$(pacstall -L | wc -l)" ;;
|
|
||||||
kiss) packages="$(kiss list | wc -l)" ;;
|
|
||||||
emerge) packages="$(qlist -I | wc -l)" ;;
|
|
||||||
pkg) packages="$(pkg info | wc -l | tr -d ' ')" ;;
|
|
||||||
cave) packages="$(cave show installed-slots | wc -l)" ;;
|
|
||||||
xbps-query) packages="$(xbps-query -l | wc -l)" ;;
|
|
||||||
nix-env) packages="$(nix-store -q --requisites /run/current-system/sw | wc -l)" ;;
|
|
||||||
apk) packages="$(apk list --installed | wc -l)" ;;
|
|
||||||
pmm) packages="$(/bedrock/libexec/pmm pacman pmm -Q 2>/dev/null | wc -l)" ;;
|
|
||||||
eopkg) packages="$(eopkg li | wc -l)" ;;
|
|
||||||
/usr/sbin/slackpkg) packages="$(ls /var/log/packages | wc -l)" ;;
|
|
||||||
bulge) packages="$(bulge list | wc -l)" ;;
|
|
||||||
birb) packages="$(birb --list-installed | wc -l)" ;;
|
|
||||||
pkg_info)
|
|
||||||
packages="$(pkg_info -A | wc -l)"
|
|
||||||
manager="pkg"
|
|
||||||
;;
|
|
||||||
pkg_info-netbsd)
|
|
||||||
packages="$(pkg_info -a | wc -l)"
|
|
||||||
manager="pkg"
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
if [ $distrotype = android ]; then
|
|
||||||
packages="$(dpkg-query -f '${"binary:Package"}\n' -W | wc -l)"
|
|
||||||
manager="dpkg"
|
|
||||||
else
|
|
||||||
packages="$(ls /usr/bin | wc -l)"
|
|
||||||
manager="bin"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
packages="''${packages#"''${packages%%[![:space:]]*}"}"
|
|
||||||
manager=$(echo $manager | sed "s/-query//; s/\/usr\/.*\///")
|
|
||||||
|
|
||||||
## UPTIME DETECTION
|
|
||||||
|
|
||||||
if [ $distrotype = android ]; then
|
|
||||||
uptime="$(echo $(uptime -p) | cut -c 4-)"
|
|
||||||
elif [ $distrotype = bsd ] || [ $distrotype = netbsd ]; then
|
|
||||||
uptime="$(uptime | sed -e 's/.* up //; s/, [0-9]* user.*//')"
|
|
||||||
else
|
|
||||||
case $ostype in
|
|
||||||
*"Linux"*)
|
|
||||||
IFS=. read -r s _ </proc/uptime
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
s=$(sysctl -n kern.boottime)
|
|
||||||
s=''${s#*=}
|
|
||||||
s=''${s%,*}
|
|
||||||
s=$(($(date +%s) - s))
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
d="$((s / 60 / 60 / 24))"
|
|
||||||
h="$((s / 60 / 60 % 24))"
|
|
||||||
m="$((s / 60 % 60))"
|
|
||||||
# Plurals
|
|
||||||
[ "$d" -gt 1 ] && dp=s
|
|
||||||
[ "$h" -gt 1 ] && hp=s
|
|
||||||
[ "$m" -gt 1 ] && mp=s
|
|
||||||
[ "$s" -gt 1 ] && sp=s
|
|
||||||
# Hide empty fields.
|
|
||||||
[ "$d" = 0 ] && d=
|
|
||||||
[ "$h" = 0 ] && h=
|
|
||||||
[ "$m" = 0 ] && m=
|
|
||||||
[ "$m" != "" ] && s=
|
|
||||||
# Make the output of uptime smaller.
|
|
||||||
[ "$d" ] && uptime="$d day$dp, "
|
|
||||||
[ "$h" ] && uptime="$uptime$h hour$hp, "
|
|
||||||
[ "$m" ] && uptime="$uptime$m min$mp"
|
|
||||||
[ "$s" ] && uptime="$uptime$s sec$sp"
|
|
||||||
uptime=''${uptime%, }
|
|
||||||
fi
|
|
||||||
|
|
||||||
## RAM DETECTION
|
|
||||||
|
|
||||||
case $ostype in
|
|
||||||
*"Linux"*)
|
|
||||||
while IFS=':k ' read -r key val _; do
|
|
||||||
case $key in
|
|
||||||
MemTotal)
|
|
||||||
mem_used=$((mem_used + val))
|
|
||||||
mem_full=$val
|
|
||||||
;;
|
|
||||||
Shmem) mem_used=$((mem_used + val)) ;;
|
|
||||||
MemFree | Buffers | Cached | SReclaimable) mem_used=$((mem_used - val)) ;;
|
|
||||||
esac
|
|
||||||
done </proc/meminfo
|
|
||||||
mem_used=$((mem_used / 1024))
|
|
||||||
mem_full=$((mem_full / 1024))
|
|
||||||
;;
|
|
||||||
*"Darwin"*)
|
|
||||||
mem_full=$(($(sysctl -n hw.memsize) / 1024 / 1024))
|
|
||||||
while IFS=:. read -r key val; do
|
|
||||||
case $key in
|
|
||||||
*' wired'* | *' active'* | *' occupied'*)
|
|
||||||
mem_used=$((mem_used + ${"val:-0"}))
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
done <-EOF
|
|
||||||
$(vm_stat)
|
|
||||||
EOF
|
|
||||||
|
|
||||||
mem_used=$((mem_used * 4 / 1024))
|
|
||||||
;;
|
|
||||||
*"BSD"*)
|
|
||||||
mem_full=$(($(sysctl -n hw.physmem) / 1024 / 1024))
|
|
||||||
if [ $distrotype = netbsd ]; then
|
|
||||||
mem_free=$(($(vmstat | awk 'NR==3 {print $4}') / 1024))
|
|
||||||
else
|
|
||||||
mem_free=$(($(sysctl -n vm.stats.vm.v_free_count) * $(sysctl -n vm.stats.vm.v_page_size) / 1024 / 1024))
|
|
||||||
fi
|
|
||||||
mem_used=$((mem_full - mem_free))
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
mem_full=1
|
|
||||||
mem_used=0
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
memstat="''${mem_used}/''${mem_full} MiB"
|
|
||||||
if which expr >/dev/null 2>&1; then
|
|
||||||
mempercent="($(expr $(expr ''${mem_used} \* 100 / ''${mem_full}))%)"
|
|
||||||
fi
|
|
||||||
|
|
||||||
## DEFINE COLORS
|
|
||||||
|
|
||||||
bold='[1m'
|
|
||||||
black='[30m'
|
|
||||||
red='[31m'
|
|
||||||
green='[32m'
|
|
||||||
yellow='[33m'
|
|
||||||
blue='[34m'
|
|
||||||
magenta='[35m'
|
|
||||||
cyan='[36m'
|
|
||||||
white='[37m'
|
|
||||||
grey='[90m'
|
|
||||||
reset='[0m'
|
|
||||||
|
|
||||||
## USER VARIABLES -- YOU CAN CHANGE THESE
|
|
||||||
|
|
||||||
lc="$reset$bold$magenta" # labels
|
|
||||||
nc="$reset$bold$magenta" # labels
|
|
||||||
hn="$reset$bold$magenta" # labels
|
|
||||||
ic="$reset$white" # info
|
|
||||||
c0="$reset$grey" # first color
|
|
||||||
c1="$reset$white" # second color
|
|
||||||
c2="$reset$yellow" # third color
|
|
||||||
|
|
||||||
## OUTPUT
|
|
||||||
|
|
||||||
echo """
|
|
||||||
''${c0} ___ ''${nc}''${USER}''${grey}@''${reset}''${hn}''${host}''${reset}
|
|
||||||
''${c0} (''${c1}.. ''${c0}\ ''${lc}''${osi} ''${ic}''${os}''${reset}
|
|
||||||
''${c0} (''${c2}<> ''${c0}| ''${lc}''${ki} ''${ic}''${kernel}''${reset}
|
|
||||||
''${c0} /''${c1}/ \\ ''${c0}\\ ''${lc}''${ri} ''${ic}''${RAM}''${memstat} ''${mempercent}
|
|
||||||
''${c0} ( ''${c1}| | ''${c0}/| ''${lc}''${pi} ''${ic}''${packages} (''${manager})''${reset}
|
|
||||||
''${c2} _''${c0}/\\ ''${c1}__)''${c0}/''${c2}_''${c0}) ''${lc}''${ui} ''${ic}''${uptime}''${reset}
|
|
||||||
''${c2} \/''${c0}-____''${c2}\/''${reset} ''${lc}''${ci} ''${red}███''${green}███''${yellow}███''${blue}███''${magenta}███''${cyan}███''${reset}
|
|
||||||
"""
|
|
||||||
'';
|
|
||||||
in {home.packages = [nerdfetch];}
|
|
||||||
@@ -1,19 +1,29 @@
|
|||||||
{
|
{
|
||||||
|
home.sessionVariables = {
|
||||||
|
TERMINAL = "ghostty";
|
||||||
|
TERM = "ghostty";
|
||||||
|
};
|
||||||
|
|
||||||
programs.ghostty = {
|
programs.ghostty = {
|
||||||
enable = true;
|
enable = true;
|
||||||
installVimSyntax = true;
|
installVimSyntax = true;
|
||||||
enableZshIntegration = true;
|
enableZshIntegration = true;
|
||||||
settings = {
|
settings = {
|
||||||
window-padding-x = 10;
|
window-padding-x = 10;
|
||||||
|
confirm-close-surface = false;
|
||||||
window-padding-y = 10;
|
window-padding-y = 10;
|
||||||
|
clipboard-read = "allow";
|
||||||
|
clipboard-write = "allow";
|
||||||
|
copy-on-select = "clipboard";
|
||||||
|
app-notifications = false;
|
||||||
keybind = [
|
keybind = [
|
||||||
"ctrl+j=goto_split:left"
|
"ctrl+j=goto_split:left"
|
||||||
"ctrl+i=goto_split:up"
|
"ctrl+i=goto_split:up"
|
||||||
"ctrl+k=goto_split:down"
|
"ctrl+k=goto_split:down"
|
||||||
"ctrl+l=goto_split:right"
|
"ctrl+l=goto_split:right"
|
||||||
"shift+ctrl+j=new_split:left"
|
"shift+ctrl+h=new_split:left"
|
||||||
"shift+ctrl+i=new_split:up"
|
"shift+ctrl+j=new_split:down"
|
||||||
"shift+ctrl+k=new_split:down"
|
"shift+ctrl+k=new_split:up"
|
||||||
"shift+ctrl+l=new_split:right"
|
"shift+ctrl+l=new_split:right"
|
||||||
"shift+ctrl+tab=new_tab"
|
"shift+ctrl+tab=new_tab"
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -3,10 +3,13 @@
|
|||||||
# CHANGEME: change this to your own SSH key.
|
# CHANGEME: change this to your own SSH key.
|
||||||
home.file.".ssh/allowed_signers".text = "* ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIINhWby7lUUXQNKbRu9/UOrGjWDf3fvoAwGHomWv/+lL";
|
home.file.".ssh/allowed_signers".text = "* ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIINhWby7lUUXQNKbRu9/UOrGjWDf3fvoAwGHomWv/+lL";
|
||||||
|
|
||||||
programs.git.settings = {
|
programs.git = {
|
||||||
|
signing.format = "openpgp";
|
||||||
|
settings = {
|
||||||
commit.gpgsign = true;
|
commit.gpgsign = true;
|
||||||
gpg.ssh.allowedSignersFile = "~/.ssh/allowed_signers";
|
gpg.ssh.allowedSignersFile = "~/.ssh/allowed_signers";
|
||||||
gpg.format = "ssh";
|
gpg.format = "ssh";
|
||||||
user.signingkey = "~/.ssh/key.pub";
|
user.signingkey = "~/.ssh/key.pub";
|
||||||
};
|
};
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
{
|
||||||
|
pkgs,
|
||||||
|
pkgs-stable,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
home.packages = with pkgs-stable; [
|
||||||
|
vlc # Video player
|
||||||
|
blanket # White-noise app
|
||||||
|
obsidian # Note taking app
|
||||||
|
textpieces # Manipulate texts
|
||||||
|
resources # Ressource monitor
|
||||||
|
gnome-clocks # Clocks app
|
||||||
|
gnome-text-editor # Basic graphic text editor
|
||||||
|
ticktick # Todo app
|
||||||
|
pinta # Image editor
|
||||||
|
onlyoffice-desktopeditors # Office suite
|
||||||
|
|
||||||
|
signal-desktop # Messaging app
|
||||||
|
element-desktop # Messaging app
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -1,15 +1,25 @@
|
|||||||
{
|
{
|
||||||
pkgs,
|
pkgs-stable,
|
||||||
inputs,
|
inputs,
|
||||||
...
|
...
|
||||||
}: {
|
}: {
|
||||||
home.packages = with pkgs; [
|
home.packages = with pkgs-stable; [
|
||||||
wireshark
|
wireshark
|
||||||
nmap
|
nmap
|
||||||
john
|
john
|
||||||
hashcat
|
hashcat
|
||||||
inputs.eleakxir.packages.${stdenv.hostPlatform.system}.leak-utils
|
|
||||||
|
# Web
|
||||||
caido
|
caido
|
||||||
nuclei
|
nuclei
|
||||||
|
gobuster
|
||||||
|
dirb
|
||||||
|
|
||||||
|
# Utils
|
||||||
|
inetutils
|
||||||
|
samba
|
||||||
|
openvpn
|
||||||
|
mariadb
|
||||||
|
redis
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
pkgs,
|
||||||
|
pkgs-stable,
|
||||||
|
inputs,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
home.packages =
|
||||||
|
(with pkgs; [
|
||||||
|
# Unstable: latest toolchain versions preferred for dev
|
||||||
|
go
|
||||||
|
bun
|
||||||
|
nodejs
|
||||||
|
air
|
||||||
|
duckdb
|
||||||
|
claude-code
|
||||||
|
inputs.bun2nix.packages.${stdenv.hostPlatform.system}.default
|
||||||
|
])
|
||||||
|
++ (with pkgs-stable; [
|
||||||
|
docker
|
||||||
|
python3
|
||||||
|
jq
|
||||||
|
just
|
||||||
|
]);
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
{pkgs-stable, ...}: {
|
||||||
|
home.packages = with pkgs-stable; [
|
||||||
|
peaclock
|
||||||
|
cbonsai
|
||||||
|
pipes
|
||||||
|
cmatrix
|
||||||
|
fastfetch
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
{inputs, ...}: {
|
||||||
|
imports = [inputs.nix-index-database.homeModules.default];
|
||||||
|
programs.nix-index.enable = true;
|
||||||
|
programs.nix-index-database.comma.enable = true;
|
||||||
|
}
|
||||||
@@ -5,13 +5,19 @@
|
|||||||
#- - `nixy` - UI wizard to manage the system.
|
#- - `nixy` - UI wizard to manage the system.
|
||||||
#- - `nixy rebuild` - Rebuild the system.
|
#- - `nixy rebuild` - Rebuild the system.
|
||||||
#- - `nixy ...` - ... see the script for more commands.
|
#- - `nixy ...` - ... see the script for more commands.
|
||||||
{ pkgs, config, ... }:
|
{
|
||||||
let
|
pkgs,
|
||||||
configDirectory = config.var.configDirectory;
|
config,
|
||||||
|
...
|
||||||
|
}: let
|
||||||
|
inherit (config.var) configDirectory;
|
||||||
|
|
||||||
nixy = pkgs.writeShellScriptBin "nixy"
|
nixy =
|
||||||
|
pkgs.writeShellScriptBin "nixy"
|
||||||
# bash
|
# bash
|
||||||
''
|
''
|
||||||
|
EXTRA_ARGS="''${@:2}"
|
||||||
|
|
||||||
function exec() {
|
function exec() {
|
||||||
$@
|
$@
|
||||||
}
|
}
|
||||||
@@ -27,8 +33,6 @@ let
|
|||||||
";Collect Garbage;nixy gc"
|
";Collect Garbage;nixy gc"
|
||||||
";Clean Boot Menu;nixy cb"
|
";Clean Boot Menu;nixy cb"
|
||||||
";List generation;nixy listgen"
|
";List generation;nixy listgen"
|
||||||
";Hyprland Keybindings;nvim ${configDirectory}/docs/KEYBINDINGS-HYPRLAND.md"
|
|
||||||
";Wallpapers;zen https://github.com/anotherhadi/nixy-wallpapers"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Apply default icons if empty:
|
# Apply default icons if empty:
|
||||||
@@ -49,11 +53,11 @@ let
|
|||||||
[[ $1 == "" ]] && ui
|
[[ $1 == "" ]] && ui
|
||||||
|
|
||||||
if [[ $1 == "rebuild" ]];then
|
if [[ $1 == "rebuild" ]];then
|
||||||
cd ${configDirectory} && git add . && sudo nixos-rebuild switch --flake
|
cd ${configDirectory} && git add . && sudo nixos-rebuild switch --flake . $EXTRA_ARGS
|
||||||
elif [[ $1 == "test" ]];then
|
elif [[ $1 == "test" ]];then
|
||||||
cd ${configDirectory} && git add . && sudo nixos-rebuild test --flake
|
cd ${configDirectory} && git add . && sudo nixos-rebuild test --flake . $EXTRA_ARGS
|
||||||
elif [[ $1 == "update" ]];then
|
elif [[ $1 == "update" ]];then
|
||||||
cd ${configDirectory} && nix flake update
|
cd ${configDirectory} && nix flake update $EXTRA_ARGS
|
||||||
elif [[ $1 == "gc" ]];then
|
elif [[ $1 == "gc" ]];then
|
||||||
echo "Starting Nix garbage collection..."
|
echo "Starting Nix garbage collection..."
|
||||||
cd ${configDirectory} && \
|
cd ${configDirectory} && \
|
||||||
@@ -74,4 +78,4 @@ let
|
|||||||
echo "Unknown argument"
|
echo "Unknown argument"
|
||||||
fi
|
fi
|
||||||
'';
|
'';
|
||||||
in { home.packages = [ nixy ]; }
|
in {home.packages = [nixy];}
|
||||||
|
|||||||
@@ -17,10 +17,5 @@
|
|||||||
|
|
||||||
programs.nvf = {
|
programs.nvf = {
|
||||||
enable = true;
|
enable = true;
|
||||||
settings.vim = {
|
|
||||||
startPlugins = [
|
|
||||||
pkgs.vimPlugins.vim-kitty-navigator
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,33 +32,7 @@
|
|||||||
desc = "Next Buffer";
|
desc = "Next Buffer";
|
||||||
}
|
}
|
||||||
|
|
||||||
# Kitty navigator
|
# Disable Arrow Keys in Normal Mode and Middle click
|
||||||
{
|
|
||||||
key = "<C-h>";
|
|
||||||
mode = "n";
|
|
||||||
silent = true;
|
|
||||||
action = "<cmd>KittyNavigateLeft<cr>";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
key = "<C-j>";
|
|
||||||
mode = "n";
|
|
||||||
silent = true;
|
|
||||||
action = "<cmd>KittyNavigateDown<cr>";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
key = "<C-k>";
|
|
||||||
mode = "n";
|
|
||||||
silent = true;
|
|
||||||
action = "<cmd>KittyNavigateUp<cr>";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
key = "<C-l>";
|
|
||||||
mode = "n";
|
|
||||||
silent = true;
|
|
||||||
action = "<cmd>KittyNavigateRight<cr>";
|
|
||||||
}
|
|
||||||
|
|
||||||
# Disable Arrow Keys in Normal Mode
|
|
||||||
{
|
{
|
||||||
key = "<Up>";
|
key = "<Up>";
|
||||||
mode = "n";
|
mode = "n";
|
||||||
@@ -87,6 +61,24 @@
|
|||||||
action = "<Nop>";
|
action = "<Nop>";
|
||||||
desc = "Disable Right Arrow";
|
desc = "Disable Right Arrow";
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
key = "<MiddleMouse>";
|
||||||
|
mode = ["n" "i" "v"]; # Normal, Insert, Visual
|
||||||
|
action = "<nop>"; # No Operation
|
||||||
|
silent = true;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<2-MiddleMouse>"; # Désactive aussi le double clic molette
|
||||||
|
mode = ["n" "i" "v"];
|
||||||
|
action = "<nop>";
|
||||||
|
silent = true;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<3-MiddleMouse>"; # Désactive aussi le double clic molette
|
||||||
|
mode = ["n" "i" "v"];
|
||||||
|
action = "<nop>";
|
||||||
|
silent = true;
|
||||||
|
}
|
||||||
|
|
||||||
# UI
|
# UI
|
||||||
{
|
{
|
||||||
@@ -168,6 +160,70 @@
|
|||||||
action = "<cmd>close<cr>";
|
action = "<cmd>close<cr>";
|
||||||
desc = "Close";
|
desc = "Close";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# QOL
|
||||||
|
{
|
||||||
|
key = ">";
|
||||||
|
mode = "v";
|
||||||
|
silent = true;
|
||||||
|
action = ">gv";
|
||||||
|
desc = "Indent and keep selection";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<";
|
||||||
|
mode = "v";
|
||||||
|
silent = true;
|
||||||
|
action = "<gv";
|
||||||
|
desc = "Dedent and keep selection";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Move
|
||||||
|
{
|
||||||
|
key = "<C-h>";
|
||||||
|
mode = "n";
|
||||||
|
silent = true;
|
||||||
|
action = "<C-w>h";
|
||||||
|
desc = "Move to left window";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<C-j>";
|
||||||
|
mode = "n";
|
||||||
|
silent = true;
|
||||||
|
action = "<C-w>j";
|
||||||
|
desc = "Move to bottom window";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<C-k>";
|
||||||
|
mode = "n";
|
||||||
|
silent = true;
|
||||||
|
action = "<C-w>k";
|
||||||
|
desc = "Move to top window";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "<C-l>";
|
||||||
|
mode = "n";
|
||||||
|
silent = true;
|
||||||
|
action = "<C-w>l";
|
||||||
|
desc = "Move to right window";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Save
|
||||||
|
{
|
||||||
|
key = "<C-s>";
|
||||||
|
mode = ["n" "i" "v"];
|
||||||
|
silent = true;
|
||||||
|
action = "<cmd>w<cr>";
|
||||||
|
desc = "Save file";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Deactivate "esc"
|
||||||
|
{
|
||||||
|
key = "<Esc>";
|
||||||
|
mode = ["n" "i" "v"];
|
||||||
|
silent = true;
|
||||||
|
action = "<Nop>";
|
||||||
|
desc = "Disable Escape";
|
||||||
|
}
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -87,6 +87,13 @@
|
|||||||
enableFormat = true;
|
enableFormat = true;
|
||||||
enableTreesitter = true;
|
enableTreesitter = true;
|
||||||
|
|
||||||
|
python = {
|
||||||
|
enable = true;
|
||||||
|
lsp = {
|
||||||
|
enable = true;
|
||||||
|
servers = ["pyright"];
|
||||||
|
};
|
||||||
|
};
|
||||||
astro.enable = true;
|
astro.enable = true;
|
||||||
go.enable = true;
|
go.enable = true;
|
||||||
markdown = {
|
markdown = {
|
||||||
|
|||||||
@@ -30,6 +30,19 @@
|
|||||||
registers = "unnamedplus";
|
registers = "unnamedplus";
|
||||||
providers.wl-copy.enable = true;
|
providers.wl-copy.enable = true;
|
||||||
};
|
};
|
||||||
|
luaConfigRC.osc52-clipboard = ''
|
||||||
|
vim.g.clipboard = {
|
||||||
|
name = 'OSC 52',
|
||||||
|
copy = {
|
||||||
|
['+'] = require('vim.ui.clipboard.osc52').copy '+',
|
||||||
|
['*'] = require('vim.ui.clipboard.osc52').copy '*',
|
||||||
|
},
|
||||||
|
paste = {
|
||||||
|
['+'] = require('vim.ui.clipboard.osc52').paste '+',
|
||||||
|
['*'] = require('vim.ui.clipboard.osc52').paste '*',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
'';
|
||||||
theme = {
|
theme = {
|
||||||
enable = true;
|
enable = true;
|
||||||
name = lib.mkForce "catppuccin";
|
name = lib.mkForce "catppuccin";
|
||||||
|
|||||||
@@ -1,4 +1,11 @@
|
|||||||
{
|
{pkgs, ...}: {
|
||||||
|
home.packages = with pkgs; [
|
||||||
|
imagemagick
|
||||||
|
tree-sitter
|
||||||
|
ghostscript
|
||||||
|
tectonic
|
||||||
|
mermaid-cli
|
||||||
|
];
|
||||||
programs.nvf.settings.vim.utility.snacks-nvim = {
|
programs.nvf.settings.vim.utility.snacks-nvim = {
|
||||||
enable = true;
|
enable = true;
|
||||||
setupOpts = {
|
setupOpts = {
|
||||||
|
|||||||
@@ -18,7 +18,6 @@
|
|||||||
enable = true;
|
enable = true;
|
||||||
sources = {
|
sources = {
|
||||||
buffer = "[Buffer]";
|
buffer = "[Buffer]";
|
||||||
nvim-cmp = null;
|
|
||||||
path = "[Path]";
|
path = "[Path]";
|
||||||
};
|
};
|
||||||
sourcePlugins = [
|
sourcePlugins = [
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
wayland.windowManager.hyprland.settings.exec-once = [
|
||||||
|
"protonvpn-app --start-minimized &"
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{pkgs, ...}: {
|
{pkgs, ...}: {
|
||||||
home.packages = with pkgs; [
|
home.packages = with pkgs; [
|
||||||
protonvpn-gui
|
proton-vpn
|
||||||
proton-pass
|
proton-pass
|
||||||
proton-authenticator
|
proton-authenticator
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -0,0 +1,466 @@
|
|||||||
|
{
|
||||||
|
config,
|
||||||
|
lib,
|
||||||
|
pkgs,
|
||||||
|
...
|
||||||
|
}: let
|
||||||
|
bookmarkList = [
|
||||||
|
{
|
||||||
|
name = "Proton Mail";
|
||||||
|
url = "https://mail.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Proton Drive";
|
||||||
|
url = "https://drive.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Proton Lumo";
|
||||||
|
url = "https://lumo.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Proton Calendar";
|
||||||
|
url = "https://calendar.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Tools";
|
||||||
|
bookmarks = [
|
||||||
|
{
|
||||||
|
name = "tldr";
|
||||||
|
url = "https://tldr.inbrowser.app/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Excalidraw";
|
||||||
|
url = "https://excalidraw.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Cobalt (downloader)";
|
||||||
|
url = "https://cobalt.meowing.de";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Mazanoke (image)";
|
||||||
|
url = "https://mazanoke.hadi.icu";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Stirling PDF";
|
||||||
|
url = "https://pdf.hadi.icu";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Vert";
|
||||||
|
url = "https://vert.sh";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Markdown to PDF";
|
||||||
|
url = "https://md2file.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Image to Vector";
|
||||||
|
url = "https://www.vectorcascade.com/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "PrivateBin";
|
||||||
|
url = "https://privatebin.net";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Social";
|
||||||
|
bookmarks = [
|
||||||
|
{
|
||||||
|
name = "Bluesky";
|
||||||
|
url = "https://bsky.app";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Reddit";
|
||||||
|
url = "https://reddit.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Youtube";
|
||||||
|
url = "https://youtube.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Instagram";
|
||||||
|
url = "https://instagram.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Github";
|
||||||
|
url = "https://github.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Discord";
|
||||||
|
url = "https://discord.com/channels/@me/";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Other";
|
||||||
|
bookmarks = [
|
||||||
|
{
|
||||||
|
name = "Startpage Config";
|
||||||
|
url = "https://www.startpage.com/do/mypage.pl?prfe=45d331deb05471d659dba933e7400df51d952bb103da6f6125c0e769a6be1d65610456a479f495ceeee7e97311cf227d7c1bb198de0ceeb193d8cddf9c455c19a409cc35c3e3f542ee27bd7cecd3";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Hyprland Wiki";
|
||||||
|
url = "https://wiki.hypr.land";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "MyNixOS";
|
||||||
|
url = "https://mynixos.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Nixpkgs";
|
||||||
|
url = "https://github.com/NixOS/nixpkgs";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Claude";
|
||||||
|
url = "https://claude.ai";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Gemini";
|
||||||
|
url = "https://gemini.google.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Medium";
|
||||||
|
url = "https://medium.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Maps";
|
||||||
|
url = "https://maps.apple.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Amazon";
|
||||||
|
url = "https://amazon.fr";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Infosec";
|
||||||
|
bookmarks = [
|
||||||
|
{
|
||||||
|
name = "Nix 4 Cyber";
|
||||||
|
url = "https://n4c.hadi.icu";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Cyberchef";
|
||||||
|
url = "https://cyberchef.hadi.icu";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "TryHackMe";
|
||||||
|
url = "https://tryhackme.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Root-Me";
|
||||||
|
url = "https://root-me.org";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Exploit-DB";
|
||||||
|
url = "https://exploit-db.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Crack Station";
|
||||||
|
url = "https://crackstation.net";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "Osint Tracker";
|
||||||
|
url = "https://app.osintracker.com";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
c = config.lib.stylix.colors;
|
||||||
|
|
||||||
|
stripProtocol = url:
|
||||||
|
lib.removePrefix "https://" (lib.removePrefix "http://" url);
|
||||||
|
|
||||||
|
mkCard = item: ''
|
||||||
|
<a href="${item.url}" class="card">
|
||||||
|
<div class="favicon-wrapper">
|
||||||
|
<span class="favicon-fallback">${builtins.substring 0 1 item.name}</span>
|
||||||
|
</div>
|
||||||
|
<div class="card-info">
|
||||||
|
<span class="card-name">${item.name}</span>
|
||||||
|
<span class="card-url">${stripProtocol item.url}</span>
|
||||||
|
</div>
|
||||||
|
</a>'';
|
||||||
|
|
||||||
|
# Group consecutive root items so they share the same .cards grid
|
||||||
|
grouped = let
|
||||||
|
step = acc: item:
|
||||||
|
if item ? url
|
||||||
|
then acc // {pending = acc.pending ++ [item];}
|
||||||
|
else {
|
||||||
|
groups =
|
||||||
|
acc.groups
|
||||||
|
++ lib.optional (acc.pending != []) {
|
||||||
|
isRoot = true;
|
||||||
|
items = acc.pending;
|
||||||
|
}
|
||||||
|
++ [
|
||||||
|
{
|
||||||
|
isRoot = false;
|
||||||
|
inherit item;
|
||||||
|
}
|
||||||
|
];
|
||||||
|
pending = [];
|
||||||
|
};
|
||||||
|
result =
|
||||||
|
lib.foldl' step {
|
||||||
|
groups = [];
|
||||||
|
pending = [];
|
||||||
|
}
|
||||||
|
bookmarkList;
|
||||||
|
in
|
||||||
|
result.groups
|
||||||
|
++ lib.optional (result.pending != []) {
|
||||||
|
isRoot = true;
|
||||||
|
items = result.pending;
|
||||||
|
};
|
||||||
|
|
||||||
|
mkSection = group:
|
||||||
|
if group.isRoot
|
||||||
|
then ''
|
||||||
|
<div class="folder-section">
|
||||||
|
<div class="cards">
|
||||||
|
${lib.concatMapStrings mkCard group.items}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
''
|
||||||
|
else ''
|
||||||
|
<div class="folder-section">
|
||||||
|
<h2 class="folder-title">${group.item.name}</h2>
|
||||||
|
<div class="cards">
|
||||||
|
${lib.concatMapStrings mkCard group.item.bookmarks}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
'';
|
||||||
|
|
||||||
|
publicBookmarks =
|
||||||
|
pkgs.writeText "qutebrowser-public-bookmarks"
|
||||||
|
(lib.concatMapStrings (
|
||||||
|
item:
|
||||||
|
if item ? url
|
||||||
|
then "${item.url} ${item.name}\n"
|
||||||
|
else lib.concatMapStrings (b: "${b.url} ${item.name}/${b.name}\n") item.bookmarks
|
||||||
|
)
|
||||||
|
bookmarkList);
|
||||||
|
|
||||||
|
inherit (config.qutebrowser) privateBookmarksPath;
|
||||||
|
in {
|
||||||
|
options.qutebrowser.privateBookmarksPath = lib.mkOption {
|
||||||
|
type = lib.types.nullOr lib.types.str;
|
||||||
|
default = null;
|
||||||
|
description = "Path to a file containing extra (private) bookmarks to append.";
|
||||||
|
};
|
||||||
|
|
||||||
|
config = {
|
||||||
|
# Fully static HTML — order is preserved, no dependency on qutebrowser's Jinja rendering
|
||||||
|
xdg.dataFile."qutebrowser/bookmarks.html".text = ''
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>Bookmarks</title>
|
||||||
|
<style>
|
||||||
|
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: #${c.base00};
|
||||||
|
color: #${c.base05};
|
||||||
|
font-family: 'Source Sans Pro', sans-serif;
|
||||||
|
font-size: 14px;
|
||||||
|
min-height: 100vh;
|
||||||
|
padding: 2.5rem 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
max-width: 1100px;
|
||||||
|
margin: 0 auto 2rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-title {
|
||||||
|
font-size: 1.3rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #${c.base0D};
|
||||||
|
letter-spacing: -0.01em;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-bar {
|
||||||
|
flex: 1;
|
||||||
|
background-color: #${c.base01};
|
||||||
|
border: 1px solid #${c.base02};
|
||||||
|
border-radius: 8px;
|
||||||
|
color: #${c.base05};
|
||||||
|
font-family: 'Source Sans Pro', sans-serif;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
padding: 0.5rem 0.9rem;
|
||||||
|
outline: none;
|
||||||
|
transition: border-color 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-bar::placeholder { color: #${c.base03}; }
|
||||||
|
.search-bar:focus { border-color: #${c.base0D}; }
|
||||||
|
|
||||||
|
.bookmarks-container {
|
||||||
|
max-width: 1100px;
|
||||||
|
margin: 0 auto;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-title {
|
||||||
|
font-size: 0.72rem;
|
||||||
|
font-weight: 700;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: #${c.base03};
|
||||||
|
margin-bottom: 0.6rem;
|
||||||
|
padding-left: 0.2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cards {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
|
||||||
|
gap: 0.4rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.7rem;
|
||||||
|
padding: 0.6rem 0.8rem;
|
||||||
|
background-color: #${c.base01};
|
||||||
|
border: 1px solid #${c.base02};
|
||||||
|
border-radius: 10px;
|
||||||
|
text-decoration: none;
|
||||||
|
transition: background-color 0.12s ease, border-color 0.12s ease;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card:hover {
|
||||||
|
background-color: #${c.base02};
|
||||||
|
border-color: #${c.base03};
|
||||||
|
}
|
||||||
|
|
||||||
|
.favicon-wrapper { position: relative; width: 18px; height: 18px; flex-shrink: 0; }
|
||||||
|
|
||||||
|
.favicon-fallback {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
background-color: #${c.base02};
|
||||||
|
border-radius: 4px;
|
||||||
|
color: #${c.base04};
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.05rem;
|
||||||
|
overflow: hidden;
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-name {
|
||||||
|
color: #${c.base05};
|
||||||
|
font-size: 0.86rem;
|
||||||
|
font-weight: 500;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
transition: color 0.12s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card:hover .card-name { color: #${c.base0A}; }
|
||||||
|
|
||||||
|
.card-url {
|
||||||
|
color: #${c.base03};
|
||||||
|
font-family: 'Maple Mono NF', monospace;
|
||||||
|
font-size: 0.7rem;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-results {
|
||||||
|
color: #${c.base03};
|
||||||
|
font-size: 0.9rem;
|
||||||
|
padding: 0.5rem 0.2rem;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="header">
|
||||||
|
<h1 class="page-title">Bookmarks</h1>
|
||||||
|
<input id="search" class="search-bar" type="text" placeholder="Search…" autocomplete="off" spellcheck="false">
|
||||||
|
</div>
|
||||||
|
<div class="bookmarks-container">
|
||||||
|
<p id="no-results" class="no-results">No results</p>
|
||||||
|
${lib.concatMapStrings mkSection grouped}
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
window.addEventListener('DOMContentLoaded', function () {
|
||||||
|
const input = document.getElementById('search');
|
||||||
|
const noResults = document.getElementById('no-results');
|
||||||
|
|
||||||
|
input.focus();
|
||||||
|
|
||||||
|
input.addEventListener('input', function () {
|
||||||
|
const query = this.value.toLowerCase().trim();
|
||||||
|
let anyVisible = false;
|
||||||
|
|
||||||
|
document.querySelectorAll('.folder-section').forEach(function (section) {
|
||||||
|
let sectionVisible = false;
|
||||||
|
section.querySelectorAll('.card').forEach(function (card) {
|
||||||
|
const name = card.querySelector('.card-name').textContent.toLowerCase();
|
||||||
|
const url = card.querySelector('.card-url').textContent.toLowerCase();
|
||||||
|
const match = !query || name.includes(query) || url.includes(query);
|
||||||
|
card.style.display = match ? ''' : 'none';
|
||||||
|
if (match) { sectionVisible = true; anyVisible = true; }
|
||||||
|
});
|
||||||
|
section.style.display = sectionVisible ? ''' : 'none';
|
||||||
|
});
|
||||||
|
|
||||||
|
noResults.style.display = anyVisible || !query ? 'none' : 'block';
|
||||||
|
});
|
||||||
|
|
||||||
|
input.addEventListener('keydown', function (e) {
|
||||||
|
if (e.key === 'Enter') {
|
||||||
|
const first = document.querySelector('.card:not([style*="none"])');
|
||||||
|
if (first) {
|
||||||
|
location.href = first.href;
|
||||||
|
} else {
|
||||||
|
const q = encodeURIComponent(this.value.trim());
|
||||||
|
if (q) location.href = 'https://www.startpage.com/sp/search?q=' + q;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
this.value = ''';
|
||||||
|
this.dispatchEvent(new Event('input'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
'';
|
||||||
|
|
||||||
|
home.activation.qutebrowserBookmarks = lib.hm.dag.entryAfter ["setupSecrets" "writeBoundary"] ''
|
||||||
|
mkdir -p ${config.home.homeDirectory}/.config/qutebrowser/bookmarks
|
||||||
|
cat ${publicBookmarks} ${lib.optionalString (privateBookmarksPath != null) ''"${privateBookmarksPath}"''} \
|
||||||
|
> ${config.home.homeDirectory}/.config/qutebrowser/bookmarks/urls
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
{...}: {
|
||||||
|
imports = [
|
||||||
|
./bookmarks.nix
|
||||||
|
./search.nix
|
||||||
|
./keybindings.nix
|
||||||
|
./settings.nix
|
||||||
|
./userscripts.nix
|
||||||
|
];
|
||||||
|
|
||||||
|
home.sessionVariables = {
|
||||||
|
DEFAULT_BROWSER = "qutebrowser";
|
||||||
|
BROWSER = "qutebrowser";
|
||||||
|
};
|
||||||
|
|
||||||
|
xdg.mimeApps.defaultApplications = {
|
||||||
|
"text/html" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"text/xml" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"x-scheme-handler/http" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"x-scheme-handler/https" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"x-scheme-handler/unknown" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"x-scheme-handler/about" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
"x-scheme-handler/qute" = ["org.qutebrowser.qutebrowser.desktop"];
|
||||||
|
};
|
||||||
|
|
||||||
|
xdg.desktopEntries.qutebrowser-private = {
|
||||||
|
name = "Qutebrowser (Temp session)";
|
||||||
|
genericName = "Web Browser";
|
||||||
|
exec = "qutebrowser --temp-basedir %U";
|
||||||
|
icon = "qutebrowser";
|
||||||
|
categories = ["Network" "WebBrowser"];
|
||||||
|
};
|
||||||
|
|
||||||
|
programs.qutebrowser = {
|
||||||
|
enable = true;
|
||||||
|
loadAutoconfig = true;
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,826 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Don't track me Google
|
||||||
|
// @namespace Rob W
|
||||||
|
// @description Removes the annoying link-conversion at Google Search/maps/...
|
||||||
|
// @version 4.28
|
||||||
|
// @icon https://raw.githubusercontent.com/Rob--W/dont-track-me-google/master/icon48.png
|
||||||
|
// @supportURL https://github.com/Rob--W/dont-track-me-google/issues
|
||||||
|
// @license MIT
|
||||||
|
// @run-at document-start
|
||||||
|
// @match *://*.google.com/*
|
||||||
|
// @match *://*.google.ad/*
|
||||||
|
// @match *://*.google.ae/*
|
||||||
|
// @match *://*.google.com.af/*
|
||||||
|
// @match *://*.google.com.ag/*
|
||||||
|
// @match *://*.google.com.ai/*
|
||||||
|
// @match *://*.google.al/*
|
||||||
|
// @match *://*.google.am/*
|
||||||
|
// @match *://*.google.co.ao/*
|
||||||
|
// @match *://*.google.com.ar/*
|
||||||
|
// @match *://*.google.as/*
|
||||||
|
// @match *://*.google.at/*
|
||||||
|
// @match *://*.google.com.au/*
|
||||||
|
// @match *://*.google.az/*
|
||||||
|
// @match *://*.google.ba/*
|
||||||
|
// @match *://*.google.com.bd/*
|
||||||
|
// @match *://*.google.be/*
|
||||||
|
// @match *://*.google.bf/*
|
||||||
|
// @match *://*.google.bg/*
|
||||||
|
// @match *://*.google.com.bh/*
|
||||||
|
// @match *://*.google.bi/*
|
||||||
|
// @match *://*.google.bj/*
|
||||||
|
// @match *://*.google.com.bn/*
|
||||||
|
// @match *://*.google.com.bo/*
|
||||||
|
// @match *://*.google.com.br/*
|
||||||
|
// @match *://*.google.bs/*
|
||||||
|
// @match *://*.google.bt/*
|
||||||
|
// @match *://*.google.co.bw/*
|
||||||
|
// @match *://*.google.by/*
|
||||||
|
// @match *://*.google.com.bz/*
|
||||||
|
// @match *://*.google.ca/*
|
||||||
|
// @match *://*.google.cd/*
|
||||||
|
// @match *://*.google.cf/*
|
||||||
|
// @match *://*.google.cg/*
|
||||||
|
// @match *://*.google.ch/*
|
||||||
|
// @match *://*.google.ci/*
|
||||||
|
// @match *://*.google.co.ck/*
|
||||||
|
// @match *://*.google.cl/*
|
||||||
|
// @match *://*.google.cm/*
|
||||||
|
// @match *://*.google.cn/*
|
||||||
|
// @match *://*.google.com.co/*
|
||||||
|
// @match *://*.google.co.cr/*
|
||||||
|
// @match *://*.google.com.cu/*
|
||||||
|
// @match *://*.google.cv/*
|
||||||
|
// @match *://*.google.com.cy/*
|
||||||
|
// @match *://*.google.cz/*
|
||||||
|
// @match *://*.google.de/*
|
||||||
|
// @match *://*.google.dj/*
|
||||||
|
// @match *://*.google.dk/*
|
||||||
|
// @match *://*.google.dm/*
|
||||||
|
// @match *://*.google.com.do/*
|
||||||
|
// @match *://*.google.dz/*
|
||||||
|
// @match *://*.google.com.ec/*
|
||||||
|
// @match *://*.google.ee/*
|
||||||
|
// @match *://*.google.com.eg/*
|
||||||
|
// @match *://*.google.es/*
|
||||||
|
// @match *://*.google.com.et/*
|
||||||
|
// @match *://*.google.fi/*
|
||||||
|
// @match *://*.google.com.fj/*
|
||||||
|
// @match *://*.google.fm/*
|
||||||
|
// @match *://*.google.fr/*
|
||||||
|
// @match *://*.google.ga/*
|
||||||
|
// @match *://*.google.ge/*
|
||||||
|
// @match *://*.google.gg/*
|
||||||
|
// @match *://*.google.com.gh/*
|
||||||
|
// @match *://*.google.com.gi/*
|
||||||
|
// @match *://*.google.gl/*
|
||||||
|
// @match *://*.google.gm/*
|
||||||
|
// @match *://*.google.gp/*
|
||||||
|
// @match *://*.google.gr/*
|
||||||
|
// @match *://*.google.com.gt/*
|
||||||
|
// @match *://*.google.gy/*
|
||||||
|
// @match *://*.google.com.hk/*
|
||||||
|
// @match *://*.google.hn/*
|
||||||
|
// @match *://*.google.hr/*
|
||||||
|
// @match *://*.google.ht/*
|
||||||
|
// @match *://*.google.hu/*
|
||||||
|
// @match *://*.google.co.id/*
|
||||||
|
// @match *://*.google.ie/*
|
||||||
|
// @match *://*.google.co.il/*
|
||||||
|
// @match *://*.google.im/*
|
||||||
|
// @match *://*.google.co.in/*
|
||||||
|
// @match *://*.google.iq/*
|
||||||
|
// @match *://*.google.is/*
|
||||||
|
// @match *://*.google.it/*
|
||||||
|
// @match *://*.google.je/*
|
||||||
|
// @match *://*.google.com.jm/*
|
||||||
|
// @match *://*.google.jo/*
|
||||||
|
// @match *://*.google.co.jp/*
|
||||||
|
// @match *://*.google.co.ke/*
|
||||||
|
// @match *://*.google.com.kh/*
|
||||||
|
// @match *://*.google.ki/*
|
||||||
|
// @match *://*.google.kg/*
|
||||||
|
// @match *://*.google.co.kr/*
|
||||||
|
// @match *://*.google.com.kw/*
|
||||||
|
// @match *://*.google.kz/*
|
||||||
|
// @match *://*.google.la/*
|
||||||
|
// @match *://*.google.com.lb/*
|
||||||
|
// @match *://*.google.li/*
|
||||||
|
// @match *://*.google.lk/*
|
||||||
|
// @match *://*.google.co.ls/*
|
||||||
|
// @match *://*.google.lt/*
|
||||||
|
// @match *://*.google.lu/*
|
||||||
|
// @match *://*.google.lv/*
|
||||||
|
// @match *://*.google.com.ly/*
|
||||||
|
// @match *://*.google.co.ma/*
|
||||||
|
// @match *://*.google.md/*
|
||||||
|
// @match *://*.google.me/*
|
||||||
|
// @match *://*.google.mg/*
|
||||||
|
// @match *://*.google.mk/*
|
||||||
|
// @match *://*.google.ml/*
|
||||||
|
// @match *://*.google.com.mm/*
|
||||||
|
// @match *://*.google.mn/*
|
||||||
|
// @match *://*.google.ms/*
|
||||||
|
// @match *://*.google.com.mt/*
|
||||||
|
// @match *://*.google.mu/*
|
||||||
|
// @match *://*.google.mv/*
|
||||||
|
// @match *://*.google.mw/*
|
||||||
|
// @match *://*.google.com.mx/*
|
||||||
|
// @match *://*.google.com.my/*
|
||||||
|
// @match *://*.google.co.mz/*
|
||||||
|
// @match *://*.google.com.na/*
|
||||||
|
// @match *://*.google.com.nf/*
|
||||||
|
// @match *://*.google.com.ng/*
|
||||||
|
// @match *://*.google.com.ni/*
|
||||||
|
// @match *://*.google.ne/*
|
||||||
|
// @match *://*.google.nl/*
|
||||||
|
// @match *://*.google.no/*
|
||||||
|
// @match *://*.google.com.np/*
|
||||||
|
// @match *://*.google.nr/*
|
||||||
|
// @match *://*.google.nu/*
|
||||||
|
// @match *://*.google.co.nz/*
|
||||||
|
// @match *://*.google.com.om/*
|
||||||
|
// @match *://*.google.com.pa/*
|
||||||
|
// @match *://*.google.com.pe/*
|
||||||
|
// @match *://*.google.com.pg/*
|
||||||
|
// @match *://*.google.com.ph/*
|
||||||
|
// @match *://*.google.com.pk/*
|
||||||
|
// @match *://*.google.pl/*
|
||||||
|
// @match *://*.google.pn/*
|
||||||
|
// @match *://*.google.com.pr/*
|
||||||
|
// @match *://*.google.ps/*
|
||||||
|
// @match *://*.google.pt/*
|
||||||
|
// @match *://*.google.com.py/*
|
||||||
|
// @match *://*.google.com.qa/*
|
||||||
|
// @match *://*.google.ro/*
|
||||||
|
// @match *://*.google.ru/*
|
||||||
|
// @match *://*.google.rw/*
|
||||||
|
// @match *://*.google.com.sa/*
|
||||||
|
// @match *://*.google.com.sb/*
|
||||||
|
// @match *://*.google.sc/*
|
||||||
|
// @match *://*.google.se/*
|
||||||
|
// @match *://*.google.com.sg/*
|
||||||
|
// @match *://*.google.sh/*
|
||||||
|
// @match *://*.google.si/*
|
||||||
|
// @match *://*.google.sk/*
|
||||||
|
// @match *://*.google.com.sl/*
|
||||||
|
// @match *://*.google.sn/*
|
||||||
|
// @match *://*.google.so/*
|
||||||
|
// @match *://*.google.sm/*
|
||||||
|
// @match *://*.google.sr/*
|
||||||
|
// @match *://*.google.st/*
|
||||||
|
// @match *://*.google.com.sv/*
|
||||||
|
// @match *://*.google.td/*
|
||||||
|
// @match *://*.google.tg/*
|
||||||
|
// @match *://*.google.co.th/*
|
||||||
|
// @match *://*.google.com.tj/*
|
||||||
|
// @match *://*.google.tk/*
|
||||||
|
// @match *://*.google.tl/*
|
||||||
|
// @match *://*.google.tm/*
|
||||||
|
// @match *://*.google.tn/*
|
||||||
|
// @match *://*.google.to/*
|
||||||
|
// @match *://*.google.com.tr/*
|
||||||
|
// @match *://*.google.tt/*
|
||||||
|
// @match *://*.google.com.tw/*
|
||||||
|
// @match *://*.google.co.tz/*
|
||||||
|
// @match *://*.google.com.ua/*
|
||||||
|
// @match *://*.google.co.ug/*
|
||||||
|
// @match *://*.google.co.uk/*
|
||||||
|
// @match *://*.google.com.uy/*
|
||||||
|
// @match *://*.google.co.uz/*
|
||||||
|
// @match *://*.google.com.vc/*
|
||||||
|
// @match *://*.google.co.ve/*
|
||||||
|
// @match *://*.google.vg/*
|
||||||
|
// @match *://*.google.co.vi/*
|
||||||
|
// @match *://*.google.com.vn/*
|
||||||
|
// @match *://*.google.vu/*
|
||||||
|
// @match *://*.google.ws/*
|
||||||
|
// @match *://*.google.rs/*
|
||||||
|
// @match *://*.google.co.za/*
|
||||||
|
// @match *://*.google.co.zm/*
|
||||||
|
// @match *://*.google.co.zw/*
|
||||||
|
// @match *://*.google.cat/*
|
||||||
|
// @match *://*.google.ng/*
|
||||||
|
// @downloadURL https://update.greasyfork.org/scripts/428243/Don%27t%20track%20me%20Google.user.js
|
||||||
|
// @updateURL https://update.greasyfork.org/scripts/428243/Don%27t%20track%20me%20Google.meta.js
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
document.addEventListener('mousedown', handlePointerPress, true);
|
||||||
|
document.addEventListener('touchstart', handlePointerPress, true);
|
||||||
|
document.addEventListener('click', handleClick, true);
|
||||||
|
var scriptCspNonce;
|
||||||
|
var needsCspNonce = typeof browser !== 'undefined'; // Firefox.
|
||||||
|
var preferenceObservers = [];
|
||||||
|
setupAggresiveUglyLinkPreventer();
|
||||||
|
|
||||||
|
var forceNoReferrer = true;
|
||||||
|
var noping = true;
|
||||||
|
if (typeof chrome == 'object' && chrome.storage) {
|
||||||
|
(chrome.storage.sync || chrome.storage.local).get({
|
||||||
|
forceNoReferrer: true,
|
||||||
|
// From version 4.7 until 4.11, the preference was the literal value of
|
||||||
|
// the referrer policy.
|
||||||
|
referrerPolicy: 'no-referrer',
|
||||||
|
noping: true,
|
||||||
|
}, function(items) {
|
||||||
|
if (items) {
|
||||||
|
// Migration code (to be removed in the future).
|
||||||
|
if (items.referrerPolicy === '') {
|
||||||
|
// User explicitly allowed referrers to be sent, respect that.
|
||||||
|
items.forceNoReferrer = false;
|
||||||
|
}
|
||||||
|
forceNoReferrer = items.forceNoReferrer;
|
||||||
|
noping = items.noping;
|
||||||
|
callPreferenceObservers();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
chrome.storage.onChanged.addListener(function(changes) {
|
||||||
|
if (changes.forceNoReferrer) {
|
||||||
|
forceNoReferrer = changes.forceNoReferrer.newValue;
|
||||||
|
}
|
||||||
|
if (changes.noping) {
|
||||||
|
noping = changes.noping.newValue;
|
||||||
|
}
|
||||||
|
callPreferenceObservers();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function callImmediatelyAndOnPreferenceUpdate(callback) {
|
||||||
|
callback();
|
||||||
|
preferenceObservers.push(callback);
|
||||||
|
}
|
||||||
|
function callPreferenceObservers() {
|
||||||
|
// This method is usually once, and occasionally more than once if the user
|
||||||
|
// changes a preference. For simplicity we don't check whether a pref was
|
||||||
|
// changed before calling a callback - these are cheap anyway.
|
||||||
|
preferenceObservers.forEach(function(callback) {
|
||||||
|
callback();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getReferrerPolicy() {
|
||||||
|
return forceNoReferrer ? 'origin' : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateReferrerPolicy(a) {
|
||||||
|
if (a.referrerPolicy === 'no-referrer') {
|
||||||
|
// "no-referrer" is more privacy-friendly than "origin".
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var referrerPolicy = getReferrerPolicy();
|
||||||
|
if (referrerPolicy) {
|
||||||
|
a.referrerPolicy = referrerPolicy;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handlePointerPress(e) {
|
||||||
|
var a = e.target;
|
||||||
|
while (a && !a.href) {
|
||||||
|
a = a.parentElement;
|
||||||
|
}
|
||||||
|
if (!a) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var inlineMousedown = a.getAttribute('onmousedown');
|
||||||
|
// return rwt(....); // E.g Google search results.
|
||||||
|
// return google.rwt(...); // E.g. sponsored search results
|
||||||
|
// return google.arwt(this); // E.g. sponsored search results (dec 2016).
|
||||||
|
if (inlineMousedown && /\ba?rwt\(/.test(inlineMousedown)) {
|
||||||
|
a.removeAttribute('onmousedown');
|
||||||
|
// Just in case:
|
||||||
|
a.removeAttribute('ping');
|
||||||
|
// In Chrome, removing onmousedown during event dispatch does not
|
||||||
|
// prevent the inline listener from running... So we have to cancel
|
||||||
|
// event propagation just in case.
|
||||||
|
e.stopImmediatePropagation();
|
||||||
|
}
|
||||||
|
if (noping) {
|
||||||
|
a.removeAttribute('ping');
|
||||||
|
}
|
||||||
|
var realLink = getRealLinkFromGoogleUrl(a);
|
||||||
|
if (realLink) {
|
||||||
|
a.href = realLink;
|
||||||
|
// Sometimes, two fixups are needed, on old mobile user agents:
|
||||||
|
// /url?q=https://googleweblight.com/fp?u=... -> ...
|
||||||
|
realLink = getRealLinkFromGoogleUrl(a);
|
||||||
|
if (realLink) {
|
||||||
|
a.href = realLink;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
updateReferrerPolicy(a);
|
||||||
|
|
||||||
|
if (e.eventPhase === Event.CAPTURING_PHASE) {
|
||||||
|
// Our event listener runs first, to sanitize the link.
|
||||||
|
// But the page may have an event handler that modifies the link again.
|
||||||
|
// We can append a listener to the bubbling phase of the (current)
|
||||||
|
// event dispatch to fix the link up again, provided that the page did
|
||||||
|
// not call stopPropagation() or stopImmediatePropagation().
|
||||||
|
var eventOptions = { capture: false, once: true };
|
||||||
|
a.addEventListener(e.type, handlePointerPress, eventOptions);
|
||||||
|
document.addEventListener(e.type, handlePointerPress, eventOptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is specifically designed for catching clicks in Gmail.
|
||||||
|
// Gmail binds a click handler to a <div> and cancels the event after opening
|
||||||
|
// a window with an ugly URL. It uses a blank window + meta refresh in Firefox,
|
||||||
|
// which is too crazy to patch. So we just make sure that the browser's default
|
||||||
|
// click handler is activated (=open link in new tab).
|
||||||
|
// The entry point for this crazy stuff is shown in my comment at
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/2
|
||||||
|
function handleClick(e) {
|
||||||
|
if (e.button !== 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var a = e.target;
|
||||||
|
while (a && !a.href) {
|
||||||
|
a = a.parentElement;
|
||||||
|
}
|
||||||
|
if (!a) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (a.dataset && a.dataset.url) {
|
||||||
|
var realLink = getSanitizedIntentUrl(a.dataset.url);
|
||||||
|
if (realLink) {
|
||||||
|
a.dataset.url = realLink;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!location.hostname.startsWith('mail.')) {
|
||||||
|
// This hack was designed for Gmail, but broke other Google sites:
|
||||||
|
// - https://github.com/Rob--W/dont-track-me-google/issues/6
|
||||||
|
// - https://github.com/Rob--W/dont-track-me-google/issues/19
|
||||||
|
// So let's disable it for every domain except Gmail.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// TODO: Consider using a.baseURI instead of location in case Gmail ever
|
||||||
|
// starts using <base href>?
|
||||||
|
if (a.origin === location.origin) {
|
||||||
|
// Same-origin link.
|
||||||
|
// E.g. an in-page navigation at Google Docs (#...)
|
||||||
|
// or an attachment at Gmail (https://mail.google.com/mail/u/0?ui=2&...)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (a.protocol !== 'http:' &&
|
||||||
|
a.protocol !== 'https:' &&
|
||||||
|
a.protocol !== 'ftp:') {
|
||||||
|
// Be conservative and don't block too much. E.g. Gmail has special
|
||||||
|
// handling for mailto:-URLs, and using stopPropagation now would
|
||||||
|
// cause mailto:-links to be opened by the platform's default mailto
|
||||||
|
// handler instead of Gmail's handler (=open in new window).
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (a.target === '_blank') {
|
||||||
|
e.stopPropagation();
|
||||||
|
updateReferrerPolicy(a);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {URL|HTMLHyperlinkElementUtils} a
|
||||||
|
* @returns {String} the real URL if the given link is a Google redirect URL.
|
||||||
|
*/
|
||||||
|
function getRealLinkFromGoogleUrl(a) {
|
||||||
|
if (a.protocol !== 'https:' && a.protocol !== 'http:') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var url;
|
||||||
|
if ((a.hostname === location.hostname || a.hostname === 'www.google.com') &&
|
||||||
|
(a.pathname === '/url' || a.pathname === '/local_url' ||
|
||||||
|
a.pathname === '/searchurl/rr.html' ||
|
||||||
|
a.pathname === '/linkredirect')) {
|
||||||
|
// Google Maps / Dito (/local_url?q=<url>)
|
||||||
|
// Mobile (/url?q=<url>)
|
||||||
|
// Google Meet's chat (/linkredirect?authuser=0&dest=<url>)
|
||||||
|
url = /[?&](?:q|url|dest)=((?:https?|ftp)[%:][^&]+)/.exec(a.search);
|
||||||
|
if (url) {
|
||||||
|
return decodeURIComponent(url[1]);
|
||||||
|
}
|
||||||
|
// Help pages, e.g. safe browsing (/url?...&q=%2Fsupport%2Fanswer...)
|
||||||
|
url = /[?&](?:q|url)=((?:%2[Ff]|\/)[^&]+)/.exec(a.search);
|
||||||
|
if (url) {
|
||||||
|
return a.origin + decodeURIComponent(url[1]);
|
||||||
|
}
|
||||||
|
// Redirect pages for Android intents (/searchurl/rr.html#...&url=...)
|
||||||
|
// rr.html only supports http(s). So restrict to http(s) only.
|
||||||
|
url = /[#&]url=(https?[:%][^&]+)/.exec(a.hash);
|
||||||
|
if (url) {
|
||||||
|
return decodeURIComponent(url[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Google Search with old mobile UA (e.g. Firefox 41).
|
||||||
|
if (a.hostname === 'googleweblight.com' && a.pathname === '/fp') {
|
||||||
|
url = /[?&]u=((?:https?|ftp)[%:][^&]+)/.exec(a.search);
|
||||||
|
if (url) {
|
||||||
|
return decodeURIComponent(url[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {string} intentUrl
|
||||||
|
* @returns {string|undefined} The sanitized intent:-URL if it was an intent URL
|
||||||
|
* with embedded tracking link.
|
||||||
|
*/
|
||||||
|
function getSanitizedIntentUrl(intentUrl) {
|
||||||
|
if (!intentUrl.startsWith('intent:')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// https://developer.chrome.com/multidevice/android/intents#syntax
|
||||||
|
var BROWSER_FALLBACK_URL = ';S.browser_fallback_url=';
|
||||||
|
var indexStart = intentUrl.indexOf(BROWSER_FALLBACK_URL);
|
||||||
|
if (indexStart === -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
indexStart += BROWSER_FALLBACK_URL.length;
|
||||||
|
var indexEnd = intentUrl.indexOf(';', indexStart);
|
||||||
|
indexEnd = indexEnd === -1 ? intentUrl.length : indexEnd;
|
||||||
|
|
||||||
|
var url = decodeURIComponent(intentUrl.substring(indexStart, indexEnd));
|
||||||
|
var realUrl = getRealLinkFromGoogleUrl(newURL(url));
|
||||||
|
if (!realUrl) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
return intentUrl.substring(0, indexStart) +
|
||||||
|
encodeURIComponent(realUrl) +
|
||||||
|
intentUrl.substring(indexEnd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Intercept the .href setter in the page so that the page can never change the
|
||||||
|
* URL to a tracking URL. Just intercepting mousedown/touchstart is not enough
|
||||||
|
* because e.g. on Google Maps, the page rewrites the URL in the contextmenu
|
||||||
|
* event at the bubbling event stage and then stops the event propagation. So
|
||||||
|
* there is no event-driven way to fix the URL. The DOMAttrModified event could
|
||||||
|
* be used, but the event is deprecated, so not a viable long-term solution.
|
||||||
|
*/
|
||||||
|
function setupAggresiveUglyLinkPreventer() {
|
||||||
|
// This content script runs as document_start, so we can have some assurance
|
||||||
|
// that the methods in the page are reliable.
|
||||||
|
var s = document.createElement('script');
|
||||||
|
if (getScriptCspNonce()) {
|
||||||
|
s.setAttribute('nonce', scriptCspNonce);
|
||||||
|
} else if (document.readyState !== 'complete' && needsCspNonce) {
|
||||||
|
// In Firefox, a page's CSP is enforced for content scripts, so we need
|
||||||
|
// to wait for the document to be loaded (we may be at document_start)
|
||||||
|
// and find a fitting CSP nonce.
|
||||||
|
findScriptCspNonce(setupAggresiveUglyLinkPreventer);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
s.textContent = '(' + function(getRealLinkFromGoogleUrl) {
|
||||||
|
var proto = HTMLAnchorElement.prototype;
|
||||||
|
// The link target can be changed in many ways, but let's only consider
|
||||||
|
// the .href attribute since it's probably the only used setter.
|
||||||
|
var hrefProp = Object.getOwnPropertyDescriptor(proto, 'href');
|
||||||
|
var hrefGet = Function.prototype.call.bind(hrefProp.get);
|
||||||
|
var hrefSet = Function.prototype.call.bind(hrefProp.set);
|
||||||
|
|
||||||
|
Object.defineProperty(proto, 'href', {
|
||||||
|
configurable: true,
|
||||||
|
enumerable: true,
|
||||||
|
get() {
|
||||||
|
return hrefGet(this);
|
||||||
|
},
|
||||||
|
set(v) {
|
||||||
|
hrefSet(this, v);
|
||||||
|
try {
|
||||||
|
v = getRealLinkFromGoogleUrl(this);
|
||||||
|
if (v) {
|
||||||
|
hrefSet(this, v);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Not expected to happen, but don't break the setter if for
|
||||||
|
// some reason the (hostile) page broke the link APIs.
|
||||||
|
}
|
||||||
|
updateReferrerPolicy(this);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
function replaceAMethod(methodName, methodFunc) {
|
||||||
|
// Overwrite the methods without triggering setters, because that
|
||||||
|
// may inadvertently overwrite the prototype, as observed in
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/52#issuecomment-1596207655
|
||||||
|
Object.defineProperty(proto, methodName, {
|
||||||
|
configurable: true,
|
||||||
|
// All methods that we are overriding are not part of
|
||||||
|
// HTMLAnchorElement.prototype, but inherit.
|
||||||
|
enumerable: false,
|
||||||
|
writable: true,
|
||||||
|
value: methodFunc,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// proto inherits Element.prototype.setAttribute:
|
||||||
|
var setAttribute = Function.prototype.call.bind(proto.setAttribute);
|
||||||
|
replaceAMethod('setAttribute', function(name, value) {
|
||||||
|
// Attribute names are not case-sensitive, but weird capitalizations
|
||||||
|
// are unlikely, so only check all-lowercase and all-uppercase.
|
||||||
|
if (name === 'href' || name === 'HREF') {
|
||||||
|
this.href = value;
|
||||||
|
} else {
|
||||||
|
setAttribute(this, name, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// proto inherits EventTarget.prototype.dispatchEvent:
|
||||||
|
var aDispatchEvent = Function.prototype.apply.bind(proto.dispatchEvent);
|
||||||
|
replaceAMethod('dispatchEvent', function() {
|
||||||
|
updateReferrerPolicy(this);
|
||||||
|
return aDispatchEvent(this, arguments);
|
||||||
|
});
|
||||||
|
|
||||||
|
// proto inherits HTMLElement.prototype.click:
|
||||||
|
var aClick = Function.prototype.apply.bind(proto.click);
|
||||||
|
replaceAMethod('click', function() {
|
||||||
|
updateReferrerPolicy(this);
|
||||||
|
return aClick(this, arguments);
|
||||||
|
});
|
||||||
|
|
||||||
|
var rpProp = Object.getOwnPropertyDescriptor(proto, 'referrerPolicy');
|
||||||
|
var rpGet = Function.prototype.call.bind(rpProp.get);
|
||||||
|
var rpSet = Function.prototype.call.bind(rpProp.set);
|
||||||
|
|
||||||
|
var currentScript = document.currentScript;
|
||||||
|
var getReferrerPolicy = Object.getOwnPropertyDescriptor(
|
||||||
|
HTMLScriptElement.prototype,
|
||||||
|
'referrerPolicy'
|
||||||
|
).get.bind(currentScript);
|
||||||
|
|
||||||
|
function updateReferrerPolicy(a) {
|
||||||
|
try {
|
||||||
|
if (rpGet(a) === 'no-referrer') {
|
||||||
|
// "no-referrer" is more privacy-friendly than "origin".
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var referrerPolicy = getReferrerPolicy();
|
||||||
|
if (referrerPolicy) {
|
||||||
|
rpSet(a, referrerPolicy);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Not expected to happen, but don't break callers if it happens
|
||||||
|
// anyway.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
currentScript.dataset.jsEnabled = 1;
|
||||||
|
} + ')(' + getRealLinkFromGoogleUrl + ');';
|
||||||
|
callImmediatelyAndOnPreferenceUpdate(function forceNoReferrerChanged() {
|
||||||
|
// Send the desired referrerPolicy value to the injected script.
|
||||||
|
s.referrerPolicy = getReferrerPolicy();
|
||||||
|
});
|
||||||
|
(document.head || document.documentElement).appendChild(s);
|
||||||
|
s.remove();
|
||||||
|
if (!s.dataset.jsEnabled) {
|
||||||
|
cleanLinksWhenJsIsDisabled();
|
||||||
|
if (!needsCspNonce) {
|
||||||
|
needsCspNonce = true;
|
||||||
|
// This is not Firefox, but the script was blocked. Perhaps a CSP
|
||||||
|
// nonce is needed anyway.
|
||||||
|
findScriptCspNonce(function() {
|
||||||
|
if (scriptCspNonce) {
|
||||||
|
setupAggresiveUglyLinkPreventer();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Scripts enabled (not blocked by CSP), run other inline scripts.
|
||||||
|
blockTrackingBeacons();
|
||||||
|
overwriteWindowOpen();
|
||||||
|
|
||||||
|
if (location.hostname === 'docs.google.com') {
|
||||||
|
// Google Docs have simple non-JS interfaces where the ugly links
|
||||||
|
// are hard-coded in the HTML. Remove them (#51).
|
||||||
|
// https://docs.google.com/document/d/.../mobilebasic
|
||||||
|
// https://docs.google.com/spreadsheets/d/.../htmlview
|
||||||
|
cleanLinksWhenJsIsDisabled();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Block sendBeacon requests with destination /gen_204, because Google
|
||||||
|
// asynchronously sends beacon requests in response to mouse events on links:
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/20
|
||||||
|
//
|
||||||
|
// This implementation also blocks other forms of tracking via gen_204 as a side
|
||||||
|
// effect. That is not fully intentional, but given the lack of obvious ways to
|
||||||
|
// discern such link-tracking events from others, I will block all of them.
|
||||||
|
function blockTrackingBeacons() {
|
||||||
|
var s = document.createElement('script');
|
||||||
|
if (getScriptCspNonce()) {
|
||||||
|
s.setAttribute('nonce', scriptCspNonce);
|
||||||
|
}
|
||||||
|
s.textContent = '(' + function() {
|
||||||
|
var navProto = window.Navigator.prototype;
|
||||||
|
var navProtoSendBeacon = navProto.sendBeacon;
|
||||||
|
if (!navProtoSendBeacon) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var sendBeacon = Function.prototype.apply.bind(navProtoSendBeacon);
|
||||||
|
|
||||||
|
// Blocks the following:
|
||||||
|
// gen_204
|
||||||
|
// /gen_204
|
||||||
|
// https://www.google.com/gen_204
|
||||||
|
var isTrackingUrl = RegExp.prototype.test.bind(
|
||||||
|
/^(?:(?:https?:\/\/[^\/]+)?\/)?gen_204(?:[?#]|$)/);
|
||||||
|
|
||||||
|
navProto.sendBeacon = function(url, data) {
|
||||||
|
if (isTrackingUrl(url) && isNoPingEnabled()) {
|
||||||
|
// Lie that the data has been transmitted to avoid fallbacks.
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return sendBeacon(this, arguments);
|
||||||
|
};
|
||||||
|
|
||||||
|
var currentScript = document.currentScript;
|
||||||
|
var getElementId = Object.getOwnPropertyDescriptor(
|
||||||
|
Element.prototype,
|
||||||
|
'id'
|
||||||
|
).get.bind(currentScript);
|
||||||
|
function isNoPingEnabled() {
|
||||||
|
try {
|
||||||
|
return getElementId() !== '_dtmg_do_not_touch_ping';
|
||||||
|
} catch (e) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} + ')();';
|
||||||
|
callImmediatelyAndOnPreferenceUpdate(function nopingChanged() {
|
||||||
|
// Send the noping value to the injected script. The "id" property is
|
||||||
|
// mirrored and can have an arbitrary (string) value, so we use that:
|
||||||
|
s.id = noping ? '' : '_dtmg_do_not_touch_ping';
|
||||||
|
});
|
||||||
|
(document.head || document.documentElement).appendChild(s);
|
||||||
|
s.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Google sometimes uses window.open() to open ugly links.
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/18
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/41
|
||||||
|
function overwriteWindowOpen() {
|
||||||
|
var s = document.createElement('script');
|
||||||
|
if (getScriptCspNonce()) {
|
||||||
|
s.setAttribute('nonce', scriptCspNonce);
|
||||||
|
}
|
||||||
|
s.textContent = '(' + function() {
|
||||||
|
var open = window.open;
|
||||||
|
window.open = function(url, windowName, windowFeatures) {
|
||||||
|
var isBlankUrl = !url || url === "about:blank";
|
||||||
|
try {
|
||||||
|
if (!isBlankUrl) {
|
||||||
|
var a = document.createElement('a');
|
||||||
|
// Triggers getRealLinkFromGoogleUrl via the href setter in
|
||||||
|
// setupAggresiveUglyLinkPreventer.
|
||||||
|
a.href = url;
|
||||||
|
url = a.href;
|
||||||
|
// The origin check exists to avoid adding "noreferrer" to
|
||||||
|
// same-origin popups. That implies noopener and causes
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/43
|
||||||
|
// And allow any Google domain to support auth popups:
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/45
|
||||||
|
// And don't bother editing the list if it already contains
|
||||||
|
// "opener" (it would be disabled by "noreferrer").
|
||||||
|
if (a.referrerPolicy && a.origin !== location.origin &&
|
||||||
|
!/\.google\.([a-z]+)$/.test(a.hostname) &&
|
||||||
|
!/\bopener|noreferrer/.test(windowFeatures)) {
|
||||||
|
if (windowFeatures) {
|
||||||
|
windowFeatures += ',';
|
||||||
|
} else {
|
||||||
|
windowFeatures = '';
|
||||||
|
}
|
||||||
|
windowFeatures += 'noreferrer';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Not expected to happen, but don't break callers if it does.
|
||||||
|
}
|
||||||
|
var win = open(url, windowName, windowFeatures);
|
||||||
|
try {
|
||||||
|
if (isBlankUrl && win) {
|
||||||
|
// In Google Docs, sometimes a blank document is opened,
|
||||||
|
// and document.write is used to insert a redirector.
|
||||||
|
// https://github.com/Rob--W/dont-track-me-google/issues/41
|
||||||
|
var doc = win.document;
|
||||||
|
var docWrite = win.Function.prototype.call.bind(doc.write);
|
||||||
|
doc.write = function(markup) {
|
||||||
|
try {
|
||||||
|
markup = fixupDocMarkup(markup);
|
||||||
|
} catch (e) {
|
||||||
|
// Not expected, but don't break callers otherwise.
|
||||||
|
}
|
||||||
|
return docWrite(this, markup);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Not expected to happen, but don't break callers if it does.
|
||||||
|
}
|
||||||
|
return win;
|
||||||
|
};
|
||||||
|
function fixupDocMarkup(html) {
|
||||||
|
html = html || '';
|
||||||
|
html += '';
|
||||||
|
return html.replace(
|
||||||
|
/<meta [^>]*http-equiv=(["']?)refresh\1[^>]*>/i,
|
||||||
|
function(m) {
|
||||||
|
var doc = new DOMParser().parseFromString(m, 'text/html');
|
||||||
|
var meta = doc.querySelector('meta[http-equiv=refresh]');
|
||||||
|
return meta && fixupMetaUrl(meta) || m;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
function fixupMetaUrl(meta) {
|
||||||
|
var parts = /^(\d*;\s*url=)(.+)$/i.exec(meta.content);
|
||||||
|
if (!parts) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var metaPrefix = parts[1];
|
||||||
|
var url = parts[2];
|
||||||
|
var a = document.createElement('a');
|
||||||
|
// Triggers getRealLinkFromGoogleUrl via the href setter in
|
||||||
|
// setupAggresiveUglyLinkPreventer.
|
||||||
|
a.href = url;
|
||||||
|
url = a.href;
|
||||||
|
meta.content = metaPrefix + url;
|
||||||
|
|
||||||
|
var html = meta.outerHTML;
|
||||||
|
if (a.referrerPolicy) {
|
||||||
|
// Google appears to already append the no-referrer
|
||||||
|
// meta tag, but add one just in case it doesn't.
|
||||||
|
html = '<meta name="referrer" content="no-referrer">' + html;
|
||||||
|
}
|
||||||
|
return html;
|
||||||
|
}
|
||||||
|
} + ')();';
|
||||||
|
(document.head || document.documentElement).appendChild(s);
|
||||||
|
s.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
function cleanLinksWhenJsIsDisabled() {
|
||||||
|
// When JavaScript is disabled, Google sets the "href" attribute's value to
|
||||||
|
// an ugly URL. Although the link is rewritten on click, we still need to
|
||||||
|
// rewrite the link even earlier because otherwise the ugly URL is shown in
|
||||||
|
// the tooltip upon hover.
|
||||||
|
|
||||||
|
if (document.readyState == 'complete') {
|
||||||
|
cleanAllLinks();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// When JS is disabled, the links won't change after the document finishes
|
||||||
|
// loading. Until the DOM has finished loading, use the mouseover event to
|
||||||
|
// beautify links (the DOMContentLoaded may be delayed on slow networks).
|
||||||
|
document.addEventListener('mouseover', handleMouseOver);
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
document.removeEventListener('mouseover', handleMouseOver);
|
||||||
|
cleanAllLinks();
|
||||||
|
}, {once: true});
|
||||||
|
|
||||||
|
function cleanAllLinks() {
|
||||||
|
var as = document.querySelectorAll('a[href]');
|
||||||
|
for (var i = 0; i < as.length; ++i) {
|
||||||
|
var href = getRealLinkFromGoogleUrl(as[i]);
|
||||||
|
if (href) {
|
||||||
|
as[i].href = href;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleMouseOver(e) {
|
||||||
|
var a = e.target;
|
||||||
|
var href = a.href && getRealLinkFromGoogleUrl(a);
|
||||||
|
if (href) {
|
||||||
|
a.href = href;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getScriptCspNonce() {
|
||||||
|
var scripts = document.querySelectorAll('script[nonce]');
|
||||||
|
for (var i = 0; i < scripts.length && !scriptCspNonce; ++i) {
|
||||||
|
scriptCspNonce = scripts[i].nonce;
|
||||||
|
}
|
||||||
|
return scriptCspNonce;
|
||||||
|
}
|
||||||
|
|
||||||
|
function findScriptCspNonce(callback) {
|
||||||
|
var timer;
|
||||||
|
function checkDOM() {
|
||||||
|
if (getScriptCspNonce() || document.readyState === 'complete') {
|
||||||
|
document.removeEventListener('DOMContentLoaded', checkDOM, true);
|
||||||
|
if (timer) {
|
||||||
|
clearTimeout(timer);
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
timer = setTimeout(checkDOM, 50);
|
||||||
|
}
|
||||||
|
document.addEventListener('DOMContentLoaded', checkDOM, true);
|
||||||
|
checkDOM();
|
||||||
|
}
|
||||||
|
|
||||||
|
function newURL(href) {
|
||||||
|
try {
|
||||||
|
return new URL(href);
|
||||||
|
} catch (e) {
|
||||||
|
var a = document.createElement('a');
|
||||||
|
a.href = href;
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,984 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name I don't care about cookies
|
||||||
|
// @name:vi Tôi không quan tâm về cookie
|
||||||
|
// @name:zh-CN 我不关心cookie
|
||||||
|
// @name:zh-TW 我不關心cookie
|
||||||
|
// @name:ja クッキーについては気にしない
|
||||||
|
// @name:ru Я не забочусь о куки
|
||||||
|
// @namespace http://tampermonkey.net/
|
||||||
|
// @version 2025.01.03.2
|
||||||
|
// @description Remove cookie warnings from almost all websites! Auto accept cookies and remove annoying cookie popups
|
||||||
|
// @description:vi Loại bỏ cảnh báo cookie từ hầu hết các trang web! Tự động chấp nhận cookie và xóa các popup cookie phiền phức
|
||||||
|
// @description:zh-CN 自动接受cookie并移除烦人的cookie弹窗
|
||||||
|
// @description:zh-TW 自動接受cookie並移除煩人的cookie彈窗
|
||||||
|
// @description:ru Автоматическое принятие cookie и удаление надоедливых всплывающих окон
|
||||||
|
// @description:ja 自動承認cookieと迷惑なポップアップを削除
|
||||||
|
// @author Yuusei
|
||||||
|
// @match *://*/*
|
||||||
|
// @grant none
|
||||||
|
// @icon https://lh3.googleusercontent.com/sCLTYpGX0VcVootQ_XaFQ9saRIhVWu79ngSzY5eTZ5evRpJ_Q27OdvxA4RrOoZXP7Q4enFh-u6VhxObcJLfARw1g=s60
|
||||||
|
// @compatible chrome
|
||||||
|
// @compatible edge
|
||||||
|
// @compatible firefox
|
||||||
|
// @compatible safari
|
||||||
|
// @run-at document-start
|
||||||
|
// @license gpl-3.0-only
|
||||||
|
// @downloadURL https://update.greasyfork.org/scripts/522645/I%20don%27t%20care%20about%20cookies.user.js
|
||||||
|
// @updateURL https://update.greasyfork.org/scripts/522645/I%20don%27t%20care%20about%20cookies.meta.js
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const CONSENT_TEXTS = {
|
||||||
|
en: ['accept', 'accept all', 'agree', 'continue', 'got it', 'reject all', 'decline', 'necessary only', 'required only', 'manage', 'customize'],
|
||||||
|
vi: ['chấp nhận', 'chấp nhận tất cả', 'đồng ý', 'tiếp tục', 'từ chối tất cả', 'từ chối', 'chỉ cần thiết', 'tùy chỉnh', 'quản lý', 'cho phép', 'đồng ý và tiếp tục', 'tôi đồng ý', 'xác nhận', 'tôi chấp nhận', 'đồng ý tất cả', 'chấp nhận và tiếp tục', 'cho phép tất cả'],
|
||||||
|
zh: ['接受', '接受全部', '同意', '继续', '拒绝全部', '拒绝', '仅必要', '管理', '自定义'],
|
||||||
|
ru: ['принять', 'принять все', 'согласен', 'продолжить', 'отклонить все', 'отклонить', 'только необходимые', 'настроить', 'управлять'],
|
||||||
|
ja: ['承認', '同意', '続ける', 'すべて拒否', '拒否', '必要のみ', 'カスタマイズ', '管理'],
|
||||||
|
de: ['akzeptieren', 'einverstanden', 'fortfahren', 'alle ablehnen', 'ablehnen', 'nur notwendige', 'anpassen', 'verwalten'],
|
||||||
|
fr: ['accepter', 'accepter tout', 'accepte', 'continuer', 'tout refuser', 'refuser', 'uniquement nécessaire', 'personnaliser', 'gérer'],
|
||||||
|
es: ['aceptar', 'acepto todo', 'acepto', 'continuar', 'rechazar todo', 'rechazar', 'solo necesario', 'personalizar', 'gestionar'],
|
||||||
|
it: ['accetta', 'accetto tutto', 'accetto', 'continua', 'rifiuta tutto', 'rifiuta', 'solo necessari', 'personalizza', 'gestisci'],
|
||||||
|
pl: ['akceptuj', 'akceptuj wszystko', 'zgadzam się', 'kontynuuj', 'odrzuć wszystko', 'odrzuć', 'tylko niezbędne', 'dostosuj', 'zarządzaj'],
|
||||||
|
nl: ['accepteren', 'accepteren', 'doorgaan', 'alles weigeren', 'weigeren', 'alleen noodzakelijk', 'aanpassen', 'beheren'],
|
||||||
|
ko: ['동의', '모두 동의', '계속하기', '모두 거부', '거부', '필수만', '설정', '관리'],
|
||||||
|
th: ['ยอมรับ', 'ยอมรับทั้งหมด', 'ตกลง', 'ปฏิเสธทั้งหมด', 'ปฏิเสธ', 'จำเป็นเท่านั้น', 'ตั้งค่า', 'จัดการ'],
|
||||||
|
id: ['setuju', 'setuju semua', 'lanjutkan', 'tolak semua', 'tolak', 'wajib saja', 'pengaturan', 'kelola'],
|
||||||
|
ms: ['terima', 'terima semua', 'teruskan', 'tolak semua', 'tolak', 'perlu sahaja', 'tetapan', 'urus'],
|
||||||
|
pt: ['aceitar', 'aceitar tudo', 'continuar', 'rejeitar tudo', 'rejeitar', 'necessário', 'configurar', 'gerir'],
|
||||||
|
sv: ['godkänn', 'godkänn alla', 'fortsätt', 'neka alla', 'neka', 'nödvändiga', 'inställningar', 'hantera'],
|
||||||
|
da: ['accepter', 'accepter alle', 'fortsæt', 'afvis alle', 'afvis', 'nødvendige', 'indstillinger', 'administrer'],
|
||||||
|
fi: ['hyväksy', 'hyväksy kaikki', 'jatka', 'hylkää kaikki', 'hylkää', 'välttämätön', 'asetukset', 'hallitse'],
|
||||||
|
'zh-CN': ['接受', '接受全部', '同意', '继续', '我同意', '确定', '确认', '知道了', '好的', '拒绝全部', '拒绝', '仅必要', '设置', '自定义', '管理', '保存设置', '允许', '允许全部', '接受并继续', '同意并继续', '保存并继续'],
|
||||||
|
'zh-TW': ['接受', '接受全部', '同意', '繼續', '我同意', '確定', '確認', '知道了', '好的', '拒絕全部', '拒絕', '僅必要', '設置', '自定義', '管理', '保存設置', '允許', '允許全部', '接受並繼續', '同意並繼續', '保存並繼續'],
|
||||||
|
ko: ['동의', '모두 동의', '수락', '계속하기', '확인', '거부', '거부하기', '필수만', '설정', '관리', '저장', '허용', '모두 허용'],
|
||||||
|
th: ['ยอมรับ', 'ยอมรับทั้งหมด', 'ตกลง', 'ดำเนินการต่อ', 'ปฏิเสธ', 'ปฏิเสธทั้งหมด', 'จำเป็นเท่านั้น', 'ตั้งค่า', 'จัดการ', 'บันทึก', 'อนุญาต', 'อนุญาตทั้งหมด'],
|
||||||
|
id: ['terima', 'terima semua', 'setuju', 'lanjutkan', 'tolak', 'tolak semua', 'wajib saja', 'pengaturan', 'kelola', 'simpan', 'izinkan', 'izinkan semua'],
|
||||||
|
ms: ['terima', 'terima semua', 'setuju', 'teruskan', 'tolak', 'tolak semua', 'perlu sahaja', 'tetapan', 'urus', 'simpan', 'benarkan', 'benarkan semua'],
|
||||||
|
};
|
||||||
|
|
||||||
|
function matchesConsentText(element) {
|
||||||
|
const text = element.textContent.toLowerCase();
|
||||||
|
const lang = document.documentElement.lang || 'en';
|
||||||
|
const texts = CONSENT_TEXTS[lang.split('-')[0]] || CONSENT_TEXTS['en'];
|
||||||
|
return texts.some(t => text.includes(t));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Utility functions
|
||||||
|
function _sl(selector, container) {
|
||||||
|
return (container || document).querySelector(selector);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _id(id) {
|
||||||
|
return document.getElementById(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _ev(selector, container, full) {
|
||||||
|
return document.evaluate((typeof full == 'undefined' ? '//' : '') + selector, container || document, null, XPathResult.ANY_TYPE, null).iterateNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
function _if(condition, ...selectors) {
|
||||||
|
return _sl(condition) ? _chain(...selectors) : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function _if_else(condition, if_selectors, else_selectors) {
|
||||||
|
if (_sl(condition)) return _chain(...if_selectors);
|
||||||
|
|
||||||
|
return _chain(...else_selectors);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _chain(...selectors) {
|
||||||
|
let elements,
|
||||||
|
l = selectors.length;
|
||||||
|
let flagUnique = false,
|
||||||
|
flagOptional = false,
|
||||||
|
flagAllMatches = false;
|
||||||
|
|
||||||
|
for (let i = currentChainElement; i < l; i++) {
|
||||||
|
if (/^FLAG\:/.test(selectors[i])) {
|
||||||
|
selectors[i]
|
||||||
|
.split(':')[1]
|
||||||
|
.split(',')
|
||||||
|
.forEach(function (flag) {
|
||||||
|
if (flag == 'UNIQUE') flagUnique = true;
|
||||||
|
else if (flag == 'OPTIONAL') flagOptional = true;
|
||||||
|
else if (flag == 'REQUIRED') flagOptional = false;
|
||||||
|
else if (flag == 'ALL-MATCHES') flagAllMatches = true;
|
||||||
|
else if (flag == 'SINGLE-MATCH') flagAllMatches = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flagUnique) selectors[i] += selectors[i].startsWith('//') ? '[not(contains(@class, "' + classname + '"))]' : ':not(.' + classname + ')';
|
||||||
|
|
||||||
|
if (i == l - 1) return selectors[i];
|
||||||
|
|
||||||
|
elements = _sl(selectors[i], false, flagAllMatches);
|
||||||
|
|
||||||
|
if (!flagAllMatches) elements = elements ? [elements] : [];
|
||||||
|
|
||||||
|
if (!elements.length) {
|
||||||
|
if (flagOptional) {
|
||||||
|
currentChainElement++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
currentChainElement++;
|
||||||
|
|
||||||
|
elements.forEach(function (element) {
|
||||||
|
if (flagUnique) element.classList.add(classname);
|
||||||
|
|
||||||
|
if (element.nodeName == 'OPTION') element.selected = true;
|
||||||
|
else element.click();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getItem(hostname) {
|
||||||
|
switch (hostname) {
|
||||||
|
case 'youtube.com':
|
||||||
|
case 'www.youtube.com':
|
||||||
|
return { strict: true, key: 'CONSENT', value: 'PENDING+999' };
|
||||||
|
|
||||||
|
case 'google.com':
|
||||||
|
case 'www.google.com':
|
||||||
|
return { strict: true, key: 'CONSENT', value: 'YES+' };
|
||||||
|
|
||||||
|
case 'twitter.com':
|
||||||
|
case 'www.twitter.com':
|
||||||
|
return { strict: false, key: 'twtr_cookie_consent', value: '1' };
|
||||||
|
|
||||||
|
case 'pepephone.com':
|
||||||
|
case 'lyricsbox.com':
|
||||||
|
return { strict: true, key: 'cookieconsent', value: '1111' };
|
||||||
|
|
||||||
|
case 'kontaktbazar.at':
|
||||||
|
case 'hoernews.de':
|
||||||
|
return { strict: false, key: 'cookieconsent_status', value: 'dismiss' };
|
||||||
|
|
||||||
|
case 'vodafoneziggo.nl':
|
||||||
|
return { strict: false, key: 'cookies-accepted', value: 'true' };
|
||||||
|
case 'frankfurt.de':
|
||||||
|
return { strict: false, key: 'cookieAccepted', value: 'needed---piwik' };
|
||||||
|
case 'hackerrank.com':
|
||||||
|
return { strict: false, key: 'show_cookie_banner', value: 'false' };
|
||||||
|
|
||||||
|
case 'facebook.com':
|
||||||
|
case 'www.facebook.com':
|
||||||
|
return { strict: true, key: 'datr', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'instagram.com':
|
||||||
|
case 'www.instagram.com':
|
||||||
|
return { strict: true, key: 'ig_did', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'linkedin.com':
|
||||||
|
case 'www.linkedin.com':
|
||||||
|
return { strict: false, key: 'lidc', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'reddit.com':
|
||||||
|
case 'www.reddit.com':
|
||||||
|
return { strict: false, key: 'eu_cookie', value: '{"opted":true}' };
|
||||||
|
|
||||||
|
case 'tiktok.com':
|
||||||
|
case 'www.tiktok.com':
|
||||||
|
return { strict: true, key: 'tt_webid', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'netflix.com':
|
||||||
|
case 'www.netflix.com':
|
||||||
|
return { strict: false, key: 'netflix-cookie-consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'spotify.com':
|
||||||
|
case 'www.spotify.com':
|
||||||
|
return { strict: false, key: 'sp_dc', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'amazon.com':
|
||||||
|
case 'www.amazon.com':
|
||||||
|
return { strict: false, key: 'amazon-cookie-consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'pinterest.com':
|
||||||
|
case 'www.pinterest.com':
|
||||||
|
return { strict: false, key: '_pinterest_sess', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'twitch.tv':
|
||||||
|
case 'www.twitch.tv':
|
||||||
|
return { strict: false, key: 'twitch_cookie_consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'github.com':
|
||||||
|
case 'www.github.com':
|
||||||
|
return { strict: false, key: '_gh_sess', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'medium.com':
|
||||||
|
case 'www.medium.com':
|
||||||
|
return { strict: false, key: 'medium_cookie_consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'quora.com':
|
||||||
|
case 'www.quora.com':
|
||||||
|
return { strict: false, key: 'm-b', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'stackoverflow.com':
|
||||||
|
case 'www.stackoverflow.com':
|
||||||
|
return { strict: false, key: 'acct', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'microsoft.com':
|
||||||
|
case 'www.microsoft.com':
|
||||||
|
return { strict: false, key: 'MUID', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'apple.com':
|
||||||
|
case 'www.apple.com':
|
||||||
|
return { strict: false, key: 'geo', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'dropbox.com':
|
||||||
|
case 'www.dropbox.com':
|
||||||
|
return { strict: false, key: 'dbx-consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'booking.com':
|
||||||
|
case 'www.booking.com':
|
||||||
|
return { strict: false, key: 'bkng_consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'vnexpress.net':
|
||||||
|
case 'www.vnexpress.net':
|
||||||
|
return { strict: false, key: 'vnexpress_cookie_consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'thanhnien.vn':
|
||||||
|
case 'www.thanhnien.vn':
|
||||||
|
return { strict: false, key: 'thanhnien_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'tuoitre.vn':
|
||||||
|
case 'www.tuoitre.vn':
|
||||||
|
return { strict: false, key: 'tuoitre_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'tiki.vn':
|
||||||
|
case 'www.tiki.vn':
|
||||||
|
return { strict: false, key: 'tiki_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'shopee.vn':
|
||||||
|
case 'www.shopee.vn':
|
||||||
|
return { strict: false, key: 'shopee_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'lazada.vn':
|
||||||
|
case 'www.lazada.vn':
|
||||||
|
return { strict: false, key: 'lzd_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'sendo.vn':
|
||||||
|
case 'www.sendo.vn':
|
||||||
|
return { strict: false, key: 'sendo_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'thegioididong.com':
|
||||||
|
case 'www.thegioididong.com':
|
||||||
|
return { strict: false, key: 'tgdd_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'fptshop.com.vn':
|
||||||
|
case 'www.fptshop.com.vn':
|
||||||
|
return { strict: false, key: 'fpt_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'cellphones.com.vn':
|
||||||
|
case 'www.cellphones.com.vn':
|
||||||
|
return { strict: false, key: 'cps_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'aliexpress.com':
|
||||||
|
case 'www.aliexpress.com':
|
||||||
|
return { strict: false, key: 'aep_usuc_f', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'ebay.com':
|
||||||
|
case 'www.ebay.com':
|
||||||
|
return { strict: false, key: 'ebay_cookie_consent', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'coursera.org':
|
||||||
|
case 'www.coursera.org':
|
||||||
|
return { strict: false, key: 'coursera_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'udemy.com':
|
||||||
|
case 'www.udemy.com':
|
||||||
|
return { strict: false, key: 'ud_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'bachhoaxanh.com':
|
||||||
|
case 'www.bachhoaxanh.com':
|
||||||
|
return { strict: false, key: 'bhx_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'dienmayxanh.com':
|
||||||
|
case 'www.dienmayxanh.com':
|
||||||
|
return { strict: false, key: 'dmx_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'nguyenkim.com':
|
||||||
|
case 'www.nguyenkim.com':
|
||||||
|
return { strict: false, key: 'nk_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'dantri.com.vn':
|
||||||
|
case 'www.dantri.com.vn':
|
||||||
|
return { strict: false, key: 'dantri_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'vietnamnet.vn':
|
||||||
|
case 'www.vietnamnet.vn':
|
||||||
|
return { strict: false, key: 'vietnamnet_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case '24h.com.vn':
|
||||||
|
case 'www.24h.com.vn':
|
||||||
|
return { strict: false, key: '24h_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'vietcombank.com.vn':
|
||||||
|
case 'www.vietcombank.com.vn':
|
||||||
|
return { strict: false, key: 'vcb_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'techcombank.com.vn':
|
||||||
|
case 'www.techcombank.com.vn':
|
||||||
|
return { strict: false, key: 'tcb_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'mbbank.com.vn':
|
||||||
|
case 'www.mbbank.com.vn':
|
||||||
|
return { strict: false, key: 'mb_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'amazon.co.jp':
|
||||||
|
case 'www.amazon.co.jp':
|
||||||
|
return { strict: false, key: 'amazon_jp_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'rakuten.co.jp':
|
||||||
|
case 'www.rakuten.co.jp':
|
||||||
|
return { strict: false, key: 'rakuten_jp_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'taobao.com':
|
||||||
|
case 'www.taobao.com':
|
||||||
|
return { strict: false, key: 'taobao_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'line.me':
|
||||||
|
case 'www.line.me':
|
||||||
|
return { strict: false, key: 'line_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'weibo.com':
|
||||||
|
case 'www.weibo.com':
|
||||||
|
return { strict: false, key: 'weibo_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'kakao.com':
|
||||||
|
case 'www.kakao.com':
|
||||||
|
return { strict: false, key: 'kakao_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'yahoo.co.jp':
|
||||||
|
case 'www.yahoo.co.jp':
|
||||||
|
return { strict: false, key: 'yahoo_jp_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'naver.com':
|
||||||
|
case 'www.naver.com':
|
||||||
|
return { strict: false, key: 'naver_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'baidu.com':
|
||||||
|
case 'www.baidu.com':
|
||||||
|
return { strict: false, key: 'baidu_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'zalando.com':
|
||||||
|
case 'www.zalando.com':
|
||||||
|
return { strict: false, key: 'zalando_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'asos.com':
|
||||||
|
case 'www.asos.com':
|
||||||
|
return { strict: false, key: 'asos_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'zara.com':
|
||||||
|
case 'www.zara.com':
|
||||||
|
return { strict: false, key: 'zara_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'tmall.com':
|
||||||
|
case 'www.tmall.com':
|
||||||
|
return { strict: false, key: 'tmall_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'jd.com':
|
||||||
|
case 'www.jd.com':
|
||||||
|
return { strict: false, key: 'jd_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'sina.com.cn':
|
||||||
|
case 'www.sina.com.cn':
|
||||||
|
return { strict: false, key: 'sina_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'qq.com':
|
||||||
|
case 'www.qq.com':
|
||||||
|
return { strict: false, key: 'qq_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case '163.com':
|
||||||
|
case 'www.163.com':
|
||||||
|
return { strict: false, key: 'netease_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'sohu.com':
|
||||||
|
case 'www.sohu.com':
|
||||||
|
return { strict: false, key: 'sohu_cookie', value: 'accepted' };
|
||||||
|
|
||||||
|
case 'bilibili.com':
|
||||||
|
case 'www.bilibili.com':
|
||||||
|
return { strict: false, key: 'bilibili_cookie', value: 'accepted' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const parts = hostname.split('.');
|
||||||
|
if (parts.length > 2) {
|
||||||
|
parts.shift();
|
||||||
|
return getItem(parts.join('.'));
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function _parent(element) {
|
||||||
|
if (element && element.parentNode) return element.parentNode;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function _iframe(iframe_selector, selector) {
|
||||||
|
var e = _sl(iframe_selector);
|
||||||
|
return e ? _sl(selector, e.contentDocument || e.contentWindow.contentDocument) : e;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cookie consent handling logic
|
||||||
|
let searchPairs = {
|
||||||
|
'.vn-cookie-banner': ['.vn-cookie-banner__reject', '.vn-cookie-banner__customize'],
|
||||||
|
'.vn-cookie-notice': ['.vn-cookie-notice__reject', '.vn-cookie-notice__settings'],
|
||||||
|
'.vn-cookie-consent': ['.vn-cookie-consent__reject', '.vn-cookie-consent__customize'],
|
||||||
|
'.vn-cookie-popup': ['.vn-cookie-popup__reject', '.vn-cookie-popup__settings'],
|
||||||
|
|
||||||
|
'.shopee-cookie-banner': ['.shopee-cookie-banner__reject', '.shopee-cookie-banner__settings'],
|
||||||
|
'.lazada-cookie-notice': ['.lazada-cookie-notice__reject', '.lazada-cookie-notice__settings'],
|
||||||
|
'.tiki-cookie-popup': ['.tiki-cookie-popup__reject', '.tiki-cookie-popup__settings'],
|
||||||
|
'.sendo-cookie-consent': ['.sendo-cookie-consent__reject', '.sendo-cookie-consent__settings'],
|
||||||
|
|
||||||
|
'.vnexpress-cookie': ['.vnexpress-cookie__reject', '.vnexpress-cookie__settings'],
|
||||||
|
'.tuoitre-cookie': ['.tuoitre-cookie__reject', '.tuoitre-cookie__settings'],
|
||||||
|
'.thanhnien-cookie': ['.thanhnien-cookie__reject', '.thanhnien-cookie__settings'],
|
||||||
|
'.dantri-cookie': ['.dantri-cookie__reject', '.dantri-cookie__settings'],
|
||||||
|
|
||||||
|
'.tgdd-cookie-notice': ['.tgdd-cookie-notice__reject', '.tgdd-cookie-notice__settings'],
|
||||||
|
'.fpt-cookie-popup': ['.fpt-cookie-popup__reject', '.fpt-cookie-popup__settings'],
|
||||||
|
'.cps-cookie-consent': ['.cps-cookie-consent__reject', '.cps-cookie-consent__settings'],
|
||||||
|
|
||||||
|
'.vcb-cookie-notice': ['.vcb-cookie-notice__reject', '.vcb-cookie-notice__settings'],
|
||||||
|
'.tcb-cookie-popup': ['.tcb-cookie-popup__reject', '.tcb-cookie-popup__settings'],
|
||||||
|
'.mb-cookie-consent': ['.mb-cookie-consent__reject', '.mb-cookie-consent__settings'],
|
||||||
|
|
||||||
|
'div[class*="consent"]': ['button[class*="reject"]', 'button[class*="decline"]', 'button[class*="settings"]'],
|
||||||
|
'div[class*="notice"]': ['button[class*="reject"]', 'button[class*="decline"]', 'button[class*="settings"]'],
|
||||||
|
'div[class*="popup"]': ['button[class*="reject"]', 'button[class*="decline"]', 'button[class*="settings"]'],
|
||||||
|
'div[class*="banner"]': ['button[class*="reject"]', 'button[class*="decline"]', 'button[class*="settings"]'],
|
||||||
|
};
|
||||||
|
|
||||||
|
let searchGroups = [
|
||||||
|
'.qc-cmp2-summary-buttons button[mode="secondary"],\
|
||||||
|
.qc-cmp2-buttons-desktop > button:first-child,\
|
||||||
|
#didomi-popup .didomi-button-highlight:not([class*="paywall"]):not([class*="disagree"]),\
|
||||||
|
#rgpd_video .rgpd-mask a[data-rgpd-consent],\
|
||||||
|
.cli-barmodal-open #wt-cli-privacy-save-btn,\
|
||||||
|
.js--modal[style*="block"] .cookie-permission--accept-button,\
|
||||||
|
.gdpr-modal-rider .btn-cookieaccept,\
|
||||||
|
.js-cookiewall #sel-test-accept-cookies-button,\
|
||||||
|
#mpo[style*="block"] .submit.modal-privacy__btn[onclick*="privacyframe.accept"],\
|
||||||
|
.lightbox--cookie-consent .btn-cta,\
|
||||||
|
.lightbox.cookie-consent .cookie-consent-button-decline,\
|
||||||
|
.js-modal-gdpr.is-active .btn[data-level="2"],\
|
||||||
|
#CybotCookiebotDialogBodyLevelButtonLevelOptinAllowallSelection,\
|
||||||
|
#cookieNotificationModal.in .btn.accept-cookie,\
|
||||||
|
.has-ccwindow .cc-compliance .cc-dismiss,\
|
||||||
|
.ds2-cookie-disclaimer--slidedown .ds2-cookie-disclaimer-js--submit,\
|
||||||
|
#mdlCookieCompliance.in .cookieClose,\
|
||||||
|
#cookieModal.in .js-acceptDefaultCookie,\
|
||||||
|
.c-cookiebutton .c-cookiebutton__close,\
|
||||||
|
#normativa_cookies.in .btn,\
|
||||||
|
#cookiewall.in .btn-primary,\
|
||||||
|
.outerCookieBar .EuCookieBar__cookieButton,\
|
||||||
|
#TOS-POPUP .rhododendron-popup__button--agree,\
|
||||||
|
#cookie-wall #accept-cookies,\
|
||||||
|
#popup-wrapper .button[href*="/cookies.consent.php"],\
|
||||||
|
.reveal.cookies[style*="block"] button[click*="aceptaCookies"],\
|
||||||
|
.mnd-cookie-modal[style*="block"] .btn.is--primary,\
|
||||||
|
.cookieHandler.cookieHandler--modalOpen #acceptAllCookies,\
|
||||||
|
.gdpr-modal--active .btn--primary,\
|
||||||
|
#dpi-banner:not(.hidden) #btn-agree-cookie,\
|
||||||
|
.gh-banner.gh-banner-active #gh-cookiebanner-close,\
|
||||||
|
#mrktpref.notification-bar .btn-success,\
|
||||||
|
#PopinGDPRCookie[style*="block"] .jsbd-popin-ok,\
|
||||||
|
#modal-rodo.in .btn-primary,\
|
||||||
|
.cookie-compliance-modal.in .btn-primary,\
|
||||||
|
.cookieconsent.show .btn[data-dm*="accept"],\
|
||||||
|
.cookie-wall-modal.in .btn.ja,\
|
||||||
|
#modal-consent.in .modal-consent-accept,\
|
||||||
|
.rodo #cookies.in .btn-primary,\
|
||||||
|
.js-cookie-alert.in .js-cookie-alert-accept,\
|
||||||
|
#modal_gdpr_intro_popup.in #gdpr-modal-btn-ok-agree,\
|
||||||
|
#consentButtonContainer > button[onclick*="sendAndRedirect"],\
|
||||||
|
#eu-consent[style*="block"] .btn.yes,\
|
||||||
|
.modal--gdpr.is-open .js-gdpr-consent,\
|
||||||
|
#cookiePopupModal.in .cookiepopup-agreed,\
|
||||||
|
.polityka-cookie-rodo[style*="block"] .button-zgoda,\
|
||||||
|
.ui-dialog.consent-modal[style*="block"] .js-btn-agree,\
|
||||||
|
#up-cookie.active .button[onclick*="setCookiePreference"],\
|
||||||
|
.RodoModal.in .close,\
|
||||||
|
.consent-popup form[action*="cookie-consent"] .consent-popup__button,\
|
||||||
|
#consent form[action*="cookie-consent"] .one-btn,\
|
||||||
|
#cookiewall-wrapper .button[href*="accept"],\
|
||||||
|
#cookieChoiceButtonAccept,\
|
||||||
|
.mod-cookie-consent[style*="block"] .btn-all-cookies,\
|
||||||
|
.c-layer--consent .layer-button--accept,\
|
||||||
|
.button.button-ok[onclick*="acceptAVG"],\
|
||||||
|
#meredithGdprConsentFormButton,\
|
||||||
|
#advanced-cookie-modal.in .cookie-accept,\
|
||||||
|
.show-modal .cookie-settings-manager-container .initial-dialog .js-accept-button,\
|
||||||
|
.cookie-settings-manager-container .initial-dialog[style*="block"] .js-accept-button,\
|
||||||
|
.gdprLightbox[data-module="gdprLightbox"] ._type_gdpr_agree,\
|
||||||
|
.cookie.showa #Row1_Column1_Cell1_CookieSettings_AdvancedSaveAccept,\
|
||||||
|
#core-cookie-container[style*="block"] .btn--agree,\
|
||||||
|
.cookie-consent-modal._show .action-primary,\
|
||||||
|
#dsgvoModal.show #dsvgo-banner__button,\
|
||||||
|
.basicLightbox--visible #accept-all-gdpr,\
|
||||||
|
#gdpr-modal.in .gdpr-modal__btn--accept,\
|
||||||
|
.cookiehint .btn.cookieagree,\
|
||||||
|
#cookiealert .modal.in .btn[href*="accept"],\
|
||||||
|
#lml-data-consent-accept,\
|
||||||
|
#CBCookieMsg.in .btn[onclick*="approveCookies"],\
|
||||||
|
#cookiewall-container .button[name="submit"],\
|
||||||
|
#cookie_disclaimer.in .cookie_disclaimer_button,\
|
||||||
|
.m-cookie.iziModal[style*="block"] .m-cookie__save2.button,\
|
||||||
|
kamino-cookie-policy .mat-raised-button,\
|
||||||
|
#surbma-gpga-modal[style*="block"] button,\
|
||||||
|
#GDPR.overlayBox .menuButton,\
|
||||||
|
#cookiebar .cookie-selection-button.accept,\
|
||||||
|
.modal.in .btn.close-modal-cookie,\
|
||||||
|
#consent-module[style*="block"] #consent-module-text-button,\
|
||||||
|
.modal #consentButton,\
|
||||||
|
#consent-modal[style*="block"] .lm_modal__modal__content__body__buttons__ok,\
|
||||||
|
.cookiesOverlay3Box #cookiesConsentOK,\
|
||||||
|
.bemCookieOverlay--activePopup .bemCookieOverlay__btn--save,\
|
||||||
|
#root main ~ div [data-gi-selector="reject-all-cookies"] ~ div a,\
|
||||||
|
.cookies-management .cookies-deny,\
|
||||||
|
.offcanvas.is-open .js-offcanvas-cookie-submit,\
|
||||||
|
.force--consent.show--consent #cs_save__btn,\
|
||||||
|
.force--consent.show--consent #s-sv-bn,\
|
||||||
|
#cookieNoticeModal.vrm-reveal[style*="block"] .vrm-reveal__icon--close',
|
||||||
|
'.cookie-banner .cookie-banner__buttons .cookie-banner__button--reject,\
|
||||||
|
.cookie-consent-banner .cookie-consent-banner__reject,\
|
||||||
|
.cookie-notification .cookie-notification__buttons .cookie-notification__reject,\
|
||||||
|
.cookie-policy-banner .cookie-policy-banner__buttons .cookie-policy-banner__reject,\
|
||||||
|
.cookie-warning .cookie-warning__buttons .cookie-warning__reject,\
|
||||||
|
.cookie-notice .cookie-notice__buttons .cookie-notice__reject,\
|
||||||
|
.cookie-alert .cookie-alert__buttons .cookie-alert__reject,\
|
||||||
|
.cookie-popup .cookie-popup__buttons .cookie-popup__reject,\
|
||||||
|
.cookie-modal .cookie-modal__buttons .cookie-modal__reject,\
|
||||||
|
.cookie-dialog .cookie-dialog__buttons .cookie-dialog__reject,\
|
||||||
|
.gdpr-banner .gdpr-banner__buttons .gdpr-banner__reject,\
|
||||||
|
.gdpr-modal .gdpr-modal__buttons .gdpr-modal__reject,\
|
||||||
|
.gdpr-notice .gdpr-notice__buttons .gdpr-notice__reject,\
|
||||||
|
.gdpr-popup .gdpr-popup__buttons .gdpr-popup__reject,\
|
||||||
|
.gdpr-dialog .gdpr-dialog__buttons .gdpr-dialog__reject,\
|
||||||
|
.privacy-banner .privacy-banner__buttons .privacy-banner__reject,\
|
||||||
|
.privacy-modal .privacy-modal__buttons .privacy-modal__reject,\
|
||||||
|
.privacy-notice .privacy-notice__buttons .privacy-notice__reject,\
|
||||||
|
.privacy-popup .privacy-popup__buttons .privacy-popup__reject,\
|
||||||
|
.privacy-dialog .privacy-dialog__buttons .privacy-dialog__reject',
|
||||||
|
|
||||||
|
'.consent-banner .consent-banner__buttons .consent-banner__reject,\
|
||||||
|
.consent-modal .consent-modal__buttons .consent-modal__reject,\
|
||||||
|
.consent-notice .consent-notice__buttons .consent-notice__reject,\
|
||||||
|
.consent-popup .consent-popup__buttons .consent-popup__reject,\
|
||||||
|
.consent-dialog .consent-dialog__buttons .consent-dialog__reject,\
|
||||||
|
.tracking-banner .tracking-banner__buttons .tracking-banner__reject,\
|
||||||
|
.tracking-modal .tracking-modal__buttons .tracking-modal__reject,\
|
||||||
|
.tracking-notice .tracking-notice__buttons .tracking-notice__reject,\
|
||||||
|
.tracking-popup .tracking-popup__buttons .tracking-popup__reject,\
|
||||||
|
.tracking-dialog .tracking-dialog__buttons .tracking-dialog__reject',
|
||||||
|
|
||||||
|
'.cookie-settings-modal .cookie-settings__reject,\
|
||||||
|
.cookie-settings-modal .cookie-settings__customize,\
|
||||||
|
.cookie-preferences-modal .cookie-preferences__reject,\
|
||||||
|
.cookie-preferences-modal .cookie-preferences__customize,\
|
||||||
|
.cookie-manager-modal .cookie-manager__reject,\
|
||||||
|
.cookie-manager-modal .cookie-manager__customize,\
|
||||||
|
.cookie-control-modal .cookie-control__reject,\
|
||||||
|
.cookie-control-modal .cookie-control__customize,\
|
||||||
|
.cookie-options-modal .cookie-options__reject,\
|
||||||
|
.cookie-options-modal .cookie-options__customize',
|
||||||
|
|
||||||
|
'.data-privacy-modal .data-privacy__reject,\
|
||||||
|
.data-privacy-modal .data-privacy__customize,\
|
||||||
|
.data-protection-modal .data-protection__reject,\
|
||||||
|
.data-protection-modal .data-protection__customize,\
|
||||||
|
.data-consent-modal .data-consent__reject,\
|
||||||
|
.data-consent-modal .data-consent__customize,\
|
||||||
|
.data-settings-modal .data-settings__reject,\
|
||||||
|
.data-settings-modal .data-settings__customize,\
|
||||||
|
.data-preferences-modal .data-preferences__reject,\
|
||||||
|
.data-preferences-modal .data-preferences__customize',
|
||||||
|
|
||||||
|
'.privacy-manager-modal .privacy-manager__reject,\
|
||||||
|
.privacy-manager-modal .privacy-manager__customize,\
|
||||||
|
.privacy-settings-modal .privacy-settings__reject,\
|
||||||
|
.privacy-settings-modal .privacy-settings__customize,\
|
||||||
|
.privacy-options-modal .privacy-options__reject,\
|
||||||
|
.privacy-options-modal .privacy-options__customize,\
|
||||||
|
.privacy-control-modal .privacy-control__reject,\
|
||||||
|
.privacy-control-modal .privacy-control__customize,\
|
||||||
|
.privacy-preferences-modal .privacy-preferences__reject,\
|
||||||
|
.privacy-preferences-modal .privacy-preferences__customize',
|
||||||
|
|
||||||
|
'.consent-manager-modal .consent-manager__reject,\
|
||||||
|
.consent-manager-modal .consent-manager__customize,\
|
||||||
|
.consent-settings-modal .consent-settings__reject,\
|
||||||
|
.consent-settings-modal .consent-settings__customize,\
|
||||||
|
.consent-options-modal .consent-options__reject,\
|
||||||
|
.consent-options-modal .consent-options__customize,\
|
||||||
|
.consent-control-modal .consent-control__reject,\
|
||||||
|
.consent-control-modal .consent-control__customize,\
|
||||||
|
.consent-preferences-modal .consent-preferences__reject,\
|
||||||
|
.consent-preferences-modal .consent-preferences__customize',
|
||||||
|
|
||||||
|
'.tracking-manager-modal .tracking-manager__reject,\
|
||||||
|
.tracking-manager-modal .tracking-manager__customize,\
|
||||||
|
.tracking-settings-modal .tracking-settings__reject,\
|
||||||
|
.tracking-settings-modal .tracking-settings__customize,\
|
||||||
|
.tracking-options-modal .tracking-options__reject,\
|
||||||
|
.tracking-options-modal .tracking-options__customize,\
|
||||||
|
.tracking-control-modal .tracking-control__reject,\
|
||||||
|
.tracking-control-modal .tracking-control__customize,\
|
||||||
|
.tracking-preferences-modal .tracking-preferences__reject,\
|
||||||
|
.tracking-preferences-modal .tracking-preferences__customize',
|
||||||
|
|
||||||
|
'.gdpr-manager-modal .gdpr-manager__reject,\
|
||||||
|
.gdpr-manager-modal .gdpr-manager__customize,\
|
||||||
|
.gdpr-settings-modal .gdpr-settings__reject,\
|
||||||
|
.gdpr-settings-modal .gdpr-settings__customize,\
|
||||||
|
.gdpr-options-modal .gdpr-options__reject,\
|
||||||
|
.gdpr-options-modal .gdpr-options__customize,\
|
||||||
|
.gdpr-control-modal .gdpr-control__reject,\
|
||||||
|
.gdpr-control-modal .gdpr-control__customize,\
|
||||||
|
.gdpr-preferences-modal .gdpr-preferences__reject,\
|
||||||
|
.gdpr-preferences-modal .gdpr-preferences__customize',
|
||||||
|
|
||||||
|
'div[class*="cookie-banner"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-banner"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-banner"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-banner"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-banner"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-banner"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="cookie-consent"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-consent"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-consent"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-consent"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-consent"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-consent"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="cookie-notice"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-notice"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-notice"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-notice"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-notice"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-notice"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'[data-*="cookie"] button[data-*="reject"],\
|
||||||
|
[data-*="cookie"] button[data-*="decline"],\
|
||||||
|
[data-*="cookie"] button[data-*="settings"],\
|
||||||
|
[data-*="gdpr"] button[data-*="reject"],\
|
||||||
|
[data-*="gdpr"] button[data-*="decline"],\
|
||||||
|
[data-*="gdpr"] button[data-*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="cookie-layer"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-layer"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-layer"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-layer"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-layer"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-layer"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="cookie-section"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-section"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-section"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-section"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-section"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-section"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="cookie-container"] button[class*="reject"],\
|
||||||
|
div[class*="cookie-container"] button[class*="decline"],\
|
||||||
|
div[class*="cookie-container"] button[class*="settings"],\
|
||||||
|
div[id*="cookie-container"] button[class*="reject"],\
|
||||||
|
div[id*="cookie-container"] button[class*="decline"],\
|
||||||
|
div[id*="cookie-container"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'[data-purpose*="cookie"] button[data-purpose*="reject"],\
|
||||||
|
[data-purpose*="cookie"] button[data-purpose*="decline"],\
|
||||||
|
[data-purpose*="cookie"] button[data-purpose*="settings"],\
|
||||||
|
[data-ref*="cookie"] button[data-ref*="reject"],\
|
||||||
|
[data-ref*="cookie"] button[data-ref*="decline"],\
|
||||||
|
[data-ref*="cookie"] button[data-ref*="settings"]',
|
||||||
|
|
||||||
|
'[id*="cookie-manager"] button[id*="reject"],\
|
||||||
|
[id*="cookie-manager"] button[id*="decline"],\
|
||||||
|
[id*="cookie-manager"] button[id*="settings"],\
|
||||||
|
[id*="cookie-control"] button[id*="reject"],\
|
||||||
|
[id*="cookie-control"] button[id*="decline"],\
|
||||||
|
[id*="cookie-control"] button[id*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="vn-cookie"] button[class*="reject"],\
|
||||||
|
div[class*="vn-cookie"] button[class*="decline"],\
|
||||||
|
div[class*="vn-cookie"] button[class*="settings"],\
|
||||||
|
div[id*="vn-cookie"] button[class*="reject"],\
|
||||||
|
div[id*="vn-cookie"] button[class*="decline"],\
|
||||||
|
div[id*="vn-cookie"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="ecommerce"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="ecommerce"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="ecommerce"] button[class*="cookie-settings"],\
|
||||||
|
div[id*="ecommerce"] button[class*="cookie-reject"],\
|
||||||
|
div[id*="ecommerce"] button[class*="cookie-decline"],\
|
||||||
|
div[id*="ecommerce"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="news"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="news"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="news"] button[class*="cookie-settings"],\
|
||||||
|
div[id*="news"] button[class*="cookie-reject"],\
|
||||||
|
div[id*="news"] button[class*="cookie-decline"],\
|
||||||
|
div[id*="news"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="retail"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="retail"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="retail"] button[class*="cookie-settings"],\
|
||||||
|
div[id*="retail"] button[class*="cookie-reject"],\
|
||||||
|
div[id*="retail"] button[class*="cookie-decline"],\
|
||||||
|
div[id*="retail"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="gdpr"] button[class*="reject"],\
|
||||||
|
div[class*="gdpr"] button[class*="decline"],\
|
||||||
|
div[class*="gdpr"] button[class*="settings"],\
|
||||||
|
div[id*="gdpr"] button[class*="reject"],\
|
||||||
|
div[id*="gdpr"] button[class*="decline"],\
|
||||||
|
div[id*="gdpr"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="privacy"] button[class*="reject"],\
|
||||||
|
div[class*="privacy"] button[class*="decline"],\
|
||||||
|
div[class*="privacy"] button[class*="settings"],\
|
||||||
|
div[id*="privacy"] button[class*="reject"],\
|
||||||
|
div[id*="privacy"] button[class*="decline"],\
|
||||||
|
div[id*="privacy"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="bank"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="bank"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="bank"] button[class*="cookie-settings"],\
|
||||||
|
div[id*="bank"] button[class*="cookie-reject"],\
|
||||||
|
div[id*="bank"] button[class*="cookie-decline"],\
|
||||||
|
div[id*="bank"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="consent"] button[class*="reject"],\
|
||||||
|
div[class*="consent"] button[class*="decline"],\
|
||||||
|
div[class*="consent"] button[class*="settings"],\
|
||||||
|
div[id*="consent"] button[class*="reject"],\
|
||||||
|
div[id*="consent"] button[class*="decline"],\
|
||||||
|
div[id*="consent"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="notice"] button[class*="reject"],\
|
||||||
|
div[class*="notice"] button[class*="decline"],\
|
||||||
|
div[class*="notice"] button[class*="settings"],\
|
||||||
|
div[id*="notice"] button[class*="reject"],\
|
||||||
|
div[id*="notice"] button[class*="decline"],\
|
||||||
|
div[id*="notice"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="popup"] button[class*="reject"],\
|
||||||
|
div[class*="popup"] button[class*="decline"],\
|
||||||
|
div[class*="popup"] button[class*="settings"],\
|
||||||
|
div[id*="popup"] button[class*="reject"],\
|
||||||
|
div[id*="popup"] button[class*="decline"],\
|
||||||
|
div[id*="popup"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="eu-cookie-banner"] button[class*="reject"],\
|
||||||
|
div[class*="eu-cookie-banner"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="gdpr-notice"] button[class*="reject"],\
|
||||||
|
div[class*="gdpr-notice"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="privacy-consent"] button[class*="reject"],\
|
||||||
|
div[class*="privacy-consent"] button[class*="settings"]',
|
||||||
|
|
||||||
|
'div[class*="asia"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="asia"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="asia"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="europe"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="europe"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="europe"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="global"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="global"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="global"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="america"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="america"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="america"] button[class*="cookie-settings"]',
|
||||||
|
|
||||||
|
'div[class*="oceania"] button[class*="cookie-reject"],\
|
||||||
|
div[class*="oceania"] button[class*="cookie-decline"],\
|
||||||
|
div[class*="oceania"] button[class*="cookie-settings"]',
|
||||||
|
];
|
||||||
|
|
||||||
|
let currentChainElement = 0;
|
||||||
|
const classname = Math.random()
|
||||||
|
.toString(36)
|
||||||
|
.replace(/[^a-z]+/g, '');
|
||||||
|
|
||||||
|
// Search loop function
|
||||||
|
let searchGroupsLength = searchGroups.length,
|
||||||
|
searchPairsKeys = Object.keys(searchPairs),
|
||||||
|
searchPairsJoinedKeys = searchPairsKeys.join(','),
|
||||||
|
timeoutDuration = 300;
|
||||||
|
|
||||||
|
function querySelectorAllShadowRoot(selector, root = document) {
|
||||||
|
const elements = [];
|
||||||
|
const findElements = node => {
|
||||||
|
if (node.shadowRoot) {
|
||||||
|
node.shadowRoot.querySelectorAll(selector).forEach(el => elements.push(el));
|
||||||
|
node.shadowRoot.querySelectorAll('*').forEach(findElements);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
root.querySelectorAll('*').forEach(findElements);
|
||||||
|
return elements;
|
||||||
|
}
|
||||||
|
|
||||||
|
function searchLoop(counter) {
|
||||||
|
if (document.cookie.indexOf('cookie_consent') !== -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const dynamicTimeout = Math.min(timeoutDuration * (counter + 1), 2000);
|
||||||
|
|
||||||
|
setTimeout(function () {
|
||||||
|
document.querySelectorAll(searchPairsJoinedKeys).forEach(function (box) {
|
||||||
|
searchPairsKeys.forEach(function (selector) {
|
||||||
|
if (box.matches(selector)) {
|
||||||
|
const shadowElements = querySelectorAllShadowRoot(searchPairs[selector].join(','), box);
|
||||||
|
shadowElements.forEach(button => {
|
||||||
|
if (button.click && !button.classList.contains(classname)) {
|
||||||
|
button.classList.add(classname);
|
||||||
|
|
||||||
|
if (typeof chrome == 'object' && chrome.runtime) chrome.runtime.sendMessage({ command: 'cookie_warning_dismissed', url: document.location.href });
|
||||||
|
|
||||||
|
button.click();
|
||||||
|
|
||||||
|
if (selector != '.message-container' && button.getAttribute('href') != '#cookieman-settings')
|
||||||
|
setTimeout(function () {
|
||||||
|
if (button) button.click();
|
||||||
|
}, 500);
|
||||||
|
|
||||||
|
timeoutDuration += 500;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
document.querySelectorAll(searchGroups[counter % searchGroupsLength]).forEach(function (element) {
|
||||||
|
if (element.click && !element.classList.contains(classname)) {
|
||||||
|
element.classList.add(classname);
|
||||||
|
|
||||||
|
if (typeof chrome == 'object' && chrome.runtime) chrome.runtime.sendMessage({ command: 'cookie_warning_dismissed', url: document.location.href });
|
||||||
|
|
||||||
|
element.click();
|
||||||
|
|
||||||
|
if (element.getAttribute('aria-pressed') != 'true')
|
||||||
|
setTimeout(function () {
|
||||||
|
if (element) element.click();
|
||||||
|
}, 500);
|
||||||
|
|
||||||
|
timeoutDuration += 500;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (counter < 100 * searchGroupsLength) {
|
||||||
|
searchLoop(counter + 1);
|
||||||
|
}
|
||||||
|
}, dynamicTimeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Embeds handling
|
||||||
|
function handleEmbeds() {
|
||||||
|
var l = document.location,
|
||||||
|
is_audioboom = false,
|
||||||
|
is_dailymotion = false,
|
||||||
|
is_dailybuzz = false,
|
||||||
|
is_playerclipslaliga = false;
|
||||||
|
|
||||||
|
switch (l.hostname) {
|
||||||
|
case 'embeds.audioboom.com':
|
||||||
|
is_audioboom = true;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'dailymotion.com':
|
||||||
|
case 'www.dailymotion.com':
|
||||||
|
is_dailymotion = l.pathname.indexOf('/embed') === 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'geo.dailymotion.com':
|
||||||
|
is_dailymotion = l.pathname.indexOf('/player') === 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'dailybuzz.nl':
|
||||||
|
is_dailybuzz = l.pathname.indexOf('/buzz/embed') === 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'playerclipslaliga.tv':
|
||||||
|
is_playerclipslaliga = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
function searchEmbeds() {
|
||||||
|
setTimeout(function () {
|
||||||
|
if (is_audioboom) {
|
||||||
|
document.querySelectorAll('div[id^="cookie-modal"] .modal[style*="block"] .btn.mrs:not(.' + classname + ')').forEach(function (button) {
|
||||||
|
button.className += ' ' + classname;
|
||||||
|
button.click();
|
||||||
|
});
|
||||||
|
} else if (is_dailymotion) {
|
||||||
|
document.querySelectorAll('.np_DialogConsent-accept:not(.' + classname + '), .consent_screen-accept:not(.' + classname + ')').forEach(function (button) {
|
||||||
|
button.className += ' ' + classname;
|
||||||
|
button.click();
|
||||||
|
});
|
||||||
|
} else if (is_dailybuzz) {
|
||||||
|
document.querySelectorAll('#ask-consent #accept:not(.' + classname + ')').forEach(function (button) {
|
||||||
|
button.className += ' ' + classname;
|
||||||
|
button.click();
|
||||||
|
});
|
||||||
|
} else if (is_playerclipslaliga) {
|
||||||
|
document.querySelectorAll('#cookies button[onclick*="saveCookiesSelection"]:not(.' + classname + ')').forEach(function (button) {
|
||||||
|
button.className += ' ' + classname;
|
||||||
|
button.click();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
searchEmbeds();
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
searchEmbeds();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize
|
||||||
|
var start = setInterval(function () {
|
||||||
|
var html = document.querySelector('html');
|
||||||
|
|
||||||
|
if (!html || /idc0_350/.test(html.className)) return;
|
||||||
|
|
||||||
|
clearInterval(start);
|
||||||
|
|
||||||
|
html.className += ' idc0_350';
|
||||||
|
searchLoop(0);
|
||||||
|
handleEmbeds();
|
||||||
|
}, 500);
|
||||||
|
|
||||||
|
var hostname = document.location.hostname.replace(/^w{2,3}\d*\./i, ''),
|
||||||
|
items = getItem(hostname);
|
||||||
|
|
||||||
|
if (items) {
|
||||||
|
(items instanceof Array ? items : [items]).forEach(function (item) {
|
||||||
|
let value = localStorage.getItem(item.key);
|
||||||
|
if (value == null || (item.strict && value != item.value)) {
|
||||||
|
localStorage.setItem(item.key, item.value);
|
||||||
|
counter++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (counter > 0) document.location.reload();
|
||||||
|
}
|
||||||
|
})();
|
||||||
@@ -0,0 +1,704 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Return YouTube Dislike
|
||||||
|
// @namespace https://www.returnyoutubedislike.com/
|
||||||
|
// @homepage https://www.returnyoutubedislike.com/
|
||||||
|
// @version 3.1.5
|
||||||
|
// @encoding utf-8
|
||||||
|
// @description Return of the YouTube Dislike, Based off https://www.returnyoutubedislike.com/
|
||||||
|
// @icon https://github.com/Anarios/return-youtube-dislike/raw/main/Icons/Return%20Youtube%20Dislike%20-%20Transparent.png
|
||||||
|
// @author Anarios & JRWR
|
||||||
|
// @match *://*.youtube.com/*
|
||||||
|
// @exclude *://music.youtube.com/*
|
||||||
|
// @exclude *://*.music.youtube.com/*
|
||||||
|
// @compatible chrome
|
||||||
|
// @compatible firefox
|
||||||
|
// @compatible opera
|
||||||
|
// @compatible safari
|
||||||
|
// @compatible edge
|
||||||
|
// @grant GM.xmlHttpRequest
|
||||||
|
// @connect youtube.com
|
||||||
|
// @grant GM_addStyle
|
||||||
|
// @run-at document-end
|
||||||
|
// @downloadURL https://update.greasyfork.org/scripts/436115/Return%20YouTube%20Dislike.user.js
|
||||||
|
// @updateURL https://update.greasyfork.org/scripts/436115/Return%20YouTube%20Dislike.meta.js
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
const extConfig = {
|
||||||
|
// BEGIN USER OPTIONS
|
||||||
|
// You may change the following variables to allowed values listed in the corresponding brackets (* means default). Keep the style and keywords intact.
|
||||||
|
showUpdatePopup: false, // [true, false*] Show a popup tab after extension update (See what's new)
|
||||||
|
disableVoteSubmission: false, // [true, false*] Disable like/dislike submission (Stops counting your likes and dislikes)
|
||||||
|
disableLogging: true, // [true*, false] Disable Logging API Response in JavaScript Console.
|
||||||
|
coloredThumbs: false, // [true, false*] Colorize thumbs (Use custom colors for thumb icons)
|
||||||
|
coloredBar: false, // [true, false*] Colorize ratio bar (Use custom colors for ratio bar)
|
||||||
|
colorTheme: "classic", // [classic*, accessible, neon] Color theme (red/green, blue/yellow, pink/cyan)
|
||||||
|
numberDisplayFormat: "compactShort", // [compactShort*, compactLong, standard] Number format (For non-English locale users, you may be able to improve appearance with a different option. Please file a feature request if your locale is not covered)
|
||||||
|
numberDisplayRoundDown: true, // [true*, false] Round down numbers (Show rounded down numbers)
|
||||||
|
tooltipPercentageMode: "none", // [none*, dash_like, dash_dislike, both, only_like, only_dislike] Mode of showing percentage in like/dislike bar tooltip.
|
||||||
|
numberDisplayReformatLikes: false, // [true, false*] Re-format like numbers (Make likes and dislikes format consistent)
|
||||||
|
rateBarEnabled: false, // [true, false*] Enables ratio bar under like/dislike buttons
|
||||||
|
// END USER OPTIONS
|
||||||
|
};
|
||||||
|
|
||||||
|
const LIKED_STATE = "LIKED_STATE";
|
||||||
|
const DISLIKED_STATE = "DISLIKED_STATE";
|
||||||
|
const NEUTRAL_STATE = "NEUTRAL_STATE";
|
||||||
|
let previousState = 3; //1=LIKED, 2=DISLIKED, 3=NEUTRAL
|
||||||
|
let likesvalue = 0;
|
||||||
|
let dislikesvalue = 0;
|
||||||
|
let preNavigateLikeButton = null;
|
||||||
|
|
||||||
|
let isMobile = location.hostname == "m.youtube.com";
|
||||||
|
let isShorts = () => location.pathname.startsWith("/shorts");
|
||||||
|
let mobileDislikes = 0;
|
||||||
|
function cLog(text, subtext = "") {
|
||||||
|
if (!extConfig.disableLogging) {
|
||||||
|
subtext = subtext.trim() === "" ? "" : `(${subtext})`;
|
||||||
|
console.log(`[Return YouTube Dislikes] ${text} ${subtext}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function isInViewport(element) {
|
||||||
|
const rect = element.getBoundingClientRect();
|
||||||
|
const height = innerHeight || document.documentElement.clientHeight;
|
||||||
|
const width = innerWidth || document.documentElement.clientWidth;
|
||||||
|
return (
|
||||||
|
// When short (channel) is ignored, the element (like/dislike AND short itself) is
|
||||||
|
// hidden with a 0 DOMRect. In this case, consider it outside of Viewport
|
||||||
|
!(rect.top == 0 && rect.left == 0 && rect.bottom == 0 && rect.right == 0) &&
|
||||||
|
rect.top >= 0 &&
|
||||||
|
rect.left >= 0 &&
|
||||||
|
rect.bottom <= height &&
|
||||||
|
rect.right <= width
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getButtons() {
|
||||||
|
if (isShorts()) {
|
||||||
|
let elements = document.querySelectorAll(
|
||||||
|
isMobile ? "ytm-like-button-renderer" : "#like-button > ytd-like-button-renderer",
|
||||||
|
);
|
||||||
|
for (let element of elements) {
|
||||||
|
if (isInViewport(element)) {
|
||||||
|
return element;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (isMobile) {
|
||||||
|
return (
|
||||||
|
document.querySelector(".slim-video-action-bar-actions .segmented-buttons") ??
|
||||||
|
document.querySelector(".slim-video-action-bar-actions")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (document.getElementById("menu-container")?.offsetParent === null) {
|
||||||
|
return (
|
||||||
|
document.querySelector("ytd-menu-renderer.ytd-watch-metadata > div") ??
|
||||||
|
document.querySelector("ytd-menu-renderer.ytd-video-primary-info-renderer > div")
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return document.getElementById("menu-container")?.querySelector("#top-level-buttons-computed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDislikeButton() {
|
||||||
|
if (getButtons().children[0].tagName === "YTD-SEGMENTED-LIKE-DISLIKE-BUTTON-RENDERER") {
|
||||||
|
if (getButtons().children[0].children[1] === undefined) {
|
||||||
|
return document.querySelector("#segmented-dislike-button");
|
||||||
|
} else {
|
||||||
|
return getButtons().children[0].children[1];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (getButtons().querySelector("segmented-like-dislike-button-view-model")) {
|
||||||
|
const dislikeViewModel = getButtons().querySelector("dislike-button-view-model");
|
||||||
|
if (!dislikeViewModel) cLog("Dislike button wasn't added to DOM yet...");
|
||||||
|
return dislikeViewModel;
|
||||||
|
} else {
|
||||||
|
return getButtons().children[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLikeButton() {
|
||||||
|
return getButtons().children[0].tagName === "YTD-SEGMENTED-LIKE-DISLIKE-BUTTON-RENDERER"
|
||||||
|
? document.querySelector("#segmented-like-button") !== null
|
||||||
|
? document.querySelector("#segmented-like-button")
|
||||||
|
: getButtons().children[0].children[0]
|
||||||
|
: getButtons().querySelector("like-button-view-model") ?? getButtons().children[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLikeTextContainer() {
|
||||||
|
return (
|
||||||
|
getLikeButton().querySelector("#text") ??
|
||||||
|
getLikeButton().getElementsByTagName("yt-formatted-string")[0] ??
|
||||||
|
getLikeButton().querySelector("span[role='text']")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDislikeTextContainer() {
|
||||||
|
const dislikeButton = getDislikeButton();
|
||||||
|
let result =
|
||||||
|
dislikeButton?.querySelector("#text") ??
|
||||||
|
dislikeButton?.getElementsByTagName("yt-formatted-string")[0] ??
|
||||||
|
dislikeButton?.querySelector("span[role='text']");
|
||||||
|
if (result === null) {
|
||||||
|
let textSpan = document.createElement("span");
|
||||||
|
textSpan.id = "text";
|
||||||
|
textSpan.style.marginLeft = "6px";
|
||||||
|
dislikeButton?.querySelector("button").appendChild(textSpan);
|
||||||
|
if (dislikeButton) dislikeButton.querySelector("button").style.width = "auto";
|
||||||
|
result = textSpan;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
function createObserver(options, callback) {
|
||||||
|
const observerWrapper = new Object();
|
||||||
|
observerWrapper.options = options;
|
||||||
|
observerWrapper.observer = new MutationObserver(callback);
|
||||||
|
observerWrapper.observe = function (element) {
|
||||||
|
this.observer.observe(element, this.options);
|
||||||
|
};
|
||||||
|
observerWrapper.disconnect = function () {
|
||||||
|
this.observer.disconnect();
|
||||||
|
};
|
||||||
|
return observerWrapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
let shortsObserver = null;
|
||||||
|
|
||||||
|
if (isShorts() && !shortsObserver) {
|
||||||
|
cLog("Initializing shorts mutation observer");
|
||||||
|
shortsObserver = createObserver(
|
||||||
|
{
|
||||||
|
attributes: true,
|
||||||
|
},
|
||||||
|
(mutationList) => {
|
||||||
|
mutationList.forEach((mutation) => {
|
||||||
|
if (
|
||||||
|
mutation.type === "attributes" &&
|
||||||
|
mutation.target.nodeName === "TP-YT-PAPER-BUTTON" &&
|
||||||
|
mutation.target.id === "button"
|
||||||
|
) {
|
||||||
|
cLog("Short thumb button status changed");
|
||||||
|
if (mutation.target.getAttribute("aria-pressed") === "true") {
|
||||||
|
mutation.target.style.color =
|
||||||
|
mutation.target.parentElement.parentElement.id === "like-button"
|
||||||
|
? getColorFromTheme(true)
|
||||||
|
: getColorFromTheme(false);
|
||||||
|
} else {
|
||||||
|
mutation.target.style.color = "unset";
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cLog("Unexpected mutation observer event: " + mutation.target + mutation.type);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isVideoLiked() {
|
||||||
|
if (isMobile) {
|
||||||
|
return getLikeButton().querySelector("button").getAttribute("aria-label") == "true";
|
||||||
|
}
|
||||||
|
return getLikeButton().classList.contains("style-default-active");
|
||||||
|
}
|
||||||
|
|
||||||
|
function isVideoDisliked() {
|
||||||
|
if (isMobile) {
|
||||||
|
return getDislikeButton()?.querySelector("button").getAttribute("aria-label") == "true";
|
||||||
|
}
|
||||||
|
return getDislikeButton()?.classList.contains("style-default-active");
|
||||||
|
}
|
||||||
|
|
||||||
|
function isVideoNotLiked() {
|
||||||
|
if (isMobile) {
|
||||||
|
return !isVideoLiked();
|
||||||
|
}
|
||||||
|
return getLikeButton().classList.contains("style-text");
|
||||||
|
}
|
||||||
|
|
||||||
|
function isVideoNotDisliked() {
|
||||||
|
if (isMobile) {
|
||||||
|
return !isVideoDisliked();
|
||||||
|
}
|
||||||
|
return getDislikeButton()?.classList.contains("style-text");
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkForUserAvatarButton() {
|
||||||
|
if (isMobile) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (document.querySelector("#avatar-btn")) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getState() {
|
||||||
|
if (isVideoLiked()) {
|
||||||
|
return LIKED_STATE;
|
||||||
|
}
|
||||||
|
if (isVideoDisliked()) {
|
||||||
|
return DISLIKED_STATE;
|
||||||
|
}
|
||||||
|
return NEUTRAL_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setLikes(likesCount) {
|
||||||
|
if (isMobile) {
|
||||||
|
getButtons().children[0].querySelector(".button-renderer-text").innerText = likesCount;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
getLikeTextContainer().innerText = likesCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setDislikes(dislikesCount) {
|
||||||
|
if (isMobile) {
|
||||||
|
mobileDislikes = dislikesCount;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const _container = getDislikeTextContainer();
|
||||||
|
_container?.removeAttribute("is-empty");
|
||||||
|
if (_container?.innerText !== dislikesCount) {
|
||||||
|
_container.innerText = dislikesCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLikeCountFromButton() {
|
||||||
|
try {
|
||||||
|
if (isShorts()) {
|
||||||
|
//Youtube Shorts don't work with this query. It's not necessary; we can skip it and still see the results.
|
||||||
|
//It should be possible to fix this function, but it's not critical to showing the dislike count.
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
let likeButton =
|
||||||
|
getLikeButton().querySelector("yt-formatted-string#text") ?? getLikeButton().querySelector("button");
|
||||||
|
|
||||||
|
let likesStr = likeButton.getAttribute("aria-label").replace(/\D/g, "");
|
||||||
|
return likesStr.length > 0 ? parseInt(likesStr) : false;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
(typeof GM_addStyle != "undefined"
|
||||||
|
? GM_addStyle
|
||||||
|
: (styles) => {
|
||||||
|
let styleNode = document.createElement("style");
|
||||||
|
styleNode.type = "text/css";
|
||||||
|
styleNode.innerText = styles;
|
||||||
|
document.head.appendChild(styleNode);
|
||||||
|
})(`
|
||||||
|
#return-youtube-dislike-bar-container {
|
||||||
|
background: var(--yt-spec-icon-disabled);
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#return-youtube-dislike-bar {
|
||||||
|
background: var(--yt-spec-text-primary);
|
||||||
|
border-radius: 2px;
|
||||||
|
transition: all 0.15s ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ryd-tooltip {
|
||||||
|
position: absolute;
|
||||||
|
display: block;
|
||||||
|
height: 2px;
|
||||||
|
bottom: -10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ryd-tooltip-bar-container {
|
||||||
|
width: 100%;
|
||||||
|
height: 2px;
|
||||||
|
position: absolute;
|
||||||
|
padding-top: 6px;
|
||||||
|
padding-bottom: 12px;
|
||||||
|
top: -6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
ytd-menu-renderer.ytd-watch-metadata {
|
||||||
|
overflow-y: visible !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#top-level-buttons-computed {
|
||||||
|
position: relative !important;
|
||||||
|
}
|
||||||
|
`);
|
||||||
|
|
||||||
|
function createRateBar(likes, dislikes) {
|
||||||
|
if (isMobile || !extConfig.rateBarEnabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let rateBar = document.getElementById("return-youtube-dislike-bar-container");
|
||||||
|
|
||||||
|
const widthPx = getLikeButton().clientWidth + (getDislikeButton()?.clientWidth ?? 52);
|
||||||
|
|
||||||
|
const widthPercent = likes + dislikes > 0 ? (likes / (likes + dislikes)) * 100 : 50;
|
||||||
|
|
||||||
|
var likePercentage = parseFloat(widthPercent.toFixed(1));
|
||||||
|
const dislikePercentage = (100 - likePercentage).toLocaleString();
|
||||||
|
likePercentage = likePercentage.toLocaleString();
|
||||||
|
|
||||||
|
var tooltipInnerHTML;
|
||||||
|
switch (extConfig.tooltipPercentageMode) {
|
||||||
|
case "dash_like":
|
||||||
|
tooltipInnerHTML = `${likes.toLocaleString()} / ${dislikes.toLocaleString()} - ${likePercentage}%`;
|
||||||
|
break;
|
||||||
|
case "dash_dislike":
|
||||||
|
tooltipInnerHTML = `${likes.toLocaleString()} / ${dislikes.toLocaleString()} - ${dislikePercentage}%`;
|
||||||
|
break;
|
||||||
|
case "both":
|
||||||
|
tooltipInnerHTML = `${likePercentage}% / ${dislikePercentage}%`;
|
||||||
|
break;
|
||||||
|
case "only_like":
|
||||||
|
tooltipInnerHTML = `${likePercentage}%`;
|
||||||
|
break;
|
||||||
|
case "only_dislike":
|
||||||
|
tooltipInnerHTML = `${dislikePercentage}%`;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
tooltipInnerHTML = `${likes.toLocaleString()} / ${dislikes.toLocaleString()}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!rateBar && !isMobile) {
|
||||||
|
let colorLikeStyle = "";
|
||||||
|
let colorDislikeStyle = "";
|
||||||
|
if (extConfig.coloredBar) {
|
||||||
|
colorLikeStyle = "; background-color: " + getColorFromTheme(true);
|
||||||
|
colorDislikeStyle = "; background-color: " + getColorFromTheme(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
getButtons().insertAdjacentHTML(
|
||||||
|
"beforeend",
|
||||||
|
`
|
||||||
|
<div class="ryd-tooltip" style="width: ${widthPx}px">
|
||||||
|
<div class="ryd-tooltip-bar-container">
|
||||||
|
<div
|
||||||
|
id="return-youtube-dislike-bar-container"
|
||||||
|
style="width: 100%; height: 2px;${colorDislikeStyle}"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
id="return-youtube-dislike-bar"
|
||||||
|
style="width: ${widthPercent}%; height: 100%${colorDislikeStyle}"
|
||||||
|
></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<tp-yt-paper-tooltip position="top" id="ryd-dislike-tooltip" class="style-scope ytd-sentiment-bar-renderer" role="tooltip" tabindex="-1">
|
||||||
|
<!--css-build:shady-->${tooltipInnerHTML}
|
||||||
|
</tp-yt-paper-tooltip>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
|
);
|
||||||
|
let descriptionAndActionsElement = document.getElementById("top-row");
|
||||||
|
descriptionAndActionsElement.style.borderBottom = "1px solid var(--yt-spec-10-percent-layer)";
|
||||||
|
descriptionAndActionsElement.style.paddingBottom = "10px";
|
||||||
|
} else {
|
||||||
|
document.querySelector(".ryd-tooltip").style.width = widthPx + "px";
|
||||||
|
document.getElementById("return-youtube-dislike-bar").style.width = widthPercent + "%";
|
||||||
|
|
||||||
|
if (extConfig.coloredBar) {
|
||||||
|
document.getElementById("return-youtube-dislike-bar-container").style.backgroundColor = getColorFromTheme(false);
|
||||||
|
document.getElementById("return-youtube-dislike-bar").style.backgroundColor = getColorFromTheme(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setState() {
|
||||||
|
cLog("Fetching votes...");
|
||||||
|
let statsSet = false;
|
||||||
|
|
||||||
|
fetch(`https://returnyoutubedislikeapi.com/votes?videoId=${getVideoId()}`).then((response) => {
|
||||||
|
response.json().then((json) => {
|
||||||
|
if (json && !("traceId" in response) && !statsSet) {
|
||||||
|
const { dislikes, likes } = json;
|
||||||
|
cLog(`Received count: ${dislikes}`);
|
||||||
|
likesvalue = likes;
|
||||||
|
dislikesvalue = dislikes;
|
||||||
|
setDislikes(numberFormat(dislikes));
|
||||||
|
if (extConfig.numberDisplayReformatLikes === true) {
|
||||||
|
const nativeLikes = getLikeCountFromButton();
|
||||||
|
if (nativeLikes !== false) {
|
||||||
|
setLikes(numberFormat(nativeLikes));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
createRateBar(likes, dislikes);
|
||||||
|
if (extConfig.coloredThumbs === true) {
|
||||||
|
const dislikeButton = getDislikeButton();
|
||||||
|
if (isShorts()) {
|
||||||
|
// for shorts, leave deactived buttons in default color
|
||||||
|
const shortLikeButton = getLikeButton().querySelector("tp-yt-paper-button#button");
|
||||||
|
const shortDislikeButton = dislikeButton?.querySelector("tp-yt-paper-button#button");
|
||||||
|
if (shortLikeButton.getAttribute("aria-pressed") === "true") {
|
||||||
|
shortLikeButton.style.color = getColorFromTheme(true);
|
||||||
|
}
|
||||||
|
if (shortDislikeButton && shortDislikeButton.getAttribute("aria-pressed") === "true") {
|
||||||
|
shortDislikeButton.style.color = getColorFromTheme(false);
|
||||||
|
}
|
||||||
|
shortsObserver.observe(shortLikeButton);
|
||||||
|
shortsObserver.observe(shortDislikeButton);
|
||||||
|
} else {
|
||||||
|
getLikeButton().style.color = getColorFromTheme(true);
|
||||||
|
if (dislikeButton) dislikeButton.style.color = getColorFromTheme(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateDOMDislikes() {
|
||||||
|
setDislikes(numberFormat(dislikesvalue));
|
||||||
|
createRateBar(likesvalue, dislikesvalue);
|
||||||
|
}
|
||||||
|
|
||||||
|
function likeClicked() {
|
||||||
|
if (checkForUserAvatarButton() == true) {
|
||||||
|
if (previousState == 1) {
|
||||||
|
likesvalue--;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 3;
|
||||||
|
} else if (previousState == 2) {
|
||||||
|
likesvalue++;
|
||||||
|
dislikesvalue--;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 1;
|
||||||
|
} else if (previousState == 3) {
|
||||||
|
likesvalue++;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 1;
|
||||||
|
}
|
||||||
|
if (extConfig.numberDisplayReformatLikes === true) {
|
||||||
|
const nativeLikes = getLikeCountFromButton();
|
||||||
|
if (nativeLikes !== false) {
|
||||||
|
setLikes(numberFormat(nativeLikes));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function dislikeClicked() {
|
||||||
|
if (checkForUserAvatarButton() == true) {
|
||||||
|
if (previousState == 3) {
|
||||||
|
dislikesvalue++;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 2;
|
||||||
|
} else if (previousState == 2) {
|
||||||
|
dislikesvalue--;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 3;
|
||||||
|
} else if (previousState == 1) {
|
||||||
|
likesvalue--;
|
||||||
|
dislikesvalue++;
|
||||||
|
updateDOMDislikes();
|
||||||
|
previousState = 2;
|
||||||
|
if (extConfig.numberDisplayReformatLikes === true) {
|
||||||
|
const nativeLikes = getLikeCountFromButton();
|
||||||
|
if (nativeLikes !== false) {
|
||||||
|
setLikes(numberFormat(nativeLikes));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setInitialState() {
|
||||||
|
setState();
|
||||||
|
}
|
||||||
|
|
||||||
|
function getVideoId() {
|
||||||
|
const urlObject = new URL(window.location.href);
|
||||||
|
const pathname = urlObject.pathname;
|
||||||
|
if (pathname.startsWith("/clip")) {
|
||||||
|
return (document.querySelector("meta[itemprop='videoId']") || document.querySelector("meta[itemprop='identifier']")).content;
|
||||||
|
} else {
|
||||||
|
if (pathname.startsWith("/shorts")) {
|
||||||
|
return pathname.slice(8);
|
||||||
|
}
|
||||||
|
return urlObject.searchParams.get("v");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function isVideoLoaded() {
|
||||||
|
if (isMobile) {
|
||||||
|
return document.getElementById("player").getAttribute("loading") == "false";
|
||||||
|
}
|
||||||
|
const videoId = getVideoId();
|
||||||
|
|
||||||
|
return (
|
||||||
|
// desktop: spring 2024 UI
|
||||||
|
document.querySelector(`ytd-watch-grid[video-id='${videoId}']`) !== null ||
|
||||||
|
// desktop: older UI
|
||||||
|
document.querySelector(`ytd-watch-flexy[video-id='${videoId}']`) !== null ||
|
||||||
|
// mobile: no video-id attribute
|
||||||
|
document.querySelector('#player[loading="false"]:not([hidden])') !== null
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function roundDown(num) {
|
||||||
|
if (num < 1000) return num;
|
||||||
|
const int = Math.floor(Math.log10(num) - 2);
|
||||||
|
const decimal = int + (int % 3 ? 1 : 0);
|
||||||
|
const value = Math.floor(num / 10 ** decimal);
|
||||||
|
return value * 10 ** decimal;
|
||||||
|
}
|
||||||
|
|
||||||
|
function numberFormat(numberState) {
|
||||||
|
let numberDisplay;
|
||||||
|
if (extConfig.numberDisplayRoundDown === false) {
|
||||||
|
numberDisplay = numberState;
|
||||||
|
} else {
|
||||||
|
numberDisplay = roundDown(numberState);
|
||||||
|
}
|
||||||
|
return getNumberFormatter(extConfig.numberDisplayFormat).format(numberDisplay);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNumberFormatter(optionSelect) {
|
||||||
|
let userLocales;
|
||||||
|
if (document.documentElement.lang) {
|
||||||
|
userLocales = document.documentElement.lang;
|
||||||
|
} else if (navigator.language) {
|
||||||
|
userLocales = navigator.language;
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
userLocales = new URL(
|
||||||
|
Array.from(document.querySelectorAll("head > link[rel='search']"))
|
||||||
|
?.find((n) => n?.getAttribute("href")?.includes("?locale="))
|
||||||
|
?.getAttribute("href"),
|
||||||
|
)?.searchParams?.get("locale");
|
||||||
|
} catch {
|
||||||
|
cLog("Cannot find browser locale. Use en as default for number formatting.");
|
||||||
|
userLocales = "en";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let formatterNotation;
|
||||||
|
let formatterCompactDisplay;
|
||||||
|
switch (optionSelect) {
|
||||||
|
case "compactLong":
|
||||||
|
formatterNotation = "compact";
|
||||||
|
formatterCompactDisplay = "long";
|
||||||
|
break;
|
||||||
|
case "standard":
|
||||||
|
formatterNotation = "standard";
|
||||||
|
formatterCompactDisplay = "short";
|
||||||
|
break;
|
||||||
|
case "compactShort":
|
||||||
|
default:
|
||||||
|
formatterNotation = "compact";
|
||||||
|
formatterCompactDisplay = "short";
|
||||||
|
}
|
||||||
|
|
||||||
|
const formatter = Intl.NumberFormat(userLocales, {
|
||||||
|
notation: formatterNotation,
|
||||||
|
compactDisplay: formatterCompactDisplay,
|
||||||
|
});
|
||||||
|
return formatter;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getColorFromTheme(voteIsLike) {
|
||||||
|
let colorString;
|
||||||
|
switch (extConfig.colorTheme) {
|
||||||
|
case "accessible":
|
||||||
|
if (voteIsLike === true) {
|
||||||
|
colorString = "dodgerblue";
|
||||||
|
} else {
|
||||||
|
colorString = "gold";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "neon":
|
||||||
|
if (voteIsLike === true) {
|
||||||
|
colorString = "aqua";
|
||||||
|
} else {
|
||||||
|
colorString = "magenta";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "classic":
|
||||||
|
default:
|
||||||
|
if (voteIsLike === true) {
|
||||||
|
colorString = "lime";
|
||||||
|
} else {
|
||||||
|
colorString = "red";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return colorString;
|
||||||
|
}
|
||||||
|
|
||||||
|
let smartimationObserver = null;
|
||||||
|
|
||||||
|
function setEventListeners(evt) {
|
||||||
|
let jsInitChecktimer;
|
||||||
|
|
||||||
|
function checkForJS_Finish() {
|
||||||
|
//console.log();
|
||||||
|
if (isShorts() || (getButtons()?.offsetParent && isVideoLoaded())) {
|
||||||
|
const buttons = getButtons();
|
||||||
|
const dislikeButton = getDislikeButton();
|
||||||
|
|
||||||
|
if (preNavigateLikeButton !== getLikeButton() && dislikeButton) {
|
||||||
|
cLog("Registering button listeners...");
|
||||||
|
try {
|
||||||
|
getLikeButton().addEventListener("click", likeClicked);
|
||||||
|
dislikeButton?.addEventListener("click", dislikeClicked);
|
||||||
|
getLikeButton().addEventListener("touchstart", likeClicked);
|
||||||
|
dislikeButton?.addEventListener("touchstart", dislikeClicked);
|
||||||
|
dislikeButton?.addEventListener("focusin", updateDOMDislikes);
|
||||||
|
dislikeButton?.addEventListener("focusout", updateDOMDislikes);
|
||||||
|
preNavigateLikeButton = getLikeButton();
|
||||||
|
|
||||||
|
if (!smartimationObserver) {
|
||||||
|
smartimationObserver = createObserver(
|
||||||
|
{
|
||||||
|
attributes: true,
|
||||||
|
subtree: true,
|
||||||
|
childList: true,
|
||||||
|
},
|
||||||
|
updateDOMDislikes,
|
||||||
|
);
|
||||||
|
smartimationObserver.container = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const smartimationContainer = buttons.querySelector("yt-smartimation");
|
||||||
|
if (smartimationContainer && smartimationObserver.container != smartimationContainer) {
|
||||||
|
cLog("Initializing smartimation mutation observer");
|
||||||
|
smartimationObserver.disconnect();
|
||||||
|
smartimationObserver.observe(smartimationContainer);
|
||||||
|
smartimationObserver.container = smartimationContainer;
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
return;
|
||||||
|
} //Don't spam errors into the console
|
||||||
|
}
|
||||||
|
if (dislikeButton) {
|
||||||
|
setInitialState();
|
||||||
|
clearInterval(jsInitChecktimer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cLog("Setting up...");
|
||||||
|
jsInitChecktimer = setInterval(checkForJS_Finish, 111);
|
||||||
|
}
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
"use strict";
|
||||||
|
window.addEventListener("yt-navigate-finish", setEventListeners, true);
|
||||||
|
setEventListeners();
|
||||||
|
})();
|
||||||
|
if (isMobile) {
|
||||||
|
let originalPush = history.pushState;
|
||||||
|
history.pushState = function (...args) {
|
||||||
|
window.returnDislikeButtonlistenersSet = false;
|
||||||
|
setEventListeners(args[2]);
|
||||||
|
return originalPush.apply(history, args);
|
||||||
|
};
|
||||||
|
setInterval(() => {
|
||||||
|
const dislikeButton = getDislikeButton();
|
||||||
|
if (dislikeButton?.querySelector(".button-renderer-text") === null) {
|
||||||
|
getDislikeTextContainer().innerText = mobileDislikes;
|
||||||
|
} else {
|
||||||
|
if (dislikeButton) dislikeButton.querySelector(".button-renderer-text").innerText = mobileDislikes;
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,18 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Startpage - Hide Ads
|
||||||
|
// @match https://www.startpage.com/*
|
||||||
|
// @run-at document-start
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
new MutationObserver(function(mutations) {
|
||||||
|
mutations.forEach((mutation) => {
|
||||||
|
if (mutation.type === 'childList') {
|
||||||
|
mutation.addedNodes.forEach((node) => {
|
||||||
|
if (node.nodeType === 1 && node.nodeName === 'DIV' && node.id === 'gcsa-top') {
|
||||||
|
node.remove();
|
||||||
|
this.disconnect();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}).observe(document, { childList: true, subtree: true });
|
||||||
@@ -0,0 +1,191 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name TrackingTokenStripper
|
||||||
|
// @version 1.4
|
||||||
|
// @description Remove most of the annoying tracking token from URL parameters
|
||||||
|
// @license MIT
|
||||||
|
// @homepage https://blog.miniasp.com/
|
||||||
|
// @homepageURL https://blog.miniasp.com/
|
||||||
|
// @website https://www.facebook.com/will.fans
|
||||||
|
// @source https://github.com/doggy8088/TrackingTokenStripper/raw/refs/heads/master/TrackingTokenStripper.user.js
|
||||||
|
// @namespace https://github.com/doggy8088/TrackingTokenStripper
|
||||||
|
// @author Will Huang
|
||||||
|
// @match *://*/*
|
||||||
|
// @run-at document-start
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const oldReplaceState = history.replaceState;
|
||||||
|
history.replaceState = function replaceState() {
|
||||||
|
let ret = oldReplaceState.apply(this, arguments);
|
||||||
|
window.dispatchEvent(new Event('replacestate'));
|
||||||
|
window.dispatchEvent(new Event('locationchange'));
|
||||||
|
return ret;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('popstate', () => {
|
||||||
|
window.dispatchEvent(new Event('locationchange'));
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener('locationchange', function () {
|
||||||
|
executeActions();
|
||||||
|
});
|
||||||
|
|
||||||
|
executeActions();
|
||||||
|
|
||||||
|
let id = setInterval(executeActions, 500);
|
||||||
|
|
||||||
|
setTimeout(() => { clearInterval(id); }, 2000);
|
||||||
|
|
||||||
|
function executeActions() {
|
||||||
|
|
||||||
|
var s = TrackingTokenStripper(location.href)
|
||||||
|
// facebook
|
||||||
|
.remove('fbclid')
|
||||||
|
.removeByDomain('www.facebook.com', 'privacy_mutation_token')
|
||||||
|
.removeByDomain('www.facebook.com', 'acontext')
|
||||||
|
.removeByDomain('www.facebook.com', '__xts__[0]')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_t')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_id')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_ids[0]')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_ids[1]')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_ids[2]')
|
||||||
|
.removeByDomain('www.facebook.com', 'notif_ids[3]')
|
||||||
|
.removeByDomain('www.facebook.com', 'ref', 'notif')
|
||||||
|
.removeByDomain('www.facebook.com', 'ref=watch_permalink')
|
||||||
|
|
||||||
|
// Dropbox
|
||||||
|
.removeByDomain('www.dropbox.com', '_ad')
|
||||||
|
.removeByDomain('www.dropbox.com', '_camp')
|
||||||
|
.removeByDomain('www.dropbox.com', '_tk')
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
// https://youtu.be/4f-Y9G5ENPc?si=SHSu2hEdSbXGy4_Q
|
||||||
|
// https://www.youtube.com/embed/4f-Y9G5ENPc?si=GQFJV_nKMXxpiQb6
|
||||||
|
.removeByDomain('youtu.be', 'si')
|
||||||
|
.removeByDomain('www.youtube.com', 'si')
|
||||||
|
|
||||||
|
// Google Analytics
|
||||||
|
// https://support.google.com/analytics/answer/1033863?hl=en
|
||||||
|
.remove('utm_id')
|
||||||
|
.remove('utm_source')
|
||||||
|
.remove('utm_medium')
|
||||||
|
.remove('utm_campaign')
|
||||||
|
.remove('utm_term')
|
||||||
|
.remove('utm_content')
|
||||||
|
.remove('_ga')
|
||||||
|
|
||||||
|
// GA - others
|
||||||
|
.remove('utm_campaignid')
|
||||||
|
.remove('utm_cid')
|
||||||
|
.remove('utm_reader')
|
||||||
|
.remove('utm_referrer')
|
||||||
|
.remove('utm_name')
|
||||||
|
.remove('utm_social')
|
||||||
|
.remove('utm_social-type')
|
||||||
|
.remove('gclid')
|
||||||
|
.remove('igshid')
|
||||||
|
.remove('_hsenc')
|
||||||
|
.remove('_hsmi')
|
||||||
|
.remove('mc_cid')
|
||||||
|
.remove('mc_eid')
|
||||||
|
.remove('mkt_tok')
|
||||||
|
.remove('yclid')
|
||||||
|
.remove('_openstat')
|
||||||
|
|
||||||
|
// devblogs.microsoft.com
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_issue')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_position')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_topic')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_section')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_cta')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'utm_description')
|
||||||
|
.removeByDomain('devblogs.microsoft.com', 'ocid')
|
||||||
|
|
||||||
|
// Microsoft
|
||||||
|
.remove('wt.mc_id')
|
||||||
|
.removeByDomain('learn.microsoft.com', 'ocid')
|
||||||
|
.removeByDomain('learn.microsoft.com', 'redirectedfrom')
|
||||||
|
|
||||||
|
.removeByDomain('azure.microsoft.com', 'OCID')
|
||||||
|
.removeByDomain('azure.microsoft.com', 'ef_id')
|
||||||
|
|
||||||
|
.removeByDomain('www.msn.com', 'ocid')
|
||||||
|
.removeByDomain('www.msn.com', 'cvid')
|
||||||
|
|
||||||
|
// bilibili
|
||||||
|
.removeByDomain('www.bilibili.com', 'share_source')
|
||||||
|
.removeByDomain('www.bilibili.com', 'share_medium')
|
||||||
|
|
||||||
|
// Others
|
||||||
|
.remove('__tn__')
|
||||||
|
.remove('gclsrc')
|
||||||
|
.remove('itm_source')
|
||||||
|
.remove('itm_medium')
|
||||||
|
.remove('itm_campaign')
|
||||||
|
.remove('mc') // sendgrid.com
|
||||||
|
.remove('mcd') // sendgrid.com
|
||||||
|
.remove('cvosrc') // sendgrid.com
|
||||||
|
.remove('cr_cc') // https://blogs.microsoft.com/
|
||||||
|
|
||||||
|
.remove('sc_channel')
|
||||||
|
.remove('sc_campaign')
|
||||||
|
.remove('sc_geo')
|
||||||
|
.remove('trk')
|
||||||
|
.remove('sc_publisher')
|
||||||
|
.remove('trkCampaign')
|
||||||
|
.remove('sc_outcome')
|
||||||
|
.remove('sc_country')
|
||||||
|
|
||||||
|
.remove('__hstc')
|
||||||
|
.remove('__hssc')
|
||||||
|
.remove('__hsfp')
|
||||||
|
.remove('_gl')
|
||||||
|
|
||||||
|
// Yahoo News
|
||||||
|
.remove('guce_referrer')
|
||||||
|
.remove('guce_referrer_sig')
|
||||||
|
|
||||||
|
.toString();
|
||||||
|
|
||||||
|
if (s && location.href !== s) {
|
||||||
|
// console.log('Changing URL', s);
|
||||||
|
// location.href = s;
|
||||||
|
oldReplaceState.apply(history, [{}, '', s]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function TrackingTokenStripper(url) {
|
||||||
|
const parsedUrl = new URL(url);
|
||||||
|
return {
|
||||||
|
remove(name, value) {
|
||||||
|
if (parsedUrl.searchParams.has(name)) {
|
||||||
|
if (value && value === parsedUrl.searchParams.get(name)) {
|
||||||
|
parsedUrl.searchParams.delete(name);
|
||||||
|
}
|
||||||
|
if (!value) {
|
||||||
|
parsedUrl.searchParams.delete(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TrackingTokenStripper(parsedUrl.toString());
|
||||||
|
},
|
||||||
|
removeByDomain(domain, name) {
|
||||||
|
if (parsedUrl.hostname.toLocaleLowerCase() === domain.toLocaleLowerCase()) {
|
||||||
|
if (name.indexOf('=') >= 0) {
|
||||||
|
var [key, value] = name.split("=");
|
||||||
|
return this.remove(key, value);
|
||||||
|
} else {
|
||||||
|
return this.remove(name);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
toString() {
|
||||||
|
return parsedUrl.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
})();
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
{config, ...}: {
|
||||||
|
programs.qutebrowser.keyBindings = {
|
||||||
|
normal."<Ctrl-w>" = "tab-close";
|
||||||
|
normal."<Ctrl-Tab>" = "tab-next";
|
||||||
|
normal."<Ctrl-Shift-Tab>" = "tab-prev";
|
||||||
|
normal."<Ctrl-b>" = "open file://${config.xdg.dataHome}/qutebrowser/bookmarks.html";
|
||||||
|
|
||||||
|
# Ctrl+c is used to leave the current mode and return to normal mode.
|
||||||
|
insert."<Ctrl-c>" = "mode-leave";
|
||||||
|
hint."<Ctrl-c>" = "mode-leave";
|
||||||
|
caret."<Ctrl-c>" = "mode-leave";
|
||||||
|
command."<Ctrl-c>" = "mode-leave";
|
||||||
|
prompt."<Ctrl-c>" = "mode-leave";
|
||||||
|
yesno."<Ctrl-c>" = "mode-leave";
|
||||||
|
register."<Ctrl-c>" = "mode-leave";
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
programs.qutebrowser.searchEngines = rec {
|
||||||
|
startpage = "https://www.startpage.com/sp/search?q={}";
|
||||||
|
|
||||||
|
mynixos = "https://mynixos.com/search?q={}";
|
||||||
|
duckduckgo = "https://duckduckgo.com/?q={}";
|
||||||
|
google = "https://google.com/search?hl=en&q={}";
|
||||||
|
yandex = "https://yandex.com/search/?text={}";
|
||||||
|
bing = "https://bing.com/search?q={}";
|
||||||
|
youtube = "https://youtube.com/results?search_query={}";
|
||||||
|
google-images = "https://google.com/search?hl=en&tbm=isch&q={}";
|
||||||
|
|
||||||
|
# AI
|
||||||
|
chatgpt = "https://chatgpt.com/?q={}";
|
||||||
|
claude = "https://claude.ai/new?q={}";
|
||||||
|
gemini = "https://gemini.google.com/app?q={}";
|
||||||
|
|
||||||
|
# Maps
|
||||||
|
amap = "https://maps.apple.com/?q={}";
|
||||||
|
gmap = "https://www.google.com/maps/search/{}";
|
||||||
|
|
||||||
|
# shortcuts
|
||||||
|
g = google;
|
||||||
|
n = mynixos;
|
||||||
|
gpt = chatgpt;
|
||||||
|
gem = gemini;
|
||||||
|
cla = claude;
|
||||||
|
DEFAULT = startpage;
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
{config, ...}: {
|
||||||
|
programs.qutebrowser = {
|
||||||
|
settings = {
|
||||||
|
url = rec {
|
||||||
|
default_page = "file://${config.xdg.dataHome}/qutebrowser/bookmarks.html";
|
||||||
|
start_pages = [default_page];
|
||||||
|
};
|
||||||
|
new_instance_open_target = "window";
|
||||||
|
"tabs.last_close" = "close";
|
||||||
|
"statusbar.widgets" = ["keypress" "url" "progress"];
|
||||||
|
|
||||||
|
# Adblock
|
||||||
|
"content.blocking.enabled" = true;
|
||||||
|
"content.blocking.method" = "adblock";
|
||||||
|
"content.blocking.adblock.lists" = [
|
||||||
|
"https://easylist.to/easylist/easylist.txt"
|
||||||
|
"https://easylist.to/easylist/easyprivacy.txt"
|
||||||
|
"https://secure.fanboy.co.nz/fanboy-cookiemonster.txt"
|
||||||
|
];
|
||||||
|
|
||||||
|
# Clipboard access (needed for Excalidraw, Cyberchef, etc.)
|
||||||
|
"content.javascript.clipboard" = "access";
|
||||||
|
|
||||||
|
# Downloads
|
||||||
|
"downloads.location.directory" = "~/Downloads";
|
||||||
|
"downloads.location.prompt" = false;
|
||||||
|
|
||||||
|
# Editor (Ctrl+e in text fields)
|
||||||
|
"editor.command" = ["ghostty" "-e" "nvim" "{}"];
|
||||||
|
|
||||||
|
# Tabs
|
||||||
|
"tabs.show" = "switching";
|
||||||
|
"tabs.show_switching_delay" = 3000;
|
||||||
|
|
||||||
|
# Scrollbar
|
||||||
|
"scrolling.bar" = "never";
|
||||||
|
|
||||||
|
# Statusbar
|
||||||
|
"statusbar.show" = "in-mode";
|
||||||
|
|
||||||
|
# Privacy
|
||||||
|
"content.geolocation" = false;
|
||||||
|
"content.cookies.accept" = "no-3rdparty";
|
||||||
|
};
|
||||||
|
|
||||||
|
extraConfig = ''
|
||||||
|
c.statusbar.padding = {'top': 6, 'bottom': 6, 'left': 8, 'right': 8}
|
||||||
|
c.tabs.padding = {'top': 6, 'bottom': 6, 'left': 8, 'right': 8}
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
xdg.dataFile = {
|
||||||
|
# Startpage - hide sponsored results
|
||||||
|
"qutebrowser/greasemonkey/startpage-no-ads.user.js".source =
|
||||||
|
./greasemonkey/startpage-no-ads.user.js;
|
||||||
|
|
||||||
|
# Return YouTube Dislike
|
||||||
|
"qutebrowser/greasemonkey/return-youtube-dislike.user.js".source =
|
||||||
|
./greasemonkey/return-youtube-dislike.user.js;
|
||||||
|
|
||||||
|
# SponsorBlock Lite - auto-skip sponsors on YouTube
|
||||||
|
"qutebrowser/greasemonkey/sponsorblock-lite.user.js".source =
|
||||||
|
./greasemonkey/sponsorblock-lite.user.js;
|
||||||
|
|
||||||
|
# Don't Track Me Google - remove Google tracking redirects
|
||||||
|
"qutebrowser/greasemonkey/dont-track-me-google.user.js".source =
|
||||||
|
./greasemonkey/dont-track-me-google.user.js;
|
||||||
|
|
||||||
|
# I don't care about cookies - auto-dismiss cookie banners
|
||||||
|
"qutebrowser/greasemonkey/i-dont-care-about-cookies.user.js".source =
|
||||||
|
./greasemonkey/i-dont-care-about-cookies.user.js;
|
||||||
|
|
||||||
|
# TrackingTokenStripper - remove tracking params from URLs (utm_*, fbclid, etc.)
|
||||||
|
"qutebrowser/greasemonkey/tracking-token-stripper.user.js".source =
|
||||||
|
./greasemonkey/tracking-token-stripper.user.js;
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -7,11 +7,15 @@
|
|||||||
}: let
|
}: let
|
||||||
fetch = config.theme.fetch; # neofetch, nerdfetch, pfetch
|
fetch = config.theme.fetch; # neofetch, nerdfetch, pfetch
|
||||||
in {
|
in {
|
||||||
home.packages = with pkgs; [bat ripgrep tldr sesh];
|
home.packages = with pkgs; [bat ripgrep tldr witr];
|
||||||
|
|
||||||
# Add go binaries to the PATH
|
# Add go binaries to the PATH
|
||||||
home.sessionPath = ["$HOME/go/bin"];
|
home.sessionPath = ["$HOME/go/bin"];
|
||||||
|
|
||||||
|
home.sessionVariables = {
|
||||||
|
COLORTERM = "truecolor";
|
||||||
|
};
|
||||||
|
|
||||||
programs.zsh = {
|
programs.zsh = {
|
||||||
enable = true;
|
enable = true;
|
||||||
enableCompletion = true;
|
enableCompletion = true;
|
||||||
@@ -47,7 +51,6 @@ in {
|
|||||||
tree = "eza --icons=always --tree --no-quotes";
|
tree = "eza --icons=always --tree --no-quotes";
|
||||||
sl = "ls";
|
sl = "ls";
|
||||||
open = "${pkgs.xdg-utils}/bin/xdg-open";
|
open = "${pkgs.xdg-utils}/bin/xdg-open";
|
||||||
icat = "${pkgs.kitty}/bin/kitty +kitten icat";
|
|
||||||
cat = "bat --theme=base16 --color=always --paging=never --tabs=2 --wrap=never --plain";
|
cat = "bat --theme=base16 --color=always --paging=never --tabs=2 --wrap=never --plain";
|
||||||
mkdir = "mkdir -p";
|
mkdir = "mkdir -p";
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
# Spicetify is a spotify client customizer
|
# Spicetify is a spotify client customizer
|
||||||
{
|
{
|
||||||
pkgs,
|
pkgs,
|
||||||
|
pkgs-stable,
|
||||||
config,
|
config,
|
||||||
lib,
|
lib,
|
||||||
inputs,
|
inputs,
|
||||||
@@ -16,6 +17,7 @@ in {
|
|||||||
|
|
||||||
programs.spicetify = {
|
programs.spicetify = {
|
||||||
enable = true;
|
enable = true;
|
||||||
|
spotifyPackage = pkgs-stable.spotify;
|
||||||
theme = lib.mkForce spicePkgs.themes.dribbblish;
|
theme = lib.mkForce spicePkgs.themes.dribbblish;
|
||||||
|
|
||||||
colorScheme = "custom";
|
colorScheme = "custom";
|
||||||
@@ -31,7 +33,6 @@ in {
|
|||||||
|
|
||||||
enabledExtensions = with spicePkgs.extensions; [
|
enabledExtensions = with spicePkgs.extensions; [
|
||||||
playlistIcons
|
playlistIcons
|
||||||
lastfm
|
|
||||||
historyShortcut
|
historyShortcut
|
||||||
hidePodcasts
|
hidePodcasts
|
||||||
adblock
|
adblock
|
||||||
|
|||||||
@@ -1,13 +1,16 @@
|
|||||||
# Thunar is a file explorer
|
# Thunar is a file explorer
|
||||||
{
|
{
|
||||||
pkgs,
|
pkgs,
|
||||||
|
pkgs-stable,
|
||||||
config,
|
config,
|
||||||
|
lib,
|
||||||
...
|
...
|
||||||
}: let
|
}: let
|
||||||
user = config.var.username;
|
user = config.var.username;
|
||||||
in {
|
in {
|
||||||
# ctrl + m to toggle the menubar
|
# ctrl + m to toggle the menubar
|
||||||
home.packages = with pkgs; [
|
home.packages =
|
||||||
|
(with pkgs-stable; [
|
||||||
xfce.thunar
|
xfce.thunar
|
||||||
xfce.xfconf
|
xfce.xfconf
|
||||||
xfce.tumbler
|
xfce.tumbler
|
||||||
@@ -16,29 +19,40 @@ in {
|
|||||||
xfce.thunar-media-tags-plugin
|
xfce.thunar-media-tags-plugin
|
||||||
p7zip
|
p7zip
|
||||||
xarchiver
|
xarchiver
|
||||||
];
|
])
|
||||||
|
++ (with pkgs; [
|
||||||
|
# Icon themes: keep on global pkgs to avoid conflicts with other modules
|
||||||
|
papirus-icon-theme
|
||||||
|
material-icons
|
||||||
|
material-design-icons
|
||||||
|
material-symbols
|
||||||
|
]);
|
||||||
|
|
||||||
gtk = {
|
gtk = {
|
||||||
|
enable = true;
|
||||||
iconTheme = {
|
iconTheme = {
|
||||||
name = "WhiteSur";
|
name = "Papirus-Dark";
|
||||||
package = pkgs.whitesur-icon-theme.override {
|
package = pkgs.papirus-icon-theme;
|
||||||
boldPanelIcons = true;
|
|
||||||
alternativeIcons = true;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
home.sessionVariables = {
|
gtk4.theme = null;
|
||||||
XDG_ICON_DIR = "${pkgs.whitesur-icon-theme}/share/icons/WhiteSur";
|
|
||||||
};
|
|
||||||
|
|
||||||
# bookmarks for the side pane
|
# bookmarks for the side pane
|
||||||
gtk.gtk3.bookmarks = [
|
gtk3.bookmarks = [
|
||||||
"file:///home/${user}/Downloads Downloads"
|
"file:///home/${user}/Downloads Downloads"
|
||||||
"file:///home/${user}/Pictures Pictures"
|
"file:///home/${user}/Pictures Pictures"
|
||||||
"file:///home/${user}/.config/nixos NixOS"
|
"file:///home/${user}/.config/nixos NixOS"
|
||||||
"file:///home/${user}/dev Development"
|
"file:///home/${user}/dev Development"
|
||||||
];
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
qt.enable = true;
|
||||||
|
|
||||||
|
home.sessionVariables = {
|
||||||
|
XDG_ICON_DIR = "${pkgs.papirus-icon-theme}/share/icons/Papirus";
|
||||||
|
QS_ICON_THEME = "Papirus";
|
||||||
|
QT_STYLE_OVERRIDE = lib.mkForce "Fusion";
|
||||||
|
};
|
||||||
|
|
||||||
home.file.".config/xarchiver/xarchiverrc".text = ''
|
home.file.".config/xarchiver/xarchiverrc".text = ''
|
||||||
[xarchiver]
|
[xarchiver]
|
||||||
@@ -56,7 +70,7 @@ in {
|
|||||||
show_toolbar=true
|
show_toolbar=true
|
||||||
preferred_custom_cmd=
|
preferred_custom_cmd=
|
||||||
preferred_temp_dir=/tmp
|
preferred_temp_dir=/tmp
|
||||||
preferred_extract_dir=/home/${user}/Downloads
|
preferred_extract_dir=./
|
||||||
allow_sub_dir=0
|
allow_sub_dir=0
|
||||||
ensure_directory=true
|
ensure_directory=true
|
||||||
overwrite=false
|
overwrite=false
|
||||||
@@ -72,7 +86,8 @@ in {
|
|||||||
remove_files=false
|
remove_files=false
|
||||||
'';
|
'';
|
||||||
|
|
||||||
home.file.".config/xfce4/xfconf/xfce-perchannel-xml/thunar.xml".text = ''
|
home.file.".config/xfce4/xfconf/xfce-perchannel-xml/thunar.xml" = {
|
||||||
|
text = ''
|
||||||
<?xml version="1.1" encoding="UTF-8"?>
|
<?xml version="1.1" encoding="UTF-8"?>
|
||||||
|
|
||||||
<channel name="thunar" version="1.0">
|
<channel name="thunar" version="1.0">
|
||||||
@@ -119,6 +134,8 @@ in {
|
|||||||
<property name="misc-show-delete-action" type="bool" value="false"/>
|
<property name="misc-show-delete-action" type="bool" value="false"/>
|
||||||
</channel>
|
</channel>
|
||||||
'';
|
'';
|
||||||
|
force = true;
|
||||||
|
};
|
||||||
|
|
||||||
xdg.configFile."Thunar/uca.xml".text = ''
|
xdg.configFile."Thunar/uca.xml".text = ''
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
@@ -127,8 +144,8 @@ in {
|
|||||||
<icon>utilities-terminal</icon>
|
<icon>utilities-terminal</icon>
|
||||||
<name>Open Terminal Here</name>
|
<name>Open Terminal Here</name>
|
||||||
<unique-id>1700000000000001</unique-id>
|
<unique-id>1700000000000001</unique-id>
|
||||||
<command>kitty -d %f</command>
|
<command>ghostty -d %f</command>
|
||||||
<description>Opens Kitty terminal in the selected folder</description>
|
<description>Opens terminal in the selected folder</description>
|
||||||
<patterns>*</patterns>
|
<patterns>*</patterns>
|
||||||
<startup-notify/>
|
<startup-notify/>
|
||||||
<directories/>
|
<directories/>
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
{pkgs, ...}: {
|
|
||||||
programs.vivaldi = {
|
|
||||||
enable = true;
|
|
||||||
commandLineArgs = [
|
|
||||||
"--no-default-browser-check"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
home.sessionVariables = {
|
|
||||||
DEFAULT_BROWSER = "${pkgs.vivaldi}/bin/vivaldi";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
# TODO: Misc section with VLC, Zathura etc
|
|
||||||
# Zathura is a PDF viewer
|
|
||||||
{
|
|
||||||
programs.zathura = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
options = {
|
|
||||||
guioptions = "v";
|
|
||||||
adjust-open = "width";
|
|
||||||
statusbar-basename = true;
|
|
||||||
render-loading = false;
|
|
||||||
scroll-step = 120;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -29,8 +29,8 @@ in {
|
|||||||
enabled = true;
|
enabled = true;
|
||||||
maxToasts = 4;
|
maxToasts = 4;
|
||||||
toasts = {
|
toasts = {
|
||||||
audioInputChanged = true;
|
audioInputChanged = false;
|
||||||
audioOutputChanged = true;
|
audioOutputChanged = false;
|
||||||
capsLockChanged = false;
|
capsLockChanged = false;
|
||||||
chargingChanged = true;
|
chargingChanged = true;
|
||||||
configLoaded = false;
|
configLoaded = false;
|
||||||
@@ -41,6 +41,7 @@ in {
|
|||||||
kbLayoutChanged = false;
|
kbLayoutChanged = false;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
dashboard.showOnHover = false;
|
||||||
border = {
|
border = {
|
||||||
inherit rounding;
|
inherit rounding;
|
||||||
thickness = 8;
|
thickness = 8;
|
||||||
|
|||||||
@@ -1,58 +0,0 @@
|
|||||||
{
|
|
||||||
wayland.windowManager.hyprland.settings = {
|
|
||||||
bindin = [
|
|
||||||
# Launcher
|
|
||||||
"$mod, mouse:272, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse:273, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse:274, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse:275, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse:276, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse:277, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse_up, global, caelestia:launcherInterrupt"
|
|
||||||
"$mod, mouse_down, global, caelestia:launcherInterrupt"
|
|
||||||
];
|
|
||||||
bind = [
|
|
||||||
# Launcher
|
|
||||||
"$mod, SPACE, global, caelestia:launcher"
|
|
||||||
"$mod, X, global, caelestia:session" # Powermenu
|
|
||||||
|
|
||||||
# Misc
|
|
||||||
"$mod, L, global, caelestia:lock"
|
|
||||||
"$mod, N, exec, caelestia shell drawers toggle sidebar"
|
|
||||||
|
|
||||||
# Utilities
|
|
||||||
"$mod+Shift, SPACE, exec, caelestia shell gameMode toggle" # Toggle Focus/Game mode
|
|
||||||
"$mod+Shift, S, global, caelestia:screenshotFreeze" # Capture region (freeze)
|
|
||||||
"$mod+Shift+Alt, S, global, caelestia:screenshot" # Capture region
|
|
||||||
"$mod+Alt, R, exec, caelestia record -s" # Record screen with sound
|
|
||||||
"Ctrl+Alt, R, exec, caelestia record" # Record screen
|
|
||||||
"$mod+Shift+Alt, R, exec, caelestia record -r" # Record region
|
|
||||||
|
|
||||||
"$mod+Shift, E, exec, pkill fuzzel || caelestia emoji -p"
|
|
||||||
];
|
|
||||||
bindl = [
|
|
||||||
# Brightness
|
|
||||||
", XF86MonBrightnessUp, global, caelestia:brightnessUp"
|
|
||||||
", XF86MonBrightnessDown, global, caelestia:brightnessDown"
|
|
||||||
"$mod, F2, exec, nightshift-toggle"
|
|
||||||
"$mod, F3, exec, nightshift-toggle"
|
|
||||||
|
|
||||||
# Media
|
|
||||||
", XF86AudioPlay, global, caelestia:mediaToggle"
|
|
||||||
", XF86AudioPause, global, caelestia:mediaToggle"
|
|
||||||
", XF86AudioNext, global, caelestia:mediaNext"
|
|
||||||
", XF86AudioPrev, global, caelestia:mediaPrev"
|
|
||||||
", XF86AudioStop, global, caelestia:mediaStop"
|
|
||||||
|
|
||||||
# Sound
|
|
||||||
", XF86AudioMute, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ toggle"
|
|
||||||
|
|
||||||
# Utilities
|
|
||||||
", Print, exec, caelestia screenshot" # Full screen capture > clipboard
|
|
||||||
];
|
|
||||||
bindle = [
|
|
||||||
", XF86AudioRaiseVolume, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ 0; wpctl set-volume -l 1 @DEFAULT_AUDIO_SINK@ 5%+"
|
|
||||||
", XF86AudioLowerVolume, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ 0; wpctl set-volume @DEFAULT_AUDIO_SINK@ 5%-"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -7,11 +7,11 @@
|
|||||||
}: {
|
}: {
|
||||||
imports = [
|
imports = [
|
||||||
inputs.caelestia-shell.homeManagerModules.default
|
inputs.caelestia-shell.homeManagerModules.default
|
||||||
./bindings.nix
|
|
||||||
./bar.nix
|
./bar.nix
|
||||||
./launcher.nix
|
./launcher.nix
|
||||||
./appearance.nix
|
./appearance.nix
|
||||||
./scheme.nix
|
./scheme.nix
|
||||||
|
./swappy.nix
|
||||||
];
|
];
|
||||||
|
|
||||||
programs.caelestia = {
|
programs.caelestia = {
|
||||||
@@ -52,6 +52,7 @@
|
|||||||
wayland.windowManager.hyprland.settings.exec-once = [
|
wayland.windowManager.hyprland.settings.exec-once = [
|
||||||
"uwsm app -- caelestia resizer -d"
|
"uwsm app -- caelestia resizer -d"
|
||||||
"uwsm app -- caelestia shell -d"
|
"uwsm app -- caelestia shell -d"
|
||||||
|
"caelestia scheme set -n onedark"
|
||||||
"caelestia scheme set -n custom"
|
"caelestia scheme set -n custom"
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
{
|
{
|
||||||
programs.caelestia.settings = {
|
programs.caelestia.settings = {
|
||||||
session.commands = {
|
session.commands = {
|
||||||
logout = ["loginctl" "terminate-user" ""];
|
|
||||||
shutdown = ["systemctl" "poweroff"];
|
shutdown = ["systemctl" "poweroff"];
|
||||||
|
logout = ["loginctl" "lock-session"];
|
||||||
hibernate = ["systemctl" "hibernate"];
|
hibernate = ["systemctl" "hibernate"];
|
||||||
reboot = ["systemctl" "reboot"];
|
reboot = ["systemctl" "reboot"];
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -120,7 +120,10 @@
|
|||||||
onSuccessContainer ${colors.base05}
|
onSuccessContainer ${colors.base05}
|
||||||
'';
|
'';
|
||||||
|
|
||||||
|
colorsHash = builtins.hashString "sha256" (builtins.toJSON colors);
|
||||||
|
|
||||||
customCli = inputs.caelestia-cli.packages.${pkgs.stdenv.hostPlatform.system}.default.overrideAttrs (oldAttrs: {
|
customCli = inputs.caelestia-cli.packages.${pkgs.stdenv.hostPlatform.system}.default.overrideAttrs (oldAttrs: {
|
||||||
|
name = "${oldAttrs.name or "caelestia-cli"}-themed-${colorsHash}";
|
||||||
postUnpack = ''
|
postUnpack = ''
|
||||||
mkdir -p $sourceRoot/src/caelestia/data/schemes/custom/main
|
mkdir -p $sourceRoot/src/caelestia/data/schemes/custom/main
|
||||||
cp ${customSchemeFile} $sourceRoot/src/caelestia/data/schemes/custom/main/dark.txt
|
cp ${customSchemeFile} $sourceRoot/src/caelestia/data/schemes/custom/main/dark.txt
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
xdg.configFile."swappy/config".text = ''
|
||||||
|
[Default]
|
||||||
|
save_dir=~/Pictures/Screenshots
|
||||||
|
save_filename_format=screenshot-%Y%m%d-%H%M%S.png
|
||||||
|
show_panel=false
|
||||||
|
line_size=5
|
||||||
|
text_size=20
|
||||||
|
text_font=sans-serif
|
||||||
|
'';
|
||||||
|
}
|
||||||
@@ -1,31 +1,148 @@
|
|||||||
{pkgs, ...}: {
|
{
|
||||||
|
pkgs,
|
||||||
|
lib,
|
||||||
|
config,
|
||||||
|
...
|
||||||
|
}: let
|
||||||
|
colors = config.lib.stylix.colors;
|
||||||
|
|
||||||
|
mkMenu = menu: let
|
||||||
|
configFile =
|
||||||
|
pkgs.writeText "config.yaml"
|
||||||
|
(lib.generators.toYAML {} {
|
||||||
|
anchor = "bottom-right";
|
||||||
|
border = "#${colors.base0D}80";
|
||||||
|
background = "#${colors.base01}EE";
|
||||||
|
color = "#${colors.base05}";
|
||||||
|
margin_right = 15;
|
||||||
|
margin_bottom = 15;
|
||||||
|
rows_per_column = 5;
|
||||||
|
|
||||||
|
inherit menu;
|
||||||
|
});
|
||||||
|
in
|
||||||
|
pkgs.writeShellScriptBin "menu" ''
|
||||||
|
exec ${lib.getExe pkgs.wlr-which-key} ${configFile}
|
||||||
|
'';
|
||||||
|
in {
|
||||||
wayland.windowManager.hyprland.settings = {
|
wayland.windowManager.hyprland.settings = {
|
||||||
"$mod" = "SUPER";
|
"$mod" = "SUPER";
|
||||||
"$shiftMod" = "SUPER_SHIFT";
|
"$shiftMod" = "SUPER_SHIFT";
|
||||||
|
|
||||||
bind =
|
bind =
|
||||||
[
|
[
|
||||||
"$mod,RETURN, exec, uwsm app -- ${pkgs.ghostty}/bin/ghostty" # Ghostty (terminal)
|
# Applications
|
||||||
"$mod,E, exec, uwsm app -- ${pkgs.xfce.thunar}/bin/thunar" # Thunar
|
("$shiftMod, A, exec, "
|
||||||
"$mod,B, exec, uwsm app -- ${pkgs.vivaldi}/bin/vivaldi" # Browser (Vivaldi)
|
+ lib.getExe (mkMenu [
|
||||||
"$mod,K, exec, uwsm app -- ${pkgs.proton-pass}/bin/proton-pass" # Proton Pass
|
{
|
||||||
"$mod,V, exec, uwsm app -- ${pkgs.protonvpn-gui}/bin/protonvpn-app" # Proton VPN
|
key = "a";
|
||||||
"$mod,A, exec, uwsm app -- env WEBKIT_DISABLE_COMPOSITING_MODE=1 ${pkgs.proton-authenticator}/bin/proton-authenticator" # Proton Auth
|
desc = "Proton Authenticator";
|
||||||
"$mod,M, exec, uwsm app -- ${pkgs.vivaldi}/bin/vivaldi --profile-directory=Default --app-id=jnpecgipniidlgicjocehkhajgdnjekh" # Proton Mail (PWA)
|
cmd = "env WEBKIT_DISABLE_COMPOSITING_MODE=1 ${pkgs.proton-authenticator}/bin/proton-authenticator";
|
||||||
"$mod,C, exec, uwsm app -- ${pkgs.vivaldi}/bin/vivaldi --profile-directory=Default --app-id=ojibjkjikcpjonjjngfkegflhmffeemk" # Proton Calendar (PWA)
|
}
|
||||||
"$mod,I, exec, uwsm app -- ${pkgs.vivaldi}/bin/vivaldi --profile-directory=Default --app-id=lcfjlhjhpmdjimnbkdfjnkojodddgfmd" # Proton Lumo (PWA)
|
{
|
||||||
|
key = "p";
|
||||||
|
desc = "Proton Pass";
|
||||||
|
cmd = "${pkgs.proton-pass}/bin/proton-pass";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "v";
|
||||||
|
desc = "Proton VPN";
|
||||||
|
cmd = "${pkgs.proton-vpn}/bin/protonvpn-app";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "o";
|
||||||
|
desc = "Obsidian";
|
||||||
|
cmd = "${pkgs.obsidian}/bin/obsidian";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "s";
|
||||||
|
desc = "Signal";
|
||||||
|
cmd = "${pkgs.signal-desktop}/bin/signal-desktop";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "t";
|
||||||
|
desc = "TickTick";
|
||||||
|
cmd = "${pkgs.ticktick}/bin/ticktick";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "b";
|
||||||
|
desc = "Qutebrowser";
|
||||||
|
cmd = "${pkgs.qutebrowser}/bin/qutebrowser";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "i";
|
||||||
|
desc = "Qutebrowser (Temp session)";
|
||||||
|
cmd = "${pkgs.qutebrowser}/bin/qutebrowser --temp-basedir";
|
||||||
|
}
|
||||||
|
]))
|
||||||
|
|
||||||
|
# Web links
|
||||||
|
"$mod,B, exec, uwsm app -- ${pkgs.qutebrowser}/bin/qutebrowser" # Browser (Qutebrowser)
|
||||||
|
|
||||||
|
# Power
|
||||||
|
"$mod, X, global, caelestia:session" # Powermenu
|
||||||
|
("$shiftMod, X, exec, "
|
||||||
|
+ lib.getExe (mkMenu [
|
||||||
|
{
|
||||||
|
key = "l";
|
||||||
|
desc = "Lock";
|
||||||
|
cmd = "hyprctl dispatch global caelestia:lock";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "s";
|
||||||
|
desc = "Suspend";
|
||||||
|
cmd = "systemctl suspend";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "r";
|
||||||
|
desc = "Reboot";
|
||||||
|
cmd = "systemctl reboot";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "p";
|
||||||
|
desc = "Power Off";
|
||||||
|
cmd = "systemctl poweroff";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "n";
|
||||||
|
desc = "Nightshift";
|
||||||
|
cmd = "nightshift-toggle";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
key = "c";
|
||||||
|
desc = "Restart caelestia";
|
||||||
|
cmd = "hyprctl dispatch exec 'caelestia-shell kill | sleep 1 | caelestia-shell'";
|
||||||
|
}
|
||||||
|
]))
|
||||||
|
|
||||||
|
# Quick launch
|
||||||
|
"$mod,RETURN, exec, uwsm app -- ${pkgs.ghostty}/bin/ghostty" # Ghostty (terminal)
|
||||||
|
"$mod,E, exec, uwsm app -- ${pkgs.thunar}/bin/thunar" # Thunar
|
||||||
|
"$shiftMod, E, exec, pkill fuzzel || caelestia emoji -p" # Emoji picker
|
||||||
|
"$mod, SPACE, global, caelestia:launcher" # Launcher
|
||||||
|
"$mod, N, exec, caelestia shell drawers toggle sidebar" # Sidebar (Notifications, quick actions)
|
||||||
|
"$mod, D, exec, caelestia shell drawers toggle dashboard" # Dashboard
|
||||||
|
|
||||||
|
# Windows
|
||||||
"$mod,Q, killactive," # Close window
|
"$mod,Q, killactive," # Close window
|
||||||
"$mod,T, togglefloating," # Toggle Floating
|
|
||||||
"$mod,F, fullscreen" # Toggle Fullscreen
|
"$mod,F, fullscreen" # Toggle Fullscreen
|
||||||
"$mod,left, movefocus, l" # Move focus left
|
"$shiftMod,F, togglefloating," # Toggle Floating
|
||||||
"$mod,right, movefocus, r" # Move focus Right
|
|
||||||
"$mod,up, movefocus, u" # Move focus Up
|
# Focus Windows
|
||||||
"$mod,down, movefocus, d" # Move focus Down
|
"$mod,H, movefocus, l" # Move focus left
|
||||||
"$shiftMod,up, focusmonitor, -1" # Focus previous monitor
|
"$mod,J, movefocus, d" # Move focus Down
|
||||||
"$shiftMod,down, focusmonitor, 1" # Focus next monitor
|
"$mod,K, movefocus, u" # Move focus Up
|
||||||
"$shiftMod,left, layoutmsg, addmaster" # Add to master
|
"$mod,L, movefocus, r" # Move focus Right
|
||||||
"$shiftMod,right, layoutmsg, removemaster" # Remove from master
|
"$shiftMod,H, focusmonitor, -1" # Focus previous monitor
|
||||||
|
"$shiftMod,J, layoutmsg, removemaster" # Remove from master
|
||||||
|
"$shiftMod,K, layoutmsg, addmaster" # Add to master
|
||||||
|
"$shiftMod,L, focusmonitor, 1" # Focus next monitor
|
||||||
|
|
||||||
|
# Utilities
|
||||||
|
"$shiftMod, SPACE, exec, caelestia shell gameMode toggle" # Toggle Focus/Game mode
|
||||||
|
"$shiftMod, S, global, caelestia:screenshotFreeze" # Capture region (freeze)
|
||||||
|
", Print, global, caelestia:screenshotFreeze" # Capture region (freeze)
|
||||||
|
"$shiftMod+Alt, S, global, caelestia:screenshot" # Capture region
|
||||||
]
|
]
|
||||||
++ (builtins.concatLists (builtins.genList (i: let
|
++ (builtins.concatLists (builtins.genList (i: let
|
||||||
ws = i + 1;
|
ws = i + 1;
|
||||||
@@ -39,5 +156,35 @@
|
|||||||
"$mod,mouse:272, movewindow" # Move Window (mouse)
|
"$mod,mouse:272, movewindow" # Move Window (mouse)
|
||||||
"$mod,R, resizewindow" # Resize Window (mouse)
|
"$mod,R, resizewindow" # Resize Window (mouse)
|
||||||
];
|
];
|
||||||
|
|
||||||
|
bindl = [
|
||||||
|
# Brightness
|
||||||
|
", XF86MonBrightnessUp, global, caelestia:brightnessUp"
|
||||||
|
", XF86MonBrightnessDown, global, caelestia:brightnessDown"
|
||||||
|
|
||||||
|
# Media
|
||||||
|
", XF86AudioPlay, global, caelestia:mediaToggle"
|
||||||
|
", XF86AudioPause, global, caelestia:mediaToggle"
|
||||||
|
", XF86AudioNext, global, caelestia:mediaNext"
|
||||||
|
", XF86AudioPrev, global, caelestia:mediaPrev"
|
||||||
|
", XF86AudioStop, global, caelestia:mediaStop"
|
||||||
|
|
||||||
|
# Sound
|
||||||
|
", XF86AudioMute, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ toggle"
|
||||||
|
", XF86AudioRaiseVolume, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ 0; wpctl set-volume -l 1 @DEFAULT_AUDIO_SINK@ 5%+"
|
||||||
|
", XF86AudioLowerVolume, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ 0; wpctl set-volume @DEFAULT_AUDIO_SINK@ 5%-"
|
||||||
|
];
|
||||||
|
|
||||||
|
bindin = [
|
||||||
|
# Launcher
|
||||||
|
"$mod, mouse:272, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse:273, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse:274, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse:275, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse:276, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse:277, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse_up, global, caelestia:launcherInterrupt"
|
||||||
|
"$mod, mouse_down, global, caelestia:launcherInterrupt"
|
||||||
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
rounding = config.theme.rounding;
|
rounding = config.theme.rounding;
|
||||||
blur = config.theme.blur;
|
blur = config.theme.blur;
|
||||||
keyboardLayout = config.var.keyboardLayout;
|
keyboardLayout = config.var.keyboardLayout;
|
||||||
background = "rgba(" + config.lib.stylix.colors.base00 + "77)";
|
background = "rgba(" + config.lib.stylix.colors.base00 + "EE)";
|
||||||
in {
|
in {
|
||||||
imports = [
|
imports = [
|
||||||
./animations.nix
|
./animations.nix
|
||||||
@@ -27,6 +27,8 @@ in {
|
|||||||
qt6.qtwayland
|
qt6.qtwayland
|
||||||
libsForQt5.qt5ct
|
libsForQt5.qt5ct
|
||||||
qt6Packages.qt6ct
|
qt6Packages.qt6ct
|
||||||
|
xcb-util-cursor
|
||||||
|
libxcb
|
||||||
hyprland-qtutils
|
hyprland-qtutils
|
||||||
adw-gtk3
|
adw-gtk3
|
||||||
hyprshot
|
hyprshot
|
||||||
@@ -37,13 +39,9 @@ in {
|
|||||||
wlr-randr
|
wlr-randr
|
||||||
brightnessctl
|
brightnessctl
|
||||||
gnome-themes-extra
|
gnome-themes-extra
|
||||||
libva
|
|
||||||
dconf
|
dconf
|
||||||
wayland-utils
|
wayland-utils
|
||||||
wayland-protocols
|
wayland-protocols
|
||||||
glib
|
|
||||||
direnv
|
|
||||||
meson
|
|
||||||
];
|
];
|
||||||
|
|
||||||
wayland.windowManager.hyprland = {
|
wayland.windowManager.hyprland = {
|
||||||
@@ -66,33 +64,24 @@ in {
|
|||||||
monitor = [
|
monitor = [
|
||||||
"eDP-2,highres,0x0,1" # My internal laptop screen
|
"eDP-2,highres,0x0,1" # My internal laptop screen
|
||||||
"desc:AOC U34G2G1 0x00000E06,3440x1440@99.98,auto,1" # My external monitor
|
"desc:AOC U34G2G1 0x00000E06,3440x1440@99.98,auto,1" # My external monitor
|
||||||
"desc:United Microelectr Corporation UMC SHARP,3840x2160,auto,2" # TV
|
"desc:Philips Consumer Electronics Company PHL 221B8L ZV02144013987,highres,0x0,1"
|
||||||
",prefered,auto,1" # default
|
",prefered,auto,1" # default
|
||||||
];
|
];
|
||||||
|
|
||||||
env = [
|
env = [
|
||||||
"XDG_CURRENT_DESKTOP,Hyprland"
|
"XDG_CURRENT_DESKTOP,Hyprland"
|
||||||
"MOZ_ENABLE_WAYLAND,1"
|
|
||||||
"ANKI_WAYLAND,1"
|
|
||||||
"DISABLE_QT5_COMPAT,0"
|
|
||||||
"NIXOS_OZONE_WL,1"
|
|
||||||
"XDG_SESSION_TYPE,wayland"
|
"XDG_SESSION_TYPE,wayland"
|
||||||
"XDG_SESSION_DESKTOP,Hyprland"
|
"XDG_SESSION_DESKTOP,Hyprland"
|
||||||
|
"ANKI_WAYLAND,1"
|
||||||
|
"DISABLE_QT5_COMPAT,0"
|
||||||
"QT_AUTO_SCREEN_SCALE_FACTOR,1"
|
"QT_AUTO_SCREEN_SCALE_FACTOR,1"
|
||||||
"QT_QPA_PLATFORM=wayland,xcb"
|
"QT_QPA_PLATFORM,wayland;xcb"
|
||||||
|
"QT_QPA_PLATFORMTHEME,gtk3"
|
||||||
"QT_WAYLAND_DISABLE_WINDOWDECORATION,1"
|
"QT_WAYLAND_DISABLE_WINDOWDECORATION,1"
|
||||||
"ELECTRON_OZONE_PLATFORM_HINT,auto"
|
"ELECTRON_OZONE_PLATFORM_HINT,auto"
|
||||||
"__GL_GSYNC_ALLOWED,0"
|
|
||||||
"__GL_VRR_ALLOWED,0"
|
|
||||||
"DISABLE_QT5_COMPAT,0"
|
|
||||||
"DIRENV_LOG_FORMAT,"
|
"DIRENV_LOG_FORMAT,"
|
||||||
"WLR_DRM_NO_ATOMIC,1"
|
|
||||||
"WLR_BACKEND,vulkan"
|
|
||||||
"WLR_RENDERER,vulkan"
|
|
||||||
"WLR_NO_HARDWARE_CURSORS,1"
|
|
||||||
"SDL_VIDEODRIVER,wayland"
|
"SDL_VIDEODRIVER,wayland"
|
||||||
"CLUTTER_BACKEND,wayland"
|
"CLUTTER_BACKEND,wayland"
|
||||||
"AQ_DRM_DEVICES,/dev/dri/card2:/dev/dri/card1" # CHANGEME: Related to the GPU
|
|
||||||
];
|
];
|
||||||
|
|
||||||
cursor = {
|
cursor = {
|
||||||
@@ -128,7 +117,7 @@ in {
|
|||||||
};
|
};
|
||||||
|
|
||||||
master = {
|
master = {
|
||||||
new_status = true;
|
new_status = "slave";
|
||||||
allow_small_split = true;
|
allow_small_split = true;
|
||||||
mfact = 0.5;
|
mfact = 0.5;
|
||||||
};
|
};
|
||||||
@@ -136,17 +125,18 @@ in {
|
|||||||
gesture = "3, horizontal, workspace";
|
gesture = "3, horizontal, workspace";
|
||||||
|
|
||||||
windowrule = [
|
windowrule = [
|
||||||
|
"match:class .*, suppress_event maximize"
|
||||||
|
|
||||||
"match:class proton-authenticator, float on"
|
"match:class proton-authenticator, float on"
|
||||||
"match:class proton-authenticator, suppress_event maximize"
|
|
||||||
"match:class proton-authenticator, center on"
|
"match:class proton-authenticator, center on"
|
||||||
"match:class proton-authenticator, size 500 400"
|
"match:class proton-authenticator, size 500 400"
|
||||||
|
|
||||||
"match:class protonvpn-app, float on"
|
"match:class protonvpn-app, float on"
|
||||||
"match:class protonvpn-app, center on"
|
"match:class protonvpn-app, center on"
|
||||||
"match:class protonvpn-app, size 500 400"
|
"match:class protonvpn-app, size 500 400"
|
||||||
];
|
];
|
||||||
|
|
||||||
misc = {
|
misc = {
|
||||||
vfr = true;
|
|
||||||
disable_hyprland_logo = true;
|
disable_hyprland_logo = true;
|
||||||
disable_splash_rendering = true;
|
disable_splash_rendering = true;
|
||||||
disable_autoreload = true;
|
disable_autoreload = true;
|
||||||
@@ -168,6 +158,10 @@ in {
|
|||||||
clickfinger_behavior = true;
|
clickfinger_behavior = true;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ecosystem = {
|
||||||
|
no_update_news = true;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,6 @@
|
|||||||
settings = {
|
settings = {
|
||||||
ipc = "on";
|
ipc = "on";
|
||||||
splash = false;
|
splash = false;
|
||||||
splash_offset = 2.0;
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
systemd.user.services.hyprpaper.Unit.After =
|
systemd.user.services.hyprpaper.Unit.After =
|
||||||
|
|||||||
@@ -1,23 +1,64 @@
|
|||||||
# Mime type associations for the system.
|
# Mime type associations for the system.
|
||||||
{lib, ...}:
|
{
|
||||||
|
lib,
|
||||||
|
pkgs,
|
||||||
|
...
|
||||||
|
}:
|
||||||
with lib; let
|
with lib; let
|
||||||
defaultApps = {
|
defaultApps = {
|
||||||
# check desktop files here: `ls $(echo $XDG_DATA_DIRS| sed "s/:/ /g")`
|
# check desktop files here: `ls $(echo $XDG_DATA_DIRS| sed "s/:/ /g")`
|
||||||
browser = ["vivaldi-stable.desktop"];
|
text = [
|
||||||
text = ["org.gnome.TextEditor.desktop"];
|
# "org.gnome.TextEditor.desktop"
|
||||||
|
"nvim-ghostty.desktop"
|
||||||
|
];
|
||||||
|
code = ["nvim-ghostty.desktop"];
|
||||||
image = ["imv-dir.desktop"];
|
image = ["imv-dir.desktop"];
|
||||||
audio = ["mpv.desktop"];
|
audio = ["mpv.desktop"];
|
||||||
video = ["mpv.desktop"];
|
video = ["mpv.desktop"];
|
||||||
directory = ["thunar.desktop"];
|
directory = ["thunar.desktop"];
|
||||||
office = ["libreoffice.desktop"];
|
office = ["libreoffice.desktop"];
|
||||||
pdf = ["zathura.desktop"];
|
pdf = ["zathura.desktop"];
|
||||||
terminal = ["kitty.desktop"];
|
terminal = ["ghostty.desktop"];
|
||||||
discord = ["discord.desktop"];
|
|
||||||
archive = ["xarchiver.desktop"];
|
archive = ["xarchiver.desktop"];
|
||||||
};
|
};
|
||||||
|
|
||||||
mimeMap = {
|
mimeMap = {
|
||||||
text = ["text/plain"];
|
text = ["text/plain"];
|
||||||
|
code = [
|
||||||
|
"text/x-csrc"
|
||||||
|
"text/x-chdr"
|
||||||
|
"text/x-c++src"
|
||||||
|
"text/x-c++hdr"
|
||||||
|
"text/x-rust"
|
||||||
|
"text/x-go"
|
||||||
|
"text/x-java"
|
||||||
|
"text/x-csharp"
|
||||||
|
|
||||||
|
"text/x-python"
|
||||||
|
"application/x-shellscript"
|
||||||
|
"text/javascript"
|
||||||
|
"application/javascript"
|
||||||
|
"text/css"
|
||||||
|
"text/x-php"
|
||||||
|
"text/x-ruby"
|
||||||
|
|
||||||
|
"application/json"
|
||||||
|
"application/xml"
|
||||||
|
"text/xml"
|
||||||
|
"text/x-yaml"
|
||||||
|
"application/x-yaml"
|
||||||
|
"application/toml"
|
||||||
|
"text/x-nix"
|
||||||
|
"text/markdown"
|
||||||
|
|
||||||
|
"text/x-dockerfile"
|
||||||
|
"application/x-yaml"
|
||||||
|
"text/x-terraform"
|
||||||
|
|
||||||
|
"application/x-perl"
|
||||||
|
"text/x-lua"
|
||||||
|
"text/x-haskell"
|
||||||
|
];
|
||||||
image = [
|
image = [
|
||||||
"image/bmp"
|
"image/bmp"
|
||||||
"image/gif"
|
"image/gif"
|
||||||
@@ -49,13 +90,6 @@ with lib; let
|
|||||||
"video/x-msvideo"
|
"video/x-msvideo"
|
||||||
];
|
];
|
||||||
directory = ["inode/directory"];
|
directory = ["inode/directory"];
|
||||||
browser = [
|
|
||||||
"text/html"
|
|
||||||
"x-scheme-handler/about"
|
|
||||||
"x-scheme-handler/http"
|
|
||||||
"x-scheme-handler/https"
|
|
||||||
"x-scheme-handler/unknown"
|
|
||||||
];
|
|
||||||
office = [
|
office = [
|
||||||
"application/vnd.oasis.opendocument.text"
|
"application/vnd.oasis.opendocument.text"
|
||||||
"application/vnd.oasis.opendocument.spreadsheet"
|
"application/vnd.oasis.opendocument.spreadsheet"
|
||||||
@@ -76,7 +110,15 @@ with lib; let
|
|||||||
"application/7z"
|
"application/7z"
|
||||||
"application/*tar"
|
"application/*tar"
|
||||||
];
|
];
|
||||||
discord = ["x-scheme-handler/discord"];
|
};
|
||||||
|
|
||||||
|
nvim-ghostty = pkgs.makeDesktopItem {
|
||||||
|
name = "nvim-ghostty";
|
||||||
|
desktopName = "Neovim (Ghostty)";
|
||||||
|
exec = ''ghostty --title="Neovim Editor" -e nvim %F'';
|
||||||
|
terminal = false;
|
||||||
|
categories = ["Development" "TextEditor"];
|
||||||
|
mimeTypes = mimeMap.code ++ mimeMap.text;
|
||||||
};
|
};
|
||||||
|
|
||||||
associations = with lists;
|
associations = with lists;
|
||||||
@@ -84,6 +126,8 @@ with lib; let
|
|||||||
(key: map (type: attrsets.nameValuePair type defaultApps."${key}"))
|
(key: map (type: attrsets.nameValuePair type defaultApps."${key}"))
|
||||||
mimeMap));
|
mimeMap));
|
||||||
in {
|
in {
|
||||||
|
home.packages = [nvim-ghostty];
|
||||||
|
|
||||||
xdg = {
|
xdg = {
|
||||||
configFile."mimeapps.list".force = true;
|
configFile."mimeapps.list".force = true;
|
||||||
mimeApps = {
|
mimeApps = {
|
||||||
|
|||||||
@@ -11,9 +11,7 @@
|
|||||||
../../nixos/sddm.nix
|
../../nixos/sddm.nix
|
||||||
../../nixos/users.nix
|
../../nixos/users.nix
|
||||||
../../nixos/utils.nix
|
../../nixos/utils.nix
|
||||||
../../nixos/tailscale.nix
|
|
||||||
../../nixos/hyprland.nix
|
../../nixos/hyprland.nix
|
||||||
../../nixos/docker.nix
|
|
||||||
|
|
||||||
../../nixos/omen.nix # CHANGEME: For my laptop only, remove this (OMEN 16)
|
../../nixos/omen.nix # CHANGEME: For my laptop only, remove this (OMEN 16)
|
||||||
|
|
||||||
|
|||||||
+14
-52
@@ -1,26 +1,26 @@
|
|||||||
{
|
{config, ...}: {
|
||||||
pkgs,
|
|
||||||
config,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
imports = [
|
||||||
# Programs
|
# Programs
|
||||||
../../home/programs/vivaldi
|
../../home/programs/brave
|
||||||
../../home/programs/proton
|
../../home/programs/proton
|
||||||
|
../../home/programs/proton/auto-start-vpn.nix
|
||||||
../../home/programs/ghostty
|
../../home/programs/ghostty
|
||||||
../../home/programs/nvf
|
../../home/programs/nvf
|
||||||
../../home/programs/shell
|
../../home/programs/shell
|
||||||
../../home/programs/fetch
|
|
||||||
../../home/programs/git
|
../../home/programs/git
|
||||||
../../home/programs/git/lazygit.nix
|
../../home/programs/git/lazygit.nix
|
||||||
../../home/programs/git/signing.nix # Change the key or remove this file
|
../../home/programs/git/signing.nix # CHANGEME: Change the key or remove this file
|
||||||
../../home/programs/spicetify
|
../../home/programs/spicetify
|
||||||
../../home/programs/thunar
|
../../home/programs/thunar
|
||||||
../../home/programs/discord
|
|
||||||
../../home/programs/nixy
|
../../home/programs/nixy
|
||||||
../../home/programs/zathura
|
|
||||||
../../home/programs/nightshift
|
../../home/programs/nightshift
|
||||||
|
../../home/programs/qutebrowser
|
||||||
|
../../home/programs/nix-utils
|
||||||
|
|
||||||
|
../../home/programs/group/basic-apps.nix
|
||||||
../../home/programs/group/cybersecurity.nix
|
../../home/programs/group/cybersecurity.nix
|
||||||
|
../../home/programs/group/dev.nix
|
||||||
|
../../home/programs/group/misc.nix
|
||||||
|
|
||||||
# System (Desktop environment like stuff)
|
# System (Desktop environment like stuff)
|
||||||
../../home/system/hyprland
|
../../home/system/hyprland
|
||||||
@@ -34,52 +34,14 @@
|
|||||||
];
|
];
|
||||||
|
|
||||||
home = {
|
home = {
|
||||||
packages = with pkgs; [
|
|
||||||
# Apps
|
|
||||||
vlc # Video player
|
|
||||||
blanket # White-noise app
|
|
||||||
obsidian # Note taking app
|
|
||||||
textpieces # Manipulate texts
|
|
||||||
resources # Ressource monitor
|
|
||||||
gnome-clocks # Clocks app
|
|
||||||
gnome-text-editor # Basic graphic text editor
|
|
||||||
mpv # Video player
|
|
||||||
ticktick # Todo app
|
|
||||||
session-desktop # Session app, private messages
|
|
||||||
signal-desktop # Signal app, private messages
|
|
||||||
stirling-pdf # TODO: Server version
|
|
||||||
calibre
|
|
||||||
|
|
||||||
# Dev
|
|
||||||
go
|
|
||||||
bun
|
|
||||||
docker
|
|
||||||
nodejs
|
|
||||||
python3
|
|
||||||
jq
|
|
||||||
just
|
|
||||||
pnpm
|
|
||||||
air
|
|
||||||
duckdb
|
|
||||||
|
|
||||||
# Just cool
|
|
||||||
peaclock
|
|
||||||
cbonsai
|
|
||||||
pipes
|
|
||||||
cmatrix
|
|
||||||
fastfetch
|
|
||||||
|
|
||||||
# Backup
|
|
||||||
vscode
|
|
||||||
brave
|
|
||||||
];
|
|
||||||
|
|
||||||
inherit (config.var) username;
|
inherit (config.var) username;
|
||||||
homeDirectory = "/home/" + config.var.username;
|
homeDirectory = "/home/" + config.var.username;
|
||||||
|
|
||||||
# Import a profile picture, used by the caelestia dashboard
|
|
||||||
file.".face" = {source = ./profile_picture.png;};
|
file.".face" = {source = ./profile_picture.png;};
|
||||||
|
|
||||||
|
sessionVariables = {
|
||||||
|
AQ_DRM_DEVICES = "/dev/dri/card2:/dev/dri/card1"; # CHANGEME: Related to the GPU
|
||||||
|
};
|
||||||
|
|
||||||
# Don't touch this
|
# Don't touch this
|
||||||
stateVersion = "24.05";
|
stateVersion = "24.05";
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,27 +1,31 @@
|
|||||||
# Those are my secrets, encrypted with sops
|
# Those are my secrets, encrypted with sops
|
||||||
# You shouldn't import this file, unless you edit it
|
# You shouldn't import this file, unless you edit it
|
||||||
{
|
{
|
||||||
pkgs,
|
|
||||||
inputs,
|
inputs,
|
||||||
|
pkgs,
|
||||||
|
config,
|
||||||
...
|
...
|
||||||
}: {
|
}: let
|
||||||
|
home = config.home.homeDirectory;
|
||||||
|
in {
|
||||||
imports = [inputs.sops-nix.homeManagerModules.sops];
|
imports = [inputs.sops-nix.homeManagerModules.sops];
|
||||||
|
|
||||||
sops = {
|
sops = {
|
||||||
age.keyFile = "/home/hadi/.config/sops/age/keys.txt";
|
age.keyFile = "${home}/.config/sops/age/keys.txt";
|
||||||
defaultSopsFile = ./secrets.yaml;
|
defaultSopsFile = ./secrets.yaml;
|
||||||
secrets = {
|
secrets = {
|
||||||
sshconfig = {path = "/home/hadi/.ssh/config";};
|
ssh-config = {path = "${home}/.ssh/config";};
|
||||||
github-key = {path = "/home/hadi/.ssh/github";};
|
github-key = {path = "${home}/.ssh/github";};
|
||||||
jack-key = {path = "/home/hadi/.ssh/jack";};
|
jack-key = {path = "${home}/.ssh/jack";};
|
||||||
signing-key = {path = "/home/hadi/.ssh/key";};
|
signing-key = {path = "${home}/.ssh/key";};
|
||||||
signing-pub-key = {path = "/home/hadi/.ssh/key.pub";};
|
signing-pub-key = {path = "${home}/.ssh/key.pub";};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
home.file.".config/nixos/.sops.yaml".text = ''
|
home.file.".config/nixos/.sops.yaml".text = ''
|
||||||
keys:
|
keys:
|
||||||
- &primary age12yvtj49pfh3fqzqflscm0ek4yzrjhr6cqhn7x89gdxnlykq0xudq5c7334
|
- &primary age12yvtj49pfh3fqzqflscm0ek4yzrjhr6cqhn7x89gdxnlykq0xudq5c7334
|
||||||
|
- &work age1c8pawdsxptfslgrz2c56s39mrtnjzc5mm3hfzgr2wdwu2v6vfsdsupjsq6
|
||||||
creation_rules:
|
creation_rules:
|
||||||
- path_regex: hosts/laptop/secrets/secrets.yaml$
|
- path_regex: hosts/laptop/secrets/secrets.yaml$
|
||||||
key_groups:
|
key_groups:
|
||||||
@@ -31,6 +35,10 @@
|
|||||||
key_groups:
|
key_groups:
|
||||||
- age:
|
- age:
|
||||||
- *primary
|
- *primary
|
||||||
|
- path_regex: hosts/work/secrets/secrets.yaml$
|
||||||
|
key_groups:
|
||||||
|
- age:
|
||||||
|
- *work
|
||||||
'';
|
'';
|
||||||
|
|
||||||
systemd.user.services.mbsync.Unit.After = ["sops-nix.service"];
|
systemd.user.services.mbsync.Unit.After = ["sops-nix.service"];
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
pia: ENC[AES256_GCM,data:0bnhHeVqKSLHVimd78a94ShHlO3+LUoZ4oiKD3cnBYkaZsw=,iv:S+/IChlFlqdI+PyFF+Ti4AJUkch2MS0qKiqHL1Q3RMk=,tag:+v2kV70ou84eIc01dKnAhA==,type:str]
|
ssh-config: ENC[AES256_GCM,data:LE468Anl/ri1JBOdNOsMPanhHcasn68NCOHewCLOfU7NpMxCdJ9sdHFxuFJ34F8OB2qPjGTtRCv0d+75aeJNrh1hwUyyD5DBoWgM0vvQit/VpmpNgLPndWo6dIo2Xf1xPJdaADiLBIjsNNgUUmo6N8InT4eLihDOepbHwyVD0qFYFC1vze+V253aodLADfp3YbRI7MvUQrn+vjiDtuyEBMcYBqUQF1DhLYDbM/S6ez6NdscxTJvdM33u4rRehn8p3hE1mOovM/0f7gjvmtXpJqaBQ1aP7CoitI5U/eORQj51KrtIBtwQbDRA9CUWTQmm7Q6EAfDuyPsQYbwvqbl46GpbHaXLXMe04NUnxGxeDoMLw3S348iH5owfAp+NKRxK8QBc6cguIw9fwEzDixvaHzbMuby5Xxlm2O0OtZ3EE/M7Q721rZ2y9mfGksDmIYlYMO8DyrHppFXwoBr7ZSGd3R4CGg==,iv:7w9YqC3Dt7MMnEe0jFRsIhhbU+vMdmxpLM4RPL4viHY=,tag:3Tuccvume125sFgXoqXTRw==,type:str]
|
||||||
sshconfig: ENC[AES256_GCM,data:1EY0vnJ6NHb/GODYkiOKVR6i0UoojFPileQW1/ZokxDswGnz6P0eicS72wkMjwsw9tY5l0OM0NNbTIyYkqgE+q0HJw8IkRHvmvBq4AX/6o+2++YJuDTMpCcDkMAp4BNtrzqpBkKaqXJgMbxTpEPRNTnSpg96sinluCtuASEZaI8Ej+QKq/pEeELIGnkYkUr9d8/M8ZavzZ59oQmUsOxOCtYBsCi5BHMpATfVg+WqyXv3uptMB2PoTwsJiJ9IiQTJHWgkpe+WpIIsi4Dc49YpX3O2pHytimfuuy8D5DESKTOtcqg8qIL/xcR8iJl8ISJAA9IZg8eJ1FYdEpfTruwfiPYo6Ce5zH4W/BVm3iQ8B3Tuzi+gvRBwffcRpYkdAS4c7tk7cgF3U7sr1L7yLL6OmuBXaNSnALyHO2W5b3tefGY0YmwGUfh6M5FMI8/5SfxcMfSOAZKxhcckRL1CpFMzmKkAdY/PRCghmEWobU1KswBPiYWQ6vfubbdN1Yx1mng4z6zmkXmiqrc5IQfOBpbY71VnUoXh384Et9PM3cti4K6SmZFYX1uZStP7KXLhw4ZnVZ0J0fIks4RZ11A3NWtE1yak9z7QXj7k/PtdfrsNtIphVZ7LsZEvOvdgIlUToYNcyvagYsRRlreQPqhLSc7ninjIbRaL4a/s6kIdrr3qovPNSlE3R3HRGTBOjTEF9Bmp+oGb5L9HvBBu6NmNwHQ76gySAXbMnRa34JuXtzpEIZ5yo4AI5i2eDPeM2Dx6jUSl7OJm3wT6cVMF/2rLj0zHzp8B1bYOsf7VaMzHCYEhXMu7REMmmfDKwtb9zsuf/r1zlPXJcukDCrtJ3Z/mFl03,iv:0hS7g9gVVntWVSRSPCgZ3rjGcMf2RzQPjMpgSz0GlxE=,tag:mnDakNZiG28H4jmnZH00eg==,type:str]
|
|
||||||
github-key: ENC[AES256_GCM,data:xUgFMlBo8e+3eXqNscxbby1dWug3SgUagDiNUe/IGVbU6cczkaJ3uOaB0OuRBQ8AYhOLkzXj0pIKjUrElHwmYrhURtS1aF4SFEGJsjhhobNA//j3E2/5/nLVjfco+lRzyHdwmsNhEUCqEhsXrrodJMb39H4b5oip3z0rjc729YveiWUKQxXVZVPurp3nq9yNnix9R4CA6XYFRW6T6MNqgPD5qhbcDlhxLb/SN+uI5h+5eZIS24VDWlKaTaCLL5KLhZmfuA37SquOQ+edi9Yg8MnfrZkMrp/3qmAjP2rSQLMOc4QdQCLQBQSf0/snpydgLwY+FoJmMSztwtkqUdIZWOfDUJbJxegEOrAR68jLTNbp+GYiDn3thtOZDiK5p/M1amjCT+A9qeFMed5WS+aZHNTRbR9UcfiP6+48MGZFt1mr7q+/CoL04/DTp0w6tUf6/SZD31NvTJDqngkhpc0ZH9Dh5+2JcnBWpq25AM36kZTn1hIQCLNTr/oGWZXSLA2tksAhQCaUcFj4IIh6Bl96,iv:GEJsAs5NriwENYTV/VShgJF6iMmrtTwNiXOvfXyEP8E=,tag:nfZgsFqaet075GjQAoVZxQ==,type:str]
|
github-key: ENC[AES256_GCM,data:xUgFMlBo8e+3eXqNscxbby1dWug3SgUagDiNUe/IGVbU6cczkaJ3uOaB0OuRBQ8AYhOLkzXj0pIKjUrElHwmYrhURtS1aF4SFEGJsjhhobNA//j3E2/5/nLVjfco+lRzyHdwmsNhEUCqEhsXrrodJMb39H4b5oip3z0rjc729YveiWUKQxXVZVPurp3nq9yNnix9R4CA6XYFRW6T6MNqgPD5qhbcDlhxLb/SN+uI5h+5eZIS24VDWlKaTaCLL5KLhZmfuA37SquOQ+edi9Yg8MnfrZkMrp/3qmAjP2rSQLMOc4QdQCLQBQSf0/snpydgLwY+FoJmMSztwtkqUdIZWOfDUJbJxegEOrAR68jLTNbp+GYiDn3thtOZDiK5p/M1amjCT+A9qeFMed5WS+aZHNTRbR9UcfiP6+48MGZFt1mr7q+/CoL04/DTp0w6tUf6/SZD31NvTJDqngkhpc0ZH9Dh5+2JcnBWpq25AM36kZTn1hIQCLNTr/oGWZXSLA2tksAhQCaUcFj4IIh6Bl96,iv:GEJsAs5NriwENYTV/VShgJF6iMmrtTwNiXOvfXyEP8E=,tag:nfZgsFqaet075GjQAoVZxQ==,type:str]
|
||||||
gitlab-key: ENC[AES256_GCM,data:6271Tg7LIJsXAw7Co32vva+iUWv+eRAfVfXaGkoirHxiPTAgZfq8yDpGJwti2D/aFnLvbURcUgp3B5zy1aURu9puXi6QMu3orJOE2Zx0dX3lyeHyIhOZ1hAWsQxwBDkJYODuMZbIKHNrF26Mkk6smFF9SrGoADZGxY7XSou/iopw6yrLvQAXP6lN55PQvoP5/ek/QzKNwjJAbAjp7FjjtPqPUbUdj7bfIfr9avs6gpRYnRzb2H3auyhb5qvyuRWnKu4+FeZUofC7HnfRU1eSI8UzG49Rg4f2b1Xqr0JtsjvEmuRx7lbptINXEAjW7zaSZ2B9NgotWPIzkNZRNZkuTEaOulBf5iH3wJ9GM1FcbUu2gpCt6y+CMIk5QkVvETYtKuALLbcPx9/sc90doiH1411LeKQCs5l88BBILG9KPkKmXiGGUVNxzFtjbRYQIUEmmyl95kPPEdCkkzGzHXNfE3F9HRu4MCGKPK2yZPZtCXd3Jp1b21wKSIRNYqLqxru+J8eLSljd3czO0SuTG/Vx,iv:E+y3aVLaiStlPLTQBqyfJuEXtAQyOvDUYS+XfuOHuA0=,tag:6d1IAsGcmcpgL1pE9dDcbw==,type:str]
|
|
||||||
jack-key: ENC[AES256_GCM,data:VfCl3wH0MMBc8QDyjLDFeSvzSEsf7uGpfJvRjFrmjW+bPRUXBpZhJV8a9VQIAz7z7zZXvzARMfCeI0ydyC57CW81GH5/H5pneJ4b+xreINjVfdLbL1nC1thelo/O64jda/L+xVKhgE+QQi8/zt4JmXGghkP+74nYcTTaMpmcbgWw354J1ybXqyCEY+88nsJ1d2s+M7M2bplx4fGb7sLUs6sqdsad3sENzhH/0HQCFXreHTtgsLbIs8ccmdRgFNKM8/wD0OoW76rOQsJoA9JY4yOTQNVoX5M8+Olj6+wVlt6QBrWrYRuEztGnHrHvzxiHXtmEkMwVNfoPpEflQyRYRa0rVp/66REOkMckGx6/LbxKFgrxnifRlsK3kWd28v2bRGVQOghUluYUtVkaJ+eh6o6ik0NQKx8/H6BznBSDE6MjDwbLv434LHBfDtAqhWN1eMbOlunFivsl5Hb/6rl9kydHlcCS6FY8cUHoKQ90gDaUuDrvUifwmdO5hU0GH5tgvGi1ReK9ndcpQsrHptG6,iv:oC1xU5Tu3The105VYRmxIw4kEwDoqe8T/EH6mmqpqwQ=,tag:Pu8c536u6W7ALrqjRsvXDw==,type:str]
|
jack-key: ENC[AES256_GCM,data:VfCl3wH0MMBc8QDyjLDFeSvzSEsf7uGpfJvRjFrmjW+bPRUXBpZhJV8a9VQIAz7z7zZXvzARMfCeI0ydyC57CW81GH5/H5pneJ4b+xreINjVfdLbL1nC1thelo/O64jda/L+xVKhgE+QQi8/zt4JmXGghkP+74nYcTTaMpmcbgWw354J1ybXqyCEY+88nsJ1d2s+M7M2bplx4fGb7sLUs6sqdsad3sENzhH/0HQCFXreHTtgsLbIs8ccmdRgFNKM8/wD0OoW76rOQsJoA9JY4yOTQNVoX5M8+Olj6+wVlt6QBrWrYRuEztGnHrHvzxiHXtmEkMwVNfoPpEflQyRYRa0rVp/66REOkMckGx6/LbxKFgrxnifRlsK3kWd28v2bRGVQOghUluYUtVkaJ+eh6o6ik0NQKx8/H6BznBSDE6MjDwbLv434LHBfDtAqhWN1eMbOlunFivsl5Hb/6rl9kydHlcCS6FY8cUHoKQ90gDaUuDrvUifwmdO5hU0GH5tgvGi1ReK9ndcpQsrHptG6,iv:oC1xU5Tu3The105VYRmxIw4kEwDoqe8T/EH6mmqpqwQ=,tag:Pu8c536u6W7ALrqjRsvXDw==,type:str]
|
||||||
signing-key: ENC[AES256_GCM,data:NvLqmt7NzrWkbQQqFfosmSMvEv8C8+MDDpxSoDo3zZ3MR6WXr9B+6CnUc6rtevM230wgE17VC9XlmcQxX+PjJsWq6gZteK5THTIcrR5zPJVNlVCEyeLKoFj/6m7qBgyyoN3OjCNjgMkhsm009jwBgNk0qJMAYebOGo8eoP/al+4ytm7dhna+iX5WZabAg0J4z4JMDQonqQl3SaDnCEdHvk9m7ZEP2vUFscPkbLj4ewgFq+bUCHOOQb2uqRJoEgcR/NetRFcQfzcU6lp7JxobaICaO3zdmomUm7oabIUTrc3Kom3Wjw8ryqfqC1/SC0SHr5XGk2ygk0WnlQ4kNshqriL2dwbWAzy1Z/cTX9+aB/KNtC0U6zWG1bpnL3dgSgvhRiocIis/eNg1HWsLWVicmdebL/lXHztxFMdTuX/gWgQBotggTmx+OFGZfP9ZGlF3635mDxM/fEOCtTyA195dMicCUsjEpBegNtXsp+oOtxyRI0jSW47MBcXFP09f5ywELTPyz4eOUHO8sQ6UfJ6X,iv:KNQUlMPaiR4x1Fw+HZe/EOmh/gfsrqlefpq23uflz/8=,tag:illu42HKunQXnijjsUIvnQ==,type:str]
|
signing-key: ENC[AES256_GCM,data:NvLqmt7NzrWkbQQqFfosmSMvEv8C8+MDDpxSoDo3zZ3MR6WXr9B+6CnUc6rtevM230wgE17VC9XlmcQxX+PjJsWq6gZteK5THTIcrR5zPJVNlVCEyeLKoFj/6m7qBgyyoN3OjCNjgMkhsm009jwBgNk0qJMAYebOGo8eoP/al+4ytm7dhna+iX5WZabAg0J4z4JMDQonqQl3SaDnCEdHvk9m7ZEP2vUFscPkbLj4ewgFq+bUCHOOQb2uqRJoEgcR/NetRFcQfzcU6lp7JxobaICaO3zdmomUm7oabIUTrc3Kom3Wjw8ryqfqC1/SC0SHr5XGk2ygk0WnlQ4kNshqriL2dwbWAzy1Z/cTX9+aB/KNtC0U6zWG1bpnL3dgSgvhRiocIis/eNg1HWsLWVicmdebL/lXHztxFMdTuX/gWgQBotggTmx+OFGZfP9ZGlF3635mDxM/fEOCtTyA195dMicCUsjEpBegNtXsp+oOtxyRI0jSW47MBcXFP09f5ywELTPyz4eOUHO8sQ6UfJ6X,iv:KNQUlMPaiR4x1Fw+HZe/EOmh/gfsrqlefpq23uflz/8=,tag:illu42HKunQXnijjsUIvnQ==,type:str]
|
||||||
signing-pub-key: ENC[AES256_GCM,data:XuokZmCsnaNQ0rvVa3k81T4vtxw//r63xp1yHDLaNAMwA2r2bh6addl9WoAmm76g1rweqZrAAIw1PXDb90ubPaBP1iEHSkGZpwUpy/tOTePRdHMW2WtVvidpHQ==,iv:auB/bA89cJK6DnQi1BK2uldXRPyJfo+r7nl5qOLefUk=,tag:/I3kB6El1yesBMGOjJ+oHA==,type:str]
|
signing-pub-key: ENC[AES256_GCM,data:XuokZmCsnaNQ0rvVa3k81T4vtxw//r63xp1yHDLaNAMwA2r2bh6addl9WoAmm76g1rweqZrAAIw1PXDb90ubPaBP1iEHSkGZpwUpy/tOTePRdHMW2WtVvidpHQ==,iv:auB/bA89cJK6DnQi1BK2uldXRPyJfo+r7nl5qOLefUk=,tag:/I3kB6El1yesBMGOjJ+oHA==,type:str]
|
||||||
@@ -16,7 +14,7 @@ sops:
|
|||||||
cmVxVC8rVDlWMUdZaGw3bmdOaWZGS1EKYahBlc8XpB5UdKZQkvxbLcKQ/xkFJjWo
|
cmVxVC8rVDlWMUdZaGw3bmdOaWZGS1EKYahBlc8XpB5UdKZQkvxbLcKQ/xkFJjWo
|
||||||
FSfQWnjhe/a7BJtJEcKZkjOQU0mYqlSu+uT2RA9diCQeRUJPRF+nAw==
|
FSfQWnjhe/a7BJtJEcKZkjOQU0mYqlSu+uT2RA9diCQeRUJPRF+nAw==
|
||||||
-----END AGE ENCRYPTED FILE-----
|
-----END AGE ENCRYPTED FILE-----
|
||||||
lastmodified: "2025-09-27T13:28:57Z"
|
lastmodified: "2026-04-17T22:53:33Z"
|
||||||
mac: ENC[AES256_GCM,data:XMPDLGHwTYIxgEz9Stj7j+lWgAOpzkEsnoRdWVVs4798m1sQRIyUuvEiCgbHoAktajrAAzXq3GZ4HJ1dfE1fDWGh+B7WwRoJKxtP6qI8Ub4h7lSiDsxJhr8ieqm5bzmHZkn8VJkrqBrSSXesirLprRLR5yZOVLAgco0lg9boM1Y=,iv:pYA5oz8cldKw1Ai7k2LCQBipE2keZa49L3SHcL+eDp0=,tag:4kVsC95SSPCCSZDf6qDeqA==,type:str]
|
mac: ENC[AES256_GCM,data:dA/D1q3t+AA4M7UOSF8lhTtC71D56l6njhh/Oaxnro0zQYRTCHO+K3u43srVzTdwaCAtQtKRcpbrjANpO3sgb9iqZme5asfLFGqDr0bbWXir7BwDyZVgkw8XiYGPtpEYlgdwLwYxejpZtRlE7IeFCRkEmlvWJHq4R3SfbjKkOxE=,iv:0JHoWrQbjzW/gmj6HcBf+2O0cZeBC9kT76ue8ubJwaU=,tag:9fSVDwN2qgsre5yVr7DpMw==,type:str]
|
||||||
unencrypted_suffix: _unencrypted
|
unencrypted_suffix: _unencrypted
|
||||||
version: 3.10.2
|
version: 3.12.2
|
||||||
|
|||||||
@@ -5,11 +5,11 @@
|
|||||||
}: {
|
}: {
|
||||||
imports = [
|
imports = [
|
||||||
# Choose your theme here:
|
# Choose your theme here:
|
||||||
../../themes/rose-pine.nix
|
../../themes/zen.nix
|
||||||
];
|
];
|
||||||
|
|
||||||
config.var = {
|
config.var = {
|
||||||
hostname = "nixy";
|
hostname = "h-laptop";
|
||||||
username = "hadi";
|
username = "hadi";
|
||||||
configDirectory =
|
configDirectory =
|
||||||
"/home/"
|
"/home/"
|
||||||
|
|||||||
@@ -6,19 +6,27 @@
|
|||||||
../../nixos/systemd-boot.nix
|
../../nixos/systemd-boot.nix
|
||||||
../../nixos/users.nix
|
../../nixos/users.nix
|
||||||
../../nixos/utils.nix
|
../../nixos/utils.nix
|
||||||
../../nixos/docker.nix
|
../../nixos/amd-graphics.nix
|
||||||
../../nixos/tailscale.nix
|
|
||||||
|
|
||||||
# NixOS server modules
|
# NixOS server modules
|
||||||
../../server-modules/ssh.nix
|
../../server-modules/ssh.nix
|
||||||
# ../../server-modules/bitwarden.nix
|
|
||||||
../../server-modules/firewall.nix
|
../../server-modules/firewall.nix
|
||||||
../../server-modules/nginx.nix
|
../../server-modules/cloudflared.nix
|
||||||
../../server-modules/glance.nix
|
../../server-modules/glance
|
||||||
../../server-modules/adguardhome.nix
|
../../server-modules/adguardhome.nix
|
||||||
../../server-modules/arr.nix
|
../../server-modules/arr.nix
|
||||||
|
../../server-modules/blog.nix
|
||||||
|
../../server-modules/awesome-wallpapers.nix
|
||||||
|
../../server-modules/iknowyou.nix
|
||||||
|
../../server-modules/stirling-pdf.nix
|
||||||
|
../../server-modules/cyberchef.nix
|
||||||
|
../../server-modules/mazanoke.nix
|
||||||
|
../../server-modules/kernel-hardening.nix
|
||||||
|
../../server-modules/fail2ban.nix
|
||||||
|
../../server-modules/default-creds.nix
|
||||||
|
../../server-modules/umami.nix
|
||||||
|
../../server-modules/gitea.nix
|
||||||
../../server-modules/mealie.nix
|
../../server-modules/mealie.nix
|
||||||
../../server-modules/eleakxir.nix
|
|
||||||
|
|
||||||
# You should let those lines as is
|
# You should let those lines as is
|
||||||
./hardware-configuration.nix
|
./hardware-configuration.nix
|
||||||
|
|||||||
+5
-30
@@ -1,8 +1,4 @@
|
|||||||
{
|
{config, ...}: {
|
||||||
pkgs,
|
|
||||||
config,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
imports = [
|
||||||
# Mostly user-specific configuration
|
# Mostly user-specific configuration
|
||||||
./variables.nix
|
./variables.nix
|
||||||
@@ -10,39 +6,18 @@
|
|||||||
# Programs
|
# Programs
|
||||||
../../home/programs/nvf
|
../../home/programs/nvf
|
||||||
../../home/programs/shell
|
../../home/programs/shell
|
||||||
../../home/programs/fetch
|
|
||||||
../../home/programs/git
|
../../home/programs/git
|
||||||
../../home/programs/lazygit
|
../../home/programs/git/lazygit.nix
|
||||||
|
../../home/programs/nixy
|
||||||
|
../../home/programs/nix-utils
|
||||||
|
|
||||||
# Scripts
|
../../home/programs/group/dev.nix
|
||||||
../../home/scripts # All scripts
|
|
||||||
];
|
];
|
||||||
|
|
||||||
home = {
|
home = {
|
||||||
inherit (config.var) username;
|
inherit (config.var) username;
|
||||||
homeDirectory = "/home/" + config.var.username;
|
homeDirectory = "/home/" + config.var.username;
|
||||||
|
|
||||||
packages = with pkgs; [
|
|
||||||
# Dev
|
|
||||||
go
|
|
||||||
nodejs
|
|
||||||
python3
|
|
||||||
jq
|
|
||||||
just
|
|
||||||
pnpm
|
|
||||||
wireguard-tools
|
|
||||||
duckdb
|
|
||||||
|
|
||||||
# Utils
|
|
||||||
zip
|
|
||||||
unzip
|
|
||||||
optipng
|
|
||||||
pfetch
|
|
||||||
btop
|
|
||||||
fastfetch
|
|
||||||
tailscale
|
|
||||||
];
|
|
||||||
|
|
||||||
# Don't touch this
|
# Don't touch this
|
||||||
stateVersion = "24.05";
|
stateVersion = "24.05";
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,13 +1,12 @@
|
|||||||
|
umami-secret: ENC[AES256_GCM,data:tImpd4sD92Omf/YFB8YE4gxAu+g801wQNR+k5rhY6AbzIIYOzpVmQL4XGjfp9Teky3olii4s3XTcmTyuMoxMWg==,iv:QFAEzYnAnxOOtrHWiM2IkvSs0Aqk3s1T5X7j5WC+tO8=,tag:FIbgHLfRVMJ2qZ6dOJ8zLw==,type:str]
|
||||||
sshconfig: ENC[AES256_GCM,data:R54HVxqAyj9yGO/AYL8p6cnXgYxkQKW9XveHlBMTnDXBJ7r/4HgnefdymprnXmdlbNWcWrRqmaLEuzJs/0BfixXfMvmGTUrmJ0ASVuDrz9k6rOLADAKFikQh0dib7NU4JmPgmUzMncXc2WuCd3BCG3kwBQ==,iv:Ro9FA+MzTAp+ERQMT88z8ioCox/dTj2vWcqCDOSLag4=,tag:5XiXIyz5/pjGFOB5ZjdOVg==,type:str]
|
sshconfig: ENC[AES256_GCM,data:R54HVxqAyj9yGO/AYL8p6cnXgYxkQKW9XveHlBMTnDXBJ7r/4HgnefdymprnXmdlbNWcWrRqmaLEuzJs/0BfixXfMvmGTUrmJ0ASVuDrz9k6rOLADAKFikQh0dib7NU4JmPgmUzMncXc2WuCd3BCG3kwBQ==,iv:Ro9FA+MzTAp+ERQMT88z8ioCox/dTj2vWcqCDOSLag4=,tag:5XiXIyz5/pjGFOB5ZjdOVg==,type:str]
|
||||||
github-key: ENC[AES256_GCM,data:NRYhcBIwGJEV13+YECLR+2IErsn/7clbnkx0Mltr7dQajSb5WHZ3QDH0KQPylEHhplE5IVS0h4I0z+Pb1B0UteCxFmJ5wZq+2BKZkvE7G3dojqBpgHcVqJV2GLEJkRjlHfRgsbq/OBe8xcsPh20P1KUyP0WIwVbpt+9dFWGxEGYkp2uSyuBIJ98kElt0zuVgl7WcYoDO7v5WmGzZfla+yZwURvMk8zcM3gopo+4KL6YnYUs+UA3VlBBn6VK4Nvbqy6X0R0+ZA5HHAXg+OFgGmfWnENZmsyQJHXEchGGgEldzThkQ4r8yMkgN/ax+AGouLyzbITapGE4sE11FFgL6Hmp4pSXxl3UAGF+cvV5pIujbb28CXmSPRMyYpoNxI93PSYz/txAzE6Cr2dgwxR4zpMelv4i6IaGnY8NgpY8jp2Y6C0uuJxJCN0RtnjQw1rM2uRnm7vMGyU7XXz9DEVfGnYpTWnykXsEjHE5DVGy80ejYQlc6dtmf3vdTWpt+YYdCPw8/cd0PIx2D6geh1c28,iv:wl+RG24mXYMklD8CBGXVD36DMhlWT/7zh8ZMvr7vgOk=,tag:OJhqF8PoXotr7IsyFW6q1g==,type:str]
|
github-key: ENC[AES256_GCM,data:NRYhcBIwGJEV13+YECLR+2IErsn/7clbnkx0Mltr7dQajSb5WHZ3QDH0KQPylEHhplE5IVS0h4I0z+Pb1B0UteCxFmJ5wZq+2BKZkvE7G3dojqBpgHcVqJV2GLEJkRjlHfRgsbq/OBe8xcsPh20P1KUyP0WIwVbpt+9dFWGxEGYkp2uSyuBIJ98kElt0zuVgl7WcYoDO7v5WmGzZfla+yZwURvMk8zcM3gopo+4KL6YnYUs+UA3VlBBn6VK4Nvbqy6X0R0+ZA5HHAXg+OFgGmfWnENZmsyQJHXEchGGgEldzThkQ4r8yMkgN/ax+AGouLyzbITapGE4sE11FFgL6Hmp4pSXxl3UAGF+cvV5pIujbb28CXmSPRMyYpoNxI93PSYz/txAzE6Cr2dgwxR4zpMelv4i6IaGnY8NgpY8jp2Y6C0uuJxJCN0RtnjQw1rM2uRnm7vMGyU7XXz9DEVfGnYpTWnykXsEjHE5DVGy80ejYQlc6dtmf3vdTWpt+YYdCPw8/cd0PIx2D6geh1c28,iv:wl+RG24mXYMklD8CBGXVD36DMhlWT/7zh8ZMvr7vgOk=,tag:OJhqF8PoXotr7IsyFW6q1g==,type:str]
|
||||||
cloudflare-dns-token: ENC[AES256_GCM,data:JIXUtVDpYS9B74W0ooj50kd4v2+PX+FdF218gvgaS04rYATu4N6w9KEFPfdQqLjUmVihzV6s/IR1fg==,iv:AlbQ86kvFQbetvmFwt/hEyUcqKTI+XzL/NvSMXW9wm4=,tag:1JQeK4KrTDa6Kz+JhWxkxQ==,type:str]
|
adguard-pwd: ENC[AES256_GCM,data:CAmZFpKN6FFRlCk=,iv:cz45T/z2ZYHNC46FNVp2YHmdhXD4ERhM31+Jo+9PkSk=,tag:GTUvzkYSP4flXs/HUNGYhw==,type:str]
|
||||||
nextcloud-pwd: ENC[AES256_GCM,data:2oqsNceKuwGscBN2VxAK,iv:FoSfHItgeB91fG38zqtuQzayvNjNPFQyZjZlpUq/eic=,tag:MoEpAmTj+zqVu6OLasD4kQ==,type:str]
|
recyclarr: ENC[AES256_GCM,data:eJNbGHb4SZvbF7FAHMrpRynmpVyPiTpyZqNx40Fo+lanqdm7d7oRhkLD2PfqxpTcIyV6BxJk12wGUC0uWVuAihmToL/Ih2FJWIQFUOdbaov/xNWHE9mtoRX/A3gIwD8e3DoFMK/dAk2/TyvvAfxi2eYTDJvxecEttHQGxafq1jveHFDcC9e1aFk3M8O8YlX5yF3zl90mugsUVP+iaSURnY/nglZuEDzfJ6Edge7r5OSL6WYh3OUIu1yzuuKXRZ47B/gpXO/cP89JMmxdPj3FUJkk7Htf0s9ALzpVE9fRohDGcpPIkR6COCLTL+mqvVmyopA1zPoX8/X8eiuocpaT3vqJit7Bc6InzflY/5rlg1x0SyoXWQfEsperfNBeiZGngp382I7rrZ8QSwF4AIsmngNeyDRAnfasAU5OVwK+yPIVhRueEFLNrJPG3j2hpArlzpvY2EPBVJDrYwFslis8CASoUdvu6C7VxtJXFcOZR8QemwJarJFU7v1bVhQoIkarn3V7FfAXg/8RfYrn9HAIq5AiFFbQRNHWXzIceIgNKQFb5yfagtMtMq6XKhjsV07jeYt9X/YU80XRfkPm05yiLdLQHcDNmlB3R+SFPD7qjTE7LuSH1Kdc3DvxwrKfHJLsJ+nSIL1po8gDMW8BPc0g4n4dpQ6CFK5BaDFK4eovNpgAXoRC9orkGw/16YmMozp6J78VJLzu3jcYMFbySFznDUuWcFf/7dVV+7pLAj2ffWXzScLISRBhZ9RNUMaZAKAk7WTmxBhw0yzNiih8OFUkbpp1FFLuU0bnBY0iEz+8/OSwv8H7t0HYe29HO61p1huH34B/ZkPA5VLKivFh2mOPNzbvMqtIAkq1ro82Psvc6t88gVIc+VAOZ9zDLohp0FXrNVXwnrs7fxevQa0pUClDivO1ib5G4Znor7q+13jZglmh3G7nHiniO607dRoxszzVEQdTn8ku3XAfIPStsSfQwH46nXhr2kDkspcA7BqwJph0dsKKlfzf22C0OJkDY0L1DgKFvJRmmKAhOdB6XxXYwS+BcDT+8ugejTKpulyPSsbdb0m9p9lVAVu0GMxFEbUsuE+QHiAAP5Vle7swpl4h3i8uHSfKFra3qEswwRnPAHhPSdLSaSU6O6T5gN/b0hhO6dAiNR0UOkOG59fPasL0MRpeb3SkM4Nd6g5b/hY8dXaXR7pQHvh+4GmfJFWYaWm/cZ8k9JAYAw9Sn/spWnjRmpv9bnSxADGbkdV8NDHquDDDFVIIoDnWhQJpqrL61KZITx2Ete1SYdNiy++p3mOYisJgHDsRR1RNQNxhRW4EY8c5nCzgc8x+rU432pCOfMbAuGEfQXMwMGFGeoUg6EL5xEyUEJ/aqymxI+Vjk8h7KfDnOyyL/+1BKEd/GccrjeQfeZGRLI0Tu5gVgL6twc+u4av6inNMMhPFzZuVrfeo6bip5C8R0dK6Pbdz2L/fsgn3E4NNi14NM/DgLLhJRLvKQdCQx+BT+3soNRw5htTzuNS48MD2YDwZYvNfu6mo0r4j4I7Tj8zuCjddbGVa2/5XnhcfdUBx1VOd90H1RKkk8kgSoR0bgo7mLls=,iv:HT3a2YnFy0cF9qUO671kwqxuzFfNnfqynCUVGHf+3dM=,tag:nrA++7fsnc4Cx0OPSDKaRw==,type:str]
|
||||||
adguard-pwd: ENC[AES256_GCM,data:QavwLWENAURnRrFwiLntkiM=,iv:bxdQfBxNL5rwUr7CEKbwXtv5mUUXZHhvyqQL2KoPwEY=,tag:T+cSyzbGeo7E5smSsuFlHw==,type:str]
|
|
||||||
hoarder: ENC[AES256_GCM,data:8A3eGqIlHJ2XpC2OdMNBXPm+5BdfMlOfTSgiibPtM+SFyiPtGhjWQNmVCD9REf0P2C4pikZ3R7vtwyKpjrraaoSAY7ztAk9eqqikorIzD8hn8wbHz/y+Eko=,iv:ngoVgF348IxokWGQVpbpTGhdIwjOOA6T8qLb1wX6GEU=,tag:+v9HLUksQJ1e2vRR/5fzEg==,type:str]
|
|
||||||
recyclarr: ENC[AES256_GCM,data:3rZgs4Z/XaQPxbueepPQlUthHMSKn1e92FyIOpzn1MsGmEL8LBSUJvWp903n1BXDI+SK4ph9DaxOFavPkfCnaNMascX13TkGf03zGTEbd3xDZCL5GoXl3yobt0EMRYyRlYu5C1gidj9CjTujIaEXsF1AT2x0xtYweuC1xhpMhq2gj8eBy06sGnyKV/9Kdxo3H80skHVH6ZokndDyv6LfpkCDKQFsSBsEKBfVBDws7IgGFXCv6toWJPNgidReWzMmwKiClkviWAJiqAvI2IABRe4rEZ3Gkr3XFuoxOX2yfvQ1O7PH6O0dXvtfyBOTpGTaZihOqM1WhIutSr9SWX7QivIwtsVMGEmAmKx8JyDrnjhbCCbzjG+4Y6YoHawHYFFwKyOCxtJcsLBFvvkiuUt2ZVUVYCwhEIp3Xk2xb4cxYOIYRStxhiTuhEb8a9PkvkjmQQ4eAMB8bNPlo+vGqAQGPc6FOhm67dh3JbdQpsr9/U+P/AGsJM60SAhx+OzBcDSIS1U6/qJ6udXPOw58wXhJMD+F+gqV7blVBkXITQJ7AaLl/aWmDlkoqkjFVTIjE980e6ap6BuNbRGsN9x3EiVwqfDnqnQ2udGiLOcu/WguLzNjCvXWILoSYWwpEMwEW3SWASbGgk0wUpwZfNKY1HagdcVFO17OW9DrlTuGuJs4eJgVXm6e3glS3CS0dCL8FRs0EH4Rh8mUzhf9VP0/GlkN0kxJEM8PnjqGl0JMnaeQsSr9FDJdqv/2z4hjQBeq8//ER1BPIRSbXNQqZv5HcumSZs+JEttYUgInWL+LZhQrtpkba3sKQGkQF6X1kGpPZmiVSRgM+7PAt3ezpUKS8nMTS880l03cSaxMm+vebPhYtGC1O+4XLhv7E8MPULMkCOQoNLNcQCqKU82fa3vEtS28X6+izRMi56vbzG6Kjls85/NMGVz6rDySnQHMP5vSapcszSHwqGeAXJKfXYPzXQo1SIy0RlvRbllPt7zfOkTDMr52oH/pbtuxLSrSDe4ZjHkmdDzO+t0yAyG9xsvzldYUbDkfAWbnFXay3TICi5JOGe/Otp62sv3/7fjSmKm+9Um5AFj5F5Zo58LKCrH8qy2omIh80uFW0/yNgbuXmA4GqnDmYWaWlccc7yhPsmdAWHIypNS7GxRyW66+d4P7mhccvKGT7mx/gKuHgdUIRjy8bKtjLN2vuKfIiJR6bqWZGtbNEqY4XPfhdYdGdnox8vdrT/8DiR+6Jq+3qvJswTXhg+wYzOVTNjl8qK215VA2/FF25+T8xmSfV+cnPrLIrOrD8t1D34U4eEMrVzx+6dQ2dN1qDwj5o5GSYuYSQ5o5hy2BKt8PPVDI564MzjPfLaAf8cLK+0PKsbi2E4jGRI5BejmpMWHoLAUJvX4kdajd8Ggh5wZFS07pje3T1adzeS/UUnpy/IeLnEz8FfoH9ro7Qz0XGmtItVSjumppBg6jwbf1se9sP13pHfIh/SvVJhOOH6KiFs0lECSetuhIupsqHuJg6ZLa+V8GO9otYiJGZnOKGiZNrBYSek9l4Ncn/2Q+dqOYaCkZ8Xx6gY+qKHgiwavbsYYiJUPK3DLNhTzH,iv:00efefMekG2TsnDH3yNNdMBIITDyxo/qKN85qVGA6f8=,tag:hmwsyJRiT1MbFPKm4Kt5lw==,type:str]
|
|
||||||
wireguard-pia: ENC[AES256_GCM,data:2IvJARGhesMuH9RdWzsyrwA7eqrhLyacQqZ1RNEkGOPUkQGX4uimKBSzkxXRy/haZ4V2k73JdLSaB9rAuI0n65GmWHmarwZekOyhRZSNb+zvFgw5BPZmywG1wR2HiTGR/qILovAaz47q/VnohUnjbbMCUvarC4PytWGxMUH96GIgZar8HjHFtK8grCSxlvpHKiDeKx8VSXnY/Pxj1EplBtIqwmtAeZdf/VjtwOL0nY54doPwHdIAvJ0B8Cu0a1zJIGEbV1NlKIHEJ1YA7rmv1ODkBnbXbIHMxAR3jeqR/UDqhDmXe41KujhiJI7nNeO7FKo2v92jK3fSbxYKatLrzXktHpE9JsMYVBXzTK7yAXPgoDdgLXzWH0OrJGBSisPrvqmxUko7MPreuwVYfFlKpll6JLifk8sML4A+94UPR8b89guXn7kBkLg1Y1oIAyguCdKpNOD31nXBMFF0nTcmCwyshDySaGTfJDgox65/77AiN1wH,iv:cdu6lBjLnEEfSFmWMC4Vn2sLKsvpCaatzXlgRNkEMeA=,tag:y1rAeNPB+DNGTpnP94iQrA==,type:str]
|
wireguard-pia: ENC[AES256_GCM,data:2IvJARGhesMuH9RdWzsyrwA7eqrhLyacQqZ1RNEkGOPUkQGX4uimKBSzkxXRy/haZ4V2k73JdLSaB9rAuI0n65GmWHmarwZekOyhRZSNb+zvFgw5BPZmywG1wR2HiTGR/qILovAaz47q/VnohUnjbbMCUvarC4PytWGxMUH96GIgZar8HjHFtK8grCSxlvpHKiDeKx8VSXnY/Pxj1EplBtIqwmtAeZdf/VjtwOL0nY54doPwHdIAvJ0B8Cu0a1zJIGEbV1NlKIHEJ1YA7rmv1ODkBnbXbIHMxAR3jeqR/UDqhDmXe41KujhiJI7nNeO7FKo2v92jK3fSbxYKatLrzXktHpE9JsMYVBXzTK7yAXPgoDdgLXzWH0OrJGBSisPrvqmxUko7MPreuwVYfFlKpll6JLifk8sML4A+94UPR8b89guXn7kBkLg1Y1oIAyguCdKpNOD31nXBMFF0nTcmCwyshDySaGTfJDgox65/77AiN1wH,iv:cdu6lBjLnEEfSFmWMC4Vn2sLKsvpCaatzXlgRNkEMeA=,tag:y1rAeNPB+DNGTpnP94iQrA==,type:str]
|
||||||
signing-key: ENC[AES256_GCM,data:FrJzuTgH/ooZkcnYL55uQcc4u+QzNnFvNVs2wDSE4nnwku+EuCJBlb8pd/6W0KPwIXzcki/8CY0YfRcRrzjExMgMa4hwxrlxS9bk3LNPzJsrRK5RJgPg3iA8L791f1zcDxNf8RuWatIqm1TCK+Vhdk/p+221zy1Gcq1dW8X+o4XzbPBzHdLagcIdB0wpjYTtIoGP2X8GoL/NJpuzIiQBK1HdGNKvUI2+ztqCQZOsxm+Fki57NteX3/Llw8AwABjdZvviOBZ62OvJ/SsOQ9NYAvKfAkog5zCn8DLvaqAPGSxRBQEYWM8GyL2imgs54YfEsOpGa4DzMiv4Sc5m398E/asaPq357eksUqh3EYpzoKZ5bIbd5+Vs3KBWKHltUCzXLHaLrIX0CuJFQFi7DCxEbYqlb04x5t3jc/c+/7uwqBHv1Y5gwAjd8JswDWmE7Q3xSk96Za57SCxWPYTo2ErsA2XdL+yxXdhmqkhDZKtUzrcHExhnYe7YLpSlBEclJ/G2BeTOFIWoAmN+1y4rh21R,iv:VaZrv5/41ZyIax702Yae4QmFKpcEaWwPmTo2Mxao3bU=,tag:HC0eqDNit7jQKeeDAKWXKg==,type:str]
|
signing-key: ENC[AES256_GCM,data:FrJzuTgH/ooZkcnYL55uQcc4u+QzNnFvNVs2wDSE4nnwku+EuCJBlb8pd/6W0KPwIXzcki/8CY0YfRcRrzjExMgMa4hwxrlxS9bk3LNPzJsrRK5RJgPg3iA8L791f1zcDxNf8RuWatIqm1TCK+Vhdk/p+221zy1Gcq1dW8X+o4XzbPBzHdLagcIdB0wpjYTtIoGP2X8GoL/NJpuzIiQBK1HdGNKvUI2+ztqCQZOsxm+Fki57NteX3/Llw8AwABjdZvviOBZ62OvJ/SsOQ9NYAvKfAkog5zCn8DLvaqAPGSxRBQEYWM8GyL2imgs54YfEsOpGa4DzMiv4Sc5m398E/asaPq357eksUqh3EYpzoKZ5bIbd5+Vs3KBWKHltUCzXLHaLrIX0CuJFQFi7DCxEbYqlb04x5t3jc/c+/7uwqBHv1Y5gwAjd8JswDWmE7Q3xSk96Za57SCxWPYTo2ErsA2XdL+yxXdhmqkhDZKtUzrcHExhnYe7YLpSlBEclJ/G2BeTOFIWoAmN+1y4rh21R,iv:VaZrv5/41ZyIax702Yae4QmFKpcEaWwPmTo2Mxao3bU=,tag:HC0eqDNit7jQKeeDAKWXKg==,type:str]
|
||||||
signing-pub-key: ENC[AES256_GCM,data:CB7uU2Q4oTEKihpTIXGLaV0fJ1cv/p4oJJ5kjaU6BZiKhsiMA1JILUw2oVIDTDb+80WPzolDzZwWM8v31d5QIrZpHcPrdRLyV0X2USfG9U4aQ/ls79QAyOOJXA==,iv:/Eb5/+p86tw3tqNiDVHGu7HS1KBtFiYIgasRYJsAiEo=,tag:dGdJlcrnuU73s+IMQ3w3hA==,type:str]
|
signing-pub-key: ENC[AES256_GCM,data:CB7uU2Q4oTEKihpTIXGLaV0fJ1cv/p4oJJ5kjaU6BZiKhsiMA1JILUw2oVIDTDb+80WPzolDzZwWM8v31d5QIrZpHcPrdRLyV0X2USfG9U4aQ/ls79QAyOOJXA==,iv:/Eb5/+p86tw3tqNiDVHGu7HS1KBtFiYIgasRYJsAiEo=,tag:dGdJlcrnuU73s+IMQ3w3hA==,type:str]
|
||||||
|
cloudflared-token: ENC[AES256_GCM,data:7DwWFL6JbJXuc6fSyrZkMt/+DAcqDYBQOzJ1QhBL4yfCpQ57mjDV2beRZ8H7SuyoKY3caifHfAVVPDGG1YnffZ9anLRyDW383aHKmgaeztOXj44eVQcHi3evnSiH6h0HasiyO8pNzhnKpYew8CwHSvYZMraX3ITz8UKOTtsanO9/8ZD6Y9gyxfLoySqbeBy2x/0Ql5hrDfrX854C+vYuXM9VJbtQqMBa7+IBydUvddk=,iv:jZTk5ptuwc+1aP2rimqGnyCHR4/J+W4Kyn3LKXXkv/I=,tag:xr/LhoNH5aWdIJr0Py6nAg==,type:str]
|
||||||
sops:
|
sops:
|
||||||
age:
|
age:
|
||||||
- recipient: age12yvtj49pfh3fqzqflscm0ek4yzrjhr6cqhn7x89gdxnlykq0xudq5c7334
|
- recipient: age12yvtj49pfh3fqzqflscm0ek4yzrjhr6cqhn7x89gdxnlykq0xudq5c7334
|
||||||
@@ -19,7 +18,7 @@ sops:
|
|||||||
TEc5d01RaVFGNXc3dlljM0FTTHpENjQKOqwI+pl8UxVIVl43glnOYvW660/PsDGY
|
TEc5d01RaVFGNXc3dlljM0FTTHpENjQKOqwI+pl8UxVIVl43glnOYvW660/PsDGY
|
||||||
yefODJGVtHrOm3yeXC2xlTi3sFW+c5wUl2yPqddbvcBt5Ud/yd4iXQ==
|
yefODJGVtHrOm3yeXC2xlTi3sFW+c5wUl2yPqddbvcBt5Ud/yd4iXQ==
|
||||||
-----END AGE ENCRYPTED FILE-----
|
-----END AGE ENCRYPTED FILE-----
|
||||||
lastmodified: "2025-09-14T19:20:44Z"
|
lastmodified: "2026-04-12T18:24:10Z"
|
||||||
mac: ENC[AES256_GCM,data:nJ5lnPSVPyfMKhlNwzhxYGWY32i60P3N+jpBZKo8oEh8sqjsb4zHAECG/vMXrGTPwYzZ46m5PQQURCyeOvjuMaXK8184zMwFkehXtMJWI7/aKYbSpQqOchl8BN7QdlxH58kqtCwUkdldiW6t6cr4/VAUUPPLqpK19GDrwUYIVrY=,iv:JZBz5X8PdCFXonSPBd1hYiFG+t0aMQDmgCmAbclnpis=,tag:7Pm7V96xMRQZa/JAiDGYmQ==,type:str]
|
mac: ENC[AES256_GCM,data:D7gWC1fxZkpG8dCr7aWhcyLT92NK1OefS4SIAq5ZImMYRJzzvhXtyJM9tzUYVWPU0ojhgOJyawlWEFk1P/DWjOrafBsBHdxUl5FtZrRb4Z/rquODhR8OT8UYFVKyAmjSfomUEDzYetJdXdAxhX3eNlT+eKI51IjqD9y1dYqcjMw=,iv:ZPIkMf/fUETeUYQdqu4TpS/ZE752079jIW9BlehhWlw=,tag:Z3i/Zc0zkEF5rsNnkmUB9A==,type:str]
|
||||||
unencrypted_suffix: _unencrypted
|
unencrypted_suffix: _unencrypted
|
||||||
version: 3.10.2
|
version: 3.12.2
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
}: {
|
}: {
|
||||||
imports = [
|
imports = [
|
||||||
# Choose your theme here:
|
# Choose your theme here:
|
||||||
../../themes/nixy.nix
|
../../themes/rose-pine.nix
|
||||||
];
|
];
|
||||||
|
|
||||||
config.var = {
|
config.var = {
|
||||||
@@ -30,6 +30,10 @@
|
|||||||
|
|
||||||
autoUpgrade = false;
|
autoUpgrade = false;
|
||||||
autoGarbageCollector = true;
|
autoGarbageCollector = true;
|
||||||
|
|
||||||
|
domain = "hadi.icu";
|
||||||
|
tunnelId = "a1dfa315-7fc3-4a65-8c02-8387932c35c3";
|
||||||
|
networkInterface = "enp3s0";
|
||||||
};
|
};
|
||||||
|
|
||||||
# Let this here
|
# Let this here
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
{pkgs, ...}: {
|
||||||
|
hardware.graphics = {
|
||||||
|
enable = true;
|
||||||
|
extraPackages = with pkgs; [
|
||||||
|
rocmPackages.clr.icd
|
||||||
|
# Support VA-API pour AMD
|
||||||
|
libvdpau-va-gl
|
||||||
|
libva-vdpau-driver
|
||||||
|
];
|
||||||
|
};
|
||||||
|
}
|
||||||
+1
-3
@@ -1,9 +1,7 @@
|
|||||||
# Bluetooth configuration for NixOS
|
# Bluetooth configuration for NixOS
|
||||||
{pkgs, ...}: {
|
{
|
||||||
environment.systemPackages = with pkgs; [blueman];
|
|
||||||
hardware.bluetooth = {
|
hardware.bluetooth = {
|
||||||
enable = true;
|
enable = true;
|
||||||
powerOnBoot = true;
|
powerOnBoot = true;
|
||||||
};
|
};
|
||||||
services.blueman.enable = true;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,15 @@
|
|||||||
# Home-manager configuration for NixOS
|
# Home-manager configuration for NixOS
|
||||||
{inputs, ...}: {
|
{inputs, pkgs, ...}: {
|
||||||
home-manager = {
|
home-manager = {
|
||||||
useGlobalPkgs = true;
|
useGlobalPkgs = true;
|
||||||
useUserPackages = true;
|
useUserPackages = true;
|
||||||
backupFileExtension = "hm-backup";
|
backupFileExtension = "hm-backup";
|
||||||
extraSpecialArgs = {inherit inputs;};
|
extraSpecialArgs = {
|
||||||
|
inherit inputs;
|
||||||
|
pkgs-stable = import inputs.nixpkgs-stable {
|
||||||
|
system = pkgs.stdenv.hostPlatform.system;
|
||||||
|
config.allowUnfree = true;
|
||||||
|
};
|
||||||
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
+6
-1
@@ -6,6 +6,11 @@
|
|||||||
}: let
|
}: let
|
||||||
autoGarbageCollector = config.var.autoGarbageCollector;
|
autoGarbageCollector = config.var.autoGarbageCollector;
|
||||||
in {
|
in {
|
||||||
|
# Ask for password once per SSH session (tied to the tty, expires when session closes)
|
||||||
|
security.sudo.extraConfig = ''
|
||||||
|
Defaults timestamp_type=tty,timestamp_timeout=-1
|
||||||
|
'';
|
||||||
|
|
||||||
security.sudo.extraRules = [
|
security.sudo.extraRules = [
|
||||||
{
|
{
|
||||||
users = [config.var.username];
|
users = [config.var.username];
|
||||||
@@ -19,7 +24,7 @@ in {
|
|||||||
];
|
];
|
||||||
nixpkgs.config = {
|
nixpkgs.config = {
|
||||||
allowUnfree = true;
|
allowUnfree = true;
|
||||||
allowBroken = true;
|
allowBroken = false;
|
||||||
};
|
};
|
||||||
nix = {
|
nix = {
|
||||||
nixPath = ["nixpkgs=${inputs.nixpkgs}"];
|
nixPath = ["nixpkgs=${inputs.nixpkgs}"];
|
||||||
|
|||||||
+4
-6
@@ -6,7 +6,7 @@
|
|||||||
...
|
...
|
||||||
}: let
|
}: let
|
||||||
# Using beta driver for recent GPUs like RTX 4070
|
# Using beta driver for recent GPUs like RTX 4070
|
||||||
nvidiaDriverChannel = config.boot.kernelPackages.nvidiaPackages.beta;
|
nvidiaDriverChannel = config.boot.kernelPackages.nvidiaPackages.production;
|
||||||
in {
|
in {
|
||||||
# Video drivers configuration for Xorg and Wayland
|
# Video drivers configuration for Xorg and Wayland
|
||||||
services.xserver.videoDrivers = ["nvidia"]; # Simplified - other modules are loaded automatically
|
services.xserver.videoDrivers = ["nvidia"]; # Simplified - other modules are loaded automatically
|
||||||
@@ -15,7 +15,6 @@ in {
|
|||||||
boot.kernelParams = [
|
boot.kernelParams = [
|
||||||
"nvidia-drm.modeset=1" # Enable mode setting for Wayland
|
"nvidia-drm.modeset=1" # Enable mode setting for Wayland
|
||||||
"nvidia.NVreg_PreserveVideoMemoryAllocations=1" # Improves resume after sleep
|
"nvidia.NVreg_PreserveVideoMemoryAllocations=1" # Improves resume after sleep
|
||||||
"nvidia.NVreg_RegistryDwords=PowerMizerEnable=0x1;PerfLevelSrc=0x2222;PowerMizerLevel=0x3;PowerMizerDefault=0x3;PowerMizerDefaultAC=0x3" # Performance/power optimizations
|
|
||||||
];
|
];
|
||||||
|
|
||||||
# Blacklist nouveau to avoid conflicts
|
# Blacklist nouveau to avoid conflicts
|
||||||
@@ -24,14 +23,11 @@ in {
|
|||||||
# Environment variables for better compatibility
|
# Environment variables for better compatibility
|
||||||
environment.variables = {
|
environment.variables = {
|
||||||
LIBVA_DRIVER_NAME = "nvidia"; # Hardware video acceleration
|
LIBVA_DRIVER_NAME = "nvidia"; # Hardware video acceleration
|
||||||
XDG_SESSION_TYPE = "wayland"; # Force Wayland
|
|
||||||
GBM_BACKEND = "nvidia-drm"; # Graphics backend for Wayland
|
GBM_BACKEND = "nvidia-drm"; # Graphics backend for Wayland
|
||||||
__GLX_VENDOR_LIBRARY_NAME = "nvidia"; # Use Nvidia driver for GLX
|
__GLX_VENDOR_LIBRARY_NAME = "nvidia"; # Use Nvidia driver for GLX
|
||||||
WLR_NO_HARDWARE_CURSORS = "1"; # Fix for cursors on Wayland
|
|
||||||
NIXOS_OZONE_WL = "1"; # Wayland support for Electron apps
|
NIXOS_OZONE_WL = "1"; # Wayland support for Electron apps
|
||||||
__GL_GSYNC_ALLOWED = "1"; # Enable G-Sync if available
|
__GL_GSYNC_ALLOWED = "1"; # Enable G-Sync if available
|
||||||
__GL_VRR_ALLOWED = "1"; # Enable VRR (Variable Refresh Rate)
|
__GL_VRR_ALLOWED = "1"; # Enable VRR (Variable Refresh Rate)
|
||||||
WLR_DRM_NO_ATOMIC = "1"; # Fix for some issues with Hyprland
|
|
||||||
NVD_BACKEND = "direct"; # Configuration for new driver
|
NVD_BACKEND = "direct"; # Configuration for new driver
|
||||||
MOZ_ENABLE_WAYLAND = "1"; # Wayland support for Firefox
|
MOZ_ENABLE_WAYLAND = "1"; # Wayland support for Firefox
|
||||||
};
|
};
|
||||||
@@ -73,7 +69,6 @@ in {
|
|||||||
# Enhanced graphics support
|
# Enhanced graphics support
|
||||||
graphics = {
|
graphics = {
|
||||||
enable = true;
|
enable = true;
|
||||||
package = nvidiaDriverChannel;
|
|
||||||
enable32Bit = true;
|
enable32Bit = true;
|
||||||
extraPackages = with pkgs; [
|
extraPackages = with pkgs; [
|
||||||
nvidia-vaapi-driver
|
nvidia-vaapi-driver
|
||||||
@@ -102,4 +97,7 @@ in {
|
|||||||
mesa-demos
|
mesa-demos
|
||||||
libva-utils # VA-API debugging tools
|
libva-utils # VA-API debugging tools
|
||||||
];
|
];
|
||||||
|
|
||||||
|
# Enable Nvidia container toolkit for GPU acceleration in containers (docker)
|
||||||
|
hardware.nvidia-container-toolkit.enable = false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,6 +37,7 @@
|
|||||||
in {
|
in {
|
||||||
boot.extraModulePackages = [hp-omen-linux-module];
|
boot.extraModulePackages = [hp-omen-linux-module];
|
||||||
boot.kernelModules = ["hp-wmi"];
|
boot.kernelModules = ["hp-wmi"];
|
||||||
|
boot.kernelParams = ["hp_wmi.force_slow_fan_control=1"];
|
||||||
|
|
||||||
users.groups.omen-rgb = {};
|
users.groups.omen-rgb = {};
|
||||||
users.users.${config.var.username}.extraGroups = ["omen-rgb"];
|
users.users.${config.var.username}.extraGroups = ["omen-rgb"];
|
||||||
|
|||||||
@@ -1,41 +0,0 @@
|
|||||||
# Tailscale is a VPN service that makes it easy to connect your devices between each other.
|
|
||||||
{
|
|
||||||
config,
|
|
||||||
inputs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
username = config.var.username;
|
|
||||||
in {
|
|
||||||
security.sudo.extraRules = [
|
|
||||||
{
|
|
||||||
users = [username];
|
|
||||||
# Allow running Tailscale commands without a password
|
|
||||||
commands = [
|
|
||||||
{
|
|
||||||
command = "/etc/profiles/per-user/${username}/bin/tailscale";
|
|
||||||
options = ["NOPASSWD"];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
command = "/run/current-system/sw/bin/tailscale";
|
|
||||||
options = ["NOPASSWD"];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
environment.systemPackages = with inputs.nixpkgs-stable.legacyPackages.x86_64-linux; [
|
|
||||||
tailscale
|
|
||||||
];
|
|
||||||
|
|
||||||
services.tailscale = {
|
|
||||||
enable = true;
|
|
||||||
package = inputs.nixpkgs-stable.legacyPackages.x86_64-linux.tailscale;
|
|
||||||
openFirewall = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
networking.firewall = {
|
|
||||||
trustedInterfaces = ["tailscale0"];
|
|
||||||
# required to connect to Tailscale exit nodes
|
|
||||||
checkReversePath = "loose";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
+2
-3
@@ -57,9 +57,6 @@ in {
|
|||||||
XDG_DATA_HOME = "$HOME/.local/share";
|
XDG_DATA_HOME = "$HOME/.local/share";
|
||||||
PASSWORD_STORE_DIR = "$HOME/.local/share/password-store";
|
PASSWORD_STORE_DIR = "$HOME/.local/share/password-store";
|
||||||
EDITOR = "nvim";
|
EDITOR = "nvim";
|
||||||
TERMINAL = "kitty";
|
|
||||||
TERM = "kitty";
|
|
||||||
BROWSER = "zen-beta";
|
|
||||||
};
|
};
|
||||||
|
|
||||||
services.libinput.enable = true;
|
services.libinput.enable = true;
|
||||||
@@ -109,6 +106,8 @@ in {
|
|||||||
jpegoptim
|
jpegoptim
|
||||||
pfetch
|
pfetch
|
||||||
btop
|
btop
|
||||||
|
unrar
|
||||||
|
p7zip
|
||||||
];
|
];
|
||||||
|
|
||||||
xdg.portal = {
|
xdg.portal = {
|
||||||
|
|||||||
@@ -1,20 +1,22 @@
|
|||||||
# Adguard is a network-wide ad blocker
|
# Adguard is a network-wide ad blocker
|
||||||
# When installed, open localhost:3000 to setup
|
# When installed, open localhost:3000 to setup
|
||||||
{config, ...}: let
|
{config, ...}: {
|
||||||
domain = "adguard.hadi.diy";
|
services.adguardhome = {
|
||||||
in {
|
|
||||||
services = {
|
|
||||||
adguardhome = {
|
|
||||||
enable = true;
|
enable = true;
|
||||||
port = 3000;
|
port = 3000;
|
||||||
};
|
};
|
||||||
|
|
||||||
nginx.virtualHosts."${domain}" = {
|
networking.firewall = {
|
||||||
useACMEHost = "hadi.diy";
|
allowedTCPPorts = [53];
|
||||||
forceSSL = true;
|
allowedUDPPorts = [53];
|
||||||
locations."/" = {
|
# Allow containers to reach adguard UI (for glance dns-stats widget)
|
||||||
proxyPass = "http://127.0.0.1:${toString config.services.adguardhome.port}";
|
extraCommands = ''
|
||||||
};
|
iptables -I INPUT 1 -s 10.233.0.0/16 -p tcp --dport 3000 -j ACCEPT
|
||||||
};
|
'';
|
||||||
|
extraStopCommands = ''
|
||||||
|
iptables -D INPUT -s 10.233.0.0/16 -p tcp --dport 3000 -j ACCEPT 2>/dev/null || true
|
||||||
|
'';
|
||||||
};
|
};
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."adguard.${config.var.domain}" = "http://localhost:${toString config.services.adguardhome.port}";
|
||||||
}
|
}
|
||||||
|
|||||||
+15
-17
@@ -2,12 +2,6 @@
|
|||||||
# See https://github.com/rasmus-kirk/nixarr
|
# See https://github.com/rasmus-kirk/nixarr
|
||||||
# Setup guide: https://nixarr.com/wiki/setup/
|
# Setup guide: https://nixarr.com/wiki/setup/
|
||||||
{config, ...}: let
|
{config, ...}: let
|
||||||
domain = "hadi.diy";
|
|
||||||
mkVirtualHost = port: {
|
|
||||||
useACMEHost = domain;
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {proxyPass = "http://127.0.0.1:${toString port}";};
|
|
||||||
};
|
|
||||||
username = config.var.username;
|
username = config.var.username;
|
||||||
in {
|
in {
|
||||||
# Add my secrets
|
# Add my secrets
|
||||||
@@ -40,10 +34,13 @@ in {
|
|||||||
radarr.enable = true;
|
radarr.enable = true;
|
||||||
sonarr.enable = true;
|
sonarr.enable = true;
|
||||||
bazarr.enable = true;
|
bazarr.enable = true;
|
||||||
readarr.enable = true;
|
|
||||||
transmission = {
|
transmission = {
|
||||||
enable = true;
|
enable = true;
|
||||||
extraSettings = {trash-original-torrent-files = true;};
|
extraSettings = {
|
||||||
|
trash-original-torrent-files = true;
|
||||||
|
rpc-whitelist-enabled = false;
|
||||||
|
rpc-host-whitelist-enabled = false;
|
||||||
|
};
|
||||||
vpn.enable = true;
|
vpn.enable = true;
|
||||||
};
|
};
|
||||||
recyclarr = {
|
recyclarr = {
|
||||||
@@ -52,14 +49,15 @@ in {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services.nginx.virtualHosts = {
|
users.users.jellyfin.extraGroups = ["video" "render"];
|
||||||
"jellyfin.${domain}" = mkVirtualHost 8096;
|
|
||||||
"jellyseerr.${domain}" = mkVirtualHost 5055;
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress = {
|
||||||
"bazarr.${domain}" = mkVirtualHost 6767;
|
"media.${config.var.domain}" = "http://localhost:8096";
|
||||||
"prowlarr.${domain}" = mkVirtualHost 9696;
|
"demandemedia.${config.var.domain}" = "http://localhost:5055";
|
||||||
"radarr.${domain}" = mkVirtualHost 7878;
|
"bazarr.${config.var.domain}" = "http://localhost:6767";
|
||||||
"sonarr.${domain}" = mkVirtualHost 8989;
|
"prowlarr.${config.var.domain}" = "http://localhost:9696";
|
||||||
"transmission.${domain}" = mkVirtualHost 9091;
|
"radarr.${config.var.domain}" = "http://localhost:7878";
|
||||||
"readarr.${domain}" = mkVirtualHost 8787;
|
"sonarr.${config.var.domain}" = "http://localhost:8989";
|
||||||
|
"transmission.${config.var.domain}" = "http://localhost:9091";
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
{ config, inputs, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "wallpapers";
|
||||||
|
hostIp = "10.233.4.1";
|
||||||
|
containerIp = "10.233.4.2";
|
||||||
|
nixosConfig = { pkgs, ... }: {
|
||||||
|
services.nginx = {
|
||||||
|
enable = true;
|
||||||
|
virtualHosts."wallpapers" = {
|
||||||
|
root = "${inputs.awesome-wallpapers.packages.${pkgs.system}.default}/share/awesome-wallpapers";
|
||||||
|
listen = [{ addr = "0.0.0.0"; port = 8080; }];
|
||||||
|
locations."/" = {
|
||||||
|
tryFiles = "$uri $uri/ /index.html";
|
||||||
|
};
|
||||||
|
extraConfig = ''
|
||||||
|
port_in_redirect off;
|
||||||
|
absolute_redirect off;
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."wallpapers.${config.var.domain}" = "http://10.233.4.2:8080";
|
||||||
|
}
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
# Bitwarden (or vaultwarden) is a self-hosted password manager.
|
|
||||||
{config, ...}: let
|
|
||||||
domain = "vault.hadi.diy";
|
|
||||||
in {
|
|
||||||
services = {
|
|
||||||
vaultwarden = {
|
|
||||||
enable = true;
|
|
||||||
config = {
|
|
||||||
DOMAIN = "https://" + domain;
|
|
||||||
SIGNUPS_ALLOWED = true;
|
|
||||||
ROCKET_ADDRESS = "127.0.0.1";
|
|
||||||
ROCKET_PORT = 8222;
|
|
||||||
ROCKET_LOG = "critical";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
nginx.virtualHosts."${domain}" = {
|
|
||||||
useACMEHost = "hadi.diy";
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {
|
|
||||||
proxyPass = "http://127.0.0.1:${
|
|
||||||
toString config.services.vaultwarden.config.ROCKET_PORT
|
|
||||||
}";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,57 @@
|
|||||||
|
{
|
||||||
|
config,
|
||||||
|
inputs,
|
||||||
|
lib,
|
||||||
|
...
|
||||||
|
}: let
|
||||||
|
inherit (import ./mk-container.nix {inherit lib config;}) mkContainer;
|
||||||
|
domain = config.var.domain;
|
||||||
|
in {
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "blog";
|
||||||
|
hostIp = "10.233.3.1";
|
||||||
|
containerIp = "10.233.3.2";
|
||||||
|
nixosConfig = {pkgs, ...}: {
|
||||||
|
services.nginx = {
|
||||||
|
enable = true;
|
||||||
|
virtualHosts = {
|
||||||
|
"blog" = {
|
||||||
|
root = "${inputs.blog.packages.${pkgs.system}.default}/share/blog";
|
||||||
|
listen = [
|
||||||
|
{
|
||||||
|
addr = "0.0.0.0";
|
||||||
|
port = 8080;
|
||||||
|
}
|
||||||
|
];
|
||||||
|
locations."/" = {
|
||||||
|
tryFiles = "$uri $uri/ /index.html";
|
||||||
|
};
|
||||||
|
extraConfig = ''
|
||||||
|
port_in_redirect off;
|
||||||
|
absolute_redirect off;
|
||||||
|
add_header Content-Security-Policy "default-src 'self'; script-src 'self' 'unsafe-inline' data: https://umami.${domain}; style-src 'self' 'unsafe-inline'; img-src 'self' data: https://git.${domain}; connect-src 'self' https://umami.${domain};" always;
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
"www-redirect" = {
|
||||||
|
listen = [
|
||||||
|
{
|
||||||
|
addr = "0.0.0.0";
|
||||||
|
port = 8081;
|
||||||
|
}
|
||||||
|
];
|
||||||
|
extraConfig = "return 301 https://${domain}$request_uri;";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [8080 8081];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress = {
|
||||||
|
"${config.var.domain}" = "http://10.233.3.2:8080";
|
||||||
|
"www.${config.var.domain}" = "http://10.233.3.2:8081";
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
# Cloudflared tunnel configuration for NixOS
|
||||||
|
# It allows exposing services securely via Cloudflare Tunnel
|
||||||
|
{
|
||||||
|
config,
|
||||||
|
pkgs,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
sops.secrets.cloudflared-token.mode = "0400";
|
||||||
|
|
||||||
|
# To setup cloudflared, run:
|
||||||
|
# - `cloudflared tunnel login`
|
||||||
|
# - `cloudflared tunnel create YourTunnelName`
|
||||||
|
#
|
||||||
|
# This will create a credentials file & give you the tunnel ID to use below.
|
||||||
|
services.cloudflared = {
|
||||||
|
enable = true;
|
||||||
|
tunnels."${config.var.tunnelId}" = {
|
||||||
|
credentialsFile = config.sops.secrets."cloudflared-token".path;
|
||||||
|
default = "http_status:404";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
environment.systemPackages = with pkgs; [
|
||||||
|
cloudflared
|
||||||
|
];
|
||||||
|
|
||||||
|
systemd.services."cloudflared-tunnel-${config.var.tunnelId}" = {
|
||||||
|
wantedBy = ["multi-user.target"];
|
||||||
|
after = ["network-online.target"];
|
||||||
|
wants = ["network-online.target"];
|
||||||
|
};
|
||||||
|
|
||||||
|
# At the moment (2025), for support of browser rendering of the tunnels, this line is required:
|
||||||
|
services.openssh.settings.Macs = [
|
||||||
|
"hmac-sha2-256"
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
{ config, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "cyberchef";
|
||||||
|
hostIp = "10.233.5.1";
|
||||||
|
containerIp = "10.233.5.2";
|
||||||
|
nixosConfig = { pkgs, ... }: {
|
||||||
|
services.nginx = {
|
||||||
|
enable = true;
|
||||||
|
virtualHosts."cyberchef" = {
|
||||||
|
root = "${pkgs.cyberchef}/share/cyberchef";
|
||||||
|
listen = [{ addr = "0.0.0.0"; port = 8080; }];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."cyberchef.${config.var.domain}" = "http://10.233.5.2:8080";
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
{ config, inputs, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
domain = config.var.domain;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "def-creds";
|
||||||
|
hostIp = "10.233.6.1";
|
||||||
|
containerIp = "10.233.6.2";
|
||||||
|
nixosConfig = { ... }: {
|
||||||
|
imports = [ inputs.default-creds.nixosModules.default ];
|
||||||
|
services.default-creds = {
|
||||||
|
enable = true;
|
||||||
|
port = 8087;
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8087 ];
|
||||||
|
systemd.services.default-creds.environment = {
|
||||||
|
HOST = lib.mkForce "0.0.0.0";
|
||||||
|
PUBLIC_UMAMI_URL = "https://umami.${domain}";
|
||||||
|
PUBLIC_UMAMI_WEBSITE_ID = "7197484c-01ad-488e-9caa-5ab7b7595f08";
|
||||||
|
UMAMI_URL = "https://umami.${domain}";
|
||||||
|
UMAMI_WEBSITE_ID = "7197484c-01ad-488e-9caa-5ab7b7595f08";
|
||||||
|
};
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.default-creds.enable = lib.mkForce false;
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."default-creds.${config.var.domain}" = "http://10.233.6.2:8087";
|
||||||
|
}
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
{
|
|
||||||
config,
|
|
||||||
inputs,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
users.groups.eleakxir = {};
|
|
||||||
users.users.hadi.extraGroups = ["eleakxir"];
|
|
||||||
|
|
||||||
services.eleakxir = {
|
|
||||||
enable = true;
|
|
||||||
port = 9198;
|
|
||||||
user = "eleakxir";
|
|
||||||
group = "eleakxir";
|
|
||||||
limit = 1000;
|
|
||||||
folders = ["/var/lib/eleakxir/leaks/" "/mnt/data/clean-leak/"];
|
|
||||||
debug = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = [
|
|
||||||
inputs.eleakxir.packages.${pkgs.stdenv.hostPlatform.system}.leak-utils
|
|
||||||
];
|
|
||||||
|
|
||||||
services.nginx.virtualHosts."eleakxir-back.hadi.diy" = {
|
|
||||||
useACMEHost = "hadi.diy";
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {
|
|
||||||
proxyPass = "http://127.0.0.1:${
|
|
||||||
toString config.services.eleakxir.port
|
|
||||||
}";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
# Fail2Ban is a log-parsing application that protects Linux servers from brute-force attacks.
|
||||||
|
{
|
||||||
|
services.fail2ban = {
|
||||||
|
enable = true;
|
||||||
|
maxretry = 5;
|
||||||
|
bantime = "24h"; # Ban IPs for one day on the first ban
|
||||||
|
bantime-increment = {
|
||||||
|
enable = true; # Enable increment of bantime after each violation
|
||||||
|
multipliers = "1 2 4 8 16 32 64";
|
||||||
|
maxtime = "168h"; # Do not ban for more than 1 week
|
||||||
|
overalljails = true; # Calculate the bantime based on all the violations
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,79 @@
|
|||||||
|
{ config, pkgs, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
domain = config.var.domain;
|
||||||
|
catppuccin-gitea = pkgs.fetchzip {
|
||||||
|
url = "https://github.com/catppuccin/gitea/releases/download/v1.0.2/catppuccin-gitea.tar.gz";
|
||||||
|
sha256 = "sha256-rZHLORwLUfIFcB6K9yhrzr+UwdPNQVSadsw6rg8Q7gs=";
|
||||||
|
stripRoot = false;
|
||||||
|
};
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "gitea";
|
||||||
|
hostIp = "10.233.11.1";
|
||||||
|
containerIp = "10.233.11.2";
|
||||||
|
internet = true;
|
||||||
|
bindMounts."/var/lib/gitea" = {
|
||||||
|
hostPath = "/var/lib/gitea";
|
||||||
|
isReadOnly = false;
|
||||||
|
};
|
||||||
|
nixosConfig = { lib, ... }: {
|
||||||
|
users.users.gitea.uid = lib.mkForce 978;
|
||||||
|
users.groups.gitea.gid = lib.mkForce 968;
|
||||||
|
|
||||||
|
services.postgresql = {
|
||||||
|
enable = true;
|
||||||
|
ensureDatabases = [ "gitea" ];
|
||||||
|
ensureUsers = [{
|
||||||
|
name = "gitea";
|
||||||
|
ensureDBOwnership = true;
|
||||||
|
}];
|
||||||
|
};
|
||||||
|
|
||||||
|
services.gitea = {
|
||||||
|
enable = true;
|
||||||
|
database.type = "postgres";
|
||||||
|
settings = {
|
||||||
|
server = {
|
||||||
|
HTTP_ADDR = "0.0.0.0";
|
||||||
|
HTTP_PORT = 3002;
|
||||||
|
ROOT_URL = "https://git.${domain}/";
|
||||||
|
DOMAIN = "git.${domain}";
|
||||||
|
LANDING_PAGE = "/anotherhadi";
|
||||||
|
};
|
||||||
|
service = {
|
||||||
|
REGISTER_MANUAL_CONFIRM = true;
|
||||||
|
DISABLE_REGISTRATION = true;
|
||||||
|
DEFAULT_KEEP_EMAIL_PRIVATE = true;
|
||||||
|
SHOW_REGISTRATION_BUTTON = false;
|
||||||
|
};
|
||||||
|
ui = {
|
||||||
|
DEFAULT_THEME = "catppuccin-mocha-mauve";
|
||||||
|
THEMES = "catppuccin-latte-mauve,catppuccin-frappe-mauve,catppuccin-macchiato-mauve,catppuccin-mocha-mauve";
|
||||||
|
};
|
||||||
|
explore = {
|
||||||
|
DISABLE_USERS_PAGE = true;
|
||||||
|
DISABLE_ORGANIZATIONS_PAGE = true;
|
||||||
|
};
|
||||||
|
repository.DISABLE_STARS = true;
|
||||||
|
mailer.ENABLED = false;
|
||||||
|
api.ENABLE_SWAGGER = false;
|
||||||
|
other.SHOW_FOOTER_VERSION = false;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
systemd.services.gitea.preStart = lib.mkAfter ''
|
||||||
|
mkdir -p /var/lib/gitea/custom/public/assets
|
||||||
|
ln -sfn ${catppuccin-gitea} /var/lib/gitea/custom/public/assets/css
|
||||||
|
'';
|
||||||
|
|
||||||
|
networking.firewall.allowedTCPPorts = [ 3002 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."git.${domain}" = "http://10.233.11.2:3002";
|
||||||
|
}
|
||||||
@@ -1,353 +0,0 @@
|
|||||||
# Glance is a self-hosted homepage/dashboard service.
|
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
domain = "start.hadi.diy";
|
|
||||||
|
|
||||||
rgb-to-hsl = color: let
|
|
||||||
r = ((lib.toInt config.lib.stylix.colors."${color}-rgb-r") * 100.0) / 255;
|
|
||||||
g = ((lib.toInt config.lib.stylix.colors."${color}-rgb-g") * 100.0) / 255;
|
|
||||||
b = ((lib.toInt config.lib.stylix.colors."${color}-rgb-b") * 100.0) / 255;
|
|
||||||
max = lib.max r (lib.max g b);
|
|
||||||
min = lib.min r (lib.min g b);
|
|
||||||
delta = max - min;
|
|
||||||
fmod = base: int: base - (int * builtins.floor (base / int));
|
|
||||||
h =
|
|
||||||
if delta == 0
|
|
||||||
then 0
|
|
||||||
else if max == r
|
|
||||||
then 60 * (fmod ((g - b) / delta) 6)
|
|
||||||
else if max == g
|
|
||||||
then 60 * (((b - r) / delta) + 2)
|
|
||||||
else if max == b
|
|
||||||
then 60 * (((r - g) / delta) + 4)
|
|
||||||
else 0;
|
|
||||||
l = (max + min) / 2;
|
|
||||||
s =
|
|
||||||
if delta == 0
|
|
||||||
then 0
|
|
||||||
else 100 * delta / (100 - lib.max (2 * l - 100) (100 - (2 * l)));
|
|
||||||
roundToString = value: toString (builtins.floor (value + 0.5));
|
|
||||||
in
|
|
||||||
lib.concatMapStringsSep " " roundToString [h s l];
|
|
||||||
in {
|
|
||||||
# TODO: Add tailscale custom widget
|
|
||||||
services = {
|
|
||||||
glance = {
|
|
||||||
enable = true;
|
|
||||||
settings = {
|
|
||||||
theme = {
|
|
||||||
contrast-multiplier = lib.mkForce 1.4;
|
|
||||||
};
|
|
||||||
pages = [
|
|
||||||
{
|
|
||||||
hide-desktop-navigation = true;
|
|
||||||
columns = [
|
|
||||||
{
|
|
||||||
size = "small";
|
|
||||||
widgets = [
|
|
||||||
{
|
|
||||||
type = "clock";
|
|
||||||
hour-format = "24h";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "weather";
|
|
||||||
location = "Paris, France";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "markets";
|
|
||||||
markets = [
|
|
||||||
{
|
|
||||||
symbol = "BTC-USD";
|
|
||||||
name = "Bitcoin";
|
|
||||||
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:BTCUSD";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
symbol = "SOL-USD";
|
|
||||||
name = "Solana";
|
|
||||||
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:SOLUSD";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
symbol = "ETH-USD";
|
|
||||||
name = "Ethereum";
|
|
||||||
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:ETHUSD";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "dns-stats";
|
|
||||||
service = "adguard";
|
|
||||||
url = "https://adguard.hadi.diy";
|
|
||||||
username = "hadi";
|
|
||||||
password = "\${secret:adguard-pwd}";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
size = "full";
|
|
||||||
widgets = [
|
|
||||||
{
|
|
||||||
type = "search";
|
|
||||||
search-engine = "duckduckgo";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "bookmarks";
|
|
||||||
groups = [
|
|
||||||
{
|
|
||||||
title = "";
|
|
||||||
same-tab = true;
|
|
||||||
color = "200 50 50";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "ProtonMail";
|
|
||||||
url = "https://proton.me/mail";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Github";
|
|
||||||
url = "https://github.com";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Youtube";
|
|
||||||
url = "https://youtube.com";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Figma";
|
|
||||||
url = "https://figma.com";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Docs";
|
|
||||||
same-tab = true;
|
|
||||||
color = "200 50 50";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "Nixpkgs repo";
|
|
||||||
url = "https://github.com/NixOS/nixpkgs";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Nixvim";
|
|
||||||
url = "https://nix-community.github.io/nixvim/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Hyprland wiki";
|
|
||||||
url = "https://wiki.hyprland.org/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Search NixOS";
|
|
||||||
url = "https://search-nixos.hadi.diy";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Homelab";
|
|
||||||
same-tab = true;
|
|
||||||
color = "100 50 50";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "Router";
|
|
||||||
url = "http://192.168.1.254/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Cloudflare";
|
|
||||||
url = "https://dash.cloudflare.com/";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Work";
|
|
||||||
same-tab = true;
|
|
||||||
color = "50 50 50";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "Outlook";
|
|
||||||
url = "https://outlook.office.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Teams";
|
|
||||||
url = "https://teams.microsoft.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Office";
|
|
||||||
url = "https://www.office.com/";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Cyber";
|
|
||||||
same-tab = true;
|
|
||||||
color = rgb-to-hsl "base09";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "CyberChef";
|
|
||||||
url = "https://cyberchef.org/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "TryHackMe";
|
|
||||||
url = "https://tryhackme.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "RootMe";
|
|
||||||
url = "https://www.root-me.org/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Exploit-DB";
|
|
||||||
url = "https://www.exploit-db.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "CrackStation";
|
|
||||||
url = "https://crackstation.net/";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Misc";
|
|
||||||
same-tab = true;
|
|
||||||
color = rgb-to-hsl "base01";
|
|
||||||
links = [
|
|
||||||
{
|
|
||||||
title = "Svgl";
|
|
||||||
url = "https://svgl.app/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Excalidraw";
|
|
||||||
url = "https://excalidraw.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Cobalt (Downloader)";
|
|
||||||
url = "https://cobalt.tools/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Mazanoke (Image optimizer)";
|
|
||||||
url = "https://mazanoke.com/";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Vert (File converter)";
|
|
||||||
url = "https://vert.sh/";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "server-stats";
|
|
||||||
servers = [
|
|
||||||
{
|
|
||||||
type = "local";
|
|
||||||
name = "Jack";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "group";
|
|
||||||
widgets = [
|
|
||||||
{
|
|
||||||
type = "monitor";
|
|
||||||
title = "Services";
|
|
||||||
cache = "1m";
|
|
||||||
sites = [
|
|
||||||
{
|
|
||||||
title = "Vaultwarden";
|
|
||||||
url = "https://vault.hadi.diy";
|
|
||||||
icon = "si:bitwarden";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Nextcloud";
|
|
||||||
url = "https://cloud.hadi.diy";
|
|
||||||
icon = "si:nextcloud";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Adguard";
|
|
||||||
url = "https://adguard.hadi.diy";
|
|
||||||
icon = "si:adguard";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Mealie";
|
|
||||||
url = "https://mealie.hadi.diy";
|
|
||||||
icon = "si:mealie";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "monitor";
|
|
||||||
title = "*arr";
|
|
||||||
cache = "1m";
|
|
||||||
sites = [
|
|
||||||
{
|
|
||||||
title = "Jellyfin";
|
|
||||||
url = "https://jellyfin.hadi.diy";
|
|
||||||
icon = "si:jellyfin";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Jellyseerr";
|
|
||||||
url = "https://jellyseerr.hadi.diy";
|
|
||||||
icon = "si:odysee";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Radarr";
|
|
||||||
url = "https://radarr.hadi.diy";
|
|
||||||
icon = "si:radarr";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Sonarr";
|
|
||||||
url = "https://sonarr.hadi.diy";
|
|
||||||
icon = "si:sonarr";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Prowlarr";
|
|
||||||
url = "https://prowlarr.hadi.diy";
|
|
||||||
icon = "si:podcastindex";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
title = "Transmission";
|
|
||||||
url = "https://transmission.hadi.diy";
|
|
||||||
icon = "si:transmission";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
name = "Home";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
server = {port = 5678;};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
nginx.virtualHosts."${domain}" = {
|
|
||||||
useACMEHost = "hadi.diy";
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {
|
|
||||||
proxyPass = "http://127.0.0.1:${
|
|
||||||
toString config.services.glance.settings.server.port
|
|
||||||
}";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
systemd.services.glance = {
|
|
||||||
serviceConfig = {
|
|
||||||
DynamicUser = lib.mkForce false;
|
|
||||||
User = "glance";
|
|
||||||
Group = "glance";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
users = {
|
|
||||||
groups.glance = {};
|
|
||||||
users.glance = {
|
|
||||||
isSystemUser = true;
|
|
||||||
description = "Glance user";
|
|
||||||
group = "glance";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
sops.secrets.adguard-pwd = {
|
|
||||||
owner = "glance";
|
|
||||||
mode = "0600";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,100 @@
|
|||||||
|
{ config, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ../mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
domain = config.var.domain;
|
||||||
|
hostIp = "10.233.12.1";
|
||||||
|
|
||||||
|
# Convert 6-char hex color to "H S L" string for glance (integers, no % sign)
|
||||||
|
hexToGlanceHsl = hex:
|
||||||
|
let
|
||||||
|
h = lib.toLower hex;
|
||||||
|
d = c:
|
||||||
|
if c == "a" then 10 else if c == "b" then 11 else if c == "c" then 12
|
||||||
|
else if c == "d" then 13 else if c == "e" then 14 else if c == "f" then 15
|
||||||
|
else lib.toInt c;
|
||||||
|
byte = pos: d (builtins.substring pos 1 h) * 16 + d (builtins.substring (pos + 1) 1 h);
|
||||||
|
ri = byte 0; gi = byte 2; bi = byte 4;
|
||||||
|
r = ri * 1.0 / 255.0;
|
||||||
|
g = gi * 1.0 / 255.0;
|
||||||
|
b = bi * 1.0 / 255.0;
|
||||||
|
mx = if r >= g && r >= b then "r" else if g >= b then "g" else "b";
|
||||||
|
mn = if r <= g && r <= b then "r" else if g <= b then "g" else "b";
|
||||||
|
cmax = if mx == "r" then r else if mx == "g" then g else b;
|
||||||
|
cmin = if mn == "r" then r else if mn == "g" then g else b;
|
||||||
|
delta = cmax - cmin;
|
||||||
|
l = (cmax + cmin) / 2.0;
|
||||||
|
s = if delta < 0.0001 then 0.0
|
||||||
|
else if l <= 0.5 then delta / (cmax + cmin)
|
||||||
|
else delta / (2.0 - cmax - cmin);
|
||||||
|
hue =
|
||||||
|
if delta < 0.0001 then 0.0
|
||||||
|
else if mx == "r" then let raw = 60.0 * (g - b) / delta; in if raw < 0.0 then raw + 360.0 else raw
|
||||||
|
else if mx == "g" then 60.0 * ((b - r) / delta + 2.0)
|
||||||
|
else 60.0 * ((r - g) / delta + 4.0);
|
||||||
|
in "${toString (builtins.floor (hue + 0.5))} ${toString (builtins.floor (s * 100.0 + 0.5))} ${toString (builtins.floor (l * 100.0 + 0.5))}";
|
||||||
|
|
||||||
|
c = config.stylix.base16Scheme;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
# 0444 so the glance user inside the container can read the bind-mounted file
|
||||||
|
sops.secrets.adguard-pwd.mode = "0444";
|
||||||
|
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "glance";
|
||||||
|
hostIp = hostIp;
|
||||||
|
containerIp = "10.233.12.2";
|
||||||
|
internet = true;
|
||||||
|
bindMounts."/run/secrets/adguard-pwd" = {
|
||||||
|
hostPath = config.sops.secrets.adguard-pwd.path;
|
||||||
|
isReadOnly = true;
|
||||||
|
};
|
||||||
|
nixosConfig = { lib, ... }: {
|
||||||
|
_module.args.domain = domain;
|
||||||
|
_module.args.adguardUrl = "http://${hostIp}:3000";
|
||||||
|
imports = [ ./home.nix ./server.nix ];
|
||||||
|
|
||||||
|
services.glance = {
|
||||||
|
enable = true;
|
||||||
|
settings = {
|
||||||
|
server = {
|
||||||
|
port = 5678;
|
||||||
|
host = "127.0.0.1";
|
||||||
|
};
|
||||||
|
theme = {
|
||||||
|
light = false;
|
||||||
|
background-color = hexToGlanceHsl c.base00; # background
|
||||||
|
primary-color = hexToGlanceHsl c.base0D; # accent (iris/purple)
|
||||||
|
positive-color = hexToGlanceHsl c.base0B; # positive (pine/teal)
|
||||||
|
negative-color = hexToGlanceHsl c.base08; # negative (love/rose)
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
services.nginx = {
|
||||||
|
enable = true;
|
||||||
|
appendHttpConfig = ''
|
||||||
|
proxy_cache_path /var/cache/nginx/glance levels=1:2 keys_zone=glance:1m inactive=30m max_size=100m;
|
||||||
|
'';
|
||||||
|
virtualHosts."glance" = {
|
||||||
|
listen = [{ addr = "0.0.0.0"; port = 8080; }];
|
||||||
|
locations."/" = {
|
||||||
|
proxyPass = "http://127.0.0.1:5678";
|
||||||
|
extraConfig = ''
|
||||||
|
proxy_cache glance;
|
||||||
|
proxy_cache_valid 200 30m;
|
||||||
|
proxy_cache_use_stale error timeout updating http_500 http_502 http_503 http_504;
|
||||||
|
add_header X-Cache-Status $upstream_cache_status;
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."home.${domain}" = "http://10.233.12.2:8080";
|
||||||
|
}
|
||||||
@@ -0,0 +1,245 @@
|
|||||||
|
{ domain, ... }: {
|
||||||
|
services.glance.settings.pages = [
|
||||||
|
{
|
||||||
|
name = "Home";
|
||||||
|
hide-desktop-navigation = false;
|
||||||
|
columns = [
|
||||||
|
{
|
||||||
|
size = "small";
|
||||||
|
widgets = [
|
||||||
|
{
|
||||||
|
type = "clock";
|
||||||
|
hour-format = "24h";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "weather";
|
||||||
|
location = "Paris, France";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "markets";
|
||||||
|
markets = [
|
||||||
|
{
|
||||||
|
symbol = "BTC-USD";
|
||||||
|
name = "Bitcoin";
|
||||||
|
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:BTCUSD";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
symbol = "SOL-USD";
|
||||||
|
name = "Solana";
|
||||||
|
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:SOLUSD";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
symbol = "ETH-USD";
|
||||||
|
name = "Ethereum";
|
||||||
|
chart-link = "https://www.tradingview.com/chart/?symbol=INDEX:ETHUSD";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
size = "full";
|
||||||
|
widgets = [
|
||||||
|
{
|
||||||
|
type = "search";
|
||||||
|
search-engine = "startpage";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "group";
|
||||||
|
widgets = [
|
||||||
|
{
|
||||||
|
type = "bookmarks";
|
||||||
|
title = "Bookmarks";
|
||||||
|
groups = [
|
||||||
|
{
|
||||||
|
title = "";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Mail";
|
||||||
|
url = "https://mail.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Drive";
|
||||||
|
url = "https://drive.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Lumo";
|
||||||
|
url = "https://lumo.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Calendar";
|
||||||
|
url = "https://calendar.proton.me";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Maps";
|
||||||
|
url = "https://maps.apple.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Amazon";
|
||||||
|
url = "https://amazon.fr";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Tools";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Excalidraw";
|
||||||
|
url = "https://excalidraw.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cobalt (downloader)";
|
||||||
|
url = "https://cobalt.meowing.de";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Mazanoke (image downgrading)";
|
||||||
|
url = "https://mazanoke.${domain}";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Stirling PDF";
|
||||||
|
url = "https://pdf.${domain}";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Vert (file converter)";
|
||||||
|
url = "https://vert.sh";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Markdown to pdf";
|
||||||
|
url = "https://md2file.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Image to Vector";
|
||||||
|
url = "https://www.vectorcascade.com/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "PrivateBin";
|
||||||
|
url = "https://privatebin.net";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Social";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Bsky";
|
||||||
|
url = "https://bsky.app";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Reddit";
|
||||||
|
url = "https://reddit.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Youtube";
|
||||||
|
url = "https://youtube.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Instagram";
|
||||||
|
url = "https://instagram.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Github";
|
||||||
|
url = "https://github.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Discord";
|
||||||
|
url = "https://discord.com/channels/@me/";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Other";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Startpage config";
|
||||||
|
url = "https://www.startpage.com/do/mypage.pl?prfe=45d331deb05471d659dba933e7400df51d952bb103da6f6125c0e769a6be1d65610456a479f495ceeee7e97311cf227d7c1bb198de0ceeb193d8cddf9c455c19a409cc35c3e3f542ee27bd7cecd3";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Hyprland Wiki";
|
||||||
|
url = "https://wiki.hypr.land";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Search NixOS";
|
||||||
|
url = "https://mynixos.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Nixpkgs";
|
||||||
|
url = "https://github.com/NixOS/nixpkgs";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Claude";
|
||||||
|
url = "https://claude.ai";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Gemini";
|
||||||
|
url = "https://gemini.google.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Medium";
|
||||||
|
url = "https://medium.com";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "bookmarks";
|
||||||
|
title = "Infosec";
|
||||||
|
groups = [
|
||||||
|
{
|
||||||
|
title = "";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Nix 4 Cyber";
|
||||||
|
url = "https://n4c.${domain}";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cyberchef";
|
||||||
|
url = "https://cyberchef.${domain}";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "TryHackMe";
|
||||||
|
url = "https://tryhackme.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Root-Me";
|
||||||
|
url = "https://root-me.org";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Exploit-DB";
|
||||||
|
url = "https://exploit-db.com";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Crack Station";
|
||||||
|
url = "https://crackstation.net";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Osint Tracker";
|
||||||
|
url = "https://app.osintracker.com";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "hacker-news";
|
||||||
|
limit = 15;
|
||||||
|
collapse-after = 5;
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -0,0 +1,179 @@
|
|||||||
|
{
|
||||||
|
domain,
|
||||||
|
adguardUrl,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
services.glance.settings.pages = [
|
||||||
|
{
|
||||||
|
name = "Server";
|
||||||
|
hide-desktop-navigation = false;
|
||||||
|
columns = [
|
||||||
|
{
|
||||||
|
size = "full";
|
||||||
|
widgets = [
|
||||||
|
{
|
||||||
|
type = "server-stats";
|
||||||
|
servers = [
|
||||||
|
{
|
||||||
|
type = "local";
|
||||||
|
name = "Jack";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "group";
|
||||||
|
widgets = [
|
||||||
|
{
|
||||||
|
type = "monitor";
|
||||||
|
title = "Services";
|
||||||
|
cache = "1m";
|
||||||
|
sites = [
|
||||||
|
{
|
||||||
|
title = "Adguard";
|
||||||
|
url = "https://adguard.${domain}";
|
||||||
|
icon = "si:adguard";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Blog";
|
||||||
|
url = "https://${domain}";
|
||||||
|
icon = "si:blogger";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Gitea";
|
||||||
|
url = "https://git.${domain}";
|
||||||
|
icon = "si:gitea";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Mealie";
|
||||||
|
url = "https://mealie.${domain}";
|
||||||
|
icon = "si:mealie";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Umami";
|
||||||
|
url = "https://umami.${domain}";
|
||||||
|
icon = "si:umami";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Iknowyou";
|
||||||
|
url = "https://iknowyou.${domain}";
|
||||||
|
icon = "sh:iknowyou";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Iknowyou Prod";
|
||||||
|
url = "https://iknowyou-prod.${domain}";
|
||||||
|
icon = "sh:iknowyou";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Wallpapers";
|
||||||
|
url = "https://wallpapers.${domain}";
|
||||||
|
icon = "si:unsplash";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Mazanoke";
|
||||||
|
url = "https://mazanoke.${domain}";
|
||||||
|
icon = "sh:mazanoke";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Stirling PDF";
|
||||||
|
url = "https://pdf.${domain}";
|
||||||
|
icon = "sh:stirling-pdf";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Default-creds";
|
||||||
|
url = "https://default-creds.${domain}";
|
||||||
|
icon = "si:passbolt";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cyberchef";
|
||||||
|
url = "https://cyberchef.${domain}";
|
||||||
|
icon = "si:codechef";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
type = "monitor";
|
||||||
|
title = "*arr";
|
||||||
|
cache = "1m";
|
||||||
|
sites = [
|
||||||
|
{
|
||||||
|
title = "Jellyfin";
|
||||||
|
url = "https://media.${domain}";
|
||||||
|
icon = "si:jellyfin";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Jellyseerr";
|
||||||
|
url = "https://demandemedia.${domain}";
|
||||||
|
icon = "si:odysee";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Radarr";
|
||||||
|
url = "https://radarr.${domain}";
|
||||||
|
icon = "si:radarr";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Sonarr";
|
||||||
|
url = "https://sonarr.${domain}";
|
||||||
|
icon = "si:sonarr";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Bazarr";
|
||||||
|
url = "https://bazarr.${domain}";
|
||||||
|
icon = "si:subtitleedit";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Prowlarr";
|
||||||
|
url = "https://prowlarr.${domain}";
|
||||||
|
icon = "si:podcastindex";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Transmission";
|
||||||
|
url = "https://transmission.${domain}";
|
||||||
|
icon = "si:transmission";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
type = "dns-stats";
|
||||||
|
service = "adguard";
|
||||||
|
url = adguardUrl;
|
||||||
|
username = "hadi";
|
||||||
|
password = "\${secret:adguard-pwd}";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
type = "bookmarks";
|
||||||
|
groups = [
|
||||||
|
{
|
||||||
|
title = "";
|
||||||
|
same-tab = true;
|
||||||
|
color = "245 50 64";
|
||||||
|
links = [
|
||||||
|
{
|
||||||
|
title = "Router";
|
||||||
|
url = "http://192.168.1.254/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cloudflare";
|
||||||
|
url = "https://dash.cloudflare.com/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cloudflare Zero Trust";
|
||||||
|
url = "https://one.dash.cloudflare.com/";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
title = "Cloudflare Access";
|
||||||
|
url = "https://anotherhadi.cloudflareaccess.com";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
{
|
||||||
|
config,
|
||||||
|
inputs,
|
||||||
|
lib,
|
||||||
|
...
|
||||||
|
}: let
|
||||||
|
inherit (import ./mk-container.nix {inherit lib config;}) mkContainer;
|
||||||
|
in {
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "iky-prod";
|
||||||
|
hostIp = "10.233.1.1";
|
||||||
|
containerIp = "10.233.1.2";
|
||||||
|
internet = true;
|
||||||
|
bindMounts."/etc/iky/config.yaml" = {
|
||||||
|
hostPath = "/var/lib/iknowyou-prod/config.yaml";
|
||||||
|
isReadOnly = false;
|
||||||
|
};
|
||||||
|
nixosConfig = {...}: {
|
||||||
|
imports = [inputs.iknowyou.nixosModules.default];
|
||||||
|
users.users.iknowyou.uid = 999;
|
||||||
|
users.groups.iknowyou.gid = 999;
|
||||||
|
services.iknowyou = {
|
||||||
|
enable = true;
|
||||||
|
port = 8080;
|
||||||
|
openFirewall = true;
|
||||||
|
};
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
|
||||||
|
(mkContainer {
|
||||||
|
name = "iky-demo";
|
||||||
|
hostIp = "10.233.2.1";
|
||||||
|
containerIp = "10.233.2.2";
|
||||||
|
nixosConfig = {...}: {
|
||||||
|
imports = [inputs.iknowyou.nixosModules.default];
|
||||||
|
services.iknowyou = {
|
||||||
|
enable = true;
|
||||||
|
port = 8080;
|
||||||
|
openFirewall = true;
|
||||||
|
};
|
||||||
|
systemd.services.iknowyou.environment.IKY_DEMO = "true";
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
users.users.iknowyou = {
|
||||||
|
isSystemUser = true;
|
||||||
|
group = "iknowyou";
|
||||||
|
uid = 999;
|
||||||
|
};
|
||||||
|
users.groups.iknowyou.gid = 999;
|
||||||
|
|
||||||
|
systemd.tmpfiles.rules = [
|
||||||
|
"f /var/lib/iknowyou-prod/config.yaml 0600 iknowyou iknowyou -"
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress = {
|
||||||
|
"iknowyou-prod.${config.var.domain}" = "http://10.233.1.2:8080";
|
||||||
|
"iknowyou.${config.var.domain}" = "http://10.233.2.2:8080";
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
# Kernel hardening for the server
|
||||||
|
{
|
||||||
|
boot.kernel.sysctl = {
|
||||||
|
# Restrict access to kernel logs and pointers
|
||||||
|
"kernel.dmesg_restrict" = 1;
|
||||||
|
"kernel.kptr_restrict" = 2;
|
||||||
|
|
||||||
|
# BPF hardening
|
||||||
|
"net.core.bpf_jit_harden" = 2;
|
||||||
|
"kernel.unprivileged_bpf_disabled" = 1;
|
||||||
|
|
||||||
|
# Reverse path filtering (anti-spoofing)
|
||||||
|
"net.ipv4.conf.all.rp_filter" = 1;
|
||||||
|
"net.ipv4.conf.default.rp_filter" = 1;
|
||||||
|
|
||||||
|
# SYN flood protection
|
||||||
|
"net.ipv4.tcp_syncookies" = 1;
|
||||||
|
|
||||||
|
# Disable IP source routing
|
||||||
|
"net.ipv4.conf.all.accept_source_route" = 0;
|
||||||
|
"net.ipv4.conf.default.accept_source_route" = 0;
|
||||||
|
|
||||||
|
# Ignore ICMP redirects (prevent MITM)
|
||||||
|
"net.ipv4.conf.all.accept_redirects" = 0;
|
||||||
|
"net.ipv4.conf.default.accept_redirects" = 0;
|
||||||
|
"net.ipv4.conf.all.secure_redirects" = 0;
|
||||||
|
"net.ipv6.conf.all.accept_redirects" = 0;
|
||||||
|
|
||||||
|
# Don't send ICMP redirects
|
||||||
|
"net.ipv4.conf.all.send_redirects" = 0;
|
||||||
|
|
||||||
|
# Restrict ptrace to parent processes only
|
||||||
|
"kernel.yama.ptrace_scope" = 1;
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
{ config, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "mazanoke";
|
||||||
|
hostIp = "10.233.7.1";
|
||||||
|
containerIp = "10.233.7.2";
|
||||||
|
nixosConfig = { pkgs, ... }:
|
||||||
|
let
|
||||||
|
version = "1.1.5";
|
||||||
|
mazanoke-pkg = pkgs.stdenv.mkDerivation {
|
||||||
|
inherit version;
|
||||||
|
pname = "mazanoke";
|
||||||
|
src = pkgs.fetchFromGitHub {
|
||||||
|
owner = "civilblur";
|
||||||
|
repo = "mazanoke";
|
||||||
|
rev = "v${version}";
|
||||||
|
hash = "sha256-B/AF4diMNxN94BzpZP/C+K8kNj9q+4SDKWa/qd4LrVU=";
|
||||||
|
};
|
||||||
|
installPhase = ''
|
||||||
|
mkdir -p $out/share/mazanoke
|
||||||
|
cp -r ./index.html ./favicon.ico ./manifest.json ./service-worker.js ./assets $out/share/mazanoke/
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
in
|
||||||
|
{
|
||||||
|
services.nginx = {
|
||||||
|
enable = true;
|
||||||
|
virtualHosts."mazanoke" = {
|
||||||
|
root = "${mazanoke-pkg}/share/mazanoke";
|
||||||
|
listen = [{ addr = "0.0.0.0"; port = 8080; }];
|
||||||
|
locations."/" = { index = "index.html"; };
|
||||||
|
};
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."mazanoke.${config.var.domain}" = "http://10.233.7.2:8080";
|
||||||
|
}
|
||||||
+20
-15
@@ -1,19 +1,24 @@
|
|||||||
# Mealie is a recipe management and meal planning application.
|
{ config, lib, ... }:
|
||||||
{config, ...}: let
|
let
|
||||||
domain = "mealie.hadi.diy";
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
in {
|
in
|
||||||
services = {
|
{
|
||||||
mealie = {
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "mealie";
|
||||||
|
hostIp = "10.233.8.1";
|
||||||
|
containerIp = "10.233.8.2";
|
||||||
|
internet = true;
|
||||||
|
nixosConfig = { ... }: {
|
||||||
|
services.mealie = {
|
||||||
enable = true;
|
enable = true;
|
||||||
port = 8092;
|
port = 8080;
|
||||||
};
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
nginx.virtualHosts."${domain}" = {
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."mealie.${config.var.domain}" = "http://10.233.8.2:8080";
|
||||||
useACMEHost = "hadi.diy";
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {
|
|
||||||
proxyPass = "http://127.0.0.1:${toString config.services.mealie.port}";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
{ lib, config }:
|
||||||
|
|
||||||
|
# Returns a NixOS module (attrset), to be used in `imports`.
|
||||||
|
#
|
||||||
|
# Options:
|
||||||
|
# internet - allow outbound internet access via NAT (default: false)
|
||||||
|
# externalInterface - WAN interface for NAT, required when internet = true
|
||||||
|
# bindMounts - host paths to mount into the container (see containers.<name>.bindMounts)
|
||||||
|
# config - NixOS module for the container
|
||||||
|
|
||||||
|
let
|
||||||
|
nginxHardening = { config, ... }: lib.mkIf config.services.nginx.enable {
|
||||||
|
services.nginx.serverTokens = false;
|
||||||
|
};
|
||||||
|
in
|
||||||
|
|
||||||
|
{
|
||||||
|
mkContainer =
|
||||||
|
{
|
||||||
|
name,
|
||||||
|
hostIp,
|
||||||
|
containerIp,
|
||||||
|
internet ? false,
|
||||||
|
externalInterface ? config.var.networkInterface,
|
||||||
|
bindMounts ? {},
|
||||||
|
nixosConfig,
|
||||||
|
}:
|
||||||
|
assert lib.assertMsg
|
||||||
|
(lib.stringLength "ve-${name}" <= 15)
|
||||||
|
"mkContainer: interface name 've-${name}' is ${toString (lib.stringLength "ve-${name}")} chars, max is 15";
|
||||||
|
{
|
||||||
|
containers.${name} = {
|
||||||
|
autoStart = true;
|
||||||
|
privateNetwork = true;
|
||||||
|
hostAddress = hostIp;
|
||||||
|
localAddress = containerIp;
|
||||||
|
inherit bindMounts;
|
||||||
|
config = { ... }: {
|
||||||
|
imports = [ nixosConfig nginxHardening ];
|
||||||
|
networking.nameservers = lib.mkIf internet [ "1.1.1.1" "1.0.0.1" ];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// (lib.optionalAttrs internet {
|
||||||
|
boot.kernel.sysctl."net.ipv4.ip_forward" = lib.mkDefault true;
|
||||||
|
networking.nat = {
|
||||||
|
enable = true;
|
||||||
|
externalInterface = externalInterface;
|
||||||
|
internalInterfaces = [ "ve-${name}" ];
|
||||||
|
};
|
||||||
|
# CONTAINER-FWD (defined by another module) blocks all forwarding by default.
|
||||||
|
# Insert rules in FORWARD before it: allow return traffic, block LAN, allow internet.
|
||||||
|
networking.firewall.extraCommands = ''
|
||||||
|
iptables -I FORWARD 1 -s ${containerIp} -m conntrack --ctstate NEW -j ACCEPT
|
||||||
|
iptables -I FORWARD 1 -s ${containerIp} -d 192.168.0.0/16 -j DROP
|
||||||
|
iptables -I FORWARD 1 -s ${containerIp} -d 172.16.0.0/12 -j DROP
|
||||||
|
iptables -I FORWARD 1 -s ${containerIp} -d 10.0.0.0/8 -j DROP
|
||||||
|
iptables -I FORWARD 1 -d ${containerIp} -m conntrack --ctstate RELATED,ESTABLISHED -j ACCEPT
|
||||||
|
'';
|
||||||
|
networking.firewall.extraStopCommands = ''
|
||||||
|
iptables -D FORWARD -s ${containerIp} -m conntrack --ctstate NEW -j ACCEPT 2>/dev/null || true
|
||||||
|
iptables -D FORWARD -s ${containerIp} -d 192.168.0.0/16 -j DROP 2>/dev/null || true
|
||||||
|
iptables -D FORWARD -s ${containerIp} -d 172.16.0.0/12 -j DROP 2>/dev/null || true
|
||||||
|
iptables -D FORWARD -s ${containerIp} -d 10.0.0.0/8 -j DROP 2>/dev/null || true
|
||||||
|
iptables -D FORWARD -d ${containerIp} -m conntrack --ctstate RELATED,ESTABLISHED -j ACCEPT 2>/dev/null || true
|
||||||
|
'';
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
# Nginx is a web server that can also be used as a reverse proxy, load balancer, and HTTP cache.
|
|
||||||
{config, ...}: let
|
|
||||||
domain = "hadi.diy";
|
|
||||||
in {
|
|
||||||
security.acme = {
|
|
||||||
acceptTerms = true;
|
|
||||||
defaults.email = config.var.git.email;
|
|
||||||
certs."${domain}" = {
|
|
||||||
domain = "${domain}";
|
|
||||||
extraDomainNames = ["*.${domain}"];
|
|
||||||
group = "nginx";
|
|
||||||
dnsProvider = "cloudflare";
|
|
||||||
dnsPropagationCheck = true;
|
|
||||||
credentialsFile = config.sops.secrets.cloudflare-dns-token.path;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Return 444 for all requests not matching a used subdomain.
|
|
||||||
services.nginx = {
|
|
||||||
enable = true;
|
|
||||||
virtualHosts = {
|
|
||||||
"default" = {
|
|
||||||
default = true;
|
|
||||||
locations."/" = {return = 444;};
|
|
||||||
};
|
|
||||||
"*.${domain}" = {
|
|
||||||
useACMEHost = domain;
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {return = 444;};
|
|
||||||
};
|
|
||||||
"aaaaaa.${domain}" = {
|
|
||||||
useACMEHost = domain;
|
|
||||||
forceSSL = true;
|
|
||||||
locations."/" = {return = 444;};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
networking.firewall = {
|
|
||||||
allowedTCPPorts = [80 443];
|
|
||||||
allowedUDPPorts = [80 443];
|
|
||||||
};
|
|
||||||
|
|
||||||
sops.secrets.cloudflare-dns-token = {path = "/etc/cloudflare/dnskey.txt";};
|
|
||||||
}
|
|
||||||
+12
-3
@@ -10,13 +10,22 @@ in {
|
|||||||
PermitRootLogin = "no";
|
PermitRootLogin = "no";
|
||||||
PasswordAuthentication = false;
|
PasswordAuthentication = false;
|
||||||
AllowUsers = [username];
|
AllowUsers = [username];
|
||||||
|
MaxAuthTries = 3;
|
||||||
|
LoginGraceTime = 20;
|
||||||
|
X11Forwarding = false;
|
||||||
|
AllowAgentForwarding = false;
|
||||||
|
AllowTcpForwarding = false;
|
||||||
|
ClientAliveInterval = 300;
|
||||||
|
ClientAliveCountMax = 2;
|
||||||
|
KexAlgorithms = ["curve25519-sha256" "curve25519-sha256@libssh.org"];
|
||||||
|
Ciphers = ["chacha20-poly1305@openssh.com" "aes256-gcm@openssh.com"];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
# Add my public SSH key to my user
|
# Add my public SSH key to my user
|
||||||
users.users."${username}" = {
|
users.users."${username}".openssh.authorizedKeys.keys = [
|
||||||
openssh.authorizedKeys.keys = [
|
|
||||||
"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIPG9SE80ZyBcXZK/f5ypSKudaM5Jo3XtQikCnGo0jI5E hadi@nixy"
|
"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIPG9SE80ZyBcXZK/f5ypSKudaM5Jo3XtQikCnGo0jI5E hadi@nixy"
|
||||||
];
|
];
|
||||||
};
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."ssh.${config.var.domain}" = "ssh://localhost:22";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
{ config, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "stirling-pdf";
|
||||||
|
hostIp = "10.233.9.1";
|
||||||
|
containerIp = "10.233.9.2";
|
||||||
|
nixosConfig = { ... }: {
|
||||||
|
services.stirling-pdf = {
|
||||||
|
enable = true;
|
||||||
|
environment."SERVER_PORT" = "8080";
|
||||||
|
};
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."pdf.${config.var.domain}" = "http://10.233.9.2:8080";
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
{ config, lib, ... }:
|
||||||
|
let
|
||||||
|
inherit (import ./mk-container.nix { inherit lib config; }) mkContainer;
|
||||||
|
in
|
||||||
|
{
|
||||||
|
sops.secrets.umami-secret.mode = "0400";
|
||||||
|
|
||||||
|
imports = [
|
||||||
|
(mkContainer {
|
||||||
|
name = "umami";
|
||||||
|
hostIp = "10.233.10.1";
|
||||||
|
containerIp = "10.233.10.2";
|
||||||
|
bindMounts."/run/secrets/umami-secret" = {
|
||||||
|
hostPath = config.sops.secrets.umami-secret.path;
|
||||||
|
isReadOnly = true;
|
||||||
|
};
|
||||||
|
nixosConfig = { ... }: {
|
||||||
|
services.umami = {
|
||||||
|
enable = true;
|
||||||
|
settings = {
|
||||||
|
PORT = 8080;
|
||||||
|
HOSTNAME = "0.0.0.0";
|
||||||
|
APP_SECRET_FILE = "/run/secrets/umami-secret";
|
||||||
|
DISABLE_TELEMETRY = true;
|
||||||
|
DISABLE_BOT_CHECK = true;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
# PrivateUsers breaks systemd-creds inside nspawn containers (nested user namespaces)
|
||||||
|
systemd.services.umami.serviceConfig.PrivateUsers = lib.mkForce false;
|
||||||
|
networking.firewall.allowedTCPPorts = [ 8080 ];
|
||||||
|
system.stateVersion = "24.05";
|
||||||
|
};
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
services.cloudflared.tunnels."${config.var.tunnelId}".ingress."umami.${config.var.domain}" = "http://10.233.10.2:8080";
|
||||||
|
}
|
||||||
@@ -53,8 +53,8 @@
|
|||||||
|
|
||||||
fonts = {
|
fonts = {
|
||||||
monospace = {
|
monospace = {
|
||||||
package = pkgs.nerd-fonts.jetbrains-mono;
|
package = pkgs.maple-mono.NF;
|
||||||
name = "JetBrains Mono Nerd Font";
|
name = "Maple Mono NF";
|
||||||
};
|
};
|
||||||
sansSerif = {
|
sansSerif = {
|
||||||
package = pkgs.source-sans-pro;
|
package = pkgs.source-sans-pro;
|
||||||
|
|||||||
@@ -0,0 +1,82 @@
|
|||||||
|
{
|
||||||
|
lib,
|
||||||
|
pkgs,
|
||||||
|
config,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
options.theme = lib.mkOption {
|
||||||
|
type = lib.types.attrs;
|
||||||
|
default = {
|
||||||
|
rounding = 30;
|
||||||
|
gaps-in = 12;
|
||||||
|
gaps-out = 12 * 2;
|
||||||
|
active-opacity = 0.99;
|
||||||
|
inactive-opacity = 0.98;
|
||||||
|
blur = false;
|
||||||
|
border-size = 4;
|
||||||
|
animation-speed = "medium"; # "fast" | "medium" | "slow"
|
||||||
|
fetch = "none"; # "nerdfetch" | "neofetch" | "pfetch" | "none"
|
||||||
|
textColorOnWallpaper =
|
||||||
|
config.lib.stylix.colors.base00; # Color of the text displayed on the wallpaper (Lockscreen, display manager, ...)
|
||||||
|
};
|
||||||
|
description = "Theme configuration options";
|
||||||
|
};
|
||||||
|
|
||||||
|
config.stylix = {
|
||||||
|
enable = true;
|
||||||
|
|
||||||
|
# See https://tinted-theming.github.io/tinted-gallery/ for more schemes
|
||||||
|
base16Scheme = {
|
||||||
|
base00 = "090E13"; # Default Background
|
||||||
|
base01 = "12171E"; # Lighter Background (Used for status bars, line number and folding marks)
|
||||||
|
base02 = "393B44"; # Selection Background
|
||||||
|
base03 = "54546D"; # Comments, Invisibles, Line Highlighting
|
||||||
|
base04 = "A4A7A4"; # Dark Foreground (Used for status bars)
|
||||||
|
base05 = "C5C9C7"; # Default Foreground, Caret, Delimiters, Operators
|
||||||
|
base06 = "DCD7BA"; # Light Foreground (Not often used)
|
||||||
|
base07 = "C8C093"; # Light Background (Not often used)
|
||||||
|
base08 = "C4746E"; # Variables, XML Tags, Markup Link Text, Markup Lists, Diff Deleted
|
||||||
|
base09 = "B6927B"; # Integers, Boolean, Constants, XML Attributes, Markup Link Url
|
||||||
|
base0A = "C4B28A"; # Classes, Markup Bold, Search Text Background
|
||||||
|
base0B = "8A9A7B"; # Strings, Inherited Class, Markup Code, Diff Inserted
|
||||||
|
base0C = "8EA4A2"; # Support, Regular Expressions, Escape Characters, Markup Quotes
|
||||||
|
base0D = "8BA4B0"; # Functions, Methods, Attribute IDs, Headings, Accent color
|
||||||
|
base0E = "A292A3"; # Keywords, Storage, Selector, Markup Italic, Diff Changed
|
||||||
|
base0F = "B98D7B"; # Deprecated, Opening/Closing Embedded Language Tags, e.g. <?php ?>
|
||||||
|
};
|
||||||
|
|
||||||
|
cursor = {
|
||||||
|
name = "BreezeX-RosePine-Linux";
|
||||||
|
package = pkgs.rose-pine-cursor;
|
||||||
|
size = 20;
|
||||||
|
};
|
||||||
|
|
||||||
|
fonts = {
|
||||||
|
monospace = {
|
||||||
|
package = pkgs.maple-mono.NF;
|
||||||
|
name = "Maple Mono NF";
|
||||||
|
};
|
||||||
|
sansSerif = {
|
||||||
|
package = pkgs.source-sans-pro;
|
||||||
|
name = "Source Sans Pro";
|
||||||
|
};
|
||||||
|
serif = config.stylix.fonts.sansSerif;
|
||||||
|
emoji = {
|
||||||
|
package = pkgs.noto-fonts-color-emoji;
|
||||||
|
name = "Noto Color Emoji";
|
||||||
|
};
|
||||||
|
sizes = {
|
||||||
|
applications = 13;
|
||||||
|
desktop = 13;
|
||||||
|
popups = 13;
|
||||||
|
terminal = 13;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
polarity = "dark";
|
||||||
|
image = pkgs.fetchurl {
|
||||||
|
url = "https://raw.githubusercontent.com/anotherhadi/awesome-wallpapers/refs/heads/main/app/static/wallpapers/nepal.png";
|
||||||
|
sha256 = "sha256-0o2+B3+yA++PCyfNC+VqmaX959aKk3GuD7XpOq3SGcM=";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user