2020-10-12 00:22:58 +00:00
|
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
|
|
inherit (pkgs.stdenv.hostPlatform) isDarwin;
|
|
|
|
|
|
|
|
|
|
cfg = config.programs.firefox;
|
|
|
|
|
|
2023-01-10 09:35:00 +00:00
|
|
|
|
jsonFormat = pkgs.formats.json { };
|
|
|
|
|
|
2020-10-12 00:22:58 +00:00
|
|
|
|
mozillaConfigPath =
|
2021-11-04 16:42:44 +00:00
|
|
|
|
if isDarwin then "Library/Application Support/Mozilla" else ".mozilla";
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
firefoxConfigPath = if isDarwin then
|
|
|
|
|
"Library/Application Support/Firefox"
|
|
|
|
|
else
|
|
|
|
|
"${mozillaConfigPath}/firefox";
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
|
|
|
|
profilesPath =
|
2021-11-04 16:42:44 +00:00
|
|
|
|
if isDarwin then "${firefoxConfigPath}/Profiles" else firefoxConfigPath;
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
|
|
|
|
# The extensions path shared by all profiles; will not be supported
|
|
|
|
|
# by future Firefox versions.
|
|
|
|
|
extensionPath = "extensions/{ec8030f7-c20a-464f-9b0e-13a3a9e97384}";
|
|
|
|
|
|
|
|
|
|
extensionsEnvPkg = pkgs.buildEnv {
|
|
|
|
|
name = "hm-firefox-extensions";
|
|
|
|
|
paths = cfg.extensions;
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
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;
|
|
|
|
|
}) // {
|
|
|
|
|
General = { StartWithLastProfile = 1; };
|
2020-10-12 00:22:58 +00:00
|
|
|
|
};
|
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
profilesIni = generators.toINI { } profiles;
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
mkUserJs = prefs: extraPrefs: bookmarks:
|
|
|
|
|
let
|
2023-01-10 09:35:00 +00:00
|
|
|
|
prefs' = lib.optionalAttrs ([ ] != bookmarks) {
|
2021-11-04 16:42:44 +00:00
|
|
|
|
"browser.bookmarks.file" = toString (firefoxBookmarksFile bookmarks);
|
|
|
|
|
"browser.places.importBookmarksHTML" = true;
|
|
|
|
|
} // prefs;
|
|
|
|
|
in ''
|
|
|
|
|
// Generated by Home Manager.
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
${concatStrings (mapAttrsToList (name: value: ''
|
|
|
|
|
user_pref("${name}", ${builtins.toJSON value});
|
|
|
|
|
'') prefs')}
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
${extraPrefs}
|
|
|
|
|
'';
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
firefoxBookmarksFile = bookmarks:
|
|
|
|
|
let
|
2023-01-10 09:35:00 +00:00
|
|
|
|
indent = level:
|
|
|
|
|
lib.concatStringsSep "" (map (lib.const " ") (lib.range 1 level));
|
|
|
|
|
|
|
|
|
|
bookmarkToHTML = indentLevel: bookmark:
|
|
|
|
|
''
|
|
|
|
|
${indent indentLevel}<DT><A HREF="${
|
|
|
|
|
escapeXML bookmark.url
|
|
|
|
|
}" ADD_DATE="0" LAST_MODIFIED="0"${
|
|
|
|
|
lib.optionalString (bookmark.keyword != null)
|
|
|
|
|
" SHORTCUTURL=\"${escapeXML bookmark.keyword}\""
|
|
|
|
|
}>${escapeXML bookmark.name}</A>'';
|
|
|
|
|
|
|
|
|
|
directoryToHTML = indentLevel: directory: ''
|
|
|
|
|
${indent indentLevel}<DT>${
|
|
|
|
|
if directory.toolbar then
|
|
|
|
|
''<H3 PERSONAL_TOOLBAR_FOLDER="true">Bookmarks Toolbar''
|
|
|
|
|
else
|
|
|
|
|
"<H3>${escapeXML directory.name}"
|
|
|
|
|
}</H3>
|
|
|
|
|
${indent indentLevel}<DL><p>
|
|
|
|
|
${allItemsToHTML (indentLevel + 1) directory.bookmarks}
|
|
|
|
|
${indent indentLevel}</p></DL>'';
|
|
|
|
|
|
|
|
|
|
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;
|
2021-11-04 16:42:44 +00:00
|
|
|
|
in pkgs.writeText "firefox-bookmarks.html" ''
|
|
|
|
|
<!DOCTYPE NETSCAPE-Bookmark-file-1>
|
|
|
|
|
<!-- This is an automatically generated file.
|
|
|
|
|
It will be read and overwritten.
|
|
|
|
|
DO NOT EDIT! -->
|
|
|
|
|
<META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=UTF-8">
|
|
|
|
|
<TITLE>Bookmarks</TITLE>
|
|
|
|
|
<H1>Bookmarks Menu</H1>
|
|
|
|
|
<DL><p>
|
2023-01-10 09:35:00 +00:00
|
|
|
|
${bookmarkEntries}
|
2021-11-04 16:42:44 +00:00
|
|
|
|
</p></DL>
|
|
|
|
|
'';
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
in {
|
2023-01-10 09:35:00 +00:00
|
|
|
|
meta.maintainers = [ maintainers.rycee maintainers.kira-bruneau ];
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
|
|
|
|
imports = [
|
2021-11-04 16:42:44 +00:00
|
|
|
|
(mkRemovedOptionModule [ "programs" "firefox" "enableAdobeFlash" ]
|
2021-07-02 22:36:30 +00:00
|
|
|
|
"Support for this option has been removed.")
|
2021-11-04 16:42:44 +00:00
|
|
|
|
(mkRemovedOptionModule [ "programs" "firefox" "enableGoogleTalk" ]
|
2020-10-12 00:22:58 +00:00
|
|
|
|
"Support for this option has been removed.")
|
2021-11-04 16:42:44 +00:00
|
|
|
|
(mkRemovedOptionModule [ "programs" "firefox" "enableIcedTea" ]
|
2020-10-12 00:22:58 +00:00
|
|
|
|
"Support for this option has been removed.")
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
programs.firefox = {
|
|
|
|
|
enable = mkEnableOption "Firefox";
|
|
|
|
|
|
|
|
|
|
package = mkOption {
|
|
|
|
|
type = types.package;
|
2021-11-04 16:42:44 +00:00
|
|
|
|
default = if versionAtLeast config.home.stateVersion "19.09" then
|
|
|
|
|
pkgs.firefox
|
|
|
|
|
else
|
|
|
|
|
pkgs.firefox-unwrapped;
|
|
|
|
|
defaultText = literalExpression "pkgs.firefox";
|
|
|
|
|
example = literalExpression ''
|
2021-07-02 22:36:30 +00:00
|
|
|
|
pkgs.firefox.override {
|
|
|
|
|
# See nixpkgs' firefox/wrapper.nix to check which options you can use
|
|
|
|
|
cfg = {
|
|
|
|
|
# Gnome shell native connector
|
|
|
|
|
enableGnomeExtensions = true;
|
|
|
|
|
# Tridactyl native connector
|
|
|
|
|
enableTridactylNative = true;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
'';
|
2020-10-12 00:22:58 +00:00
|
|
|
|
description = ''
|
|
|
|
|
The Firefox package to use. If state version ≥ 19.09 then
|
|
|
|
|
this should be a wrapped Firefox package. For earlier state
|
|
|
|
|
versions it should be an unwrapped Firefox package.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
extensions = mkOption {
|
|
|
|
|
type = types.listOf types.package;
|
2021-11-04 16:42:44 +00:00
|
|
|
|
default = [ ];
|
|
|
|
|
example = literalExpression ''
|
2020-10-12 00:22:58 +00:00
|
|
|
|
with pkgs.nur.repos.rycee.firefox-addons; [
|
|
|
|
|
privacy-badger
|
|
|
|
|
]
|
|
|
|
|
'';
|
|
|
|
|
description = ''
|
|
|
|
|
List of Firefox add-on packages to install. Some
|
|
|
|
|
pre-packaged add-ons are accessible from NUR,
|
|
|
|
|
<link xlink:href="https://github.com/nix-community/NUR"/>.
|
|
|
|
|
Once you have NUR installed run
|
|
|
|
|
|
|
|
|
|
<screen language="console">
|
|
|
|
|
<prompt>$</prompt> <userinput>nix-env -f '<nixpkgs>' -qaP -A nur.repos.rycee.firefox-addons</userinput>
|
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
to list the available Firefox add-ons.
|
|
|
|
|
|
|
|
|
|
</para><para>
|
|
|
|
|
|
|
|
|
|
Note that it is necessary to manually enable these
|
|
|
|
|
extensions inside Firefox after the first installation.
|
|
|
|
|
|
|
|
|
|
</para><para>
|
|
|
|
|
|
|
|
|
|
Extensions listed here will only be available in Firefox
|
|
|
|
|
profiles managed through the
|
2023-01-10 09:35:00 +00:00
|
|
|
|
<xref linkend="opt-programs.firefox.profiles"/>
|
2020-10-12 00:22:58 +00:00
|
|
|
|
option. This is due to recent changes in the way Firefox
|
|
|
|
|
handles extension side-loading.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
profiles = mkOption {
|
2021-11-04 16:42:44 +00:00
|
|
|
|
type = types.attrsOf (types.submodule ({ config, name, ... }: {
|
2020-10-12 00:22:58 +00:00
|
|
|
|
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 = with types; attrsOf (either bool (either int str));
|
2021-11-04 16:42:44 +00:00
|
|
|
|
default = { };
|
|
|
|
|
example = literalExpression ''
|
2020-10-12 00:22:58 +00:00
|
|
|
|
{
|
|
|
|
|
"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;
|
|
|
|
|
}
|
|
|
|
|
'';
|
|
|
|
|
description = "Attribute set of Firefox preferences.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
extraConfig = mkOption {
|
|
|
|
|
type = types.lines;
|
|
|
|
|
default = "";
|
|
|
|
|
description = ''
|
|
|
|
|
Extra preferences to add to <filename>user.js</filename>.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
userChrome = mkOption {
|
|
|
|
|
type = types.lines;
|
|
|
|
|
default = "";
|
|
|
|
|
description = "Custom Firefox user chrome CSS.";
|
|
|
|
|
example = ''
|
|
|
|
|
/* Hide tab bar in FF Quantum */
|
|
|
|
|
@-moz-document url("chrome://browser/content/browser.xul") {
|
|
|
|
|
#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 Firefox user content CSS.";
|
|
|
|
|
example = ''
|
|
|
|
|
/* Hide scrollbar in FF Quantum */
|
|
|
|
|
*{scrollbar-width:none !important}
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
bookmarks = mkOption {
|
2023-01-10 09:35:00 +00:00
|
|
|
|
type = let
|
|
|
|
|
bookmarkSubmodule = types.submodule ({ config, name, ... }: {
|
|
|
|
|
options = {
|
|
|
|
|
name = mkOption {
|
|
|
|
|
type = types.str;
|
|
|
|
|
default = name;
|
|
|
|
|
description = "Bookmark name.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
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.";
|
|
|
|
|
};
|
2021-11-04 16:42:44 +00:00
|
|
|
|
};
|
2023-01-10 09:35:00 +00:00
|
|
|
|
}) // {
|
|
|
|
|
description = "bookmark submodule";
|
|
|
|
|
};
|
2021-11-04 16:42:44 +00:00
|
|
|
|
|
2023-01-10 09:35:00 +00:00
|
|
|
|
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 = "If directory should be shown in toolbar.";
|
|
|
|
|
};
|
2021-11-04 16:42:44 +00:00
|
|
|
|
};
|
2023-01-10 09:35:00 +00:00
|
|
|
|
}) // {
|
|
|
|
|
description = "directory submodule";
|
2021-11-04 16:42:44 +00:00
|
|
|
|
};
|
2023-01-10 09:35:00 +00:00
|
|
|
|
|
|
|
|
|
nodeType = types.either bookmarkType directoryType;
|
|
|
|
|
in with types;
|
|
|
|
|
coercedTo (attrsOf nodeType) attrValues (listOf nodeType);
|
|
|
|
|
default = [ ];
|
2021-11-04 16:42:44 +00:00
|
|
|
|
example = literalExpression ''
|
2023-01-10 09:35:00 +00:00
|
|
|
|
[
|
|
|
|
|
{
|
|
|
|
|
name = "wikipedia";
|
2021-11-04 16:42:44 +00:00
|
|
|
|
keyword = "wiki";
|
|
|
|
|
url = "https://en.wikipedia.org/wiki/Special:Search?search=%s&go=Go";
|
2023-01-10 09:35:00 +00:00
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
name = "kernel.org";
|
2021-11-04 16:42:44 +00:00
|
|
|
|
url = "https://www.kernel.org";
|
2023-01-10 09:35:00 +00:00
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
name = "Nix sites";
|
|
|
|
|
bookmarks = [
|
|
|
|
|
{
|
|
|
|
|
name = "homepage";
|
|
|
|
|
url = "https://nixos.org/";
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
name = "wiki";
|
|
|
|
|
url = "https://nixos.wiki/";
|
|
|
|
|
}
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
]
|
2021-11-04 16:42:44 +00:00
|
|
|
|
'';
|
|
|
|
|
description = ''
|
|
|
|
|
Preloaded bookmarks. Note, this may silently overwrite any
|
|
|
|
|
previously existing bookmarks!
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2020-10-12 00:22:58 +00:00
|
|
|
|
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.";
|
|
|
|
|
};
|
2023-01-10 09:35:00 +00:00
|
|
|
|
|
|
|
|
|
search = {
|
|
|
|
|
force = mkOption {
|
|
|
|
|
type = with types; bool;
|
|
|
|
|
default = false;
|
|
|
|
|
description = ''
|
|
|
|
|
Whether to force replace the existing search
|
|
|
|
|
configuration. This is recommended since Firefox 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.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
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://nixos.wiki/index.php?search={searchTerms}"; }];
|
|
|
|
|
iconUpdateURL = "https://nixos.wiki/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 <varname>metaData</varname> specified will
|
|
|
|
|
be treated as builtin to Firefox.
|
|
|
|
|
</para><para>
|
|
|
|
|
See <link xlink:href=
|
|
|
|
|
"https://searchfox.org/mozilla-central/rev/669329e284f8e8e2bb28090617192ca9b4ef3380/toolkit/components/search/SearchEngine.jsm#1138-1177">SearchEngine.jsm</link>
|
|
|
|
|
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.
|
|
|
|
|
</para><para>
|
|
|
|
|
Note, <varname>icon</varname> is also a special option
|
|
|
|
|
added by Home Manager to make it convenient to specify
|
|
|
|
|
absolute icon paths.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
};
|
2020-10-12 00:22:58 +00:00
|
|
|
|
};
|
|
|
|
|
}));
|
2021-11-04 16:42:44 +00:00
|
|
|
|
default = { };
|
2020-10-12 00:22:58 +00:00
|
|
|
|
description = "Attribute set of Firefox profiles.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
enableGnomeExtensions = mkOption {
|
|
|
|
|
type = types.bool;
|
|
|
|
|
default = false;
|
|
|
|
|
description = ''
|
|
|
|
|
Whether to enable the GNOME Shell native host connector. Note, you
|
|
|
|
|
also need to set the NixOS option
|
2023-01-10 09:35:00 +00:00
|
|
|
|
<literal>services.gnome.gnome-browser-connector.enable</literal> to
|
2020-10-12 00:22:58 +00:00
|
|
|
|
<literal>true</literal>.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
|
assertions = [
|
2021-11-04 16:42:44 +00:00
|
|
|
|
(let
|
|
|
|
|
defaults =
|
|
|
|
|
catAttrs "name" (filter (a: a.isDefault) (attrValues cfg.profiles));
|
|
|
|
|
in {
|
|
|
|
|
assertion = cfg.profiles == { } || length defaults == 1;
|
|
|
|
|
message = "Must have exactly one default Firefox profile but found "
|
|
|
|
|
+ toString (length defaults) + optionalString (length defaults > 1)
|
|
|
|
|
(", namely " + concatStringsSep ", " defaults);
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
(let
|
|
|
|
|
duplicates = filterAttrs (_: v: length v != 1) (zipAttrs
|
|
|
|
|
(mapAttrsToList (n: v: { "${toString v.id}" = n; }) (cfg.profiles)));
|
|
|
|
|
|
|
|
|
|
mkMsg = n: v: " - ID ${n} is used by ${concatStringsSep ", " v}";
|
|
|
|
|
in {
|
|
|
|
|
assertion = duplicates == { };
|
|
|
|
|
message = ''
|
|
|
|
|
Must not have Firefox profiles with duplicate IDs but
|
|
|
|
|
'' + concatStringsSep "\n" (mapAttrsToList mkMsg duplicates);
|
|
|
|
|
})
|
2020-10-12 00:22:58 +00:00
|
|
|
|
];
|
|
|
|
|
|
2021-07-02 22:36:30 +00:00
|
|
|
|
warnings = optional (cfg.enableGnomeExtensions or false) ''
|
|
|
|
|
Using 'programs.firefox.enableGnomeExtensions' has been deprecated and
|
|
|
|
|
will be removed in the future. Please change to overriding the package
|
|
|
|
|
configuration using 'programs.firefox.package' instead. You can refer to
|
|
|
|
|
its example for how to do this.
|
|
|
|
|
'';
|
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
home.packages = let
|
|
|
|
|
# The configuration expected by the Firefox wrapper.
|
|
|
|
|
fcfg = { enableGnomeExtensions = cfg.enableGnomeExtensions; };
|
|
|
|
|
|
|
|
|
|
# A bit of hackery to force a config into the wrapper.
|
|
|
|
|
browserName = cfg.package.browserName or (builtins.parseDrvName
|
|
|
|
|
cfg.package.name).name;
|
|
|
|
|
|
|
|
|
|
# The configuration expected by the Firefox wrapper builder.
|
|
|
|
|
bcfg = setAttrByPath [ browserName ] fcfg;
|
|
|
|
|
|
|
|
|
|
package = if isDarwin then
|
|
|
|
|
cfg.package
|
|
|
|
|
else if versionAtLeast config.home.stateVersion "19.09" then
|
|
|
|
|
cfg.package.override (old: { cfg = old.cfg or { } // fcfg; })
|
|
|
|
|
else
|
|
|
|
|
(pkgs.wrapFirefox.override { config = bcfg; }) cfg.package { };
|
|
|
|
|
in [ package ];
|
|
|
|
|
|
|
|
|
|
home.file = mkMerge ([{
|
|
|
|
|
"${mozillaConfigPath}/${extensionPath}" = mkIf (cfg.extensions != [ ]) {
|
|
|
|
|
source = "${extensionsEnvPkg}/share/mozilla/${extensionPath}";
|
|
|
|
|
recursive = true;
|
|
|
|
|
};
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
"${firefoxConfigPath}/profiles.ini" =
|
|
|
|
|
mkIf (cfg.profiles != { }) { text = profilesIni; };
|
|
|
|
|
}] ++ flip mapAttrsToList cfg.profiles (_: profile: {
|
|
|
|
|
"${profilesPath}/${profile.path}/.keep".text = "";
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
"${profilesPath}/${profile.path}/chrome/userChrome.css" =
|
|
|
|
|
mkIf (profile.userChrome != "") { text = profile.userChrome; };
|
2021-07-02 22:36:30 +00:00
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
"${profilesPath}/${profile.path}/chrome/userContent.css" =
|
|
|
|
|
mkIf (profile.userContent != "") { text = profile.userContent; };
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2021-12-03 05:30:34 +00:00
|
|
|
|
"${profilesPath}/${profile.path}/user.js" = mkIf (profile.settings != { }
|
2023-01-10 09:35:00 +00:00
|
|
|
|
|| profile.extraConfig != "" || profile.bookmarks != [ ]) {
|
2021-11-04 16:42:44 +00:00
|
|
|
|
text =
|
|
|
|
|
mkUserJs profile.settings profile.extraConfig profile.bookmarks;
|
|
|
|
|
};
|
2020-10-12 00:22:58 +00:00
|
|
|
|
|
2023-01-10 09:35:00 +00:00
|
|
|
|
"${profilesPath}/${profile.path}/search.json.mozlz4" = mkIf
|
|
|
|
|
(profile.search.default != null || profile.search.order != [ ]
|
|
|
|
|
|| profile.search.engines != { }) {
|
|
|
|
|
force = profile.search.force;
|
|
|
|
|
source = let
|
|
|
|
|
settings = {
|
|
|
|
|
version = 6;
|
|
|
|
|
|
|
|
|
|
engines = let
|
|
|
|
|
allEngines = (profile.search.engines //
|
|
|
|
|
# If search.default isn't in search.engines, assume it's app
|
|
|
|
|
# provided and include it in the set of all engines
|
|
|
|
|
optionalAttrs (profile.search.default != null
|
|
|
|
|
&& !(hasAttr profile.search.default
|
|
|
|
|
profile.search.engines)) {
|
|
|
|
|
${profile.search.default} = { };
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
# Map allEngines to a list and order by search.order
|
|
|
|
|
orderedEngineList = (imap (order: name:
|
|
|
|
|
let engine = allEngines.${name} or { };
|
|
|
|
|
in engine // {
|
|
|
|
|
inherit name;
|
|
|
|
|
metaData = (engine.metaData or { }) // { inherit order; };
|
|
|
|
|
}) profile.search.order) ++ (mapAttrsToList
|
|
|
|
|
(name: config: config // { inherit name; })
|
|
|
|
|
(removeAttrs allEngines profile.search.order));
|
|
|
|
|
|
|
|
|
|
engines = map (config:
|
|
|
|
|
let
|
|
|
|
|
name = config.name;
|
|
|
|
|
isAppProvided = removeAttrs config [ "name" "metaData" ]
|
|
|
|
|
== { };
|
|
|
|
|
metaData = config.metaData or { };
|
|
|
|
|
in mapAttrs' (name: value: {
|
|
|
|
|
# Map nice field names to internal field names. This is
|
|
|
|
|
# intended to be exhaustive, but any future fields will
|
|
|
|
|
# either have to be specified with an underscore, or added
|
|
|
|
|
# to this map.
|
|
|
|
|
name = ((genAttrs [
|
|
|
|
|
"name"
|
|
|
|
|
"isAppProvided"
|
|
|
|
|
"loadPath"
|
|
|
|
|
"hasPreferredIcon"
|
|
|
|
|
"updateInterval"
|
|
|
|
|
"updateURL"
|
|
|
|
|
"iconUpdateURL"
|
|
|
|
|
"iconURL"
|
|
|
|
|
"iconMapObj"
|
|
|
|
|
"metaData"
|
|
|
|
|
"orderHint"
|
|
|
|
|
"definedAliases"
|
|
|
|
|
"urls"
|
|
|
|
|
] (name: "_${name}")) // {
|
|
|
|
|
"searchForm" = "__searchForm";
|
|
|
|
|
}).${name} or name;
|
|
|
|
|
|
|
|
|
|
inherit value;
|
|
|
|
|
}) ((removeAttrs config [ "icon" ])
|
|
|
|
|
// (optionalAttrs (!isAppProvided)
|
|
|
|
|
(optionalAttrs (config ? iconUpdateURL) {
|
|
|
|
|
# Convenience to default iconURL to iconUpdateURL so
|
|
|
|
|
# the icon is immediately downloaded from the URL
|
|
|
|
|
iconURL = config.iconURL or config.iconUpdateURL;
|
|
|
|
|
} // optionalAttrs (config ? icon) {
|
|
|
|
|
# Convenience to specify absolute path to icon
|
|
|
|
|
iconURL = "file://${config.icon}";
|
|
|
|
|
} // {
|
|
|
|
|
# Required for custom engine configurations, loadPaths
|
|
|
|
|
# are unique identifiers that are generally formatted
|
|
|
|
|
# like: [source]/path/to/engine.xml
|
|
|
|
|
loadPath = ''
|
|
|
|
|
[home-manager]/programs.firefox.profiles.${profile.name}.search.engines."${
|
|
|
|
|
replaceStrings [ "\\" ] [ "\\\\" ] name
|
|
|
|
|
}"'';
|
|
|
|
|
})) // {
|
|
|
|
|
# Required fields for all engine configurations
|
|
|
|
|
inherit name isAppProvided metaData;
|
|
|
|
|
})) orderedEngineList;
|
|
|
|
|
in engines;
|
|
|
|
|
|
|
|
|
|
metaData = optionalAttrs (profile.search.default != null) {
|
|
|
|
|
current = profile.search.default;
|
|
|
|
|
hash = "@hash@";
|
|
|
|
|
} // {
|
|
|
|
|
useSavedOrder = profile.search.order != [ ];
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
# Home Manager doesn't circumvent user consent and isn't acting
|
|
|
|
|
# maliciously. We're modifying the search outside of Firefox, 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 = profile.path + profile.search.default
|
|
|
|
|
+ disclaimer "Firefox";
|
|
|
|
|
in pkgs.runCommand "search.json.mozlz4" {
|
|
|
|
|
nativeBuildInputs = with pkgs; [ mozlz4a openssl ];
|
|
|
|
|
json = builtins.toJSON settings;
|
|
|
|
|
inherit salt;
|
|
|
|
|
} ''
|
|
|
|
|
export hash=$(echo -n "$salt" | openssl dgst -sha256 -binary | base64)
|
|
|
|
|
mozlz4a <(substituteStream json search.json.in --subst-var hash) "$out"
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-04 16:42:44 +00:00
|
|
|
|
"${profilesPath}/${profile.path}/extensions" =
|
|
|
|
|
mkIf (cfg.extensions != [ ]) {
|
2020-10-12 00:22:58 +00:00
|
|
|
|
source = "${extensionsEnvPkg}/share/mozilla/${extensionPath}";
|
|
|
|
|
recursive = true;
|
|
|
|
|
force = true;
|
|
|
|
|
};
|
2021-11-04 16:42:44 +00:00
|
|
|
|
}));
|
2020-10-12 00:22:58 +00:00
|
|
|
|
};
|
|
|
|
|
}
|