/* global dirtyReporter showHelp toggleContextMenuDelete createSection CodeMirror linter createLivePreview showCodeMirrorPopup sectionsToMozFormat messageBox clipString rerouteHotkeys $ $$ $create t FIREFOX API debounce */ /* exported createSectionsEditor */ 'use strict'; function createSectionsEditor(style) { let INC_ID = 0; // an increment id that is used by various object to track the order const dirty = dirtyReporter(); dirty.onChange(updateTitle); const container = $('#sections'); const sections = []; const nameEl = $('#name'); nameEl.addEventListener('input', () => { dirty.modify('name', style.name, nameEl.value); style.name = nameEl.value; updateTitle(); }); const enabledEl = $('#enabled'); enabledEl.addEventListener('change', () => { dirty.modify('enabled', style.enabled, enabledEl.checked); style.enabled = enabledEl.checked; updateLivePreview(); }); $('#to-mozilla').addEventListener('click', showMozillaFormat); $('#to-mozilla-help').addEventListener('click', showToMozillaHelp); $('#from-mozilla').addEventListener('click', () => showMozillaFormatImport()); $('#save-button').addEventListener('click', saveStyle); document.addEventListener('wheel', scrollEntirePageOnCtrlShift); if (!FIREFOX) { $$([ 'input:not([type])', 'input[type="text"]', 'input[type="search"]', 'input[type="number"]', ].join(',')) .forEach(e => e.addEventListener('mousedown', toggleContextMenuDelete)); } let sectionOrder = ''; const initializing = new Promise(resolve => initSection({ sections: style.sections.slice(), done:() => { // FIXME: implement this with CSS? // https://github.com/openstyles/stylus/commit/2895ce11e271788df0e4f7314b3b981fde086574 dirty.clear(); rerouteHotkeys(true); resolve(); updateHeader(); } })); const livePreview = createLivePreview(); livePreview.show(Boolean(style.id)); return { ready: () => initializing, replaceStyle, isDirty: dirty.isDirty, getStyle: () => style, getEditors, scrollToEditor, getStyleId: () => style.id, getEditorTitle: cm => { const index = sections.filter(s => !s.isRemoved()).findIndex(s => s.cm === cm); return `${t('sectionCode')} ${index + 1}`; }, save: saveStyle, toggleStyle, nextEditor, prevEditor, closestVisible, getSearchableInputs, }; function genId() { return INC_ID++; } function setGlobalProgress(done, total) { const progressElement = $('#global-progress') || total && document.body.appendChild($create('#global-progress')); if (total) { const progress = (done / Math.max(done, total) * 100).toFixed(1); progressElement.style.borderLeftWidth = progress + 'vw'; setTimeout(() => { progressElement.title = progress + '%'; }); } else { $.remove(progressElement); } } function showToMozillaHelp(event) { event.preventDefault(); showHelp(t('styleMozillaFormatHeading'), t('styleToMozillaFormatHelp')); } function getSearchableInputs(cm) { return sections.find(s => s.cm === cm).appliesTo.map(a => a.valueEl).filter(Boolean); } // priority: // 1. associated CM for applies-to element // 2. last active if visible // 3. first visible function closestVisible(nearbyElement) { const cm = nearbyElement instanceof CodeMirror ? nearbyElement : nearbyElement instanceof Node && (nearbyElement.closest('#sections > .section') || {}).CodeMirror || getLastActivatedEditor(); if (nearbyElement instanceof Node && cm) { const {left, top} = nearbyElement.getBoundingClientRect(); const bounds = cm.display.wrapper.getBoundingClientRect(); if (top >= 0 && top >= bounds.top && left >= 0 && left >= bounds.left) { return cm; } } // closest editor should have at least 2 lines visible const lineHeight = sections[0].cm.defaultTextHeight(); const scrollY = window.scrollY; const windowBottom = scrollY + window.innerHeight - 2 * lineHeight; const allSectionsContainerTop = scrollY + $('#sections').getBoundingClientRect().top; const distances = []; const alreadyInView = cm && offscreenDistance(null, cm) === 0; return alreadyInView ? cm : findClosest(); function offscreenDistance(index, cm) { if (index >= 0 && distances[index] !== undefined) { return distances[index]; } const section = cm.display.wrapper.closest('.section'); if (!section) { return 1e9; } const top = allSectionsContainerTop + section.offsetTop; if (top < scrollY + lineHeight) { return Math.max(0, scrollY - top - lineHeight); } if (top < windowBottom) { return 0; } const distance = top - windowBottom + section.offsetHeight; if (index >= 0) { distances[index] = distance; } return distance; } function findClosest() { const editors = getEditors(); const last = editors.length - 1; let a = 0; let b = last; let c; let distance; while (a < b - 1) { c = (a + b) / 2 | 0; distance = offscreenDistance(c); if (!distance || !c) { break; } const distancePrev = offscreenDistance(c - 1); const distanceNext = c < last ? offscreenDistance(c + 1) : 1e20; if (distancePrev <= distance && distance <= distanceNext) { b = c; } else { a = c; } } while (b && offscreenDistance(b - 1) <= offscreenDistance(b)) { b--; } const cm = editors[b]; if (distances[b] > 0) { scrollToEditor(cm); } return cm; } } function getEditors() { return sections.filter(s => !s.isRemoved()).map(s => s.cm); } function toggleStyle() { const newValue = !style.enabled; dirty.modify('enabled', style.enabled, newValue); style.enabled = newValue; enabledEl.checked = newValue; } function nextEditor(cm, cycle = true) { if (!cycle && findLast(sections, s => !s.isRemoved()).cm === cm) { return; } return nextPrevEditor(cm, 1); } function prevEditor(cm, cycle = true) { if (!cycle && sections.find(s => !s.isRemoved()).cm === cm) { return; } return nextPrevEditor(cm, -1); } function findLast(arr, match) { for (let i = arr.length - 1; i >= 0; i--) { if (match(arr[i])) { return arr[i]; } } } function nextPrevEditor(cm, direction) { const editors = getEditors(); cm = editors[(editors.indexOf(cm) + direction + editors.length) % editors.length]; scrollToEditor(cm); cm.focus(); return cm; } function scrollToEditor(cm) { const section = sections.find(s => s.cm === cm).el; const bounds = section.getBoundingClientRect(); if ( (bounds.bottom > window.innerHeight && bounds.top > 0) || (bounds.top < 0 && bounds.bottom < window.innerHeight) ) { if (bounds.top < 0) { window.scrollBy(0, bounds.top - 1); } else { window.scrollBy(0, bounds.bottom - window.innerHeight + 1); } } } function getLastActivatedEditor() { let result; for (const section of sections) { if (section.isRemoved()) { continue; } // .lastActive is initiated by codemirror-factory if (!result || section.cm.lastActive > result.lastActive) { result = section.cm; } } return result; } function scrollEntirePageOnCtrlShift(event) { // make Shift-Ctrl-Wheel scroll entire page even when mouse is over a code editor if (event.shiftKey && event.ctrlKey && !event.altKey && !event.metaKey) { // Chrome scrolls horizontally when Shift is pressed but on some PCs this might be different window.scrollBy(0, event.deltaX || event.deltaY); event.preventDefault(); } } function showMozillaFormat() { const popup = showCodeMirrorPopup(t('styleToMozillaFormatTitle'), '', {readOnly: true}); popup.codebox.setValue(sectionsToMozFormat(getModel())); popup.codebox.execCommand('selectAll'); } function showMozillaFormatImport(text = '') { const popup = showCodeMirrorPopup(t('styleFromMozillaFormatPrompt'), $create('.buttons', [ $create('button', { name: 'import-replace', textContent: t('importReplaceLabel'), title: 'Ctrl-Shift-Enter:\n' + t('importReplaceTooltip'), onclick: () => doImport({replaceOldStyle: true}), }), $create('button', { name: 'import-append', textContent: t('importAppendLabel'), title: 'Ctrl-Enter:\n' + t('importAppendTooltip'), onclick: doImport, }), ])); const contents = $('.contents', popup); contents.insertBefore(popup.codebox.display.wrapper, contents.firstElementChild); popup.codebox.focus(); popup.codebox.on('changes', cm => { popup.classList.toggle('ready', !cm.isBlank()); cm.markClean(); }); if (text) { popup.codebox.setValue(text); popup.codebox.clearHistory(); popup.codebox.markClean(); } // overwrite default extraKeys as those are inapplicable in popup context popup.codebox.options.extraKeys = { 'Ctrl-Enter': doImport, 'Shift-Ctrl-Enter': () => doImport({replaceOldStyle: true}), }; function doImport({replaceOldStyle = false}) { lockPageUI(true); API.parseCss({code: popup.codebox.getValue().trim()}) .then(({sections, errors}) => { // shouldn't happen but just in case if (!sections.length || errors.length) { throw errors; } if (replaceOldStyle) { return replaceSections(sections); } return new Promise(resolve => initSection({sections, done: resolve, focusOn: false})); }) .then(() => { $('.dismiss').dispatchEvent(new Event('click')); }) .catch(showError) .then(() => lockPageUI(false)); } function lockPageUI(locked) { document.documentElement.style.pointerEvents = locked ? 'none' : ''; if (popup.codebox) { popup.classList.toggle('ready', locked ? false : !popup.codebox.isBlank()); popup.codebox.options.readOnly = locked; popup.codebox.display.wrapper.style.opacity = locked ? '.5' : ''; } } function showError(errors) { messageBox({ className: 'center danger', title: t('styleFromMozillaFormatError'), contents: $create('pre', Array.isArray(errors) ? errors.join('\n') : errors), buttons: [t('confirmClose')], }); } } function updateSectionOrder() { const oldOrder = sectionOrder; const validSections = sections.filter(s => !s.isRemoved()); sectionOrder = validSections.map(s => s.id).join(','); dirty.modify('sectionOrder', oldOrder, sectionOrder); container.dataset.sectionCount = validSections.length; linter.refreshReport(); } function getModel() { return Object.assign({}, style, { sections: sections.filter(s => !s.isRemoved()).map(s => s.getModel()) }); } function validate() { if (!nameEl.reportValidity()) { messageBox.alert(t('styleMissingName')); return false; } for (const section of sections) { for (const apply of section.appliesTo) { if (apply.getType() !== 'regexp') { continue; } if (!apply.valueEl.reportValidity()) { messageBox.alert(t('styleBadRegexp')); return false; } } } return true; } function saveStyle() { if (!dirty.isDirty()) { return; } const newStyle = getModel(); if (!validate(newStyle)) { return; } API.editSave(newStyle) .then(newStyle => { destroyRemovedSections(); sessionStorage.justEditedStyleId = newStyle.id; replaceStyle(newStyle, false); }); } function destroyRemovedSections() { for (let i = 0; i < sections.length;) { if (!sections[i].isRemoved()) { i++; continue; } sections[i].destroy(); sections[i].el.remove(); sections.splice(i, 1); } } function updateHeader() { nameEl.value = style.name || ''; enabledEl.checked = style.enabled !== false; $('#url').href = style.url || ''; updateTitle(); } function updateLivePreview() { debounce(_updateLivePreview, 200); } function _updateLivePreview() { livePreview.update(getModel()); } function updateTitle() { const name = style.name; const clean = !dirty.isDirty(); const title = !style.id ? t('addStyleTitle') : name; document.title = (clean ? '' : '* ') + title; $('#save-button').disabled = clean; } function initSection({ sections: originalSections, total = originalSections.length, focusOn = 0, done }) { container.classList.add('hidden'); chunk(); function chunk() { if (!originalSections.length) { setGlobalProgress(); if (focusOn !== false) { sections[focusOn].cm.focus(); } container.classList.remove('hidden'); for (const section of sections) { section.cm.refreshOnView(); } if (done) { done(); } return; } const t0 = performance.now(); while (originalSections.length && performance.now() - t0 < 100) { insertSectionAfter(originalSections.shift()); } setGlobalProgress(total - originalSections.length, total); setTimeout(chunk); } } function removeSection(section) { if (sections.every(s => s.isRemoved() || s === section)) { // TODO: hide remove button when `#sections[data-section-count=1]` throw new Error('Cannot remove last section'); } if (section.cm.isBlank()) { const index = sections.indexOf(section); sections.splice(index, 1); section.el.remove(); section.remove(); section.destroy(); } else { const lines = []; const MAX_LINES = 10; section.cm.doc.iter(0, MAX_LINES + 1, ({text}) => lines.push(text) && false); const title = t('sectionCode') + '\n' + '-'.repeat(20) + '\n' + lines.slice(0, MAX_LINES).map(s => clipString(s, 100)).join('\n') + (lines.length > MAX_LINES ? '\n...' : ''); $('.deleted-section', section.el).title = title; section.remove(); } dirty.remove(section, section); updateSectionOrder(); section.off(updateLivePreview); updateLivePreview(); } function restoreSection(section) { section.restore(); updateSectionOrder(); section.onChange(updateLivePreview); updateLivePreview(); } function insertSectionAfter(init, base) { if (!init) { init = {code: '', urlPrefixes: ['http://example.com']}; } const section = createSection({ originalSection: init, genId, dirty, showMozillaFormatImport, removeSection, restoreSection, insertSectionAfter, moveSectionUp, moveSectionDown, prevEditor, nextEditor }); if (base) { const index = sections.indexOf(base); sections.splice(index + 1, 0, section); container.insertBefore(section.el, base.el.nextSibling); } else { sections.push(section); container.appendChild(section.el); } section.render(); updateSectionOrder(); section.onChange(updateLivePreview); updateLivePreview(); } function moveSectionUp(section) { const index = sections.indexOf(section); if (index === 0) { return; } container.insertBefore(section.el, sections[index - 1].el); sections[index] = sections[index - 1]; sections[index - 1] = section; updateSectionOrder(); } function moveSectionDown(section) { const index = sections.indexOf(section); if (index === sections.length - 1) { return; } container.insertBefore(sections[index + 1].el, section.el); sections[index] = sections[index + 1]; sections[index + 1] = section; updateSectionOrder(); } function replaceSections(originalSections) { for (const section of sections) { section.remove(true); } sections.length = 0; container.textContent = ''; return new Promise(resolve => initSection({sections: originalSections, done: resolve})); } function replaceStyle(newStyle, codeIsUpdated) { // FIXME: avoid recreating all editors? reinit().then(() => { Object.assign(style, newStyle); updateHeader(); dirty.clear(); // Go from new style URL to edit style URL if (location.href.indexOf('id=') === -1 && style.id) { history.replaceState({}, document.title, 'edit.html?id=' + style.id); $('#heading').textContent = t('editStyleHeading'); } livePreview.show(Boolean(style.id)); }); function reinit() { if (codeIsUpdated !== false) { return replaceSections(newStyle.sections.slice()); } return Promise.resolve(); } } }