sway
This commit is contained in:
		
							parent
							
								
									492f085d52
								
							
						
					
					
						commit
						a1b4228627
					
				
					 23 changed files with 373 additions and 331 deletions
				
			
		|  | @ -47,7 +47,7 @@ | ||||||
|     home-manager.users.geoffrey = |     home-manager.users.geoffrey = | ||||||
|       { ... }: |       { ... }: | ||||||
|       { |       { | ||||||
|         xsession.windowManager.i3.config.modifier = "Mod1"; |         wayland.windowManager.sway.config.modifier = "Mod1"; | ||||||
|       }; |       }; | ||||||
| 
 | 
 | ||||||
|     # 8 makes it run out of memory when rebuilding. |     # 8 makes it run out of memory when rebuilding. | ||||||
|  |  | ||||||
|  | @ -6,18 +6,9 @@ | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   displays = { |   displays = { | ||||||
|     embedded = { |     embedded = "Chimei Innolux Corporation 0x1738 Unknown"; | ||||||
|       output = "eDP-1"; |     deskLeft = "Samsung Electric Company S24B420 H4MCB03534"; # Internal HDMI | ||||||
|       edid = "00ffffffffffff000dae381700000000011c01049526157802a155a556519d280b505400000001010101010101010101010101010101b43b804a71383440302035007dd61000001ac32f804a71383440302035007dd61000001a000000fe003059395747803137334843450a00000000000041319e001000000a010a2020004f"; |     deskRight = "Samsung Electric Company S24B420 H4MC800865"; # DisplayLink DVI | ||||||
|     }; |  | ||||||
|     deskLeft = { |  | ||||||
|       output = "HDMI-1-3"; # Internal HDMI port |  | ||||||
|       edid = "00ffffffffffff004c2d7b09333032302f160103803420782a01f1a257529f270a505423080081c0810081809500a9c0b300d1c00101283c80a070b023403020360006442100001a000000fd00353f1e5111000a202020202020000000fc00533234423432300a2020202020000000ff0048344d434230333533340a2020010702010400023a80d072382d40102c458006442100001e011d007251d01e206e28550006442100001e011d00bc52d01e20b828554006442100001e8c0ad090204031200c4055000644210000188c0ad08a20e02d10103e9600064421000018000000000000000000000000000000000000000000000000000000000000000000d2"; |  | ||||||
|     }; |  | ||||||
|     deskRight = { |  | ||||||
|       output = "DVI-I-2-1"; # DisplayLink |  | ||||||
|       edid = "00ffffffffffff004c2d7b093330323020160103803420782a01f1a257529f270a505423080081c0810081809500a9c0b300d1c00101283c80a070b023403020360006442100001a000000fd00353f1e5111000a202020202020000000fc00533234423432300a2020202020000000ff0048344d433830303836350a2020011c02010400023a80d072382d40102c458006442100001e011d007251d01e206e28550006442100001e011d00bc52d01e20b828554006442100001e8c0ad090204031200c4055000644210000188c0ad08a20e02d10103e9600064421000018000000000000000000000000000000000000000000000000000000000000000000d2"; |  | ||||||
|     }; |  | ||||||
|   }; |   }; | ||||||
| in | in | ||||||
| { | { | ||||||
|  | @ -33,8 +24,6 @@ in | ||||||
|         "rtsx_usb_sdmmc" |         "rtsx_usb_sdmmc" | ||||||
|       ]; |       ]; | ||||||
|       kernelModules = [ "kvm-intel" ]; |       kernelModules = [ "kvm-intel" ]; | ||||||
|       kernelPackages = pkgs.linuxKernel.packages.linux_6_6; |  | ||||||
|       # displaylink doesn't seem to be working for kernels >= 6.9? |  | ||||||
| 
 | 
 | ||||||
