diff --git a/edit.html b/edit.html index b162045a..8907cd5b 100644 --- a/edit.html +++ b/edit.html @@ -271,13 +271,16 @@
-
- +
+
+ +
- -
- -
diff --git a/edit/edit.css b/edit/edit.css index 0553f5bf..0ad6eec7 100644 --- a/edit/edit.css +++ b/edit/edit.css @@ -260,23 +260,13 @@ input:invalid { margin-top: .5rem; } -#actions > * { - display: inline-flex; - flex-wrap: wrap; -} - -#mozilla-format-buttons { +#actions .buttons { display: flex; flex-wrap: wrap; align-items: center; } -#actions > div > a { - height: min-content; -} - -#actions button, -#actions > div > a { +#actions .buttons > * { margin: 0 .2rem .5rem 0; } @@ -1065,11 +1055,7 @@ body.linter-disabled .hidden-unless-compact { padding: .25rem 0 .5rem; } #actions { - display: flex; - flex-wrap: wrap; white-space: nowrap; - margin: 0; - box-sizing: border-box; } #header input[type="checkbox"] { vertical-align: middle; diff --git a/edit/source-editor.js b/edit/source-editor.js index b9ce6c66..fb118b1c 100644 --- a/edit/source-editor.js +++ b/edit/source-editor.js @@ -23,7 +23,7 @@ function SourceEditor() { $('#header').on('wheel', headerOnScroll); $('#sections').textContent = ''; $('#sections').appendChild($create('.single-editor')); - $('[i18n-text="saveAsTemplate"]').onclick = saveTemplate; + $('#save-button').onauxclick = e => e.detail === 'tpl' && saveTemplate(); const cm = cmFactory.create($('.single-editor')); const sectionFinder = MozSectionFinder(cm); diff --git a/global.css b/global.css index 95440557..0f094b85 100644 --- a/global.css +++ b/global.css @@ -6,6 +6,7 @@ html#stylus #header *:not(#\1transition-suppressor) { } :root { --family: Arial, "Helvetica Neue", Helvetica, system-ui, sans-serif; + --bg: #fff; } body { font: normal 12px var(--family); @@ -33,6 +34,7 @@ button { border: 1px solid hsl(0, 0%, 62%); font: inherit; font-size: 13px; + line-height: 1.2; color: #000; background-color: hsl(0, 0%, 100%); background-image: url(''); @@ -307,6 +309,39 @@ body.resizing-v > * { } /* header resizer - end */ +.split-btn { + position: relative; +} +.split-btn-pedal { + margin-left: -1px !important; + padding-left: .2em !important; + padding-right: .2em !important; +} +.split-btn-pedal::after { + content: '\25BC'; /* down triangle */ + font-size: 90%; +} +.split-btn-pedal.active { + box-shadow: inset 0 0 100px rgba(0, 0, 0, .2); +} +.split-btn-menu { + background: #fff; + position: absolute; + box-shadow: 2px 3px 7px rgba(0, 0, 0, .5); + border: 1px solid hsl(180deg, 50%, 50%); + white-space: nowrap; + cursor: pointer; + padding: .25em 0; +} +.split-btn-menu > * { + padding: .5em 1em; + display: block; +} +.split-btn-menu > :hover { + background-color: hsla(180deg, 50%, 50%, .25); + color: #000; +} + @supports (-moz-appearance: none) { .moz-appearance-bug .svg-icon.checked, .moz-appearance-bug .onoffswitch input, diff --git a/js/dom-on-load.js b/js/dom-on-load.js index 828d14c8..634ae5dd 100644 --- a/js/dom-on-load.js +++ b/js/dom-on-load.js @@ -1,4 +1,4 @@ -/* global $ $$ focusAccessibility getEventKeyName */// dom.js +/* global $$ $ $create focusAccessibility getEventKeyName moveFocus */// dom.js /* global debounce */// toolbox.js /* global t */// localization.js 'use strict'; @@ -6,13 +6,14 @@ /** DOM housekeeping after a page finished loading */ (() => { + const SPLIT_BTN_MENU = '.split-btn-menu'; splitLongTooltips(); addTooltipsToEllipsized(); window.on('mousedown', suppressFocusRingOnClick, {passive: true}); window.on('keydown', keepFocusRingOnTabbing, {passive: true}); window.on('keypress', clickDummyLinkOnEnter); window.on('wheel', changeFocusedInputOnWheel, {capture: true, passive: false}); - window.on('click', showTooltipNote); + window.on('click', e => splitMenu(e) || showTooltipNote(e)); window.on('resize', () => debounce(addTooltipsToEllipsized, 100)); // Removing transition-suppressor rule const {sheet} = $('link[href$="global.css"]'); @@ -78,19 +79,44 @@ let el = document.activeElement; if (el) { el = el.closest('[data-focused-via-click]'); - if (el) delete el.dataset.focusedViaClick; + focusAccessibility.toggle(el, false); } }); } } + function splitMenu(event) { + const prevMenu = $(SPLIT_BTN_MENU); + const prevPedal = (prevMenu || {}).previousElementSibling; + const pedal = event.target.closest('.split-btn-pedal'); + const entry = prevMenu && event.target.closest(SPLIT_BTN_MENU + '>*'); + if (prevMenu) prevMenu.remove(); + if (prevPedal) prevPedal.classList.remove('active'); + if (pedal && pedal !== prevPedal) { + const menu = $create(SPLIT_BTN_MENU, + Array.from(pedal.attributes, ({name, value}) => + name.startsWith('menu-') && + $create('a', {tabIndex: 0, __cmd: name.split('-').pop()}, value) + )); + menu.on('focusout', e => e.target === menu && splitMenu(e)); + pedal.classList.toggle('active'); + pedal.after(menu); + moveFocus(menu, 0); + focusAccessibility.toggle(menu.firstChild, focusAccessibility.get(pedal)); + } + if (entry) { + prevPedal.previousElementSibling.dispatchEvent(new CustomEvent('auxclick', { + detail: entry.__cmd, + bubbles: true, + })); + } + } + function suppressFocusRingOnClick({target}) { const el = focusAccessibility.closest(target); if (el) { focusAccessibility.lastFocusedViaClick = true; - if (el.dataset.focusedViaClick === undefined) { - el.dataset.focusedViaClick = ''; - } + focusAccessibility.toggle(el, true); } } diff --git a/js/dom.js b/js/dom.js index 5cb50aeb..a650a387 100644 --- a/js/dom.js +++ b/js/dom.js @@ -31,6 +31,8 @@ Object.assign(EventTarget.prototype, { const focusAccessibility = { // last event's focusedViaClick lastFocusedViaClick: false, + get: el => el && el.dataset.focusedViaClick != null, + toggle: (el, state) => el && toggleDataset(el, 'focusedViaClick', state), // to avoid a full layout recalc due to changes on body/root // we modify the closest focusable element (like input or button or anything with tabindex=0) closest(el) {