feat: Add learning for omnibox commands, p=#10564

This commit is contained in:
mr. m
2025-09-27 00:53:25 +02:00
committed by GitHub
parent df1e759c8d
commit 403d5ae92e
24 changed files with 419 additions and 349 deletions

View File

@@ -62,3 +62,5 @@ zen-icons-picker-emoji =
.label = Emojis .label = Emojis
zen-icons-picker-svg = zen-icons-picker-svg =
.label = Icons .label = Icons
urlbar-search-mode-zen_actions = Actions

View File

@@ -5,6 +5,7 @@
"lint-staged": { "lint-staged": {
"**/*": "" "**/*": ""
}, },
"type": "module",
"scripts": { "scripts": {
"build": "surfer build", "build": "surfer build",
"build:ui": "surfer build --ui", "build:ui": "surfer build --ui",

View File

@@ -1,8 +1,8 @@
diff --git a/browser/components/sessionstore/TabState.sys.mjs b/browser/components/sessionstore/TabState.sys.mjs diff --git a/browser/components/sessionstore/TabState.sys.mjs b/browser/components/sessionstore/TabState.sys.mjs
index 82721356d191055bec0d4b0ca49e481221988801..9c8a2b1791e780e0fcd3a9bfc7efdadf35d52165 100644 index 82721356d191055bec0d4b0ca49e481221988801..1ea5c394c704da295149443d7794961a12f2060b 100644
--- a/browser/components/sessionstore/TabState.sys.mjs --- a/browser/components/sessionstore/TabState.sys.mjs
+++ b/browser/components/sessionstore/TabState.sys.mjs +++ b/browser/components/sessionstore/TabState.sys.mjs
@@ -85,6 +85,18 @@ class _TabState { @@ -85,7 +85,22 @@ class _TabState {
tabData.groupId = tab.group.id; tabData.groupId = tab.group.id;
} }
@@ -19,5 +19,9 @@ index 82721356d191055bec0d4b0ca49e481221988801..9c8a2b1791e780e0fcd3a9bfc7efdadf
+ tabData.zenIsGlance = tab.hasAttribute("zen-glance-tab"); + tabData.zenIsGlance = tab.hasAttribute("zen-glance-tab");
+ +
tabData.searchMode = tab.ownerGlobal.gURLBar.getSearchMode(browser, true); tabData.searchMode = tab.ownerGlobal.gURLBar.getSearchMode(browser, true);
+ if (tabData.searchMode?.source === tab.ownerGlobal.UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS) {
+ delete tabData.searchMode;
+ }
tabData.userContextId = tab.userContextId || 0; tabData.userContextId = tab.userContextId || 0;

View File

@@ -1,13 +1,13 @@
diff --git a/browser/components/urlbar/UrlbarController.sys.mjs b/browser/components/urlbar/UrlbarController.sys.mjs diff --git a/browser/components/urlbar/UrlbarController.sys.mjs b/browser/components/urlbar/UrlbarController.sys.mjs
index 36e3ab4a5a153230bb488b66dda7e3e7c763ca23..28b59c7c3a95febafc3f2a6e0ac3493b9785ff1a 100644 index 36e3ab4a5a153230bb488b66dda7e3e7c763ca23..81f2944b939ac0963c129f86aab0b55817349401 100644
--- a/browser/components/urlbar/UrlbarController.sys.mjs --- a/browser/components/urlbar/UrlbarController.sys.mjs
+++ b/browser/components/urlbar/UrlbarController.sys.mjs +++ b/browser/components/urlbar/UrlbarController.sys.mjs
@@ -411,7 +411,7 @@ export class UrlbarController { @@ -434,6 +434,8 @@ export class UrlbarController {
// When there's no search string and no view selection, we want to focus });
// the next toolbar item instead, for accessibility reasons. }
let allowTabbingThroughResults = event.preventDefault();
- this.input.focusedViaMousedown || + } else {
+ true || + this.browserWindow.gZenUIManager.enableCommandsMode(event);
this.input.searchMode?.isPreview || }
this.input.searchMode?.source == break;
lazy.UrlbarUtils.RESULT_SOURCE.ACTIONS || }

View File

@@ -1,5 +1,5 @@
diff --git a/browser/components/urlbar/UrlbarInput.sys.mjs b/browser/components/urlbar/UrlbarInput.sys.mjs diff --git a/browser/components/urlbar/UrlbarInput.sys.mjs b/browser/components/urlbar/UrlbarInput.sys.mjs
index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c563851ae89385 100644 index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..1dc520f63b240cccda7be074346d2079774eed27 100644
--- a/browser/components/urlbar/UrlbarInput.sys.mjs --- a/browser/components/urlbar/UrlbarInput.sys.mjs
+++ b/browser/components/urlbar/UrlbarInput.sys.mjs +++ b/browser/components/urlbar/UrlbarInput.sys.mjs
@@ -74,6 +74,13 @@ ChromeUtils.defineLazyGetter(lazy, "logger", () => @@ -74,6 +74,13 @@ ChromeUtils.defineLazyGetter(lazy, "logger", () =>
@@ -184,7 +184,20 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
let controller = let controller =
this.document.commandDispatcher.getControllerForCommand("cmd_paste"); this.document.commandDispatcher.getControllerForCommand("cmd_paste");
@@ -4130,6 +4195,7 @@ export class UrlbarInput { @@ -3836,7 +3901,11 @@ export class UrlbarInput {
if (!engineName && !source && !this.hasAttribute("searchmode")) {
return;
}
-
+ this.window.dispatchEvent(
+ new CustomEvent("Zen:UrlbarSearchModeChanged", {
+ detail: { searchMode },
+ })
+ );
this._searchModeIndicatorTitle.textContent = "";
this._searchModeIndicatorTitle.removeAttribute("data-l10n-id");
@@ -4130,6 +4199,7 @@ export class UrlbarInput {
this.document.l10n.setAttributes( this.document.l10n.setAttributes(
this.inputField, this.inputField,
@@ -192,7 +205,7 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
l10nId, l10nId,
l10nId == "urlbar-placeholder-with-name" ? { name } : undefined l10nId == "urlbar-placeholder-with-name" ? { name } : undefined
); );
@@ -4241,6 +4307,11 @@ export class UrlbarInput { @@ -4241,6 +4311,11 @@ export class UrlbarInput {
} }
_on_click(event) { _on_click(event) {
@@ -204,7 +217,7 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
if ( if (
event.target == this.inputField || event.target == this.inputField ||
event.target == this._inputContainer event.target == this._inputContainer
@@ -4311,7 +4382,7 @@ export class UrlbarInput { @@ -4311,7 +4386,7 @@ export class UrlbarInput {
} }
} }
@@ -213,7 +226,7 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
this.view.autoOpen({ event }); this.view.autoOpen({ event });
} else { } else {
if (this._untrimOnFocusAfterKeydown) { if (this._untrimOnFocusAfterKeydown) {
@@ -4351,9 +4422,16 @@ export class UrlbarInput { @@ -4351,9 +4426,16 @@ export class UrlbarInput {
} }
_on_mousedown(event) { _on_mousedown(event) {
@@ -231,7 +244,7 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
if ( if (
event.target != this.inputField && event.target != this.inputField &&
@@ -4364,6 +4442,10 @@ export class UrlbarInput { @@ -4364,6 +4446,10 @@ export class UrlbarInput {
this.focusedViaMousedown = !this.focused; this.focusedViaMousedown = !this.focused;
this._preventClickSelectsAll = this.focused; this._preventClickSelectsAll = this.focused;
@@ -242,7 +255,7 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
// Keep the focus status, since the attribute may be changed // Keep the focus status, since the attribute may be changed
// upon calling this.focus(). // upon calling this.focus().
@@ -4399,7 +4481,7 @@ export class UrlbarInput { @@ -4399,7 +4485,7 @@ export class UrlbarInput {
} }
// Don't close the view when clicking on a tab; we may want to keep the // Don't close the view when clicking on a tab; we may want to keep the
// view open on tab switch, and the TabSelect event arrived earlier. // view open on tab switch, and the TabSelect event arrived earlier.
@@ -251,3 +264,12 @@ index 1c447bd31de854d1522dbcfb5d7ad557c84f1388..2341d04c5afee303ce4150c3c2c56385
break; break;
} }
@@ -4716,7 +4802,7 @@ export class UrlbarInput {
// When we are in actions search mode we can show more results so
// increase the limit.
let maxResults =
- this.searchMode?.source != lazy.UrlbarUtils.RESULT_SOURCE.ACTIONS
+ this.searchMode?.source != lazy.UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS
? lazy.UrlbarPrefs.get("maxRichResults")
: UNLIMITED_MAX_RESULTS;
let options = {

View File

@@ -1,5 +1,5 @@
diff --git a/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs b/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs diff --git a/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs b/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs
index cdc476a3eb2ee2cb6193d215513b65ed375f6153..a2b106916d6cca25096d37b80bea45f016ad82a5 100644 index cdc476a3eb2ee2cb6193d215513b65ed375f6153..bc66d9651e521bda75a3bb9e7f1e4b3bb325be90 100644
--- a/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs --- a/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs
+++ b/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs +++ b/browser/components/urlbar/UrlbarMuxerStandard.sys.mjs
@@ -855,6 +855,7 @@ class MuxerUnifiedComplete extends UrlbarMuxer { @@ -855,6 +855,7 @@ class MuxerUnifiedComplete extends UrlbarMuxer {
@@ -10,3 +10,13 @@ index cdc476a3eb2ee2cb6193d215513b65ed375f6153..a2b106916d6cca25096d37b80bea45f0
// Discard the result if a tab-to-search result was added already. // Discard the result if a tab-to-search result was added already.
if (!state.canAddTabToSearch) { if (!state.canAddTabToSearch) {
return false; return false;
@@ -1501,7 +1502,9 @@ class MuxerUnifiedComplete extends UrlbarMuxer {
usedLimits.maxResultCount++;
}
+ if (!(result.heuristic && result.source == UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS)) {
state.usedResultSpan += span;
+ }
this._updateStatePostAdd(result, state);
return true;

View File

@@ -1,12 +1,12 @@
diff --git a/browser/components/urlbar/UrlbarPrefs.sys.mjs b/browser/components/urlbar/UrlbarPrefs.sys.mjs diff --git a/browser/components/urlbar/UrlbarPrefs.sys.mjs b/browser/components/urlbar/UrlbarPrefs.sys.mjs
index 3c179db3b310c43f8c6c06b1ecbcf5ed59feefe6..693bef15401cd4428c8a0222de57b83b78564194 100644 index 3c179db3b310c43f8c6c06b1ecbcf5ed59feefe6..d9d2ce116ebcee8d403e165066c3a569bb952cd2 100644
--- a/browser/components/urlbar/UrlbarPrefs.sys.mjs --- a/browser/components/urlbar/UrlbarPrefs.sys.mjs
+++ b/browser/components/urlbar/UrlbarPrefs.sys.mjs +++ b/browser/components/urlbar/UrlbarPrefs.sys.mjs
@@ -719,6 +719,7 @@ function makeResultGroups({ showSearchSuggestionsFirst }) { @@ -719,6 +719,7 @@ function makeResultGroups({ showSearchSuggestionsFirst }) {
*/ */
let rootGroup = { let rootGroup = {
children: [ children: [
+ { group: lazy.UrlbarUtils.RESULT_GROUP.ZEN_ACTION }, + { children: [{ group: lazy.UrlbarUtils.RESULT_GROUP.ZEN_ACTION }] },
// heuristic // heuristic
{ {
maxResultCount: 1, maxResultCount: 1,

View File

@@ -1,5 +1,5 @@
diff --git a/browser/components/urlbar/UrlbarUtils.sys.mjs b/browser/components/urlbar/UrlbarUtils.sys.mjs diff --git a/browser/components/urlbar/UrlbarUtils.sys.mjs b/browser/components/urlbar/UrlbarUtils.sys.mjs
index 0bc15c02f56dd8f46a21fed02b4e21a741f27f41..b69a4d34f700637bd352620c520b989cf00fa39f 100644 index 0bc15c02f56dd8f46a21fed02b4e21a741f27f41..40da868f68f21d8411107fb8a95e2d0b74337b51 100644
--- a/browser/components/urlbar/UrlbarUtils.sys.mjs --- a/browser/components/urlbar/UrlbarUtils.sys.mjs
+++ b/browser/components/urlbar/UrlbarUtils.sys.mjs +++ b/browser/components/urlbar/UrlbarUtils.sys.mjs
@@ -75,6 +75,7 @@ export var UrlbarUtils = { @@ -75,6 +75,7 @@ export var UrlbarUtils = {
@@ -10,12 +10,20 @@ index 0bc15c02f56dd8f46a21fed02b4e21a741f27f41..b69a4d34f700637bd352620c520b989c
}), }),
// Defines provider types. // Defines provider types.
@@ -576,6 +577,8 @@ export var UrlbarUtils = { @@ -134,6 +135,7 @@ export var UrlbarUtils = {
return this.RESULT_GROUP.INPUT_HISTORY; OTHER_NETWORK: 6,
case "UrlbarProviderQuickSuggest": ADDON: 7,
return this.RESULT_GROUP.GENERAL_PARENT; ACTIONS: 8,
+ ZEN_ACTIONS: 9,
}),
// Per-result exposure telemetry.
@@ -553,6 +555,8 @@ export var UrlbarUtils = {
return this.RESULT_GROUP.HEURISTIC_SEARCH_TIP;
case "HistoryUrlHeuristic":
return this.RESULT_GROUP.HEURISTIC_HISTORY_URL;
+ case "ZenUrlbarProviderGlobalActions": + case "ZenUrlbarProviderGlobalActions":
+ return this.RESULT_GROUP.ZEN_ACTION; + return this.RESULT_GROUP.ZEN_ACTION;
default: default:
break; if (result.providerName.startsWith("TestProvider")) {
} return this.RESULT_GROUP.HEURISTIC_TEST;

View File

@@ -1,5 +1,5 @@
diff --git a/browser/components/urlbar/UrlbarView.sys.mjs b/browser/components/urlbar/UrlbarView.sys.mjs diff --git a/browser/components/urlbar/UrlbarView.sys.mjs b/browser/components/urlbar/UrlbarView.sys.mjs
index fdbab8806fd320f4aacec46a42c8ef953580d00c..e23fae0d7e0b71d74899c11c229359864cd7e427 100644 index fdbab8806fd320f4aacec46a42c8ef953580d00c..031a615ad09274c578184b129434bbd93b49353d 100644
--- a/browser/components/urlbar/UrlbarView.sys.mjs --- a/browser/components/urlbar/UrlbarView.sys.mjs
+++ b/browser/components/urlbar/UrlbarView.sys.mjs +++ b/browser/components/urlbar/UrlbarView.sys.mjs
@@ -613,7 +613,7 @@ export class UrlbarView { @@ -613,7 +613,7 @@ export class UrlbarView {
@@ -11,39 +11,7 @@ index fdbab8806fd320f4aacec46a42c8ef953580d00c..e23fae0d7e0b71d74899c11c22935986
// Try to reuse the cached top-sites context. If it's not cached, then // Try to reuse the cached top-sites context. If it's not cached, then
// there will be a gap of time between when the input is focused and // there will be a gap of time between when the input is focused and
// when the view opens that can be perceived as flicker. // when the view opens that can be perceived as flicker.
@@ -824,6 +824,31 @@ export class UrlbarView { @@ -2706,6 +2706,8 @@ export class UrlbarView {
// still associated with the first result.
this.input.setResultForCurrentValue(firstResult);
}
+ if (queryContext.results[0].payload.zenAction) {
+ this.#selectElement(this.getFirstSelectableElement(), {
+ updateInput: false,
+ setAccessibleFocus:
+ this.controller._userSelectionBehavior == "arrow",
+ });
+ this.window.setTimeout(() => {
+ this.window.setTimeout(() => {
+ this.#selectElement(this.getFirstSelectableElement(), {
+ updateInput: false,
+ setAccessibleFocus:
+ this.controller._userSelectionBehavior == "arrow",
+ });
+ });
+ });
+ }
+ this.window.setTimeout(() => {
+ if (queryContext.results[0].payload.zenAction) {
+ this.#selectElement(this.getFirstSelectableElement(), {
+ updateInput: false,
+ setAccessibleFocus:
+ this.controller._userSelectionBehavior == "arrow",
+ });
+ }
+ }, 220);
}
// Announce tab-to-search results to screen readers as the user types.
@@ -2706,6 +2731,8 @@ export class UrlbarView {
if (row?.hasAttribute("row-selectable")) { if (row?.hasAttribute("row-selectable")) {
row?.toggleAttribute("selected", true); row?.toggleAttribute("selected", true);
} }
@@ -52,7 +20,7 @@ index fdbab8806fd320f4aacec46a42c8ef953580d00c..e23fae0d7e0b71d74899c11c22935986
if (element != row) { if (element != row) {
row?.toggleAttribute("descendant-selected", true); row?.toggleAttribute("descendant-selected", true);
} }
@@ -3189,7 +3216,7 @@ export class UrlbarView { @@ -3189,7 +3191,7 @@ export class UrlbarView {
} }
#enableOrDisableRowWrap() { #enableOrDisableRowWrap() {

View File

@@ -103,6 +103,7 @@ var gZenCommonActions = {
gZenUIManager.showToast('zen-copy-current-url-confirmation', { button, timeout: 3000 }); gZenUIManager.showToast('zen-copy-current-url-confirmation', { button, timeout: 3000 });
} }
}, },
copyCurrentURLAsMarkdownToClipboard() { copyCurrentURLAsMarkdownToClipboard() {
const currentUrl = gBrowser.currentURI.spec; const currentUrl = gBrowser.currentURI.spec;
const tabTitle = gBrowser.selectedTab.label; const tabTitle = gBrowser.selectedTab.label;

View File

@@ -29,8 +29,12 @@
background.appendChild(grain); background.appendChild(grain);
document.getElementById('browser').prepend(background); document.getElementById('browser').prepend(background);
const toolbarBackground = background.cloneNode(true); const toolbarBackground = background.cloneNode(true);
toolbarBackground.id = 'zen-toolbar-background'; toolbarBackground.removeAttribute('id');
toolbarBackground.classList.add('zen-toolbar-background');
document.getElementById('titlebar').prepend(toolbarBackground); document.getElementById('titlebar').prepend(toolbarBackground);
document
.getElementById('zen-appcontent-navbar-container')
.prepend(toolbarBackground.cloneNode(true));
} }
#zenInitBrowserLayout() { #zenInitBrowserLayout() {

View File

@@ -59,6 +59,10 @@ var gZenUIManager = {
}); });
window.addEventListener('TabClose', this.onTabClose.bind(this)); window.addEventListener('TabClose', this.onTabClose.bind(this));
window.addEventListener(
'Zen:UrlbarSearchModeChanged',
this.onUrlbarSearchModeChanged.bind(this)
);
gZenMediaController.init(); gZenMediaController.init();
gZenVerticalTabsManager.init(); gZenVerticalTabsManager.init();
@@ -241,6 +245,53 @@ var gZenUIManager = {
// Section: URL bar // Section: URL bar
onUrlbarSearchModeChanged(event) {
const { searchMode } = event.detail;
const input = gURLBar.textbox;
if (gURLBar.hasAttribute('breakout-extend') && !this._animatingSearchMode) {
this._animatingSearchMode = true;
this.motion
.animate(input, { scale: [1, 0.98, 1] }, { duration: 0.25, delay: 0.1 })
.then(() => {
delete this._animatingSearchMode;
});
if (searchMode) {
gURLBar.setAttribute('animate-searchmode', 'true');
this._animatingSearchModeTimeout = setTimeout(() => {
requestAnimationFrame(() => {
gURLBar.removeAttribute('animate-searchmode');
delete this._animatingSearchModeTimeout;
});
}, 700);
}
}
},
enableCommandsMode(event) {
event.preventDefault();
if (!gURLBar.hasAttribute('breakout-extend') || this._animatingSearchMode) {
return;
}
const currentSearchMode = gURLBar.getSearchMode(gBrowser.selectedBrowser);
let searchMode = null;
if (!currentSearchMode) {
searchMode = {
source: UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS,
isPreview: true,
};
}
gURLBar.removeAttribute('animate-searchmode');
if (this._animatingSearchModeTimeout) {
clearTimeout(this._animatingSearchModeTimeout);
delete this._animatingSearchModeTimeout;
}
gURLBar.searchMode = searchMode;
gURLBar.startQuery({
allowAutofill: false,
event,
});
},
get newtabButtons() { get newtabButtons() {
return document.querySelectorAll('#tabs-newtab-button'); return document.querySelectorAll('#tabs-newtab-button');
}, },

View File

@@ -57,113 +57,6 @@
} }
} }
@keyframes better-sidebar-pinned-hide {
0% {
opacity: 1;
transform: scale(1) rotateX(0deg);
}
100% {
transform: scale(0.99) rotateX(-5deg);
opacity: 0;
}
}
@keyframes better-sidebar-pinned-show {
0% {
opacity: 0;
transform: scale(0.99) rotateX(-5deg);
}
100% {
transform: scale(1) rotateX(0deg);
opacity: 1;
}
}
@keyframes better-sidebar-hide {
0% {
opacity: 1;
transform: scale(1) rotateX(0deg);
}
100% {
transform: scale(0.99) rotateX(-5deg);
opacity: 0;
}
}
@keyframes better-sidebar-show {
0% {
opacity: 0;
}
100% {
transform: scale(1);
opacity: 1;
}
}
@keyframes better-sidebar-intro-show {
0% {
opacity: 0;
transform: translateY(5px);
}
100% {
transform: translateY(0px);
opacity: 1;
}
}
@keyframes zen-vtabs-animation {
0% {
opacity: 0;
transform: translateX(-10px);
}
20% {
opacity: 1;
}
100% {
transform: translateX(0);
}
}
@keyframes zen-sidebar-panel-animation-right {
0% {
opacity: 0;
transform: translateX(10px);
}
20% {
opacity: 1;
}
100% {
transform: translateX(0);
}
}
@keyframes zen-sidebar-panel-animation-reverse {
0% {
opacity: 1;
transform: translateX(0) scale3d(1, 1, 1);
}
99% {
opacity: 0;
transform: translateX(-50px) scale3d(0.95, 0.95, 0.95);
}
100% {
display: none !important;
}
}
/* Mark: Zen Glance */ /* Mark: Zen Glance */
@keyframes zen-glance-overlay-animation { @keyframes zen-glance-overlay-animation {
from { from {
@@ -187,82 +80,6 @@
} }
} }
@keyframes zen-rice-form-out {
0% {
transform: translateX(0);
max-height: 350px;
opacity: 1;
position: relative;
}
50% {
transform: translateX(-100%);
opacity: 0;
position: relative;
}
99% {
transform: translateX(-100%);
opacity: 0;
max-height: 15px;
position: relative;
}
100% {
transform: translateX(-100%);
opacity: 0;
pointer-events: none;
position: absolute;
}
}
@keyframes zen-rice-form-in {
0% {
position: absolute;
transform: translateX(100%);
opacity: 0;
}
99% {
position: absolute;
transform: translateX(0);
opacity: 1;
}
100% {
position: relative;
}
}
@keyframes zen-rice-form-in-2 {
from {
opacity: 0;
transform: translateX(100%);
max-height: 50px;
}
to {
opacity: 1;
transform: translateX(0);
max-height: 450px;
}
}
@keyframes zen-rice-submit-animation {
/* Scale and shake the dialog */
0% {
transform: scale(1);
}
50% {
transform: scale(1.1);
}
100% {
transform: scale(1);
}
}
@keyframes zen-back-and-forth-text { @keyframes zen-back-and-forth-text {
0%, 0%,
10% { 10% {
@@ -282,3 +99,14 @@
left: 0; left: 0;
} }
} }
/* Mark: URL Bar */
@keyframes zen-urlbar-searchmode {
0% {
box-shadow: 0 0 20px color-mix(in srgb, var(--zen-primary-color), var(--toolbox-textcolor) 40%);
}
100% {
box-shadow: 0 0 300px color-mix(in srgb, var(--zen-primary-color), transparent 100%);
}
}

View File

@@ -64,7 +64,7 @@
transition: 0s; transition: 0s;
} }
&:is(#zen-toolbar-background) { &:is(.zen-toolbar-background) {
&::after { &::after {
background: var(--zen-main-browser-background-toolbar); background: var(--zen-main-browser-background-toolbar);
} }
@@ -76,7 +76,7 @@
transition: 0s; transition: 0s;
} }
&:is(#zen-toolbar-background) { &:is(.zen-toolbar-background) {
&::before { &::before {
background: var(--zen-main-browser-background-toolbar-old); background: var(--zen-main-browser-background-toolbar-old);
} }

View File

@@ -155,15 +155,15 @@
order: 2 !important; order: 2 !important;
} }
#urlbar[breakout] {
position: fixed;
}
#urlbar[breakout-extend='true'] { #urlbar[breakout-extend='true'] {
position: fixed;
z-index: 2; z-index: 2;
& .urlbar-input-container { & .urlbar-input-container {
font-weight: 400; font-weight: 400;
@media (-moz-platform: windows) {
font-weight: 500;
}
} }
& #identity-box { & #identity-box {
@@ -188,13 +188,28 @@
) !important; ) !important;
box-shadow: 0px 0px 90px -10px light-dark(rgba(0, 0, 0, 0.8), rgba(0, 0, 0, 0.75)) !important; box-shadow: 0px 0px 90px -10px light-dark(rgba(0, 0, 0, 0.8), rgba(0, 0, 0, 0.75)) !important;
backdrop-filter: none !important; backdrop-filter: none !important;
border-radius: 12px !important;
outline: 0.5px solid light-dark(rgba(0, 0, 0, 0.2), rgba(255, 255, 255, 0.2)) !important; outline: 0.5px solid light-dark(rgba(0, 0, 0, 0.2), rgba(255, 255, 255, 0.2)) !important;
outline-offset: var(--zen-urlbar-outline-offset) !important; outline-offset: var(--zen-urlbar-outline-offset) !important;
@media -moz-pref('zen.theme.acrylic-elements') { @media -moz-pref('zen.theme.acrylic-elements') {
backdrop-filter: blur(42px) saturate(110%) brightness(0.25) contrast(100%) !important; backdrop-filter: blur(42px) saturate(110%) brightness(0.25) contrast(100%) !important;
} }
} }
&,
.urlbar-background {
border-radius: 12px !important;
}
&[breakout-extend][animate-searchmode='true']::before {
position: absolute;
content: '';
inset: 0;
width: 100%;
height: 100%;
border-radius: inherit;
pointer-events: none;
animation: zen-urlbar-searchmode ease-out 0.7s forwards;
}
} }
#urlbar-go-button { #urlbar-go-button {
@@ -444,7 +459,7 @@
font-size: 1.5em !important; font-size: 1.5em !important;
} }
top: var(--zen-urlbar-top) !important; top: var(--zen-urlbar-top) !important;
transform: translateX(-50%); translate: -50% 0%;
left: 50% !important; left: 50% !important;
#urlbar-container:has(&) { #urlbar-container:has(&) {
@@ -617,7 +632,7 @@
var(--zen-primary-color) 50%, var(--zen-primary-color) 50%,
light-dark(rgba(0, 0, 0, 0.5), rgba(255, 255, 255, 0.2)) 50% light-dark(rgba(0, 0, 0, 0.5), rgba(255, 255, 255, 0.2)) 50%
); );
--zen-selected-color: color-mix(in srgb, var(--zen-selected-bg), light-dark(white, black) 40%); --zen-selected-color: color-mix(in srgb, var(--zen-selected-bg), black 30%);
background-color: var(--zen-selected-bg) !important; background-color: var(--zen-selected-bg) !important;
& *, & *,
@@ -627,6 +642,7 @@
& .urlbarView-favicon { & .urlbarView-favicon {
fill: currentColor !important; fill: currentColor !important;
stroke: currentColor !important;
} }
& .urlbarView-shortcutContent { & .urlbarView-shortcutContent {
@@ -656,8 +672,14 @@
& .action-contextualidentity { & .action-contextualidentity {
display: none; display: none;
} }
}
.urlbarView[noresults] > .urlbarView-body-outer > .urlbarView-body-inner > & { /* Hide the results body when there are no results, to avoid showing an empty box */
.urlbarView[noresults] > .urlbarView-body-outer > .urlbarView-body-inner,
#urlbar-search-mode-indicator-close {
display: none; display: none;
} }
#urlbar-search-mode-indicator-title {
font-weight: 600;
} }

