diff --git a/extensions/reviewed/DoubleClick.json b/extensions/reviewed/DoubleClick.json index 60431c869..d87a13d59 100644 --- a/extensions/reviewed/DoubleClick.json +++ b/extensions/reviewed/DoubleClick.json @@ -2,18 +2,22 @@ "author": "Silver-Streak", "category": "Input", "extensionNamespace": "", + "fullName": "Double-click and tap", "gdevelopVersion": ">=5.5.222", - "fullName": "Double-click", "helpPath": "", "iconUrl": "", "name": "DoubleClick", "previewIconUrl": "https://resources.gdevelop-app.com/assets/Icons/Line Hero Pack/Master/SVG/Computers and Hardware/Computers and Hardware_mouse_wireless_pc.svg", - "shortDescription": "Check for a double-click with a mouse, or a double-tap on a touchscreen.", - "version": "1.1.0", + "shortDescription": "Check for a double-click or a tap.", + "version": "2.0.0", "description": [ - "Check for a double-click with a mouse, or a double-tap on a touchscreen.", + "Check for a double-click with a mouse, or a tap and double-click on a touchscreen.", "", - "Please note: Touch devices do not have alternatives for middle/right clicks. Taps on a touch device will get counted for _any_ use of these conditions, so you should either design your controls accordingly, or build out separate events if the device has a touch screen." + "Please note: Touch devices do not have alternatives for middle/right clicks. Taps on a touch device will get counted for _any_ use of these conditions, so you should either design your controls accordingly, or build out separate events if the device has a touch screen.", + "", + "Breaking changes from 2.0.0:", + "- Mouse button value now needs quotes", + "- The double-click sensitivity now relies on the device settings" ], "origin": { "identifier": "DoubleClick", @@ -23,11 +27,12 @@ "mouse", "cursor", "pointer", - "double-click", - "double-tap" + "click", + "tap" ], "authorIds": [ - "8Ih1aa8f5gWUp4UB2BdhQ2iXWxJ3" + "8Ih1aa8f5gWUp4UB2BdhQ2iXWxJ3", + "IWykYNRvhCZBN3vEgKEbBPOR3Oc2" ], "dependencies": [], "globalVariables": [ @@ -37,31 +42,139 @@ "value": 0.5 } ], - "sceneVariables": [ + "sceneVariables": [], + "eventsFunctions": [ { - "name": "LastPressTime", - "type": "structure", - "children": [ + "fullName": "", + "functionType": "Action", + "name": "onFirstSceneLoaded", + "sentence": "", + "events": [ { - "name": "Left", - "type": "number", - "value": 0 + "type": "BuiltinCommonInstructions::JsCode", + "inlineCode": [ + "class ClickCounter {", + " // Multiple clicks", + " multipleClickButton = -1;", + " count = 0;", + "", + " // Simple click", + " lastButton = -1;", + " lastPointerId = 0;", + " lastTime = 0;", + " hasMoved = false;", + " simpleClickButton = -1;", + "", + " constructor() {", + " window.addEventListener(", + " 'click',", + " event => {", + " if (event.detail > 1) {", + " this.multipleClickButton = convertHtmlMouseButtonToInputManagerMouseButton(event.button);", + " this.count = event.detail;", + " }", + " }", + " );", + "", + " // The 'click' event doesn't do any constraint on the 1st click.", + " // It doesn't allow to differenciate a dragging from a click.", + " // So, we check it manually.", + "", + " window.addEventListener(", + " 'pointerdown',", + " event => {", + " this.lastButton = event.button;", + " this.lastPointerId = event.pointerId;", + " this.lastTime = Date.now();", + " this.hasMoved = false;", + " }", + " );", + "", + " window.addEventListener(", + " 'pointermove',", + " event => {", + " if (event.pointerId === this.lastPointerId) {", + " this.hasMoved = true;", + " }", + " }", + " );", + "", + " window.addEventListener(", + " 'pointerup',", + " (event) => {", + " if (event.button === this.lastButton &&", + " event.pointerId === this.lastPointerId &&", + " (!this.hasMoved || Date.now() - this.lastTime < 500)) {", + " this.simpleClickButton = convertHtmlMouseButtonToInputManagerMouseButton(event.button);", + " this.lastButton = -1;", + " }", + " }", + " );", + "", + " }", + "", + " reset() {", + " this.multipleClickButton = -1;", + " this.simpleClickButton = -1;", + " }", + "", + " hasClicked(buttonName, count) {", + " const button = gdjs.evtTools.input.mouseButtonsNameToCode[buttonName];", + " if (count === 1) {", + " return button === this.simpleClickButton;", + " }", + " else {", + " return button === this.multipleClickButton && clickCount === this.count", + " }", + " }", + "}", + "", + "// Converts HTML mouse button to InputManager mouse button.", + "// This function is used to align HTML button values with GDevelop 3 C++ SFML Mouse button enum values,", + "// notably the middle and right buttons.", + "function convertHtmlMouseButtonToInputManagerMouseButton(button) {", + " switch (button) {", + " case 1: // Middle button", + " return gdjs.InputManager.MOUSE_MIDDLE_BUTTON;", + " case 2: // Right button", + " return gdjs.InputManager.MOUSE_RIGHT_BUTTON;", + " }", + " return button;", + "}", + "", + "gdjs._DoubleClickExtension = { clickCounter: new ClickCounter() };", + "" + ], + "parameterObjects": "", + "useStrict": true, + "eventsSheetExpanded": true } - ] + ], + "parameters": [], + "objectGroups": [] }, { - "name": "IsReleased", - "type": "structure", - "children": [ + "fullName": "", + "functionType": "Action", + "name": "onScenePostEvents", + "sentence": "", + "events": [ { - "name": "Left", - "type": "boolean", - "value": false + "type": "BuiltinCommonInstructions::JsCode", + "inlineCode": [ + "const { clickCounter } = gdjs._DoubleClickExtension;\r", + "\r", + "clickCounter.reset();\r", + "" + ], + "parameterObjects": "", + "useStrict": true, + "eventsSheetExpanded": false } - ] - } - ], - "eventsFunctions": [ + ], + "parameters": [], + "objectGroups": [] + }, { "description": "Check if the specified mouse button is clicked twice in a short amount of time.", "fullName": "Double-clicked (or double-tapped)", @@ -69,176 +182,72 @@ "name": "HasDoubleClicked", "sentence": "_PARAM1_ mouse button is double-clicked", "events": [ - { - "type": "BuiltinCommonInstructions::Comment", - "color": { - "b": 109, - "g": 230, - "r": 255, - "textB": 0, - "textG": 0, - "textR": 0 - }, - "comment": "Trigger once must never be used in functions as every call would share the same one." - }, { "type": "BuiltinCommonInstructions::Standard", "conditions": [ { "type": { - "inverted": true, - "value": "MouseButtonFromTextPressed" + "value": "DoubleClick::HasClicked" }, "parameters": [ "", - "MouseButton" + "MouseButton", + "2", + "" ] } ], "actions": [ { "type": { - "value": "SetBooleanVariable" + "value": "SetReturnBoolean" }, "parameters": [ - "IsReleased[MouseButton]", - "True", - "1" + "True" ] } ] - }, + } + ], + "parameters": [ + { + "description": "Mouse button to track", + "longDescription": "As touch devices do not have middle/right tap equivalents, you will need to account for this within your events if you're not using the left mouse button and building for touch devices.", + "name": "MouseButton", + "type": "mouseButton" + } + ], + "objectGroups": [] + }, + { + "description": "Check if the specified mouse button is clicked.", + "fullName": "Clicked (or tapped)", + "functionType": "Condition", + "name": "HasSimpleClicked", + "sentence": "_PARAM1_ mouse button is clicked", + "events": [ { "type": "BuiltinCommonInstructions::Standard", "conditions": [ { "type": { - "value": "MouseButtonFromTextPressed" + "value": "DoubleClick::HasClicked" }, "parameters": [ "", - "MouseButton" - ] - }, - { - "type": { - "value": "BooleanVariable" - }, - "parameters": [ - "IsReleased[MouseButton]", - "True", - "1" - ] - }, - { - "type": { - "value": "BuiltinCommonInstructions::CompareNumbers" - }, - "parameters": [ - "TimeFromStart()", - ">=", - "MaxDelay" + "MouseButton", + "1", + "" ] } ], "actions": [ { "type": { - "value": "SetBooleanVariable" + "value": "SetReturnBoolean" }, "parameters": [ - "IsReleased[MouseButton]", - "False", - "1" - ] - } - ], - "events": [ - { - "type": "BuiltinCommonInstructions::Standard", - "conditions": [], - "actions": [ - { - "type": { - "value": "SetNumberVariable" - }, - "parameters": [ - "Delay", - "=", - "TimeFromStart() - LastPressTime[MouseButton]" - ] - } - ], - "events": [ - { - "type": "BuiltinCommonInstructions::Standard", - "conditions": [ - { - "type": { - "value": "NumberVariable" - }, - "parameters": [ - "Delay", - "<=", - "MaxDelay" - ] - } - ], - "actions": [ - { - "type": { - "value": "SetNumberVariable" - }, - "parameters": [ - "LastPressTime[MouseButton]", - "=", - "0" - ] - }, - { - "type": { - "value": "SetReturnBoolean" - }, - "parameters": [ - "True" - ] - } - ] - }, - { - "type": "BuiltinCommonInstructions::Standard", - "conditions": [ - { - "type": { - "value": "NumberVariable" - }, - "parameters": [ - "Delay", - ">", - "MaxDelay" - ] - } - ], - "actions": [ - { - "type": { - "value": "SetNumberVariable" - }, - "parameters": [ - "LastPressTime[MouseButton]", - "=", - "TimeFromStart()" - ] - } - ] - } - ], - "variables": [ - { - "name": "Delay", - "type": "number", - "value": 0 - } + "True" ] } ] @@ -249,40 +258,44 @@ "description": "Mouse button to track", "longDescription": "As touch devices do not have middle/right tap equivalents, you will need to account for this within your events if you're not using the left mouse button and building for touch devices.", "name": "MouseButton", - "type": "mouse" + "type": "mouseButton" } ], "objectGroups": [] }, { - "description": "Set the maximum delay between two clicks (or two taps) to be considered as a double click.", - "fullName": "Set the double click delay", - "functionType": "Action", - "name": "SetMaxDelay", - "sentence": "Set the double-click maximum delay between two clicks (or two taps) to _PARAM1_s", + "description": "Check if the specified mouse button is clicked.", + "fullName": "Clicked (or tapped)", + "functionType": "Condition", + "name": "HasClicked", + "private": true, + "sentence": "_PARAM1_ mouse button is clicked _PARAM2_ times", "events": [ { - "type": "BuiltinCommonInstructions::Standard", - "conditions": [], - "actions": [ - { - "type": { - "value": "SetNumberVariable" - }, - "parameters": [ - "MaxDelay", - "=", - "Value" - ] - } - ] + "type": "BuiltinCommonInstructions::JsCode", + "inlineCode": [ + "const { clickCounter } = gdjs._DoubleClickExtension;", + "", + "const buttonName = eventsFunctionContext.getArgument(\"MouseButton\");", + "const clickCount = eventsFunctionContext.getArgument(\"ClickCount\");", + "", + "eventsFunctionContext.returnValue = clickCounter.hasClicked(buttonName, clickCount);" + ], + "parameterObjects": "", + "useStrict": true, + "eventsSheetExpanded": true } ], "parameters": [ { - "description": "Maximum delay (in seconds)", - "longDescription": "By default, this value is 0.5 seconds.", - "name": "Value", + "description": "Mouse button to track", + "longDescription": "As touch devices do not have middle/right tap equivalents, you will need to account for this within your events if you're not using the left mouse button and building for touch devices.", + "name": "MouseButton", + "type": "mouseButton" + }, + { + "description": "Click count", + "name": "ClickCount", "type": "expression" } ], diff --git a/scripts/lib/ExtensionsValidatorExceptions.js b/scripts/lib/ExtensionsValidatorExceptions.js index e896e5218..a2053754f 100644 --- a/scripts/lib/ExtensionsValidatorExceptions.js +++ b/scripts/lib/ExtensionsValidatorExceptions.js @@ -234,6 +234,12 @@ const extensionsAllowedProperties = { runtimeSceneAllowedProperties: [], javaScriptObjectAllowedProperties: [], }, + DoubleClick: { + gdjsAllowedProperties: ['_DoubleClickExtension', 'InputManager'], + gdjsEvtToolsAllowedProperties: ['input'], + runtimeSceneAllowedProperties: [], + javaScriptObjectAllowedProperties: [], + }, FlexBox: { gdjsAllowedProperties: ['layoutContainers'], gdjsEvtToolsAllowedProperties: [],