diff --git a/home-mods/firefox/mkFirefoxModule.nix b/home-mods/firefox/mkFirefoxModule.nix
new file mode 100644
index 0000000..abbc1a2
--- /dev/null
+++ b/home-mods/firefox/mkFirefoxModule.nix
@@ -0,0 +1,1130 @@
+{ modulePath
+, name
+, description ? null
+, wrappedPackageName ? null
+, wrappedPkg ? null
+, unwrappedPackageName ? null
+, platforms
+, visible ? false
+,
+}: { config
+ , lib
+ , pkgs
+ , ...
+ }:
+with lib; let
+ inherit (pkgs.stdenv.hostPlatform) isDarwin;
+
+ moduleName = concatStringsSep "." modulePath;
+
+ cfg = getAttrFromPath modulePath config;
+
+ jsonFormat = pkgs.formats.json { };
+
+ supportedPlatforms = flatten (attrVals (attrNames platforms) lib.platforms);
+
+ isWrapped = versionAtLeast config.home.stateVersion "19.09" && wrappedPackageName != null;
+
+ defaultPackageName =
+ if isWrapped
+ then wrappedPackageName
+ else unwrappedPackageName;
+
+ packageName =
+ if wrappedPackageName != null
+ then wrappedPackageName
+ else unwrappedPackageName;
+
+ profilesPath =
+ if isDarwin
+ then "${cfg.configPath}/Profiles"
+ else cfg.configPath;
+
+ nativeMessagingHostsPath =
+ if isDarwin
+ then "${cfg.vendorPath}/NativeMessagingHosts"
+ else "${cfg.vendorPath}/native-messaging-hosts";
+
+ nativeMessagingHostsJoined = pkgs.symlinkJoin {
+ name = "ff_native-messaging-hosts";
+ paths =
+ [
+ # Link a .keep file to keep the directory around
+ (pkgs.writeTextDir "lib/mozilla/native-messaging-hosts/.keep" "")
+ # Link package configured native messaging hosts (entire browser actually)
+ cfg.finalPackage
+ ]
+ # Link user configured native messaging hosts
+ ++ cfg.nativeMessagingHosts;
+ };
+
+ # The extensions path shared by all profiles; will not be supported
+ # by future browser versions.
+ extensionPath = "extensions/{ec8030f7-c20a-464f-9b0e-13a3a9e97384}";
+
+ profiles =
+ flip mapAttrs' cfg.profiles
+ (
+ _: profile:
+ nameValuePair "Profile${toString profile.id}" {
+ Name = profile.name;
+ Path =
+ if isDarwin
+ then "Profiles/${profile.path}"
+ else profile.path;
+ IsRelative = 1;
+ Default =
+ if profile.isDefault
+ then 1
+ else 0;
+ ZenAvatarPath = "chrome://browser/content/zen-avatars/avatar-91.svg";
+ }
+ )
+ // {
+ General = {
+ StartWithLastProfile = 1;
+ Version = 2;
+ };
+ };
+
+ profilesIni = generators.toINI { } profiles;
+
+ userPrefValue = pref:
+ builtins.toJSON (
+ if isBool pref || isInt pref || isString pref
+ then pref
+ else builtins.toJSON pref
+ );
+
+ mkUserJs = prefs: extraPrefs: bookmarks:
+ let
+ prefs' =
+ lib.optionalAttrs ([ ] != bookmarks)
+ {
+ "browser.bookmarks.file" = toString (browserBookmarksFile bookmarks);
+ "browser.places.importBookmarksHTML" = true;
+ }
+ // prefs;
+ in
+ ''
+ // Generated by Home Manager.
+
+ ${concatStrings (
+ mapAttrsToList (name: value: ''
+ user_pref("${name}", ${userPrefValue value});
+ '')
+ prefs'
+ )}
+
+ ${extraPrefs}
+ '';
+
+ mkContainersJson = containers:
+ let
+ containerToIdentity = _: container: {
+ userContextId = container.id;
+ name = container.name;
+ icon = container.icon;
+ color = container.color;
+ public = true;
+ };
+ in
+ ''
+ ${builtins.toJSON {
+ version = 4;
+ lastUserContextId = elemAt (mapAttrsToList (_: container: container.id) containers) 0;
+ identities =
+ mapAttrsToList containerToIdentity containers
+ ++ [
+ {
+ userContextId = 4294967294; # 2^32 - 2
+ name = "userContextIdInternal.thumbnail";
+ icon = "";
+ color = "";
+ accessKey = "";
+ public = false;
+ }
+ {
+ userContextId = 4294967295; # 2^32 - 1
+ name = "userContextIdInternal.webextStorageLocal";
+ icon = "";
+ color = "";
+ accessKey = "";
+ public = false;
+ }
+ ];
+ }}
+ '';
+
+ browserBookmarksFile = bookmarks:
+ let
+ indent = level: lib.concatStringsSep "" (map (lib.const " ") (lib.range 1 level));
+
+ bookmarkToHTML = indentLevel: bookmark: ''${indent indentLevel}
${escapeXML bookmark.name}'';
+
+ directoryToHTML = indentLevel: directory: ''
+ ${indent indentLevel}${
+ if directory.toolbar
+ then ''Bookmarks Toolbar''
+ else ''${escapeXML directory.name}''
+ }
+ ${indent indentLevel}
+ ${allItemsToHTML (indentLevel + 1) directory.bookmarks}
+ ${indent indentLevel}
'';
+
+ itemToHTMLOrRecurse = indentLevel: item:
+ if item ? "url"
+ then bookmarkToHTML indentLevel item
+ else directoryToHTML indentLevel item;
+
+ allItemsToHTML = indentLevel: bookmarks: lib.concatStringsSep "\n" (map (itemToHTMLOrRecurse indentLevel) bookmarks);
+
+ bookmarkEntries = allItemsToHTML 1 bookmarks;
+ in
+ pkgs.writeText "${packageName}-bookmarks.html" ''
+
+
+
+
Bookmarks
+ Bookmarks Menu
+
+ ${bookmarkEntries}
+
+ '';
+
+ mkNoDuplicateAssertion = entities: entityKind: (
+ let
+ # Return an attribute set with entity IDs as keys and a list of
+ # entity names with corresponding ID as value. An ID is present in
+ # the result only if more than one entity has it. The argument
+ # entities is a list of AttrSet of one id/name pair.
+ findDuplicateIds = entities: filterAttrs (_entityId: entityNames: length entityNames != 1) (zipAttrs entities);
+
+ duplicates = findDuplicateIds (
+ mapAttrsToList (entityName: entity: { "${toString entity.id}" = entityName; }) entities
+ );
+
+ mkMsg = entityId: entityNames: " - ID ${entityId} is used by " + concatStringsSep ", " entityNames;
+ in
+ {
+ assertion = duplicates == { };
+ message =
+ ''
+ Must not have a ${name} ${entityKind} with an existing ID but
+ ''
+ + concatStringsSep "\n" (mapAttrsToList mkMsg duplicates);
+ }
+ );
+
+ wrapPackage = package:
+ let
+ # The configuration expected by the Firefox wrapper.
+ fcfg = {
+ enableGnomeExtensions = cfg.enableGnomeExtensions;
+ };
+
+ # A bit of hackery to force a config into the wrapper.
+ browserName = package.browserName or (builtins.parseDrvName package.name).name;
+
+ # The configuration expected by the Firefox wrapper builder.
+ bcfg = setAttrByPath [ browserName ] fcfg;
+ in
+ if package == null
+ then null
+ else if isDarwin
+ then package
+ else if isWrapped
+ then
+ package.override
+ (old: {
+ cfg = old.cfg or { } // fcfg;
+ extraPolicies = (old.extraPolicies or { }) // cfg.policies;
+ })
+ else (pkgs.wrapFirefox.override { config = bcfg; }) package { };
+in
+{
+ options = setAttrByPath modulePath {
+ enable = mkOption {
+ type = types.bool;
+ default = false;
+ example = true;
+ description = ''
+ Whether to enable ${name}.${optionalString (description != null) " ${description}"}
+ ${optionalString (!visible) "See `programs.zen-browser` for more configuration options."}
+ '';
+ };
+
+ package = mkOption {
+ inherit visible;
+ type = with types; nullOr package;
+ default = wrappedPkg;
+ defaultText = literalExpression "pkgs.${packageName}";
+ example = literalExpression ''
+ pkgs.${packageName}.override {
+ # See nixpkgs' firefox/wrapper.nix to check which options you can use
+ nativeMessagingHosts = [
+ # Gnome shell native connector
+ pkgs.gnome-browser-connector
+ # Tridactyl native connector
+ pkgs.tridactyl-native
+ ];
+ }
+ '';
+ description = ''
+ The ${name} package to use. If state version ≥ 19.09 then
+ this should be a wrapped ${name} package. For earlier state
+ versions it should be an unwrapped ${name} package.
+ Set to `null` to disable installing ${name}.
+ '';
+ };
+
+ languagePacks = mkOption {
+ type = types.listOf types.str;
+ default = [ ];
+ description = ''
+ The language packs to install. Available language codes can be found
+ on the releases page:
+ `https://releases.mozilla.org/pub/firefox/releases/''${version}/linux-x86_64/xpi/`,
+ replacing `''${version}` with the version of Firefox you have.
+ '';
+ example = [
+ "en-GB"
+ "de"
+ ];
+ };
+
+ name = mkOption {
+ internal = true;
+ type = types.str;
+ default = name;
+ example = "Firefox";
+ description = "The name of the browser.";
+ };
+
+ wrappedPackageName = mkOption {
+ internal = true;
+ type = with types; nullOr str;
+ default = wrappedPackageName;
+ description = "Name of the wrapped browser package.";
+ };
+
+ vendorPath = mkOption {
+ internal = true;
+ type = with types; nullOr str;
+ default = with platforms;
+ if isDarwin
+ then darwin.vendorPath or null
+ else linux.vendorPath or null;
+ example = ".mozilla";
+ description = "Directory containing the native messaging hosts directory.";
+ };
+
+ configPath = mkOption {
+ internal = true;
+ type = types.str;
+ default = with platforms;
+ if isDarwin
+ then darwin.configPath
+ else linux.configPath;
+ example = ".mozilla/firefox";
+ description = "Directory containing the ${name} configuration files.";
+ };
+
+ nativeMessagingHosts = optionalAttrs (cfg.vendorPath != null) (mkOption {
+ inherit visible;
+ type = types.listOf types.package;
+ default = [ ];
+ description = ''
+ Additional packages containing native messaging hosts that should be
+ made available to ${name} extensions.
+ '';
+ });
+
+ finalPackage = mkOption {
+ inherit visible;
+ type = with types; nullOr package;
+ readOnly = true;
+ description = "Resulting ${cfg.name} package.";
+ };
+
+ policies = optionalAttrs (wrappedPackageName != null) (mkOption {
+ inherit visible;
+ type = types.attrsOf jsonFormat.type;
+ default = { };
+ description = "[See list of policies](https://mozilla.github.io/policy-templates/).";
+ example = {
+ DefaultDownloadDirectory = "\${home}/Downloads";
+ BlockAboutConfig = true;
+ };
+ });
+
+ profiles = mkOption {
+ inherit visible;
+ type = types.attrsOf (
+ types.submodule (
+ { config
+ , name
+ , ...
+ }: {
+ options = {
+ name = mkOption {
+ type = types.str;
+ default = name;
+ description = "Profile name.";
+ };
+
+ id = mkOption {
+ type = types.ints.unsigned;
+ default = 0;
+ description = ''
+ Profile ID. This should be set to a unique number per profile.
+ '';
+ };
+
+ settings = mkOption {
+ type = types.attrsOf (
+ jsonFormat.type
+ // {
+ description = "${name} preference (int, bool, string, and also attrs, list, float as a JSON string)";
+ }
+ );
+ default = { };
+ example = literalExpression ''
+ {
+ "browser.startup.homepage" = "https://nixos.org";
+ "browser.search.region" = "GB";
+ "browser.search.isUS" = false;
+ "distribution.searchplugins.defaultLocale" = "en-GB";
+ "general.useragent.locale" = "en-GB";
+ "browser.bookmarks.showMobileBookmarks" = true;
+ "browser.newtabpage.pinned" = [{
+ title = "NixOS";
+ url = "https://nixos.org";
+ }];
+ }
+ '';
+ description = ''
+ Attribute set of ${name} preferences.
+
+ ${name} only supports int, bool, and string types for
+ preferences, but home-manager will automatically
+ convert all other JSON-compatible values into strings.
+ '';
+ };
+
+ extraConfig = mkOption {
+ type = types.lines;
+ default = "";
+ description = ''
+ Extra preferences to add to {file}`user.js`.
+ '';
+ };
+
+ userChrome = mkOption {
+ type = types.lines;
+ default = "";
+ description = "Custom ${name} user chrome CSS.";
+ example = ''
+ /* Hide tab bar in FF Quantum */
+ @-moz-document url(chrome://browser/content/browser.xul), url(chrome://browser/content/browser.xhtml) {
+ #TabsToolbar {
+ visibility: collapse !important;
+ margin-bottom: 21px !important;
+ }
+
+ #sidebar-box[sidebarcommand="treestyletab_piro_sakura_ne_jp-sidebar-action"] #sidebar-header {
+ visibility: collapse !important;
+ }
+ }
+ '';
+ };
+
+ userContent = mkOption {
+ type = types.lines;
+ default = "";
+ description = "Custom ${name} user content CSS.";
+ example = ''
+ /* Hide scrollbar in FF Quantum */
+ *{scrollbar-width:none !important}
+ '';
+ };
+
+ bookmarks = mkOption {
+ type =
+ let
+ bookmarkSubmodule =
+ types.submodule
+ (
+ { config
+ , name
+ , ...
+ }: {
+ options = {
+ name = mkOption {
+ type = types.str;
+ default = name;
+ description = "Bookmark name.";
+ };
+
+ tags = mkOption {
+ type = types.listOf types.str;
+ default = [ ];
+ description = "Bookmark tags.";
+ };
+
+ keyword = mkOption {
+ type = types.nullOr types.str;
+ default = null;
+ description = "Bookmark search keyword.";
+ };
+
+ url = mkOption {
+ type = types.str;
+ description = "Bookmark url, use %s for search terms.";
+ };
+ };
+ }
+ )
+ // {
+ description = "bookmark submodule";
+ };
+
+ bookmarkType = types.addCheck bookmarkSubmodule (x: x ? "url");
+
+ directoryType =
+ types.submodule
+ (
+ { config
+ , name
+ , ...
+ }: {
+ options = {
+ name = mkOption {
+ type = types.str;
+ default = name;
+ description = "Directory name.";
+ };
+
+ bookmarks = mkOption {
+ type = types.listOf nodeType;
+ default = [ ];
+ description = "Bookmarks within directory.";
+ };
+
+ toolbar = mkOption {
+ type = types.bool;
+ default = false;
+ description = ''
+ Make this the toolbar directory. Note, this does _not_
+ mean that this directory will be added to the toolbar,
+ this directory _is_ the toolbar.
+ '';
+ };
+ };
+ }
+ )
+ // {
+ description = "directory submodule";
+ };
+
+ nodeType = types.either bookmarkType directoryType;
+ in
+ with types;
+ coercedTo (attrsOf nodeType) attrValues (listOf nodeType);
+ default = [ ];
+ example = literalExpression ''
+ [
+ {
+ name = "wikipedia";
+ tags = [ "wiki" ];
+ keyword = "wiki";
+ url = "https://en.wikipedia.org/wiki/Special:Search?search=%s&go=Go";
+ }
+ {
+ name = "kernel.org";
+ url = "https://www.kernel.org";
+ }
+ {
+ name = "Nix sites";
+ toolbar = true;
+ bookmarks = [
+ {
+ name = "homepage";
+ url = "https://nixos.org/";
+ }
+ {
+ name = "wiki";
+ tags = [ "wiki" "nix" ];
+ url = "https://wiki.nixos.org/";
+ }
+ ];
+ }
+ ]
+ '';
+ description = ''
+ Preloaded bookmarks. Note, this may silently overwrite any
+ previously existing bookmarks!
+ '';
+ };
+
+ path = mkOption {
+ type = types.str;
+ default = name;
+ description = "Profile path.";
+ };
+
+ isDefault = mkOption {
+ type = types.bool;
+ default = config.id == 0;
+ defaultText = "true if profile ID is 0";
+ description = "Whether this is a default profile.";
+ };
+
+ search = {
+ force = mkOption {
+ type = with types; bool;
+ default = false;
+ description = ''
+ Whether to force replace the existing search
+ configuration. This is recommended since ${name} will
+ replace the symlink for the search configuration on every
+ launch, but note that you'll lose any existing
+ configuration by enabling this.
+ '';
+ };
+
+ default = mkOption {
+ type = with types; nullOr str;
+ default = null;
+ example = "DuckDuckGo";
+ description = ''
+ The default search engine used in the address bar and search bar.
+ '';
+ };
+
+ privateDefault = mkOption {
+ type = with types; nullOr str;
+ default = null;
+ example = "DuckDuckGo";
+ description = ''
+ The default search engine used in the Private Browsing.
+ '';
+ };
+
+ order = mkOption {
+ type = with types; uniq (listOf str);
+ default = [ ];
+ example = [
+ "DuckDuckGo"
+ "Google"
+ ];
+ description = ''
+ The order the search engines are listed in. Any engines
+ that aren't included in this list will be listed after
+ these in an unspecified order.
+ '';
+ };
+
+ engines = mkOption {
+ type = with types; attrsOf (attrsOf jsonFormat.type);
+ default = { };
+ example = literalExpression ''
+ {
+ "Nix Packages" = {
+ urls = [{
+ template = "https://search.nixos.org/packages";
+ params = [
+ { name = "type"; value = "packages"; }
+ { name = "query"; value = "{searchTerms}"; }
+ ];
+ }];
+
+ icon = "''${pkgs.nixos-icons}/share/icons/hicolor/scalable/apps/nix-snowflake.svg";
+ definedAliases = [ "@np" ];
+ };
+
+ "NixOS Wiki" = {
+ urls = [{ template = "https://wiki.nixos.org/index.php?search={searchTerms}"; }];
+ iconUpdateURL = "https://wiki.nixos.org/favicon.png";
+ updateInterval = 24 * 60 * 60 * 1000; # every day
+ definedAliases = [ "@nw" ];
+ };
+
+ "Bing".metaData.hidden = true;
+ "Google".metaData.alias = "@g"; # builtin engines only support specifying one additional alias
+ }
+ '';
+ description = ''
+ Attribute set of search engine configurations. Engines
+ that only have {var}`metaData` specified will
+ be treated as builtin to ${name}.
+
+ See [SearchEngine.jsm](https://searchfox.org/mozilla-central/rev/669329e284f8e8e2bb28090617192ca9b4ef3380/toolkit/components/search/SearchEngine.jsm#1138-1177)
+ in Firefox's source for available options. We maintain a
+ mapping to let you specify all options in the referenced
+ link without underscores, but it may fall out of date with
+ future options.
+
+ Note, {var}`icon` is also a special option
+ added by Home Manager to make it convenient to specify
+ absolute icon paths.
+ '';
+ };
+ };
+
+ containersForce = mkOption {
+ type = types.bool;
+ default = false;
+ description = ''
+ Whether to force replace the existing containers configuration.
+ This is recommended since Firefox will replace the symlink on
+ every launch, but note that you'll lose any existing configuration
+ by enabling this.
+ '';
+ };
+
+ containers = mkOption {
+ type = types.attrsOf (
+ types.submodule (
+ { name, ... }: {
+ options = {
+ name = mkOption {
+ type = types.str;
+ default = name;
+ description = "Container name, e.g., shopping.";
+ };
+
+ id = mkOption {
+ type = types.ints.unsigned;
+ default = 0;
+ description = ''
+ Container ID. This should be set to a unique number per container in this profile.
+ '';
+ };
+
+ # List of colors at
+ # https://searchfox.org/mozilla-central/rev/5ad226c7379b0564c76dc3b54b44985356f94c5a/toolkit/components/extensions/parent/ext-contextualIdentities.js#32
+ color = mkOption {
+ type = types.enum [
+ "blue"
+ "turquoise"
+ "green"
+ "yellow"
+ "orange"
+ "red"
+ "pink"
+ "purple"
+ "toolbar"
+ ];
+ default = "pink";
+ description = "Container color.";
+ };
+
+ icon = mkOption {
+ type = types.enum [
+ "briefcase"
+ "cart"
+ "circle"
+ "dollar"
+ "fence"
+ "fingerprint"
+ "gift"
+ "vacation"
+ "food"
+ "fruit"
+ "pet"
+ "tree"
+ "chill"
+ ];
+ default = "fruit";
+ description = "Container icon.";
+ };
+ };
+ }
+ )
+ );
+ default = { };
+ example = {
+ "shopping" = {
+ id = 1;
+ color = "blue";
+ icon = "cart";
+ };
+ "dangerous" = {
+ id = 2;
+ color = "red";
+ icon = "fruit";
+ };
+ };
+ description = ''
+ Attribute set of container configurations. See
+ [Multi-Account
+ Containers](https://support.mozilla.org/en-US/kb/containers)
+ for more information.
+ '';
+ };
+
+ extensions.packages = mkOption {
+ type = types.listOf types.package;
+ default = [ ];
+ example = literalExpression ''
+ with pkgs.nur.repos.rycee.firefox-addons; [
+ privacy-badger
+ ]
+ '';
+ description = ''
+ List of ${name} add-on packages to install for this profile.
+ Some pre-packaged add-ons are accessible from the
+ [Nix User Repository](https://github.com/nix-community/NUR).
+ Once you have NUR installed run
+
+ ```console
+ $ nix-env -f '' -qaP -A nur.repos.rycee.firefox-addons
+ ```
+
+ to list the available ${name} add-ons.
+
+ Note that it is necessary to manually enable these extensions
+ inside ${name} after the first installation.
+
+ To automatically enable extensions add
+ `"extensions.autoDisableScopes" = 0;`
+ to
+ [{option}`${moduleName}.profiles..settings`](#opt-${moduleName}.profiles._name_.settings)
+ '';
+ };
+ };
+ }
+ )
+ );
+ default = { };
+ description = "Attribute set of ${name} profiles.";
+ };
+
+ enableGnomeExtensions = mkOption {
+ inherit visible;
+ type = types.bool;
+ default = false;
+ description = ''
+ Whether to enable the GNOME Shell native host connector. Note, you
+ also need to set the NixOS option
+ `services.gnome.gnome-browser-connector.enable` to
+ `true`.
+ '';
+ };
+ };
+
+ config = mkIf cfg.enable (
+ {
+ assertions =
+ [
+ (hm.assertions.assertPlatform moduleName pkgs supportedPlatforms)
+
+ (
+ let
+ defaults = catAttrs "name" (filter (a: a.isDefault) (attrValues cfg.profiles));
+ in
+ {
+ assertion = cfg.profiles == { } || length defaults == 1;
+ message =
+ "Must have exactly one default ${cfg.name} profile but found "
+ + toString (length defaults)
+ + optionalString (length defaults > 1) (", namely " + concatStringsSep ", " defaults);
+ }
+ )
+
+ (
+ let
+ getContainers = profiles: flatten (mapAttrsToList (_: value: (attrValues value.containers)) profiles);
+
+ findInvalidContainerIds = profiles: filter (container: container.id >= 4294967294) (getContainers profiles);
+ in
+ {
+ assertion = cfg.profiles == { } || length (findInvalidContainerIds cfg.profiles) == 0;
+ message = "Container id must be smaller than 4294967294 (2^32 - 2)";
+ }
+ )
+
+ {
+ assertion = cfg.languagePacks == [ ] || cfg.package != null;
+ message = ''
+ 'programs.zen-browser.languagePacks' requires 'programs.zen-browser.package'
+ to be set to a non-null value.
+ '';
+ }
+
+ (mkNoDuplicateAssertion cfg.profiles "profile")
+ ]
+ ++ (mapAttrsToList
+ (
+ _: profile: mkNoDuplicateAssertion profile.containers "container"
+ )
+ cfg.profiles);
+
+ warnings = optional (cfg.enableGnomeExtensions or false) ''
+ Using '${moduleName}.enableGnomeExtensions' has been deprecated and
+ will be removed in the future. Please change to overriding the package
+ configuration using '${moduleName}.package' instead. You can refer to
+ its example for how to do this.
+ '';
+
+ programs.zen-browser.policies = {
+ ExtensionSettings = listToAttrs (
+ map
+ (
+ lang:
+ nameValuePair "langpack-${lang}@firefox.mozilla.org" {
+ installation_mode = "normal_installed";
+ install_url = "https://releases.mozilla.org/pub/firefox/releases/${cfg.package.version}/linux-x86_64/xpi/${lang}.xpi";
+ }
+ )
+ cfg.languagePacks
+ );
+ };
+
+ home.packages = lib.optional (cfg.finalPackage != null) cfg.finalPackage;
+
+ home.file = mkMerge (
+ [
+ {
+ "${cfg.configPath}/profiles.ini" = mkIf (cfg.profiles != { }) { text = profilesIni; };
+ }
+ ]
+ ++ optional (cfg.vendorPath != null) {
+ "${nativeMessagingHostsPath}" = {
+ source = "${nativeMessagingHostsJoined}/lib/mozilla/native-messaging-hosts";
+ recursive = true;
+ };
+ }
+ ++ flip mapAttrsToList cfg.profiles (
+ _: profile: {
+ "${profilesPath}/${profile.path}/.keep".text = "";
+
+ "${profilesPath}/${profile.path}/chrome/userChrome.css" = mkIf (profile.userChrome != "") {
+ text = profile.userChrome;
+ };
+
+ "${profilesPath}/${profile.path}/chrome/userContent.css" = mkIf (profile.userContent != "") {
+ text = profile.userContent;
+ };
+
+ "${profilesPath}/${profile.path}/user.js" =
+ mkIf (profile.settings != { } || profile.extraConfig != "" || profile.bookmarks != [ ])
+ {
+ text = mkUserJs profile.settings profile.extraConfig profile.bookmarks;
+ };
+
+ "${profilesPath}/${profile.path}/containers.json" = mkIf (profile.containers != { }) {
+ text = mkContainersJson profile.containers;
+ force = profile.containersForce;
+ };
+
+ "${profilesPath}/${profile.path}/search.json.mozlz4" =
+ mkIf
+ (
+ profile.search.default
+ != null
+ || profile.search.privateDefault != null
+ || profile.search.order != [ ]
+ || profile.search.engines != { }
+ )
+ {
+ force = profile.search.force;
+ source =
+ let
+ settings = {
+ version = 12;
+ engines =
+ let
+ # Map of nice field names to internal field names.
+ # This is intended to be exhaustive and should be
+ # updated at every version bump.
+ internalFieldNames =
+ (genAttrs [
+ "name"
+ "isAppProvided"
+ "loadPath"
+ "hasPreferredIcon"
+ "updateInterval"
+ "updateURL"
+ "iconUpdateURL"
+ "iconURL"
+ "iconMapObj"
+ "metaData"
+ "orderHint"
+ "definedAliases"
+ "urls"
+ ]
+ (name: "_${name}"))
+ // {
+ searchForm = "__searchForm";
+ };
+
+ processCustomEngineInput = input:
+ (removeAttrs input [ "icon" ])
+ // optionalAttrs (input ? icon) {
+ # Convenience to specify absolute path to icon
+ iconURL = "file://${input.icon}";
+ }
+ // (
+ optionalAttrs (input ? iconUpdateURL)
+ {
+ # Convenience to default iconURL to iconUpdateURL so
+ # the icon is immediately downloaded from the URL
+ iconURL = input.iconURL or input.iconUpdateURL;
+ }
+ // {
+ # Required for custom engine configurations, loadPaths
+ # are unique identifiers that are generally formatted
+ # like: [source]/path/to/engine.xml
+ loadPath = ''[home-manager]/${moduleName}.profiles.${profile.name}.search.engines."${
+ replaceStrings ["\\"] ["\\\\"] input.name
+ }"'';
+ }
+ );
+
+ processEngineInput = name: input:
+ let
+ requiredInput = {
+ inherit name;
+ isAppProvided = input.isAppProvided or removeAttrs input [ "metaData" ] == { };
+ metaData = input.metaData or { };
+ };
+ in
+ if requiredInput.isAppProvided
+ then requiredInput
+ else processCustomEngineInput (input // requiredInput);
+
+ buildEngineConfig = name: input:
+ mapAttrs'
+ (name: value: {
+ name = internalFieldNames.${name} or name;
+ inherit value;
+ })
+ (processEngineInput name input);
+
+ sortEngineConfigs = configs:
+ let
+ buildEngineConfigWithOrder = order: name:
+ let
+ config =
+ configs.${name}
+ or {
+ _name = name;
+ _isAppProvided = true;
+ _metaData = { };
+ };
+ in
+ config
+ // {
+ _metaData =
+ config._metaData
+ // {
+ inherit order;
+ };
+ };
+
+ engineConfigsWithoutOrder = attrValues (removeAttrs configs profile.search.order);
+
+ sortedEngineConfigs =
+ (imap buildEngineConfigWithOrder profile.search.order) ++ engineConfigsWithoutOrder;
+ in
+ sortedEngineConfigs;
+
+ engineInput =
+ profile.search.engines
+ // {
+ # Infer profile.search.default as an app provided
+ # engine if it's not in profile.search.engines
+ ${profile.search.default} = profile.search.engines.${profile.search.default} or { };
+ }
+ // {
+ ${profile.search.privateDefault} = profile.search.engines.${profile.search.privateDefault} or { };
+ };
+ in
+ sortEngineConfigs (mapAttrs buildEngineConfig engineInput);
+
+ metaData =
+ optionalAttrs (profile.search.default != null)
+ {
+ current = profile.search.default;
+ hash = "@hash@";
+ }
+ // optionalAttrs (profile.search.privateDefault != null) {
+ private = profile.search.privateDefault;
+ privateHash = "@privateHash@";
+ }
+ // {
+ useSavedOrder = profile.search.order != [ ];
+ };
+ };
+
+ # Home Manager doesn't circumvent user consent and isn't acting
+ # maliciously. We're modifying the search outside of the browser, but
+ # a claim by Mozilla to remove this would be very anti-user, and
+ # is unlikely to be an issue for our use case.
+ disclaimer = appName:
+ "By modifying this file, I agree that I am doing so "
+ + "only within ${appName} itself, using official, user-driven search "
+ + "engine selection processes, and in a way which does not circumvent "
+ + "user consent. I acknowledge that any attempt to change this file "
+ + "from outside of ${appName} is a malicious act, and will be responded "
+ + "to accordingly.";
+
+ salt =
+ if profile.search.default != null
+ then profile.path + profile.search.default + disclaimer cfg.name
+ else null;
+
+ privateSalt =
+ if profile.search.privateDefault != null
+ then profile.path + profile.search.privateDefault + disclaimer cfg.name
+ else null;
+ in
+ pkgs.runCommand "search.json.mozlz4"
+ {
+ nativeBuildInputs = with pkgs; [
+ mozlz4a
+ openssl
+ ];
+ json = builtins.toJSON settings;
+ inherit salt privateSalt;
+ }
+ ''
+ if [[ -n $salt ]]; then
+ export hash=$(echo -n "$salt" | openssl dgst -sha256 -binary | base64)
+ export privateHash=$(echo -n "$privateSalt" | openssl dgst -sha256 -binary | base64)
+ mozlz4a <(substituteStream json search.json.in --subst-var hash --subst-var privateHash) "$out"
+ else
+ mozlz4a <(echo "$json") "$out"
+ fi
+ '';
+ };
+
+ "${profilesPath}/${profile.path}/extensions" = mkIf (profile.extensions.packages != [ ]) {
+ source =
+ let
+ extensionsEnvPkg = pkgs.buildEnv {
+ name = "hm-firefox-extensions";
+ paths = profile.extensions.packages;
+ };
+ in
+ "${extensionsEnvPkg}/share/mozilla/${extensionPath}";
+ recursive = true;
+ force = true;
+ };
+ }
+ )
+ );
+ }
+ // setAttrByPath modulePath { finalPackage = wrapPackage cfg.package; }
+ );
+}
diff --git a/home-mods/firefox/zen-browser.nix b/home-mods/firefox/zen-browser.nix
new file mode 100644
index 0000000..2b9a31c
--- /dev/null
+++ b/home-mods/firefox/zen-browser.nix
@@ -0,0 +1,36 @@
+{ config
+, lib
+, pkgs
+, ...
+} @ args: {
+ imports =
+ let
+ modulePath = [ "programs" "zen-browser" ];
+ mkFirefoxModule = import ./mkFirefoxModule.nix;
+ in
+ [
+ (mkFirefoxModule {
+ inherit modulePath;
+ name = "Zen Browser";
+ wrappedPackageName = "zen-browser";
+ wrappedPkg = pkgs.my.zen-browser;
+ visible = true;
+ platforms.linux = rec {
+ vendorPath = ".zen";
+ configPath = "${vendorPath}";
+ };
+ platforms.darwin = rec {
+ vendorPath = "Library/Application Support/Zen";
+ configPath = "${vendorPath}";
+ };
+ })
+ ];
+
+ programs.zen-browser = import ./_base.nix args;
+
+ home.sessionVariables = {
+ BROWSER = "zen-browser";
+ MOZ_USE_XINPUT2 = "1";
+ MOZ_ENABLE_WAYLAND = "1";
+ };
+}
diff --git a/users/admin-fat.nix b/users/admin-fat.nix
index 0f2bd22..8289161 100644
--- a/users/admin-fat.nix
+++ b/users/admin-fat.nix
@@ -11,7 +11,8 @@
../home-mods/common
../home-mods/desktop
# ../home-mods/firefox
- ../home-mods/firefox/floorp.nix
+ # ../home-mods/firefox/floorp.nix
+ ../home-mods/firefox/zen-browser.nix
../home-mods/plasma
../home-mods/shell
../home-mods/virt