View File

@@ -46,12 +46,27 @@ var ZenThemeModifier = {
Services.prefs.addObserver(pref, handleEvent); Services.prefs.addObserver(pref, handleEvent);
} }
// Add fullscreen listener to update the theme when going in and out of fullscreen
const eventsForSeparation = [
'ZenViewSplitter:SplitViewDeactivated',
'ZenViewSplitter:SplitViewActivated',
'fullscreen',
'ZenCompactMode:Toggled',
];
const separationHandler = this.updateElementSeparation.bind(this);
for (let eventName of eventsForSeparation) {
window.addEventListener(eventName, separationHandler);
}
window.addEventListener( window.addEventListener(
'unload', 'unload',
() => { () => {
for (let pref of kZenThemePrefsList) { for (let pref of kZenThemePrefsList) {
Services.prefs.removeObserver(pref, handleEvent); Services.prefs.removeObserver(pref, handleEvent);
} }
for (let eventName of eventsForSeparation) {
window.removeEventListener(eventName, separationHandler);
}
}, },
{ once: true } { once: true }
); );
@@ -78,6 +93,13 @@ var ZenThemeModifier = {
updateElementSeparation() { updateElementSeparation() {
let separation = this.elementSeparation; let separation = this.elementSeparation;
if (
window.fullScreen &&
window.gZenCompactModeManager?.preference &&
!document.getElementById('tabbrowser-tabbox')?.hasAttribute('zen-split-view')
) {
separation = 0;
}
document.documentElement.style.setProperty('--zen-element-separation', separation + 'px'); document.documentElement.style.setProperty('--zen-element-separation', separation + 'px');
if (separation == 0) { if (separation == 0) {
document.documentElement.setAttribute('zen-no-padding', true); document.documentElement.setAttribute('zen-no-padding', true);

View File

@@ -246,6 +246,7 @@ var gZenCompactModeManager = {
} else { } else {
ZenHasPolyfill.disconnectObserver(this.sidebarObserverId); ZenHasPolyfill.disconnectObserver(this.sidebarObserverId);
} }
window.dispatchEvent(new CustomEvent('ZenCompactMode:Toggled', { detail: this.preference }));
}, },
// NOTE: Dont actually use event, it's just so we make sure // NOTE: Dont actually use event, it's just so we make sure

View File

@@ -5,9 +5,32 @@
*/ */
/* All overrides for compact mode go here */ /* All overrides for compact mode go here */
#zen-toolbar-background { .zen-toolbar-background {
display: none; display: none;
pointer-events: none; pointer-events: none;
overflow: clip;
z-index: -1;
background: black; /* Any color thats not white */
box-shadow: var(--zen-big-shadow);
@media -moz-pref('zen.theme.acrylic-elements') {
background: transparent;
backdrop-filter: blur(42px) saturate(110%) brightness(0.25) contrast(100%) !important;
}
&::before,
&::after {
outline: 1px solid rgba(255, 255, 255, .15);
outline-offset: -1px;
background-attachment: fixed !important;
background-size: 100vw 100vh !important;
}
&,
&::before,
&::after {
border-radius: calc(var(--zen-native-inner-radius) + var(--zen-element-separation) / 4 - var(--zen-compact-mode-no-padding-radius-fix, 0px));
}
} }
:root[zen-compact-mode='true']:not([customizing]):not([inDOMFullscreen='true']) { :root[zen-compact-mode='true']:not([customizing]):not([inDOMFullscreen='true']) {
@@ -140,30 +163,8 @@
width: calc(var(--zen-sidebar-width) + var(--zen-toolbox-padding)); width: calc(var(--zen-sidebar-width) + var(--zen-toolbox-padding));
} }
& #zen-toolbar-background { & .zen-toolbar-background {
display: flex; display: flex;
overflow: clip;
z-index: -1;
background: black; /* Any color thats not white */
box-shadow: var(--zen-big-shadow);
@media -moz-pref('zen.theme.acrylic-elements') {
background: transparent;
backdrop-filter: blur(42px) saturate(110%) brightness(0.25) contrast(100%) !important;
}
&::before,
&::after {
outline: 1px solid rgba(255, 255, 255, .15);
outline-offset: -1px;
background-attachment: fixed !important;
background-size: 100vw !important;
}
&,
&::before,
&::after {
border-radius: calc(var(--zen-native-inner-radius) + var(--zen-element-separation) / 4 - var(--zen-compact-mode-no-padding-radius-fix, 0px));
}
} }
} }
@@ -348,11 +349,6 @@
} }
transition: all 0.15s ease; transition: all 0.15s ease;
width: 100%; width: 100%;
background: var(--zen-dialog-background);
& > * {
position: relative !important;
}
} }
} }
@@ -364,19 +360,8 @@
) { ) {
& #zen-appcontent-navbar-container { & #zen-appcontent-navbar-container {
visibility: visible !important; visibility: visible !important;
:root[zen-show-grainy-background='true'] &::before { & .zen-toolbar-background {
content: ''; display: flex;
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-image: url(chrome://browser/content/zen-images/grain-bg.png);
pointer-events: none;
z-index: 0;
opacity: var(--zen-grainy-background-opacity, 0);
mix-blend-mode: overlay;
transition: opacity 0.3s ease-in-out;
} }
} }
border-top-width: 0px; border-top-width: 0px;