|       # UEFI works here, and variables can be touched |       # UEFI works here, and variables can be touched | ||||||
|       loader = { |       loader = { | ||||||
|  | @ -56,8 +45,8 @@ in | ||||||
| 
 | 
 | ||||||
|     frogeye.desktop = { |     frogeye.desktop = { | ||||||
|       x11_screens = [ |       x11_screens = [ | ||||||
|         displays.deskLeft.output |         displays.deskLeft | ||||||
|         displays.deskRight.output |         displays.deskRight | ||||||
|       ]; |       ]; | ||||||
|       maxVideoHeight = 1440; |       maxVideoHeight = 1440; | ||||||
|       numlock = true; |       numlock = true; | ||||||
|  | @ -80,44 +69,80 @@ in | ||||||
|         # TODO Display 2 doesn't work anymore? |         # TODO Display 2 doesn't work anymore? | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     services = { | 
 | ||||||
|       autorandr = { |     # Screens | ||||||
|         profiles = { |     home-manager.users.geoffrey = | ||||||
|           portable = { |       { ... }: | ||||||
|             fingerprint.${displays.embedded.output} = displays.embedded.edid; |       { | ||||||
|             config.${displays.embedded.output} = { }; |         services = { | ||||||
|           }; |           kanshi.settings = [ | ||||||
|           extOnly = { |             { | ||||||
|             fingerprint = { |               profile = { | ||||||
|               ${displays.embedded.output} = displays.embedded.edid; |                 name = "portable"; | ||||||
|               ${displays.deskLeft.output} = displays.deskLeft.edid; |                 outputs = [ | ||||||
|               ${displays.deskRight.output} = displays.deskRight.edid; |                   { criteria = displays.embedded; } | ||||||
|             }; |                 ]; | ||||||
|             config = { |  | ||||||
|               ${displays.embedded.output}.enable = false; |  | ||||||
|               ${displays.deskLeft.output} = { |  | ||||||
|                 primary = true; |  | ||||||
|                 mode = "1920x1200"; |  | ||||||
|                 rate = "59.95"; |  | ||||||
|                 position = "0x0"; |  | ||||||
|               }; |               }; | ||||||
|               ${displays.deskRight.output} = { |             } | ||||||
|                 mode = "1920x1200"; |             { | ||||||
|                 rate = "59.95"; |               profile = { | ||||||
|                 position = "1920x0"; |                 name = "extOnly"; | ||||||
|  |                 outputs = [ | ||||||
|  |                   { | ||||||
|  |                     criteria = displays.embedded; | ||||||
|  |                     status = "disable"; | ||||||
|  |                   } | ||||||
|  |                   { | ||||||
|  |                     criteria = displays.deskLeft; | ||||||
|  |                     position = "0,0"; | ||||||
|  |                   } | ||||||
|  |                   { | ||||||
|  |                     criteria = displays.deskRight; | ||||||
|  |                     position = "1920,0"; | ||||||
|  |                   } | ||||||
|  |                 ]; | ||||||
|               }; |               }; | ||||||
|             }; |             } | ||||||
|           }; |             { | ||||||
|           # TODO leftOnly and other things.Might want to abstract a few things first. |               profile = { | ||||||
|  |                 name = "leftOnly"; | ||||||
|  |                 outputs = [ | ||||||
|  |                   { | ||||||
|  |                     criteria = displays.embedded; | ||||||
|  |                     status = "disable"; | ||||||
|  |                   } | ||||||
|  |                   { | ||||||
|  |                     criteria = displays.deskLeft; | ||||||
|  |                   } | ||||||
|  |                 ]; | ||||||
|  |               }; | ||||||
|  |             } | ||||||
|  |           ]; | ||||||
|         }; |         }; | ||||||
|       }; |       }; | ||||||
|       # Needs prefetched binary blobs, see https://nixos.wiki/wiki/Displaylink | 
 | ||||||
|       xserver.videoDrivers = [ |     # Displaylink | ||||||
|         "displaylink" |     environment.variables = { | ||||||
|         "modesetting" |       WLR_EVDI_RENDER_DEVICE = "/dev/dri/card1"; | ||||||
|       ]; |  | ||||||
|       # TODO See if nvidia and DL can work together. |  | ||||||
|     }; |     }; | ||||||
|  |     nixpkgs.overlays = [ | ||||||
|  |       (final: prev: { | ||||||
|  |         wlroots_0_18 = prev.wlroots_0_18.overrideAttrs (old: { | ||||||
|  |           patches = (old.patches or [ ]) ++ [ | ||||||
|  |             (prev.fetchpatch { | ||||||
|  |               url = "https://gitlab.freedesktop.org/wlroots/wlroots/uploads/bd115aa120d20f2c99084951589abf9c/DisplayLink_v2.patch"; | ||||||
|  |               hash = "sha256-vWQc2e8a5/YZaaHe+BxfAR/Ni8HOs2sPJ8Nt9pfxqiE="; | ||||||
|  |             }) | ||||||
|  |           ]; | ||||||
|  |         }); | ||||||
|  |       }) | ||||||
|  |     ]; | ||||||
|  |     services.xserver.videoDrivers = [ | ||||||
|  |       "displaylink" | ||||||
|  |       "modesetting" | ||||||
|  |     ]; | ||||||
|  |     systemd.services.dlm.wantedBy = [ "multi-user.target" ]; | ||||||
|  |     # Needs prefetched binary blobs, see https://wiki.nixos.org/wiki/Displaylink | ||||||
|   }; |   }; | ||||||
|   imports = [ |   imports = [ | ||||||
|     nixos-hardware.nixosModules.dell-g3-3779 |     nixos-hardware.nixosModules.dell-g3-3779 | ||||||
|  |  | ||||||
							
								
								
									
										19
									
								
								flake.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										19
									
								
								flake.lock
									
										
									
										generated
									
									
									
								
							|  | @ -654,7 +654,8 @@ | ||||||
|         "nur": "nur", |         "nur": "nur", | ||||||
|         "onixpkgs": "onixpkgs", |         "onixpkgs": "onixpkgs", | ||||||
|         "stylix": "stylix", |         "stylix": "stylix", | ||||||
|         "unixpkgs": "unixpkgs" |         "unixpkgs": "unixpkgs", | ||||||
|  |         "zelbarnixpkgs": "zelbarnixpkgs" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "scss-reset": { |     "scss-reset": { | ||||||
|  | @ -909,6 +910,22 @@ | ||||||
|         "ref": "master", |         "ref": "master", | ||||||
|         "type": "indirect" |         "type": "indirect" | ||||||
|       } |       } | ||||||
|  |     }, | ||||||
|  |     "zelbarnixpkgs": { | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1748415083, | ||||||
|  |         "narHash": "sha256-jmIRxOA7kj1CFNiP6S6pg6e6XeQi3whBVvWsATpEiC4=", | ||||||
|  |         "owner": "GeoffreyFrogeye", | ||||||
|  |         "repo": "nixpkgs", | ||||||
|  |         "rev": "bc0e8ee7d08241e8fbcb5e13a7abb445329e5644", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "GeoffreyFrogeye", | ||||||
|  |         "ref": "zelbar", | ||||||
|  |         "repo": "nixpkgs", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "root": "root", |   "root": "root", | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
|     onixpkgs.url = "nixpkgs/nixos-24.11"; |     onixpkgs.url = "nixpkgs/nixos-24.11"; | ||||||
|     nixpkgs.url = "nixpkgs/nixos-25.05"; |     nixpkgs.url = "nixpkgs/nixos-25.05"; | ||||||
|     unixpkgs.url = "nixpkgs/master"; |     unixpkgs.url = "nixpkgs/master"; | ||||||
|  |     zelbarnixpkgs.url = "github:GeoffreyFrogeye/nixpkgs/zelbar"; | ||||||
|     # OS |     # OS | ||||||
|     disko = { |     disko = { | ||||||
|       url = "disko"; |       url = "disko"; | ||||||
|  | @ -41,6 +42,7 @@ | ||||||
|       self, |       self, | ||||||
|       nixpkgs, |       nixpkgs, | ||||||
|       unixpkgs, |       unixpkgs, | ||||||
|  |       zelbarnixpkgs, | ||||||
|       disko, |       disko, | ||||||
|       nix-on-droid, |       nix-on-droid, | ||||||
|       flake-utils, |       flake-utils, | ||||||
|  | @ -62,9 +64,11 @@ | ||||||
|             self: super: |             self: super: | ||||||
|             let |             let | ||||||
|               upkgs = import unixpkgs { inherit (super) system; }; |               upkgs = import unixpkgs { inherit (super) system; }; | ||||||
|  |               zelbarpkgs = import zelbarnixpkgs { inherit (super) system; }; | ||||||
|             in |             in | ||||||
|             { |             { | ||||||
|               hello = upkgs.hello; # Placeholder |               hello = upkgs.hello; # Placeholder | ||||||
|  |               zelbar = zelbarpkgs.zelbar; | ||||||
|             } |             } | ||||||
|           ) |           ) | ||||||
|         ]; |         ]; | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ let | ||||||
|       specialisation = "dark"; |       specialisation = "dark"; | ||||||
|     } |     } | ||||||
|   ]; |   ]; | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = { |   config = { | ||||||
|  | @ -48,7 +48,7 @@ in | ||||||
|       ++ (with pkgs; [ |       ++ (with pkgs; [ | ||||||
|         brightnessctl |         brightnessctl | ||||||
|       ]); |       ]); | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       XF86MonBrightnessUp = "exec ${pkgs.brightnessctl}/bin/brightnessctl set +5%"; |       XF86MonBrightnessUp = "exec ${pkgs.brightnessctl}/bin/brightnessctl set +5%"; | ||||||
|       XF86MonBrightnessDown = "exec ${pkgs.brightnessctl}/bin/brightnessctl set 5%-"; |       XF86MonBrightnessDown = "exec ${pkgs.brightnessctl}/bin/brightnessctl set 5%-"; | ||||||
|       "${mod}+F6" = "exec ${pkgs.brightnessctl}/bin/brightnessctl set 1%-"; |       "${mod}+F6" = "exec ${pkgs.brightnessctl}/bin/brightnessctl set 1%-"; | ||||||
|  |  | ||||||
|  | @ -6,7 +6,7 @@ | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   pactl = "exec ${pkgs.pulseaudio}/bin/pactl"; # TODO Use NixOS package if using NixOS |   pactl = "exec ${pkgs.pulseaudio}/bin/pactl"; # TODO Use NixOS package if using NixOS | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|  | @ -32,7 +32,7 @@ in | ||||||
|         text = ''cookie-file = .config/pulse/pulse-cookie''; |         text = ''cookie-file = .config/pulse/pulse-cookie''; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       "XF86AudioRaiseVolume" = "${pactl} set-sink-mute @DEFAULT_SINK@ false; ${pactl} set-sink-volume @DEFAULT_SINK@ +5%"; |       "XF86AudioRaiseVolume" = "${pactl} set-sink-mute @DEFAULT_SINK@ false; ${pactl} set-sink-volume @DEFAULT_SINK@ +5%"; | ||||||
|       "XF86AudioLowerVolume" = "${pactl} set-sink-mute @DEFAULT_SINK@ false; ${pactl} set-sink-volume @DEFAULT_SINK@ -5%"; |       "XF86AudioLowerVolume" = "${pactl} set-sink-mute @DEFAULT_SINK@ false; ${pactl} set-sink-volume @DEFAULT_SINK@ -5%"; | ||||||
|       "XF86AudioMute" = "${pactl} set-sink-mute @DEFAULT_SINK@ true"; |       "XF86AudioMute" = "${pactl} set-sink-mute @DEFAULT_SINK@ true"; | ||||||
|  |  | ||||||
|  | @ -14,59 +14,9 @@ let | ||||||
|     "horizontal-reverse" |     "horizontal-reverse" | ||||||
|     "vertical-reverse" |     "vertical-reverse" | ||||||
|   ]; |   ]; | ||||||
|   autorandrmenu = |  | ||||||
|     { |  | ||||||
|       title, |  | ||||||
|       option, |  | ||||||
|       builtin ? false, |  | ||||||
|     }: |  | ||||||
|     pkgs.writeShellScript "autorandrmenu" '' |  | ||||||
|       shopt -s nullglob globstar |  | ||||||
|       profiles="${ |  | ||||||
|         if builtin then lib.strings.concatLines builtin_configs else "" |  | ||||||
|       }$(${pkgs.autorandr}/bin/autorandr | ${pkgs.gawk}/bin/awk '{ print $1 }')" |  | ||||||
|       profile="$(echo "$profiles" | ${config.programs.rofi.package}/bin/rofi -dmenu -p "${title}")" |  | ||||||
|       [[ -n "$profile" ]] || exit |  | ||||||
|       ${pkgs.autorandr}/bin/autorandr ${option} "$profile" |  | ||||||
|     ''; |  | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|     frogeye.desktop.i3.bindmodes = { |     services.kanshi.enable = true; | ||||||
|       "Screen setup [A] Auto [L] Load [S] Save [R] Remove [D] Default" = { |  | ||||||
|         bindings = { |  | ||||||
|           "a" = "exec ${pkgs.autorandr}/bin/autorandr --change --force, mode default"; |  | ||||||
|           "l" = "exec ${ |  | ||||||
|             autorandrmenu { |  | ||||||
|               title = "Load profile"; |  | ||||||
|               option = "--load"; |  | ||||||
|               builtin = true; |  | ||||||
|             } |  | ||||||
|           }, mode default"; |  | ||||||
|           "s" = "exec ${ |  | ||||||
|             autorandrmenu { |  | ||||||
|               title = "Save profile"; |  | ||||||
|               option = "--save"; |  | ||||||
|             } |  | ||||||
|           }, mode default"; |  | ||||||
|           "r" = "exec ${ |  | ||||||
|             autorandrmenu { |  | ||||||
|               title = "Remove profile"; |  | ||||||
|               option = "--remove"; |  | ||||||
|             } |  | ||||||
|           }, mode default"; |  | ||||||
|           "d" = "exec ${ |  | ||||||
|             autorandrmenu { |  | ||||||
|               title = "Default profile"; |  | ||||||
|               option = "--default"; |  | ||||||
|               builtin = true; |  | ||||||
|             } |  | ||||||
|           }, mode default"; |  | ||||||
|         }; |  | ||||||
|         mod_enter = "t"; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     programs.autorandr.enable = true; |  | ||||||
|     services.autorandr.enable = true; |  | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,13 +1,8 @@ | ||||||
| { | { | ||||||
|   pkgs, |  | ||||||
|   config, |  | ||||||
|   ... |   ... | ||||||
| }: | }: | ||||||
| { | { | ||||||
|   config = { |   config = { | ||||||
|     # This correctly sets the background on some occasions, below does the rest |   # FIXME DELETE | ||||||
|     programs.autorandr.hooks.postswitch = { |  | ||||||
|       background = "${pkgs.feh}/bin/feh --no-fehbg --bg-fill ${config.stylix.image}"; |  | ||||||
|     }; |  | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -194,8 +194,8 @@ | ||||||
|         }; |         }; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       "${config.xsession.windowManager.i3.config.modifier}+m" = |       "${config.wayland.windowManager.sway.config.modifier}+m" = | ||||||
|         "exec ${config.programs.qutebrowser.package}/bin/qutebrowser --override-restore"; |         "exec ${config.programs.qutebrowser.package}/bin/qutebrowser --override-restore"; | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  | @ -22,28 +22,34 @@ | ||||||
|   ]; |   ]; | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
| 
 | 
 | ||||||
