diff --git a/packages/svgcanvas/svgcanvas.js b/packages/svgcanvas/svgcanvas.js index 17be102b7..5c4a63773 100644 --- a/packages/svgcanvas/svgcanvas.js +++ b/packages/svgcanvas/svgcanvas.js @@ -308,6 +308,9 @@ class SvgCanvas { this.contentW = this.getResolution().w this.contentH = this.getResolution().h this.clear() + + // creates custom modeEvent for editor + this.modeChangeEvent() } // End constructor getSvgOption () { @@ -827,6 +830,11 @@ class SvgCanvas { ? this.curText : this.curShape this.currentMode = name + + // fires modeChange event for the editor + if (this.modeEvent) { + document.dispatchEvent(this.modeEvent) + } } /** @@ -1328,6 +1336,14 @@ class SvgCanvas { this.decode64 = decode64 this.mergeDeep = mergeDeep } + + /** + * Creates modeChange event, adds it as an svgCanvas property + * **/ + modeChangeEvent () { + const modeEvent = new CustomEvent('modeChange', { detail: { getMode: () => this.getMode() } }) + this.modeEvent = modeEvent + } } // End class // attach utilities function to the class that are used by SvgEdit so diff --git a/src/editor/Editor.js b/src/editor/Editor.js index 17978aca7..e35c03345 100644 --- a/src/editor/Editor.js +++ b/src/editor/Editor.js @@ -388,8 +388,10 @@ class Editor extends EditorStartup { this.svgCanvas.randomizeIds(arg) } - /** @lends module:SVGEditor~Actions */ /** + * @lends module:SVGEditor~Actions */ + /** + * editor shortcuts init * @returns {void} */ setAll () { diff --git a/src/editor/EditorStartup.js b/src/editor/EditorStartup.js index 1ea9224b3..20b9aaa33 100644 --- a/src/editor/EditorStartup.js +++ b/src/editor/EditorStartup.js @@ -118,6 +118,10 @@ class EditorStartup { this.configObj.curConfig ) + // once svgCanvas is init - adding listener to the changes of the current mode + this.modeEvent = this.svgCanvas.modeEvent + document.addEventListener('modeChange', (evt) => this.modeListener(evt)) + this.leftPanel.init() this.bottomPanel.init() this.topPanel.init() @@ -278,6 +282,7 @@ class EditorStartup { let lastX = null; let lastY = null let panning = false; let keypan = false + let previousMode = 'select' $id('svgcanvas').addEventListener('mouseup', (evt) => { if (panning === false) { return true } @@ -305,7 +310,12 @@ class EditorStartup { }) $id('svgcanvas').addEventListener('mousedown', (evt) => { if (evt.button === 1 || keypan === true) { + // prDefault to avoid firing of browser's panning on mousewheel + evt.preventDefault() panning = true + previousMode = this.svgCanvas.getMode() + this.svgCanvas.setMode('ext-panning') + this.workarea.style.cursor = 'grab' lastX = evt.clientX lastY = evt.clientY return false @@ -313,10 +323,27 @@ class EditorStartup { return true }) - window.addEventListener('mouseup', () => { + // preventing browser's scaling with Ctrl+wheel + this.$container.addEventListener('wheel', (e) => { + if (e.ctrlKey) { + e.preventDefault() + } + }) + + window.addEventListener('mouseup', (evt) => { + if (evt.button === 1) { + this.svgCanvas.setMode(previousMode ?? 'select') + } panning = false }) + // Allows quick change to the select mode while panning mode is active + this.workarea.addEventListener('dblclick', (evt) => { + if (this.svgCanvas.getMode() === 'ext-panning') { + this.leftPanel.clickSelect() + } + }) + document.addEventListener('keydown', (e) => { if (e.target.nodeName !== 'BODY') return if (e.code.toLowerCase() === 'space') { @@ -332,6 +359,7 @@ class EditorStartup { if (e.target.nodeName !== 'BODY') return if (e.code.toLowerCase() === 'space') { this.svgCanvas.spaceKey = keypan = false + this.svgCanvas.setMode(previousMode === 'ext-panning' ? 'select' : previousMode ?? 'select') e.preventDefault() } else if ((e.key.toLowerCase() === 'shift') && (this.svgCanvas.getMode() === 'zoom')) { this.workarea.style.cursor = zoomInIcon @@ -695,6 +723,36 @@ class EditorStartup { console.error(err) } } + + /** + * Listens to the mode change, listener is to be added on document +* @param {Event} evt custom modeChange event +*/ + modeListener (evt) { + const mode = this.svgCanvas.getMode() + + this.setCursorStyle(mode) + } + + /** + * sets cursor styling for workarea depending on the current mode + * @param {string} mode + */ + setCursorStyle (mode) { + let cs = 'auto' + switch (mode) { + case 'ext-panning': + cs = 'grab' + break + case 'zoom': + cs = 'crosshair' + break + default: + cs = 'auto' + } + + this.workarea.style.cursor = cs + } } export default EditorStartup diff --git a/src/editor/extensions/ext-panning/ext-panning.js b/src/editor/extensions/ext-panning/ext-panning.js index df106630c..ce9fe264d 100644 --- a/src/editor/extensions/ext-panning/ext-panning.js +++ b/src/editor/extensions/ext-panning/ext-panning.js @@ -39,7 +39,7 @@ export default { return { name: svgEditor.i18next.t(`${name}:name`), callback () { - const btitle = `${name}:buttons.0.title` + const btitle = `${name} [Space / mouse wheel + drag]` // Add the button and its handler(s) const buttonTemplate = document.createElement('template') buttonTemplate.innerHTML = ` diff --git a/src/editor/panels/LeftPanel.js b/src/editor/panels/LeftPanel.js index 0a497fa78..8a3012786 100644 --- a/src/editor/panels/LeftPanel.js +++ b/src/editor/panels/LeftPanel.js @@ -45,7 +45,7 @@ class LeftPanel { */ clickSelect () { if (this.updateLeftPanel('tool_select')) { - this.editor.workarea.style.cursor = 'auto' + // this.editor.workarea.style.cursor = 'auto' this.editor.svgCanvas.setMode('select') } } diff --git a/src/editor/panels/TopPanel.html b/src/editor/panels/TopPanel.html index d3f60af4a..9cdd976a6 100644 --- a/src/editor/panels/TopPanel.html +++ b/src/editor/panels/TopPanel.html @@ -9,8 +9,8 @@