View File

@@ -297,7 +297,11 @@ class nsZenViewSplitter extends nsZenDOMOperatedFeature {
} }
// Add a min width to all the browser elements to prevent them from resizing // Add a min width to all the browser elements to prevent them from resizing
const panelsWidth = gBrowser.tabbox.getBoundingClientRect().width; const panelsWidth = gBrowser.tabbox.getBoundingClientRect().width;
const halfWidth = panelsWidth / 2; let numOfTabsToDivide = 2;
if (currentView) {
numOfTabsToDivide = currentView.tabs.length + 1;
}
const halfWidth = panelsWidth / numOfTabsToDivide;
let threshold = let threshold =
gNavToolbox.getBoundingClientRect().width * gNavToolbox.getBoundingClientRect().width *
(gZenVerticalTabsManager._prefsRightSide ? 0 : 1); (gZenVerticalTabsManager._prefsRightSide ? 0 : 1);
@@ -397,7 +401,12 @@ class nsZenViewSplitter extends nsZenDOMOperatedFeature {
return; return;
} }
const panelsWidth = panelsRect.width; const panelsWidth = panelsRect.width;
const halfWidth = panelsWidth / 2; let numOfTabsToDivide = 2;
const currentView = this._data[this._lastOpenedTab.splitViewValue];
if (currentView) {
numOfTabsToDivide = currentView.tabs.length + 1;
}
const halfWidth = panelsWidth / numOfTabsToDivide;
const padding = ZenThemeModifier.elementSeparation; const padding = ZenThemeModifier.elementSeparation;
if (!this.fakeBrowser) { if (!this.fakeBrowser) {
return; return;
@@ -1180,6 +1189,7 @@ class nsZenViewSplitter extends nsZenDOMOperatedFeature {
this.currentView = -1; this.currentView = -1;
this.toggleWrapperDisplay(false); this.toggleWrapperDisplay(false);
this.maybeDisableOpeningTabOnSplitView(); this.maybeDisableOpeningTabOnSplitView();
window.dispatchEvent(new CustomEvent('ZenViewSplitter:SplitViewDeactivated'));
} }
/** /**
@@ -1754,24 +1764,12 @@ class nsZenViewSplitter extends nsZenDOMOperatedFeature {
} else { } else {
// Create new split view with layout based on drop position // Create new split view with layout based on drop position
let gridType = 'vsep'; let gridType = 'vsep';
//switch (hoverSide) {
// case 'left':
// case 'right':
// gridType = 'vsep';
// break;
// case 'top':
// case 'bottom':
// gridType = 'hsep';
// break;
// default:
// gridType = 'grid';
//}
// Put tabs always as if it was dropped from the left // Put tabs always as if it was dropped from the left
this.splitTabs( this.splitTabs(
dropSide == 'left' ? [draggedTab, droppedOnTab] : [droppedOnTab, draggedTab], dropSide == 'left' ? [draggedTab, droppedOnTab] : [droppedOnTab, draggedTab],
gridType, gridType,
1 dropSide == 'left' ? 0 : 1
); );
} }
} }

View File

@@ -5,6 +5,7 @@
import { XPCOMUtils } from 'resource://gre/modules/XPCOMUtils.sys.mjs'; import { XPCOMUtils } from 'resource://gre/modules/XPCOMUtils.sys.mjs';
import { UrlbarProvider, UrlbarUtils } from 'resource:///modules/UrlbarUtils.sys.mjs'; import { UrlbarProvider, UrlbarUtils } from 'resource:///modules/UrlbarUtils.sys.mjs';
import { globalActions } from 'resource:///modules/ZenUBGlobalActions.sys.mjs'; import { globalActions } from 'resource:///modules/ZenUBGlobalActions.sys.mjs';
import { zenUrlbarResultsLearner } from './ZenUBResultsLearner.sys.mjs';
const lazy = {}; const lazy = {};
@@ -12,9 +13,9 @@ const DYNAMIC_TYPE_NAME = 'zen-actions';
// The suggestion index of the actions row within the urlbar results. // The suggestion index of the actions row within the urlbar results.
const MAX_RECENT_ACTIONS = 5; const MAX_RECENT_ACTIONS = 5;
const MINIMUM_QUERY_SCORE = 92;
const EN_LOCALE_MATCH = /^en(-.*)$/; const MINIMUM_QUERY_SCORE = 92;
const MINIMUM_PREFIXED_QUERY_SCORE = 50;
ChromeUtils.defineESModuleGetters(lazy, { ChromeUtils.defineESModuleGetters(lazy, {
UrlbarResult: 'resource:///modules/UrlbarResult.sys.mjs', UrlbarResult: 'resource:///modules/UrlbarResult.sys.mjs',
@@ -22,6 +23,7 @@ ChromeUtils.defineESModuleGetters(lazy, {
QueryScorer: 'resource:///modules/UrlbarProviderInterventions.sys.mjs', QueryScorer: 'resource:///modules/UrlbarProviderInterventions.sys.mjs',
BrowserWindowTracker: 'resource:///modules/BrowserWindowTracker.sys.mjs', BrowserWindowTracker: 'resource:///modules/BrowserWindowTracker.sys.mjs',
AddonManager: 'resource://gre/modules/AddonManager.sys.mjs', AddonManager: 'resource://gre/modules/AddonManager.sys.mjs',
zenUrlbarResultsLearner: 'resource:///modules/ZenUBResultsLearner.sys.mjs',
}); });
XPCOMUtils.defineLazyPreferenceGetter( XPCOMUtils.defineLazyPreferenceGetter(
@@ -35,6 +37,8 @@ XPCOMUtils.defineLazyPreferenceGetter(
* A provider that lets the user view all available global actions for a query. * A provider that lets the user view all available global actions for a query.
*/ */
export class ZenUrlbarProviderGlobalActions extends UrlbarProvider { export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
#seenCommands = new Set();
constructor() { constructor() {
super(); super();
lazy.UrlbarResult.addDynamicResultType(DYNAMIC_TYPE_NAME); lazy.UrlbarResult.addDynamicResultType(DYNAMIC_TYPE_NAME);
@@ -60,12 +64,12 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
*/ */
async isActive(queryContext) { async isActive(queryContext) {
return ( return (
lazy.enabledPref && queryContext.searchMode?.source == UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS ||
(lazy.enabledPref &&
queryContext.searchString && queryContext.searchString &&
queryContext.searchString.length < UrlbarUtils.MAX_TEXT_LENGTH && queryContext.searchString.length < UrlbarUtils.MAX_TEXT_LENGTH &&
queryContext.searchString.length > 2 && queryContext.searchString.length > 2 &&
!lazy.UrlbarTokenizer.REGEXP_LIKE_PROTOCOL.test(queryContext.searchString) && !lazy.UrlbarTokenizer.REGEXP_LIKE_PROTOCOL.test(queryContext.searchString))
EN_LOCALE_MATCH.test(Services.locale.appLocaleAsBCP47)
); );
} }
@@ -92,6 +96,7 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
prettyIcon: workspace.icon, prettyIcon: workspace.icon,
accentColor, accentColor,
}, },
commandId: `zen:workspace-${workspace.uuid}`,
icon: 'chrome://browser/skin/zen-icons/forward.svg', icon: 'chrome://browser/skin/zen-icons/forward.svg',
}); });
} }
@@ -116,6 +121,7 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
return { return {
icon: 'chrome://browser/skin/zen-icons/extension.svg', icon: 'chrome://browser/skin/zen-icons/extension.svg',
label: 'Extension', label: 'Extension',
commandId: `zen:extension-${addon.id}`,
extraPayload: { extraPayload: {
extensionId: addon.id, extensionId: addon.id,
prettyName: addon.name, prettyName: addon.name,
@@ -142,14 +148,18 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
* @param {string} query The user's search query. * @param {string} query The user's search query.
* *
*/ */
async #findMatchingActions(query) { async #findMatchingActions(query, isPrefixed) {
const window = lazy.BrowserWindowTracker.getTopWindow(); const window = lazy.BrowserWindowTracker.getTopWindow();
const actions = await this.#getAvailableActions(window); const actions = await this.#getAvailableActions(window);
let results = []; let results = [];
for (let action of actions) { for (let action of actions) {
if (isPrefixed && query.length < 1) {
results.push({ action, score: 100 });
continue;
}
const label = action.extraPayload?.prettyName || action.label; const label = action.extraPayload?.prettyName || action.label;
const score = this.#calculateFuzzyScore(label, query); const score = this.#calculateFuzzyScore(label, query);
if (score > MINIMUM_QUERY_SCORE) { if (score > (isPrefixed ? MINIMUM_PREFIXED_QUERY_SCORE : MINIMUM_QUERY_SCORE)) {
results.push({ results.push({
score, score,
action, action,
@@ -157,6 +167,10 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
} }
} }
results.sort((a, b) => b.score - a.score); results.sort((a, b) => b.score - a.score);
// We must show all we can when prefixed, to avoid showing no results.
if (isPrefixed) {
return results.map((r) => r.action);
}
return results.slice(0, MAX_RECENT_ACTIONS).map((r) => r.action); return results.slice(0, MAX_RECENT_ACTIONS).map((r) => r.action);
} }
@@ -217,16 +231,18 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
async startQuery(queryContext, addCallback) { async startQuery(queryContext, addCallback) {
const query = queryContext.trimmedLowerCaseSearchString; const query = queryContext.trimmedLowerCaseSearchString;
if (!query) { const isPrefixed = queryContext.searchMode?.source == UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS;
if (!query && !isPrefixed) {
return; return;
} }
const actionsResults = await this.#findMatchingActions(query); const actionsResults = await this.#findMatchingActions(query, isPrefixed);
if (!actionsResults.length) { if (!actionsResults.length) {
return; return;
} }
const ownerGlobal = lazy.BrowserWindowTracker.getTopWindow(); const ownerGlobal = lazy.BrowserWindowTracker.getTopWindow();
const finalResults = [];
for (const action of actionsResults) { for (const action of actionsResults) {
const [payload, payloadHighlights] = lazy.UrlbarResult.payloadAndSimpleHighlights([], { const [payload, payloadHighlights] = lazy.UrlbarResult.payloadAndSimpleHighlights([], {
suggestion: action.label, suggestion: action.label,
@@ -235,7 +251,7 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
zenCommand: action.command, zenCommand: action.command,
dynamicType: DYNAMIC_TYPE_NAME, dynamicType: DYNAMIC_TYPE_NAME,
zenAction: true, zenAction: true,
icon: action.icon || 'chrome://browser/skin/trending.svg', icon: action.icon,
shortcutContent: ownerGlobal.gZenKeyboardShortcutsManager.getShortcutDisplayFromCommand( shortcutContent: ownerGlobal.gZenKeyboardShortcutsManager.getShortcutDisplayFromCommand(
action.command action.command
), ),
@@ -245,15 +261,26 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
let result = new lazy.UrlbarResult( let result = new lazy.UrlbarResult(
UrlbarUtils.RESULT_TYPE.DYNAMIC, UrlbarUtils.RESULT_TYPE.DYNAMIC,
UrlbarUtils.RESULT_SOURCE.OTHER_LOCAL, UrlbarUtils.RESULT_SOURCE.ZEN_ACTIONS,
payload, payload,
payloadHighlights payloadHighlights
); );
if (typeof action.suggestedIndex === 'number') { if (zenUrlbarResultsLearner.shouldPrioritize(action.commandId)) {
result.suggestedIndex = action.suggestedIndex; result.heuristic = true;
} else {
result.suggestedIndex = zenUrlbarResultsLearner.getDeprioritizeIndex(action.commandId);
} }
result.commandId = action.commandId;
if (!(isPrefixed && query.length < 2)) {
// We dont want to record prefixed results, as the user explicitly asked for them.
// Selecting other results would de-prioritize these actions unfairly.
this.#seenCommands.add(action.commandId);
}
finalResults.push(result);
}
zenUrlbarResultsLearner.sortCommandsByPriority(finalResults).forEach((result) => {
addCallback(this, result); addCallback(this, result);
} });
} }
/** /**
@@ -362,6 +389,19 @@ export class ZenUrlbarProviderGlobalActions extends UrlbarProvider {
}; };
} }
onSearchSessionEnd(_queryContext, _controller, details) {
// We should only record the execution if a result was actually used.
// Otherwise we would start de-prioritizing commands that were never used.
if (details?.result) {
let usedCommand = null;
if (details?.provider === this.name) {
usedCommand = details.result?.commandId;
}
zenUrlbarResultsLearner.recordExecution(usedCommand, [...this.#seenCommands]);
}
this.#seenCommands = new Set();
}
onEngagement(queryContext, controller, details) { onEngagement(queryContext, controller, details) {
const result = details.result; const result = details.result;
const payload = result.payload; const payload = result.payload;

View File

@@ -11,7 +11,6 @@ const globalActionsTemplate = [
label: 'Toggle Compact Mode', label: 'Toggle Compact Mode',
command: 'cmd_zenCompactModeToggle', command: 'cmd_zenCompactModeToggle',
icon: 'chrome://browser/skin/zen-icons/sidebar.svg', icon: 'chrome://browser/skin/zen-icons/sidebar.svg',
suggestedIndex: 0,
}, },
{ {
label: 'Open Theme Picker', label: 'Open Theme Picker',
@@ -22,7 +21,6 @@ const globalActionsTemplate = [
label: 'New Split View', label: 'New Split View',
command: 'cmd_zenNewEmptySplit', command: 'cmd_zenNewEmptySplit',
icon: 'chrome://browser/skin/zen-icons/split.svg', icon: 'chrome://browser/skin/zen-icons/split.svg',
suggestedIndex: 0,
}, },
{ {
label: 'New Folder', label: 'New Folder',
@@ -33,7 +31,6 @@ const globalActionsTemplate = [
label: 'Copy Current URL', label: 'Copy Current URL',
command: 'cmd_zenCopyCurrentURL', command: 'cmd_zenCopyCurrentURL',
icon: 'chrome://browser/skin/zen-icons/edit-copy.svg', icon: 'chrome://browser/skin/zen-icons/edit-copy.svg',
suggestedIndex: 0,
}, },
{ {
label: 'Settings', label: 'Settings',
@@ -89,7 +86,6 @@ const globalActionsTemplate = [
label: 'Close Tab', label: 'Close Tab',
command: 'cmd_close', command: 'cmd_close',
icon: 'chrome://browser/skin/zen-icons/close.svg', icon: 'chrome://browser/skin/zen-icons/close.svg',
suggestedIndex: 1,
isAvailable: (window) => { isAvailable: (window) => {
return isNotEmptyTab(window); return isNotEmptyTab(window);
}, },
@@ -121,13 +117,11 @@ const globalActionsTemplate = [
isAvailable: (window) => { isAvailable: (window) => {
return isNotEmptyTab(window); return isNotEmptyTab(window);
}, },
suggestedIndex: 1,
}, },
{ {
label: 'Toggle Tabs on right', label: 'Toggle Tabs on right',
command: 'cmd_zenToggleTabsOnRight', command: 'cmd_zenToggleTabsOnRight',
icon: 'chrome://browser/skin/zen-icons/sidebars-right.svg', icon: 'chrome://browser/skin/zen-icons/sidebars-right.svg',
suggestedIndex: 1,
}, },
{ {
label: 'Add to Essentials', label: 'Add to Essentials',
@@ -139,14 +133,12 @@ const globalActionsTemplate = [
); );
}, },
icon: 'chrome://browser/skin/zen-icons/essential-add.svg', icon: 'chrome://browser/skin/zen-icons/essential-add.svg',
suggestedIndex: 1,
}, },
{ {
label: 'Remove from Essentials', label: 'Remove from Essentials',
command: (window) => window.gZenPinnedTabManager.removeEssentials(window.gBrowser.selectedTab), command: (window) => window.gZenPinnedTabManager.removeEssentials(window.gBrowser.selectedTab),
isAvailable: (window) => window.gBrowser.selectedTab.hasAttribute('zen-essential'), isAvailable: (window) => window.gBrowser.selectedTab.hasAttribute('zen-essential'),
icon: 'chrome://browser/skin/zen-icons/essential-remove.svg', icon: 'chrome://browser/skin/zen-icons/essential-remove.svg',
suggestedIndex: 1,
}, },
{ {
label: 'Find in Page', label: 'Find in Page',
@@ -155,13 +147,11 @@ const globalActionsTemplate = [
isAvailable: (window) => { isAvailable: (window) => {
return isNotEmptyTab(window); return isNotEmptyTab(window);
}, },
suggestedIndex: 1,
}, },
{ {
label: 'Manage Extensions', label: 'Manage Extensions',
command: 'Tools:Addons', command: 'Tools:Addons',
icon: 'chrome://browser/skin/zen-icons/extension.svg', icon: 'chrome://browser/skin/zen-icons/extension.svg',
suggestedIndex: 1,
}, },
]; ];
@@ -169,6 +159,10 @@ export const globalActions = globalActionsTemplate.map((action) => ({
isAvailable: (window) => { isAvailable: (window) => {
return window.document.getElementById(action.command)?.getAttribute('disabled') !== 'true'; return window.document.getElementById(action.command)?.getAttribute('disabled') !== 'true';
}, },
commandId:
typeof action.command === 'string'
? action.command
: `zen:global-action-${action.label.toLowerCase().replace(/\s+/g, '-')}`,
extraPayload: {}, extraPayload: {},
...action, ...action,
})); }));

View File

@@ -0,0 +1,108 @@
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
import { XPCOMUtils } from 'resource://gre/modules/XPCOMUtils.sys.mjs';
const lazy = {};
const DEFAULT_DB_DATA = '{}';
const DEPRIORITIZE_MAX = -4;
const PRIORITIZE_MAX = 5;
XPCOMUtils.defineLazyPreferenceGetter(
lazy,
'database',
'zen.urlbar.suggestions-learner',
'DEFAULT_DB_DATA'
);
/**
* A class that manages the learning of URL bar results for commands,
* can be used for any ID that can be executed in the URL bar.
*
* The schema would be something like:
* {
* "<command id>": <priority number>,
* }
*
* If the current command is not on the list is because the user
* has *seen* the command but never executed it. If the number is
* less than -2, that means the user will most likely never use it.
*
* The priority number is incremented each time the command is executed.
*/
class ZenUrlbarResultsLearner {
constructor() {}
get database() {
try {
return JSON.parse(lazy.database);
} catch {
return {};
}
}
saveDatabase(db) {
Services.prefs.setStringPref(
'zen.urlbar.suggestions-learner',
JSON.stringify(db || DEFAULT_DB_DATA)
);
}
recordExecution(commandId, seenCommands = []) {
const db = this.database;
if (commandId) {
const numberOfUsages = Math.min((db[commandId] || 0) + 1, PRIORITIZE_MAX);
db[commandId] = numberOfUsages;
}
for (const cmd of seenCommands) {
if (cmd !== commandId) {
if (!db[cmd]) {
db[cmd] = -1;
} else {
const newIndex = Math.max(db[cmd] - 1, DEPRIORITIZE_MAX);
db[cmd] = newIndex;
if (newIndex === 0) {
// Save some space by deleting commands that are not used
// and have a neutral score.
delete db[cmd];
}
}
}
}
this.saveDatabase(db);
}
shouldPrioritize(commandId) {
if (!commandId) {
return false;
}
const db = this.database;
return !!db[commandId] && db[commandId] > 0;
}
getDeprioritizeIndex(commandId) {
if (!commandId) {
return 1;
}
const db = this.database;
if (db[commandId] < 0) {
return Math.abs(db[commandId]);
}
// This will most likely never run, since
// positive commands are prioritized.
return 1;
}
/**
* Sorts the given commands by their priority in the database.
* @param {*} commands
*/
sortCommandsByPriority(commands) {
const db = this.database;
return commands.sort((a, b) => (db[b.commandId] || 0) - (db[a.commandId] || 0));
}
}
export const zenUrlbarResultsLearner = new ZenUrlbarResultsLearner();

View File

@@ -6,4 +6,5 @@ EXTRA_JS_MODULES += [
"ZenUBActionsProvider.sys.mjs", "ZenUBActionsProvider.sys.mjs",
"ZenUBGlobalActions.sys.mjs", "ZenUBGlobalActions.sys.mjs",
"ZenUBProvider.sys.mjs", "ZenUBProvider.sys.mjs",
"ZenUBResultsLearner.sys.mjs",
] ]

View File

@@ -19,7 +19,7 @@
"brandShortName": "Zen", "brandShortName": "Zen",
"brandFullName": "Zen Browser", "brandFullName": "Zen Browser",
"release": { "release": {
"displayVersion": "1.16.1b", "displayVersion": "1.16.2b",
"github": { "github": {
"repo": "zen-browser/desktop" "repo": "zen-browser/desktop"
}, },