|     xsession = { |     # xsession = { | ||||||
|  |     #   enable = true; | ||||||
|  |     #   # Not using config.xdg.configHome because it needs to be $HOME-relative paths and path manipulation is hard | ||||||
|  |     #   scriptPath = ".config/xsession"; | ||||||
|  |     #   profilePath = ".config/xprofile"; | ||||||
|  |     #   windowManager = { | ||||||
|  |     #     i3.enable = true; | ||||||
|  |     #   }; | ||||||
|  |     #   numlock.enable = config.frogeye.desktop.numlock; | ||||||
|  |     # }; | ||||||
|  | 
 | ||||||
|  |     wayland.windowManager.sway = { | ||||||
|       enable = true; |       enable = true; | ||||||
|       # Not using config.xdg.configHome because it needs to be $HOME-relative paths and path manipulation is hard |  | ||||||
|       scriptPath = ".config/xsession"; |  | ||||||
|       profilePath = ".config/xprofile"; |  | ||||||
|       windowManager = { |  | ||||||
|         i3.enable = true; |  | ||||||
|       }; |  | ||||||
|       numlock.enable = config.frogeye.desktop.numlock; |  | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     programs = { |     programs = { | ||||||
|       # Terminal |       # Terminal | ||||||
|       bash.shellAliases = { |       bash.shellAliases = { | ||||||
|         x = "startx ${config.home.homeDirectory}/${config.xsession.scriptPath}; logout"; |  | ||||||
|         lmms = "lmms --config ${config.xdg.configHome}/lmmsrc.xml"; |         lmms = "lmms --config ${config.xdg.configHome}/lmmsrc.xml"; | ||||||
|       }; |       }; | ||||||
|       rofi = { |       rofi = { | ||||||
|         # TODO This theme template, that was used for Arch, looks much better: |         # TODO This theme template, that was used for Arch, looks much better: | ||||||
|         # https://gitlab.com/jordiorlando/base16-rofi/-/blob/master/templates/default.mustache |         # https://gitlab.com/jordiorlando/base16-rofi/-/blob/master/templates/default.mustache | ||||||
|         enable = true; |         enable = true; | ||||||
|         pass.enable = true; |         pass = { | ||||||
|  |           enable = true; | ||||||
|  |           package = pkgs.rofi-pass-wayland; | ||||||
|  |         }; | ||||||
|         extraConfig = { |         extraConfig = { | ||||||
|           lazy-grab = false; |           lazy-grab = false; | ||||||
|           matching = "regex"; |           matching = "regex"; | ||||||
|  | @ -111,7 +117,6 @@ | ||||||
|     }; |     }; | ||||||
|     services = { |     services = { | ||||||
|       blueman-applet.enable = true; |       blueman-applet.enable = true; | ||||||
|       unclutter.enable = true; |  | ||||||
|       dunst = { |       dunst = { | ||||||
|         enable = true; |         enable = true; | ||||||
|         settings = with config.lib.stylix.colors.withHashtag; { |         settings = with config.lib.stylix.colors.withHashtag; { | ||||||
|  | @ -180,9 +185,10 @@ | ||||||
|         # x11-exclusive |         # x11-exclusive | ||||||
|         simplescreenrecorder |         simplescreenrecorder | ||||||
|         trayer |         trayer | ||||||
|         xclip | 
 | ||||||
|         xorg.xinit |         # wayland exclusive | ||||||
|         scrot |         wl-clipboard | ||||||
|  |         # TODO Clipboard history? | ||||||
|       ]; |       ]; | ||||||
|       sessionVariables = { |       sessionVariables = { | ||||||
|         # XAUTHORITY = "${config.xdg.configHome}/Xauthority"; # Disabled as this causes lock-ups with DMs |         # XAUTHORITY = "${config.xdg.configHome}/Xauthority"; # Disabled as this causes lock-ups with DMs | ||||||
|  |  | ||||||
|  | @ -1,23 +1,12 @@ | ||||||
| { | { | ||||||
|   pkgs ? import <nixpkgs> { |   # nixpkgs ? builtins.getFlake "github:GeoffreyFrogeye/nixpkgs/zelbar", | ||||||
|  |   nixpkgs ? /nix/store/8g86qw3c2fr56bhhvqznrlic4jig9hb3-source, | ||||||
|  |   pkgs ? import nixpkgs { | ||||||
|     config = { }; |     config = { }; | ||||||
|     overlays = [ ]; |     overlays = [ ]; | ||||||
|   }, |   }, | ||||||
|   ... |   ... | ||||||
| }: | }: | ||||||
| let |  | ||||||
|   lemonbar = ( |  | ||||||
|     pkgs.lemonbar-xft.overrideAttrs (old: { |  | ||||||
|       src = pkgs.fetchFromGitHub { |  | ||||||
|         owner = "drscream"; |  | ||||||
|         repo = "lemonbar-xft"; |  | ||||||
|         rev = "a64a2a6a6d643f4d92f9d7600722710eebce7bdb"; |  | ||||||
|         sha256 = "sha256-T5FhEPIiDt/9paJwL9Sj84CBtA0YFi1hZz0+87Hd6jU="; |  | ||||||
|         # https://github.com/drscream/lemonbar-xft/pull/2 |  | ||||||
|       }; |  | ||||||
|     }) |  | ||||||
|   ); |  | ||||||
| in |  | ||||||
| # Tried using pyproject.nix but mpd2 dependency wouldn't resolve, | # Tried using pyproject.nix but mpd2 dependency wouldn't resolve, | ||||||
| # is called pyton-mpd2 on PyPi but mpd2 in nixpkgs. | # is called pyton-mpd2 on PyPi but mpd2 in nixpkgs. | ||||||
| pkgs.python3Packages.buildPythonApplication rec { | pkgs.python3Packages.buildPythonApplication rec { | ||||||
|  | @ -31,12 +20,12 @@ pkgs.python3Packages.buildPythonApplication rec { | ||||||
|     pygobject3 |     pygobject3 | ||||||
|     rich |     rich | ||||||
|   ]; |   ]; | ||||||
|   nativeBuildInputs = |   # TODO Might just be buildInputs, maybe without the need for prefix? | ||||||
|     [ lemonbar ] |   nativeBuildInputs = with pkgs; [ | ||||||
|     ++ (with pkgs; [ |     wirelesstools | ||||||
|       wirelesstools |     playerctl | ||||||
|       playerctl |     zelbar | ||||||
|     ]); |   ]; | ||||||
|   makeWrapperArgs = [ |   makeWrapperArgs = [ | ||||||
|     "--prefix PATH : ${pkgs.lib.makeBinPath nativeBuildInputs}" |     "--prefix PATH : ${pkgs.lib.makeBinPath nativeBuildInputs}" | ||||||
|     "--prefix GI_TYPELIB_PATH : ${GI_TYPELIB_PATH}" |     "--prefix GI_TYPELIB_PATH : ${GI_TYPELIB_PATH}" | ||||||
|  |  | ||||||
|  | @ -36,9 +36,7 @@ def main() -> None: | ||||||
| 
 | 
 | ||||||
|     theme = rich.terminal_theme.TerminalTheme( |     theme = rich.terminal_theme.TerminalTheme( | ||||||
|         base16_color(0x0), |         base16_color(0x0), | ||||||
|         base16_color( |         base16_color(0x7), | ||||||
|             0x0 |  | ||||||
|         ),  # TODO should be 7, currently 0 so it's compatible with v2 |  | ||||||
|         [ |         [ | ||||||
|             base16_color(0x0),  # black |             base16_color(0x0),  # black | ||||||
|             base16_color(0x8),  # red |             base16_color(0x8),  # red | ||||||
|  | @ -68,7 +66,7 @@ def main() -> None: | ||||||
| 
 | 
 | ||||||
|     workspaces_suffixes = "▲■" |     workspaces_suffixes = "▲■" | ||||||
|     workspaces_names = { |     workspaces_names = { | ||||||
|         str(i + 1): f"{i+1} {c}" for i, c in enumerate(workspaces_suffixes) |         str(i + 1): f"{i + 1} {c}" for i, c in enumerate(workspaces_suffixes) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     color = rich.color.Color.parse |     color = rich.color.Color.parse | ||||||
|  |  | ||||||
|  | @ -4,6 +4,7 @@ import datetime | ||||||
| import enum | import enum | ||||||
| import logging | import logging | ||||||
| import signal | import signal | ||||||
|  | import sys | ||||||
| import typing | import typing | ||||||
| 
 | 
 | ||||||
| import gi | import gi | ||||||
|  | @ -62,6 +63,10 @@ def clip(text: str, length: int = 30) -> str: | ||||||
|     return text |     return text | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def color_0x(color: rich.color.ColorTriplet) -> str: | ||||||
|  |     return f"0x{color.red:02X}{color.green:02X}{color.blue:02X}" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| class ComposableText(typing.Generic[P, C]): | class ComposableText(typing.Generic[P, C]): | ||||||
|     def __init__( |     def __init__( | ||||||
|         self, |         self, | ||||||
|  | @ -73,6 +78,7 @@ class ComposableText(typing.Generic[P, C]): | ||||||
|         self.sortKey = sort_key |         self.sortKey = sort_key | ||||||
|         if parent: |         if parent: | ||||||
|             self.set_parent(parent) |             self.set_parent(parent) | ||||||
|  |             self.screen = self.get_first_parent_of_type(Screen) | ||||||
|         self.bar = self.get_first_parent_of_type(Bar) |         self.bar = self.get_first_parent_of_type(Bar) | ||||||
| 
 | 
 | ||||||
|     def set_parent(self, parent: P) -> None: |     def set_parent(self, parent: P) -> None: | ||||||
|  | @ -92,12 +98,14 @@ class ComposableText(typing.Generic[P, C]): | ||||||
|     def get_first_parent_of_type(self, typ: type[T]) -> T: |     def get_first_parent_of_type(self, typ: type[T]) -> T: | ||||||
|         parent = self |         parent = self | ||||||
|         while not isinstance(parent, typ): |         while not isinstance(parent, typ): | ||||||
|             assert parent.parent, f"{self} doesn't have a parent of {typ}" |             if not parent.parent: | ||||||
|  |                 msg = f"{self} doesn't have a parent of {typ}" | ||||||
|  |                 raise RuntimeError(msg) | ||||||
|             parent = parent.parent |             parent = parent.parent | ||||||
|         return parent |         return parent | ||||||
| 
 | 
 | ||||||
|     def update_markup(self) -> None: |     def update_markup(self) -> None: | ||||||
|         self.bar.refresh.set() |         self.parent.update_markup() | ||||||
|         # TODO OPTI See if worth caching the output |         # TODO OPTI See if worth caching the output | ||||||
| 
 | 
 | ||||||
|     def generate_markup(self) -> str: |     def generate_markup(self) -> str: | ||||||
|  | @ -191,23 +199,24 @@ class Section(ComposableText): | ||||||
|         else: |         else: | ||||||
|             self.animationTask = self.bar.taskGroup.create_task(self.animate()) |             self.animationTask = self.bar.taskGroup.create_task(self.animate()) | ||||||
| 
 | 
 | ||||||
|     def set_action( |     def set_action(self, button: Button, callback: typing.Callable | None) -> None: | ||||||
|         self, button: Button, callback: typing.Callable | None |  | ||||||
|     ) -> None: |  | ||||||
|         if button in self.actions: |         if button in self.actions: | ||||||
|             command = self.actions[button] |             command = self.actions[button] | ||||||
|             self.bar.remove_action(command) |             self.bar.remove_action(command) | ||||||
|             del self.actions[button] |             del self.actions[button] | ||||||
|         if callback: |         if callback: | ||||||
|             command = self.bar.add_action(callback) |             command = self.screen.add_action(callback) | ||||||
|             self.actions[button] = command |             self.actions[button] = command | ||||||
| 
 | 
 | ||||||
|     def generate_markup(self) -> str: |     def generate_markup(self) -> str: | ||||||
|         assert not self.is_hidden() |         assert not self.is_hidden() | ||||||
|         pad = max(0, self.size - len(self.text)) |         pad = max(0, self.size - len(self.text)) | ||||||
|         text = self.text[: self.size] + " " * pad |         text = self.text[: self.size] + " " * pad | ||||||
|         for button, command in self.actions.items(): |         if text: | ||||||
|             text = "%{A" + button.value + ":" + command + ":}" + text + "%{A}" |             for button, command in self.actions.items(): | ||||||
|  |                 # TODO zelbar doesn't support other button types | ||||||
|  |                 if button == Button.CLICK_LEFT: | ||||||
|  |                     text = "%{A:" + command + "}" + text | ||||||
|         return text |         return text | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -239,8 +248,8 @@ class Module(ComposableText): | ||||||
| 
 | 
 | ||||||
| class Alignment(enum.Enum): | class Alignment(enum.Enum): | ||||||
|     LEFT = "l" |     LEFT = "l" | ||||||
|     RIGHT = "r" |  | ||||||
|     CENTER = "c" |     CENTER = "c" | ||||||
|  |     RIGHT = "r" | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Side(ComposableText): | class Side(ComposableText): | ||||||
|  | @ -255,38 +264,64 @@ class Side(ComposableText): | ||||||
|     def generate_markup(self) -> str: |     def generate_markup(self) -> str: | ||||||
|         if not self.children: |         if not self.children: | ||||||
|             return "" |             return "" | ||||||
|         text = "%{" + self.alignment.value + "}" |         markup = "" | ||||||
|         last_section: Section | None = None |         last_section: Section | None = None | ||||||
|  | 
 | ||||||
|  |         default = self.bar.theme.background_color | ||||||
|  |         current = default  # Fallback value | ||||||
|  | 
 | ||||||
|  |         def text( | ||||||
|  |             text: str, | ||||||
|  |             bg: rich.color.ColorTriplet = default, | ||||||
|  |             fg: rich.color.ColorTriplet = default, | ||||||
|  |         ) -> None: | ||||||
|  |             if not text: | ||||||
|  |                 return "" | ||||||
|  |             return ( | ||||||
|  |                 "%{F:" | ||||||
|  |                 + color_0x(fg) | ||||||
|  |                 + "}%{B:" | ||||||
|  |                 + color_0x(bg) | ||||||
|  |                 + "}%{" | ||||||
|  |                 + self.alignment.value | ||||||
|  |                 + "}" | ||||||
|  |                 + text | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|         for module in self.children: |         for module in self.children: | ||||||
|             for section in module.get_sections(): |             for section in module.get_sections(): | ||||||
|                 if section.is_hidden(): |                 if section.is_hidden(): | ||||||
|                     continue |                     continue | ||||||
|                 hexa = section.color.get_truecolor(theme=self.bar.theme).hex |                 current = section.color.get_truecolor(theme=self.bar.theme) | ||||||
|                 if last_section is None: |                 if last_section is None: | ||||||
|                     text += ( |                     markup += ( | ||||||
|                         "%{B" + hexa + "}%{F-}" |                         text("", default, current) | ||||||
|                         if self.alignment == Alignment.LEFT |                         if self.alignment != Alignment.LEFT | ||||||
|                         else "%{B-}%{F" + hexa + "}%{R}%{F-}" |                         else "" | ||||||
|                     ) |                     ) | ||||||
|                 elif isinstance(last_section, SpacerSection): |                 elif isinstance(last_section, SpacerSection): | ||||||
|                     text += "%{B-}%{F" + hexa + "}%{R}%{F-}" |                     markup += text("", default, current) | ||||||
|  |                 elif last_section.color == section.color: | ||||||
|  |                     markup += text( | ||||||
|  |                         "" if self.alignment == Alignment.RIGHT else "", | ||||||
|  |                         current, | ||||||
|  |                         default, | ||||||
|  |                     ) | ||||||
|                 else: |                 else: | ||||||
|                     if self.alignment == Alignment.RIGHT: |                     lastone = last_section.color.get_truecolor(theme=self.bar.theme) | ||||||
|                         text += ( |                     markup += ( | ||||||
|                             "" |                         text("", lastone, current) | ||||||
|                             if last_section.color == section.color |                         if self.alignment == Alignment.RIGHT | ||||||
|                             else "%{F" + hexa + "}%{R}" |                         else text("", current, lastone) | ||||||
|                         ) |                     ) | ||||||
|                     elif last_section.color == section.color: |                 markup += text(section.get_markup(), current, default) | ||||||
|                         text += "" |  | ||||||
|                     else: |  | ||||||
|                         text += "%{R}%{B" + hexa + "}" |  | ||||||
|                     text += "%{F-}" |  | ||||||
|                 text += section.get_markup() |  | ||||||
|                 last_section = section |                 last_section = section | ||||||
|         if self.alignment != Alignment.RIGHT and last_section: |         markup += ( | ||||||
|             text += "%{R}%{B-}" |             text("", default, current) | ||||||
|         return text |             if self.alignment != Alignment.RIGHT and last_section | ||||||
|  |             else "" | ||||||
|  |         ) | ||||||
|  |         return markup | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Screen(ComposableText): | class Screen(ComposableText): | ||||||
|  | @ -296,15 +331,79 @@ class Screen(ComposableText): | ||||||
|         self.children: typing.MutableSequence[Side] |         self.children: typing.MutableSequence[Side] | ||||||
| 
 | 
 | ||||||
|         self.output = output |         self.output = output | ||||||
|  |         self.refresh = asyncio.Event() | ||||||
|  | 
 | ||||||
|  |         self.actionIndex = 0 | ||||||
|  |         self.actions: dict[str, typing.Callable] = {} | ||||||
| 
 | 
 | ||||||
|         for alignment in Alignment: |         for alignment in Alignment: | ||||||
|             Side(parent=self, alignment=alignment) |             Side(parent=self, alignment=alignment) | ||||||
| 
 | 
 | ||||||
|  |     def update_markup(self) -> str: | ||||||
|  |         self.screen.refresh.set() | ||||||
|  | 
 | ||||||
|     def generate_markup(self) -> str: |     def generate_markup(self) -> str: | ||||||
|         return ("%{Sn" + self.output + "}") + "".join( |         return "".join(side.get_markup() for side in self.children) + "\n" | ||||||
|             side.get_markup() for side in self.children | 
 | ||||||
|  |     async def run(self) -> None: | ||||||
|  |         cmd = [ | ||||||
|  |             "zelbar", | ||||||
|  |             "-btm", | ||||||
|  |             "-g", | ||||||
|  |             "0:20", | ||||||
|  |             "-fn", | ||||||
|  |             "DejaVuSansM Nerd Font:size=13", | ||||||
|  |             "-F", | ||||||
|  |             color_0x(self.bar.theme.foreground_color), | ||||||
|  |             "-B", | ||||||
|  |             color_0x(self.bar.theme.background_color), | ||||||
|  |             "-o", | ||||||
|  |             self.output, | ||||||
|  |         ] | ||||||
|  |         print(" ".join(cmd)) | ||||||
|  |         proc = await asyncio.create_subprocess_exec( | ||||||
|  |             *cmd, | ||||||
|  |             stdout=asyncio.subprocess.PIPE, | ||||||
|  |             stdin=asyncio.subprocess.PIPE, | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|  |         async def refresher() -> None: | ||||||
|  |             assert proc.stdin | ||||||
|  |             while True: | ||||||
|  |                 await self.refresh.wait() | ||||||
|  |                 self.refresh.clear() | ||||||
|  |                 markup = self.get_markup() | ||||||
|  |                 # sys.stdout.write(markup)  # DEBUG | ||||||
|  |                 proc.stdin.write(markup.encode()) | ||||||
|  | 
 | ||||||
|  |         async def action_handler() -> None: | ||||||
|  |             assert proc.stdout | ||||||
|  |             while True: | ||||||
|  |                 line = await proc.stdout.readline() | ||||||
|  |                 try: | ||||||
|  |                     command = line.decode().strip() | ||||||
|  |                 except UnicodeDecodeError: | ||||||
|  |                     # FIXME zelbar seems to have some memory issues | ||||||
|  |                     log.exception("Not unicode: %s", str(line)) | ||||||
|  |                     continue | ||||||
|  |                 callback = self.actions.get(command) | ||||||
|  |                 if callback is None: | ||||||
|  |                     log.error("Unknown command: %s", command) | ||||||
|  |                     continue | ||||||
|  |                 callback() | ||||||
|  | 
 | ||||||
|  |         self.bar.add_long_running_task(refresher()) | ||||||
|  |         self.bar.add_long_running_task(action_handler()) | ||||||
|  | 
 | ||||||
|  |     def add_action(self, callback: typing.Callable) -> str: | ||||||
|  |         command = f"com{self.actionIndex:x}" | ||||||
|  |         self.actions[command] = callback | ||||||
|  |         self.actionIndex += 1 | ||||||
|  |         return command | ||||||
|  | 
 | ||||||
|  |     def remove_action(self, command: str) -> None: | ||||||
|  |         del self.actions[command] | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| RICH_DEFAULT_THEME = rich.terminal_theme.DEFAULT_TERMINAL_THEME | RICH_DEFAULT_THEME = rich.terminal_theme.DEFAULT_TERMINAL_THEME | ||||||
| 
 | 
 | ||||||
|  | @ -322,11 +421,8 @@ class Bar(ComposableText): | ||||||
|         self.longRunningTasks: list[asyncio.Task] = [] |         self.longRunningTasks: list[asyncio.Task] = [] | ||||||
|         self.theme = theme |         self.theme = theme | ||||||
| 
 | 
 | ||||||
|         self.refresh = asyncio.Event() |  | ||||||
|         self.taskGroup = asyncio.TaskGroup() |         self.taskGroup = asyncio.TaskGroup() | ||||||
|         self.providers: list[Provider] = [] |         self.providers: list[Provider] = [] | ||||||
|         self.actionIndex = 0 |  | ||||||
|         self.actions: dict[str, typing.Callable] = {} |  | ||||||
| 
 | 
 | ||||||
|         self.periodicProviderTask: typing.Coroutine | None = None |         self.periodicProviderTask: typing.Coroutine | None = None | ||||||
| 
 | 
 | ||||||
|  | @ -343,45 +439,9 @@ class Bar(ComposableText): | ||||||
|         self.longRunningTasks.append(task) |         self.longRunningTasks.append(task) | ||||||
| 
 | 
 | ||||||
|     async def run(self) -> None: |     async def run(self) -> None: | ||||||
|         cmd = [ |  | ||||||
|             "lemonbar", |  | ||||||
|             "-b", |  | ||||||
|             "-a", |  | ||||||
|             "64", |  | ||||||
|             "-f", |  | ||||||
|             "DejaVuSansM Nerd Font:size=10", |  | ||||||
|             "-F", |  | ||||||
|             self.theme.foreground_color.hex, |  | ||||||
|             "-B", |  | ||||||
|             self.theme.background_color.hex, |  | ||||||
|         ] |  | ||||||
|         proc = await asyncio.create_subprocess_exec( |  | ||||||
|             *cmd, stdout=asyncio.subprocess.PIPE, stdin=asyncio.subprocess.PIPE |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         async def refresher() -> None: |  | ||||||
|             assert proc.stdin |  | ||||||
|             while True: |  | ||||||
|                 await self.refresh.wait() |  | ||||||
|                 self.refresh.clear() |  | ||||||
|                 markup = self.get_markup() |  | ||||||
|                 proc.stdin.write(markup.encode()) |  | ||||||
| 
 |  | ||||||
|         async def action_handler() -> None: |  | ||||||
|             assert proc.stdout |  | ||||||
|             while True: |  | ||||||
|                 line = await proc.stdout.readline() |  | ||||||
|                 command = line.decode().strip() |  | ||||||
|                 callback = self.actions.get(command) |  | ||||||
|                 if callback is None: |  | ||||||
|                     # In some conditions on start it's empty |  | ||||||
|                     log.error("Unknown command: %s", command) |  | ||||||
|                     return |  | ||||||
|                 callback() |  | ||||||
| 
 |  | ||||||
|         async with self.taskGroup: |         async with self.taskGroup: | ||||||
|             self.add_long_running_task(refresher()) |             for screen in self.children: | ||||||
|             self.add_long_running_task(action_handler()) |                 await screen.run() | ||||||
|             for provider in self.providers: |             for provider in self.providers: | ||||||
|                 self.add_long_running_task(provider.run()) |                 self.add_long_running_task(provider.run()) | ||||||
| 
 | 
 | ||||||
|  | @ -393,9 +453,6 @@ class Bar(ComposableText): | ||||||
|             loop = asyncio.get_event_loop() |             loop = asyncio.get_event_loop() | ||||||
|             loop.add_signal_handler(signal.SIGINT, finish) |             loop.add_signal_handler(signal.SIGINT, finish) | ||||||
| 
 | 
 | ||||||
|     def generate_markup(self) -> str: |  | ||||||
|         return "".join(screen.get_markup() for screen in self.children) + "\n" |  | ||||||
| 
 |  | ||||||
|     def add_provider( |     def add_provider( | ||||||
|         self, |         self, | ||||||
|         provider: "Provider", |         provider: "Provider", | ||||||
|  | @ -405,24 +462,13 @@ class Bar(ComposableText): | ||||||
|         modules = [] |         modules = [] | ||||||
|         for s, screen in enumerate(self.children): |         for s, screen in enumerate(self.children): | ||||||
|             if screen_num is None or s == screen_num: |             if screen_num is None or s == screen_num: | ||||||
|                 side = next( |                 side = next(filter(lambda s: s.alignment == alignment, screen.children)) | ||||||
|                     filter(lambda s: s.alignment == alignment, screen.children) |  | ||||||
|                 ) |  | ||||||
|                 module = Module(parent=side) |                 module = Module(parent=side) | ||||||
|                 modules.append(module) |                 modules.append(module) | ||||||
|         provider.modules = modules |         provider.modules = modules | ||||||
|         if modules: |         if modules: | ||||||
|             self.providers.append(provider) |             self.providers.append(provider) | ||||||
| 
 | 
 | ||||||
|     def add_action(self, callback: typing.Callable) -> str: |  | ||||||
|         command = f"{self.actionIndex:x}" |  | ||||||
|         self.actions[command] = callback |  | ||||||
|         self.actionIndex += 1 |  | ||||||
|         return command |  | ||||||
| 
 |  | ||||||
|     def remove_action(self, command: str) -> None: |  | ||||||
|         del self.actions[command] |  | ||||||
| 
 |  | ||||||
|     def launch(self) -> None: |     def launch(self) -> None: | ||||||
|         # Using GLib's event loop so we can run GLib's code |         # Using GLib's event loop so we can run GLib's code | ||||||
|         policy = gi.events.GLibEventLoopPolicy() |         policy = gi.events.GLibEventLoopPolicy() | ||||||
|  | @ -430,6 +476,9 @@ class Bar(ComposableText): | ||||||
|         loop = policy.get_event_loop() |         loop = policy.get_event_loop() | ||||||
|         loop.run_until_complete(self.run()) |         loop.run_until_complete(self.run()) | ||||||
| 
 | 
 | ||||||
|  |     def update_markup(self) -> None: | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| class Provider: | class Provider: | ||||||
|     section_type: type[Section] = Section |     section_type: type[Section] = Section | ||||||
|  | @ -462,9 +511,7 @@ class SingleSectionProvider(MirrorProvider): | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class StaticProvider(SingleSectionProvider): | class StaticProvider(SingleSectionProvider): | ||||||
|     def __init__( |     def __init__(self, text: str, color: rich.color.Color = RICH_DEFAULT_COLOR) -> None: | ||||||
|         self, text: str, color: rich.color.Color = RICH_DEFAULT_COLOR |  | ||||||
|     ) -> None: |  | ||||||
|         super().__init__(color=color) |         super().__init__(color=color) | ||||||
|         self.text = text |         self.text = text | ||||||
| 
 | 
 | ||||||
|  | @ -524,9 +571,7 @@ class StatefulSectionProvider(Provider): | ||||||
|     section_type = StatefulSection |     section_type = StatefulSection | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class SingleStatefulSectionProvider( | class SingleStatefulSectionProvider(StatefulSectionProvider, SingleSectionProvider): | ||||||
|     StatefulSectionProvider, SingleSectionProvider |  | ||||||
| ): |  | ||||||
|     section: StatefulSection |     section: StatefulSection | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -545,9 +590,7 @@ class MultiSectionsProvider(Provider): | ||||||
|     async def do_nothing() -> None: |     async def do_nothing() -> None: | ||||||
|         pass |         pass | ||||||
| 
 | 
 | ||||||
|     async def update_sections( |     async def update_sections(self, sections: set[Sortable], module: Module) -> None: | ||||||
|         self, sections: set[Sortable], module: Module |  | ||||||
|     ) -> None: |  | ||||||
|         module_sections = self.sectionKeys[module] |         module_sections = self.sectionKeys[module] | ||||||
|         async with asyncio.TaskGroup() as tg: |         async with asyncio.TaskGroup() as tg: | ||||||
|             for sort_key in sections: |             for sort_key in sections: | ||||||
|  | @ -556,9 +599,7 @@ class MultiSectionsProvider(Provider): | ||||||
|                     section = self.section_type( |                     section = self.section_type( | ||||||
|                         parent=module, sort_key=sort_key, color=self.color |                         parent=module, sort_key=sort_key, color=self.color | ||||||
|                     ) |                     ) | ||||||
|                     self.updaters[section] = await self.get_section_updater( |                     self.updaters[section] = await self.get_section_updater(section) | ||||||
|                         section |  | ||||||
|                     ) |  | ||||||
|                 module_sections[sort_key] = section |                 module_sections[sort_key] = section | ||||||
| 
 | 
 | ||||||
|                 updater = self.updaters[section] |                 updater = self.updaters[section] | ||||||
|  | @ -608,9 +649,7 @@ class PeriodicProvider(Provider): | ||||||
|             bar.add_long_running_task(bar.periodicProviderTask) |             bar.add_long_running_task(bar.periodicProviderTask) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class PeriodicStatefulProvider( | class PeriodicStatefulProvider(SingleStatefulSectionProvider, PeriodicProvider): | ||||||
|     SingleStatefulSectionProvider, PeriodicProvider |  | ||||||
| ): |  | ||||||
|     async def run(self) -> None: |     async def run(self) -> None: | ||||||
|         await super().run() |         await super().run() | ||||||
|         self.section.set_changed_state(self.loop) |         self.section.set_changed_state(self.loop) | ||||||
|  |  | ||||||
|  | @ -6,7 +6,7 @@ | ||||||
| }: | }: | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|     xsession.windowManager.i3.config.bars = [ ]; |     wayland.windowManager.sway.config.bars = [ ]; | ||||||
|     programs.autorandr.hooks.postswitch = { |     programs.autorandr.hooks.postswitch = { | ||||||
|       frobar = "${pkgs.systemd}/bin/systemctl --user restart frobar"; |       frobar = "${pkgs.systemd}/bin/systemctl --user restart frobar"; | ||||||
|     }; |     }; | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|       Service = { |       Service = { | ||||||
|         # Wait for i3 to start. Can't use ExecStartPre because otherwise it blocks graphical-session.target, and there's nothing i3/systemd |         # Wait for i3 to start. Can't use ExecStartPre because otherwise it blocks graphical-session.target, and there's nothing i3/systemd | ||||||
|         # TODO Do that better |         # TODO Do that better | ||||||
|         ExecStart = ''${pkgs.bash}/bin/bash -c "while ! ${pkgs.i3}/bin/i3-msg; do ${pkgs.coreutils}/bin/sleep 1; done; ${pkgs.callPackage ./. { }}/bin/frobar"''; |         ExecStart = ''${pkgs.bash}/bin/bash -c "while ! ${pkgs.sway}/bin/swaymsg; do ${pkgs.coreutils}/bin/sleep 1; done; ${pkgs.callPackage ./. { }}/bin/frobar"''; | ||||||
|       }; |       }; | ||||||
| 
 | 
 | ||||||
|       Install = { |       Install = { | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   # FOCUS |   # FOCUS | ||||||
|  |   # FIXME There should be an option in sway | ||||||
|   focus = "exec ${pkgs.writeShellScript "i3-focus-window" '' |   focus = "exec ${pkgs.writeShellScript "i3-focus-window" '' | ||||||
|     WINDOW=`${pkgs.xdotool}/bin/xdotool getwindowfocus` |     WINDOW=`${pkgs.xdotool}/bin/xdotool getwindowfocus` | ||||||
|     eval `${pkgs.xdotool}/bin/xdotool getwindowgeometry --shell $WINDOW` # this brings in variables WIDTH and HEIGHT |     eval `${pkgs.xdotool}/bin/xdotool getwindowgeometry --shell $WINDOW` # this brings in variables WIDTH and HEIGHT | ||||||
|  | @ -57,15 +58,13 @@ let | ||||||
|   forEachWorkspace = f: map (w: f w) workspaces; |   forEachWorkspace = f: map (w: f w) workspaces; | ||||||
| 
 | 
 | ||||||
|   # MISC |   # MISC | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
|   rofi = "exec --no-startup-id ${config.programs.rofi.package}/bin/rofi"; |   rofi = "exec --no-startup-id ${config.programs.rofi.package}/bin/rofi"; | ||||||
|   modes = config.frogeye.desktop.i3.bindmodes; |   modes = config.frogeye.desktop.i3.bindmodes; | ||||||
|   x11_screens = config.frogeye.desktop.x11_screens; |   x11_screens = config.frogeye.desktop.x11_screens; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = lib.mkIf config.xsession.windowManager.i3.enable { |   config = lib.mkIf config.wayland.windowManager.sway.enable { | ||||||
|     stylix.targets.i3.enable = false; |  | ||||||
|     services.picom.enable = true; |  | ||||||
|     xdg.configFile = { |     xdg.configFile = { | ||||||
|       "rofimoji.rc" = { |       "rofimoji.rc" = { | ||||||
|         text = '' |         text = '' | ||||||
|  | @ -75,10 +74,12 @@ in | ||||||
|         ''; |         ''; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config = { |     wayland.windowManager.sway.config = { | ||||||
|  |       input."*".xkb_variant = "qwerty-fr"; | ||||||
|       modifier = lib.mkDefault "Mod4"; |       modifier = lib.mkDefault "Mod4"; | ||||||
|       fonts = { |       fonts = { | ||||||
|         names = [ config.stylix.fonts.sansSerif.name ]; |         names = [ config.stylix.fonts.sansSerif.name ]; | ||||||
|  |         size = lib.mkForce 8.0; | ||||||
|       }; |       }; | ||||||
|       terminal = "alacritty"; |       terminal = "alacritty"; | ||||||
|       colors = |       colors = | ||||||
|  | @ -167,7 +168,7 @@ in | ||||||
|           "${mod}+Shift+r" = "restart"; |           "${mod}+Shift+r" = "restart"; | ||||||
|           "${mod}+Shift+e" = "exit"; |           "${mod}+Shift+e" = "exit"; | ||||||
|         } |         } | ||||||
|         // lib.mapAttrs' (k: v: lib.nameValuePair v.enter "mode ${v.name}") ( |         // lib.mapAttrs' (k: v: lib.nameValuePair v.enter ''mode "${v.name}"'') ( | ||||||
|           lib.filterAttrs (k: v: v.enter != null) modes |           lib.filterAttrs (k: v: v.enter != null) modes | ||||||
|         ) |         ) | ||||||
|         // lib.attrsets.mergeAttrsList ( |         // lib.attrsets.mergeAttrsList ( | ||||||
|  | @ -223,9 +224,12 @@ in | ||||||
|           { window_role = "task_dialog"; } |           { window_role = "task_dialog"; } | ||||||
|         ]; |         ]; | ||||||
|       }; |       }; | ||||||
|  |       seat."*" = { | ||||||
|  |         hide_cursor = "10"; | ||||||
|  |       }; | ||||||
|       startup = [ |       startup = [ | ||||||
|         { |         { | ||||||
|           notification = false; |           # notification = false; | ||||||
|           command = "${ |           command = "${ | ||||||
|             pkgs.writeShellApplication { |             pkgs.writeShellApplication { | ||||||
|               name = "batteryNotify"; |               name = "batteryNotify"; | ||||||
|  | @ -259,10 +263,10 @@ in | ||||||
|       }; |       }; | ||||||
|       "[L] Vérouillage [E] Déconnexion [S] Veille [H] Hibernation [R] Redémarrage [P] Extinction" = { |       "[L] Vérouillage [E] Déconnexion [S] Veille [H] Hibernation [R] Redémarrage [P] Extinction" = { | ||||||
|         bindings = { |         bindings = { | ||||||
|           "l" = "exec --no-startup-id exec xlock, mode default"; |           "l" = "exec --no-startup-id exec ${pkgs.systemd}/bin/loginctl lock-session, mode default"; | ||||||
|           "e" = "exit, mode default"; |           "e" = "exit, mode default"; | ||||||
|           "s" = "exec --no-startup-id exec xlock & ${pkgs.systemd}/bin/systemctl suspend --check-inhibitors=no, mode default"; |           "s" = "exec --no-startup-id exec ${pkgs.systemd}/bin/systemctl suspend --check-inhibitors=no, mode default"; | ||||||
|           "h" = "exec --no-startup-id exec xlock & ${pkgs.systemd}/bin/systemctl hibernate, mode default"; |           "h" = "exec --no-startup-id exec ${pkgs.systemd}/bin/systemctl hibernate, mode default"; | ||||||
|           "r" = "exec --no-startup-id ${pkgs.systemd}/bin/systemctl reboot, mode default"; |           "r" = "exec --no-startup-id ${pkgs.systemd}/bin/systemctl reboot, mode default"; | ||||||
|           "p" = "exec --no-startup-id ${pkgs.systemd}/bin/systemctl poweroff -i, mode default"; |           "p" = "exec --no-startup-id ${pkgs.systemd}/bin/systemctl poweroff -i, mode default"; | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|  | @ -19,68 +19,74 @@ let | ||||||
|     </svg> |     </svg> | ||||||
|   ''; |   ''; | ||||||
|   lockPng = pkgs.runCommand "lock.png" { } "${pkgs.imagemagick}/bin/convert ${lockSvg} $out"; |   lockPng = pkgs.runCommand "lock.png" { } "${pkgs.imagemagick}/bin/convert ${lockSvg} $out"; | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
|   xautolockState = "${config.xdg.cacheHome}/xautolock"; |   idleTime = 10; # minutes | ||||||
|  |   preLock = "${pkgs.writeShellScript "prelock" '' | ||||||
|  |     ${config.frogeye.hooks.lock} | ||||||
|  |     ${pkgs.procps}/bin/pkill -USR1 swayidle | ||||||
|  |   ''}"; | ||||||
| in | in | ||||||
| { | { | ||||||
|  |   # FIXME Not really working, needs test | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|     home.packages = [ |     programs.swaylock = { | ||||||
|       (pkgs.writeShellApplication { |       enable = true; | ||||||
|         name = "xlock"; |       settings = { | ||||||
|         text = '' |         color = lib.mkForce (builtins.substring 1 6 lockColors.d); | ||||||
|           ${config.frogeye.hooks.lock} |         image = lib.mkForce lockPng; | ||||||
|           # TODO Reevaluate whether we want this or not |         tiling = true; | ||||||
|           if ! ${pkgs.lightdm}/bin/dm-tool lock |         ignore-empty-password = true; | ||||||
|           then |         indicator-idle-visible = false; | ||||||
|               if [ -d ${config.xdg.cacheHome}/lockpatterns ] |         show-failed-attempts = true; | ||||||
|               then |         indicator-radius = true; | ||||||
|                   pattern=$(${pkgs.findutils} ${config.xdg.cacheHome}/lockpatterns | sort -R | head -1) |       }; | ||||||
|               else |     }; | ||||||
|                   pattern=${lockPng} |     services.swayidle = { | ||||||
|               fi |       enable = true; | ||||||
|               revert() { |       events = [ | ||||||
|                   ${pkgs.xorg.xset}/bin/xset dpms 0 0 0 |         { | ||||||
|               } |           event = "lock"; | ||||||
|               trap revert SIGHUP SIGINT SIGTERM |           command = "${preLock}"; | ||||||
|               ${pkgs.xorg.xset}/bin/xset dpms 5 5 5 |         } | ||||||
|               ${pkgs.i3lock}/bin/i3lock --nofork --color ${ |         # { | ||||||
|                 builtins.substring 1 6 lockColors.d |         #   event = "before-sleep"; | ||||||
|               } --image="$pattern" --tiling --ignore-empty-password |         #   command = "xlock"; | ||||||
|               revert |         # } | ||||||
|           fi |       ]; | ||||||
|         ''; |       timeouts = [ | ||||||
|       }) |         { | ||||||
|     ]; |           # Warn | ||||||
|     xsession.windowManager.i3.config = { |           timeout = (idleTime - 1) * 60; | ||||||
|  |           command = ''${lib.getExe pkgs.libnotify} Turning screen off in 1 minute"''; | ||||||
|  |           resumeCommand = ''${config.wayland.windowManager.sway.package}/bin/swaymsg "output * dpms on"''; | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |           # Screen off | ||||||
|  |           timeout = idleTime * 60; | ||||||
|  |           command = ''${config.wayland.windowManager.sway.package}/bin/swaymsg "output * dpms off"''; | ||||||
|  |           resumeCommand = ''${config.wayland.windowManager.sway.package}/bin/swaymsg "output * dpms off"''; | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |           # Lock | ||||||
|  |           timeout = (idleTime + 1) * 60; | ||||||
|  |           command = ''${config.wayland.windowManager.sway.package}/bin/swaymsg "output * dpms off"''; | ||||||
|  |         } | ||||||
|  |       ]; | ||||||
|  |     }; | ||||||
|  |     wayland.windowManager.sway.config = { | ||||||
|       keybindings = { |       keybindings = { | ||||||
|         # Screen off commands |         # Screen off commands | ||||||
|         "${mod}+F1" = "--release exec --no-startup-id ${pkgs.xorg.xset}/bin/xset dpms force off"; |         "${mod}+F1" = ''--release exec ${pkgs.procps}/bin/pkill -USR1 swayidle''; | ||||||
|         # Toggle to save on buttons |         # Toggle to save on buttons | ||||||
|         # xautolock -toggle doesn't allow to read state. |         "${mod}+F5" = "exec --no-startup-id ${pkgs.writeShellScript "swayidle-toggle" '' | ||||||
|         # Writing into a file also allows frobar to display a lock icon |           if ${pkgs.systemd}/bin/systemctl --user is-active swayidle --quiet | ||||||
|         "${mod}+F5" = "exec --no-startup-id ${pkgs.writeShellScript "xautolock-toggle" '' |  | ||||||
|           state="$(cat "${xautolockState}")" |  | ||||||
|           if [ "$state" = "disabled" ] |  | ||||||
|           then |           then | ||||||
|             ${pkgs.xautolock}/bin/xautolock -enable |             ${pkgs.systemd}/bin/systemctl --user stop swayidle | ||||||
|             echo enabled > ${xautolockState} |  | ||||||
|           else |           else | ||||||
|             ${pkgs.xautolock}/bin/xautolock -disable |             ${pkgs.systemd}/bin/systemctl --user start swayidle | ||||||
|             echo disabled > ${xautolockState} |  | ||||||
|           fi |           fi | ||||||
|         ''}"; |         ''}"; | ||||||
|       }; |       }; | ||||||
|       startup = [ |  | ||||||
|         # Stop screen after 10 minutes, 1 minutes after lock it |  | ||||||
|         { |  | ||||||
|           notification = false; |  | ||||||
|           command = "${pkgs.writeShellScript "xautolock-start" '' |  | ||||||
|             echo enabled > ${xautolockState} |  | ||||||
|             ${pkgs.xautolock}/bin/xautolock -time 10 -locker '${pkgs.xorg.xset}/bin/xset dpms force standby' -killtime 1 -killer xlock |  | ||||||
|           ''}"; |  | ||||||
|         } |  | ||||||
|         #  services.screen-locker.xautolock is hardcoded to use systemd for -locker (doesn't even work...) |  | ||||||
|       ]; |  | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -58,7 +58,7 @@ | ||||||
|         }; |         }; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       "XF86AudioPrev" = "exec ${lib.getExe pkgs.playerctl} previous"; |       "XF86AudioPrev" = "exec ${lib.getExe pkgs.playerctl} previous"; | ||||||
|       "XF86AudioPlay" = "exec ${lib.getExe pkgs.playerctl} play-pause"; |       "XF86AudioPlay" = "exec ${lib.getExe pkgs.playerctl} play-pause"; | ||||||
|       "XF86AudioNext" = "exec ${lib.getExe pkgs.playerctl} next"; |       "XF86AudioNext" = "exec ${lib.getExe pkgs.playerctl} next"; | ||||||
|  |  | ||||||
|  | @ -33,7 +33,7 @@ in | ||||||
|         return_bindings = false; |         return_bindings = false; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.window.commands = [ |     wayland.windowManager.sway.config.window.commands = [ | ||||||
|       # Open specific applications in floating mode |       # Open specific applications in floating mode | ||||||
|       { |       { | ||||||
|         criteria = { |         criteria = { | ||||||
|  |  | ||||||
|  | @ -6,16 +6,23 @@ | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   dir = config.xdg.userDirs.extraConfig.XDG_SCREENSHOTS_DIR; |   dir = config.xdg.userDirs.extraConfig.XDG_SCREENSHOTS_DIR; | ||||||
|   scrot = "${pkgs.scrot}/bin/scrot --exec '${pkgs.coreutils}/bin/mv $f ${dir}/ && ${pkgs.optipng}/bin/optipng ${dir}/$f'"; |   gs = | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |     mode: | ||||||
|  |     pkgs.writeShellScript "grimshot-${mode}" '' | ||||||
|  |       path="${dir}/$(date -Isec).png" | ||||||
|  |       ${lib.getExe pkgs.sway-contrib.grimshot} savecopy ${mode} "$path" | ||||||
|  |       ${pkgs.optipng}/bin/optipng "$path" | ||||||
|  |     ''; | ||||||
|  |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|     frogeye.folders.screenshots.path = "Screenshots"; |     frogeye.folders.screenshots.path = "Screenshots"; | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     home.packages = [ pkgs.sway-contrib.grimshot ]; | ||||||
|       "Print" = "exec ${scrot} --focused"; |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       "${mod}+Print" = "exec ${scrot}"; |       "Print" = "exec ${gs "active"}"; | ||||||
|       "Ctrl+Print" = "--release exec ${scrot} --select"; |       "${mod}+Print" = "exec ${gs "screen"}"; | ||||||
|  |       "Ctrl+Print" = "exec ${gs "anything"}"; | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -5,7 +5,7 @@ | ||||||
|   ... |   ... | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|  | @ -149,7 +149,7 @@ in | ||||||
|         }; |         }; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.keybindings = { |     wayland.windowManager.sway.config.keybindings = { | ||||||
|       "${mod}+Return" = "exec ${config.programs.alacritty.package}/bin/alacritty msg create-window -e zsh || exec ${config.programs.alacritty.package}/bin/alacritty -e zsh"; |       "${mod}+Return" = "exec ${config.programs.alacritty.package}/bin/alacritty msg create-window -e zsh || exec ${config.programs.alacritty.package}/bin/alacritty -e zsh"; | ||||||
|       # -e zsh is for systems where I can't configure my user's shell |       # -e zsh is for systems where I can't configure my user's shell | ||||||
|       "${mod}+Shift+Return" = "exec ${config.programs.urxvt.package}/bin/urxvt"; |       "${mod}+Shift+Return" = "exec ${config.programs.urxvt.package}/bin/urxvt"; | ||||||
|  |  | ||||||
|  | @ -5,7 +5,7 @@ | ||||||
|   ... |   ... | ||||||
| }: | }: | ||||||
| let | let | ||||||
|   mod = config.xsession.windowManager.i3.config.modifier; |   mod = config.wayland.windowManager.sway.config.modifier; | ||||||
| in | in | ||||||
| { | { | ||||||
|   config = { |   config = { | ||||||
|  | @ -67,7 +67,7 @@ in | ||||||
|       }; |       }; | ||||||
|       password-store.enable = true; |       password-store.enable = true; | ||||||
|     }; |     }; | ||||||
|     xsession.windowManager.i3.config.keybindings."${mod}+c" = "exec --no-startup-id ${config.programs.rofi.pass.package}/bin/rofi-pass --last-used"; |     wayland.windowManager.sway.config.keybindings."${mod}+c" = "exec --no-startup-id ${config.programs.rofi.pass.package}/bin/rofi-pass --last-used"; | ||||||
|     # TODO Try autopass.cr |     # TODO Try autopass.cr | ||||||
|   }; |   }; | ||||||
|   options = { |   options = { | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
|   ... |   ... | ||||||
| }: | }: | ||||||
| { | { | ||||||
|  |  # FIXME | ||||||
|   config = lib.mkIf (builtins.length config.frogeye.desktop.x11_screens > 1) { |   config = lib.mkIf (builtins.length config.frogeye.desktop.x11_screens > 1) { | ||||||
|     services = { |     services = { | ||||||
|       autorandr.enable = true; |       autorandr.enable = true; | ||||||
|  |  | ||||||
|  | @ -7,6 +7,7 @@ | ||||||
| { | { | ||||||
|   config = lib.mkIf config.frogeye.desktop.xorg { |   config = lib.mkIf config.frogeye.desktop.xorg { | ||||||
|     boot.kernelModules = [ "i2c-dev" ]; # Allows using ddcutil |     boot.kernelModules = [ "i2c-dev" ]; # Allows using ddcutil | ||||||
|  |     programs.sway.enable = true; | ||||||
|     security.rtkit.enable = true; # Recommended for pipewire |     security.rtkit.enable = true; # Recommended for pipewire | ||||||
|     services = { |     services = { | ||||||
|       blueman.enable = true; |       blueman.enable = true; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue