From 4ce4d495fc61e52246cd61d7cd6b1fd53931222a Mon Sep 17 00:00:00 2001 From: stefan-hoehn Date: Tue, 23 Jul 2024 22:44:43 +0200 Subject: [PATCH] [blockly] Add new sophisticated notification blocks (#2672) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Note that the new blocks can completely replace the old ones (though they will not be automatically converted but it must be done manually). --------- Also-by: Florian Hotze Signed-off-by: Stefan Höhn --- .../blockly/blocks-notifications.js | 428 +++++++++++++++++- .../config/controls/blockly-editor.vue | 61 ++- 2 files changed, 467 insertions(+), 22 deletions(-) diff --git a/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-notifications.js b/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-notifications.js index b602da5f6f..62508a460b 100644 --- a/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-notifications.js +++ b/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-notifications.js @@ -5,9 +5,11 @@ import Blockly from 'blockly' import { javascriptGenerator } from 'blockly/javascript.js' +import { blockGetCheckedInputType } from './utils.js' -// TODO: Add options to set icon and level (argument order should be the same as for broadcast notification etc.) export default function defineOHBlocks_Notifications (f7, isGraalJs) { + const unavailMsg = 'Extended notification blocks aren\'t supported in "application/javascript;version=ECMAScript-5.1"' + Blockly.Blocks['oh_sendNotification'] = { init: function () { this.appendValueInput('message') @@ -18,7 +20,7 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { this.setNextStatement(true, null) this.setInputsInline(false) this.setColour(0) - this.setTooltip('Send a notification message to a specific openhab user (requires openHAB Cloud Connector)') + this.setTooltip('Deprecated: send a notification message to a specific openhab user (requires openHAB Cloud Connector)') this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#send-notification-to-specific-cloud-email-user') } } @@ -27,7 +29,7 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { let email = javascriptGenerator.valueToCode(block, 'email', javascriptGenerator.ORDER_ATOMIC) let message = javascriptGenerator.valueToCode(block, 'message', javascriptGenerator.ORDER_ATOMIC) if (isGraalJs) { - return `actions.NotificationAction.sendNotification(${email}, ${message});\n` + return `actions.notificationBuilder(${message}).addUserId(${email}).send();\n` } else { const notifications = addNotificationAction() return `${notifications}.sendNotification(${email}, ${message});\n` @@ -48,7 +50,7 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { this.setNextStatement(true, null) this.setInputsInline(false) this.setColour(0) - this.setTooltip('send a notification to all clients. Provide icon name without prefix. (requires openHAB Cloud Connector)') + this.setTooltip('Deprecated: send a notification to all clients. Provide icon name without prefix. (requires openHAB Cloud Connector)') this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#send-notification-to-all-devices-and-users') } } @@ -58,7 +60,7 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { let icon = javascriptGenerator.valueToCode(block, 'icon', javascriptGenerator.ORDER_ATOMIC) let severity = block.getFieldValue('severity') if (isGraalJs) { - return `actions.NotificationAction.sendBroadcastNotification(${message}, ${icon}, '${severity}');\n` + return `actions.notificationBuilder(${message}).withIcon(${icon}).withTag('${severity}').send();\n` } else { const notifications = addNotificationAction() return `${notifications}.sendBroadcastNotification(${message}, ${icon}, '${severity}');\n` @@ -79,7 +81,7 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { this.setNextStatement(true, null) this.setInputsInline(false) this.setColour(0) - this.setTooltip('Sends a notification to the cloud log only, not to any device (requires openHAB Cloud Connector)') + this.setTooltip('Deprecated: sends a notification to the cloud log only, not to any device (requires openHAB Cloud Connector)') this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#send-notification-to-log-only') } } @@ -89,12 +91,424 @@ export default function defineOHBlocks_Notifications (f7, isGraalJs) { let icon = javascriptGenerator.valueToCode(block, 'icon', javascriptGenerator.ORDER_ATOMIC) let severity = block.getFieldValue('severity') if (isGraalJs) { - return `actions.NotificationAction.sendLogNotification(${message}, ${icon}, '${severity}');\n` + return `actions.notificationBuilder(${message}).withIcon(${icon}).withTag('${severity}').send();\n` } else { const notifications = addNotificationAction() return `${notifications}.sendLogNotification(${message}, ${icon}, '${severity}');\n` } } + + const usersImage = '' + const iconImage = '' + const tagImage = '' + const headerTitleImage = '' + const referenceImage = '' + const actionImage = '' + const actionButtonImage = '' + const actionButton1Image = '' + const actionButton2Image = '' + const actionButton3Image = '' + const mediaImage = '' + + Blockly.Blocks['oh_sendCloudNotification'] = { + init: function () { + const userImageField = new Blockly.FieldImage(usersImage, 15, 15, undefined, this.onClickUser) + userImageField.setTooltip('Comma separated users. If not provided it results into a broadcast') + const iconImageField = new Blockly.FieldImage(iconImage, 15, 15, undefined, this.onClickIcon) + iconImageField.setTooltip('Icon name') + const tagImageField = new Blockly.FieldImage(tagImage, 15, 15, undefined, this.onClickTag) + tagImageField.setTooltip('Tag') + const headerTitleImageField = new Blockly.FieldImage(headerTitleImage, 15, 15, undefined, this.onClickHeaderTitle) + headerTitleImageField.setTooltip('Header title of Notification') + const referenceImageField = new Blockly.FieldImage(referenceImage, 15, 15, undefined, this.onClickReference) + referenceImageField.setTooltip('Reference of the message') + const actionImageField = new Blockly.FieldImage(actionImage, 15, 15, undefined, this.onClickAction) + actionImageField.setTooltip('Action command to be sent') + const actionButton1ImageField = new Blockly.FieldImage(actionButton1Image, 15, 15, undefined, this.onClickActionButton1) + actionButton1ImageField.setTooltip('Action Button 1 command and action') + const actionButton2ImageField = new Blockly.FieldImage(actionButton2Image, 15, 15, undefined, this.onClickActionButton2) + actionButton2ImageField.setTooltip('Action Button 2 command and action') + const actionButton3ImageField = new Blockly.FieldImage(actionButton3Image, 15, 15, undefined, this.onClickActionButton3) + actionButton3ImageField.setTooltip('Action Button 3 command and action') + const mediaImageField = new Blockly.FieldImage(mediaImage, 15, 15, undefined, this.onClickMedia) + mediaImageField.setTooltip('A Media URL that can be clicked and can be accessed publicly to be shown with the notification') + + this.appendValueInput('message') + .setCheck('String') + .appendField(userImageField, 'imgUsers') + .appendField(headerTitleImageField, 'headerTitleIcon') + .appendField(iconImageField, 'imgIcon') + .appendField(tagImageField, 'tagIcon') + .appendField(referenceImageField, 'tagIcon') + .appendField(mediaImageField, 'mediaIcon') + .appendField(actionImageField, 'tagIcon') + .appendField(actionButton1ImageField, 'actionButton1Icon') + .appendField(actionButton2ImageField, 'actionButton2Icon') + .appendField(actionButton3ImageField, 'actionButton3Icon') + .appendField(new Blockly.FieldDropdown([['send notification', 'push'], ['send log', 'log']]), 'type') + + this.updateShape(this.hasUser, true, this.hasIconInfo, this.hasTag, this.hasHeaderTitle, this.hasReferenceId, this.hasAction, this.hasActionButton1, this.hasMedia) + + this.setPreviousStatement(true, null) + this.setNextStatement(true, null) + this.setInputsInline(false) + this.setColour(0) + this.setTooltip('Sends a notification to the cloud log only, not to any device (requires openHAB Cloud Connector)') + this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#sendCloudNotification') + }, + updateShape: function (hasUser, addTextBlock, hasIconInfo, hasTag, hasHeaderTitle, hasReferenceId, hasAction, hasActionButton1, hasActionButton2, hasActionButton3, hasMedia) { + this.hasUser = hasUser + this.hasIconInfo = hasIconInfo + this.hasTag = hasTag + this.hasHeaderTitle = hasHeaderTitle + this.hasReferenceId = hasReferenceId + this.hasMedia = hasMedia + this.hasAction = hasAction + this.hasActionButton1 = hasActionButton1 + this.hasActionButton2 = hasActionButton2 + this.hasActionButton3 = hasActionButton3 + + const blockSequence = ['usersInput', 'headerTitle', 'icon', 'tag', 'reference', 'media', 'action', 'actionButton1', 'actionButton2', 'actionButton3'] + + /** + * Adds a "shadow" text input to the parent block + * + * @param {*} parentConnection + * @param {string} tooltip + * @param {String} [blockType=text| + */ + function addBlockInput (parentConnection, tooltip, blockType = 'text') { + const textUsersBlock = this.workspace.newBlock(blockType) + textUsersBlock.setTooltip(tooltip) + textUsersBlock.initSvg() + textUsersBlock.render() + parentConnection.connect(textUsersBlock.outputConnection) + } + + /** + * Removes the block "shadow" text input from the block + * + * @param {Input} parentConnection + */ + function removeTextInput (parentConnection) { + const targetBlock = parentConnection.targetBlock() + if (targetBlock) { + targetBlock.unplug(true) + targetBlock.dispose(true) + } + } + + /** + * Show or hide this particular block + * + * @param {boolean} toggleOn show (true) / hide (false) block + * @param {string} blockName name of the block to be toggled + * @param {string} label label for the preceding label block + * @param {string} tooltip tooltip for the input block + * @param {string} [blockTypeCheck=String] checks the input by this block type + * @param {string} [shadowBlock=text] shadow block + */ + function toogleBlock (toggleOn, blockName, label, tooltip, blockTypeCheck = 'String', shadowBlock = 'text') { + if (toggleOn) { + if (!this.getInput(blockName)) { + const usersInput = this.appendValueInput(blockName) + .setCheck(blockTypeCheck) + .appendField(label) + + let startIndex = blockSequence.indexOf(blockName) + 1 + let blockAfter + // start from our block, iterate through all possible blocks and insert before the first actually existing one + for (let blockIndex = startIndex; blockIndex < blockSequence.length; blockIndex++) { + if (this.getInput(blockSequence[blockIndex])) { + blockAfter = blockSequence[blockIndex] + break + } + } + + if (blockAfter) { + this.moveInputBefore(blockName, blockAfter) + } + if (addTextBlock) { + const parentConnection = usersInput.connection + addBlockInput.call(this, parentConnection, tooltip, shadowBlock) + } + } + } else { + if (this.getInput(blockName)) { + const parentConnection = this.getInput(blockName).connection + removeTextInput(parentConnection) + this.removeInput(blockName) + } + } + } + + toogleBlock.call(this, hasUser, 'usersInput', 'to Users ', 'Add comma separated users. If not provided it results into a broadcast to everyone registered in the cloud', 'String') + toogleBlock.call(this, hasHeaderTitle, 'headerTitle', 'with Header Title ', 'Provide title', 'String') + toogleBlock.call(this, hasIconInfo, 'icon', 'with Icon ', 'Provide icon name', 'String') + toogleBlock.call(this, hasTag, 'tag', 'with Tag ', 'Provide tag', 'String') + toogleBlock.call(this, hasReferenceId, 'reference', 'with Reference ID ', 'Provide message reference ID', 'String') + toogleBlock.call(this, hasMedia, 'media', 'with Media URL ', 'Provide a public Media URL that should be shown as part of the notification', 'String') + toogleBlock.call(this, hasAction, 'action', 'with On-Click Action ', 'Provide cloud-notification-action', ['oh_notificationAction', 'String'], 'oh_cloudNotification_commandAction') + toogleBlock.call(this, hasActionButton1, 'actionButton1', 'with Action Button 1', 'Action Button 1: Title = cloud-notification-action', ['oh_notificationActionButton', 'String'], 'oh_cloudNotificationButton') + toogleBlock.call(this, hasActionButton2, 'actionButton2', 'with Action Button 2', 'Action Button 2: Title = cloud-notification-action', ['oh_notificationActionButton', 'String'], 'oh_cloudNotificationButton') + toogleBlock.call(this, hasActionButton3, 'actionButton3', 'with Action Button 3', 'Action Button 3: Title = cloud-notification-action', ['oh_notificationActionButton', 'String'], 'oh_cloudNotificationButton') + }, + onClickUser () { + let block = this.getSourceBlock() + block.hasUser = !block.hasUser + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickIcon () { + let block = this.getSourceBlock() + block.hasIconInfo = !block.hasIconInfo + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickTag () { + let block = this.getSourceBlock() + block.hasTag = !block.hasTag + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickHeaderTitle () { + let block = this.getSourceBlock() + block.hasHeaderTitle = !block.hasHeaderTitle + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickReference () { + let block = this.getSourceBlock() + block.hasReferenceId = !block.hasReferenceId + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickAction () { + let block = this.getSourceBlock() + block.hasAction = !block.hasAction + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickActionButton1 () { + let block = this.getSourceBlock() + block.hasActionButton1 = !block.hasActionButton1 + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickActionButton2 () { + let block = this.getSourceBlock() + block.hasActionButton2 = !block.hasActionButton2 + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickActionButton3 () { + let block = this.getSourceBlock() + block.hasActionButton3 = !block.hasActionButton3 + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + onClickMedia () { + let block = this.getSourceBlock() + block.hasMedia = !block.hasMedia + block.updateShape(block.hasUser, true, block.hasIconInfo, block.hasTag, block.hasHeaderTitle, block.hasReferenceId, block.hasAction, block.hasActionButton1, block.hasActionButton2, block.hasActionButton3, block.hasMedia) + }, + mutationToDom: function () { + let container = Blockly.utils.xml.createElement('mutation') + container.setAttribute('hasUser', this.hasUser) + container.setAttribute('hasIconInfo', this.hasIconInfo) + container.setAttribute('hasTag', this.hasTag) + container.setAttribute('hasHeaderTitle', this.hasHeaderTitle) + container.setAttribute('hasReferenceId', this.hasReferenceId) + container.setAttribute('hasMedia', this.hasMedia) + container.setAttribute('hasAction', this.hasAction) + container.setAttribute('hasActionButton1', this.hasActionButton1) + container.setAttribute('hasActionButton2', this.hasActionButton2) + container.setAttribute('hasActionButton3', this.hasActionButton3) + return container + }, + domToMutation: function (xmlElement) { + let hasUser = xmlElement.getAttribute('hasUser') === 'true' + let hasIconInfo = xmlElement.getAttribute('hasIconInfo') === 'true' + let hasTag = xmlElement.getAttribute('hasTag') === 'true' + let hasHeaderTitle = xmlElement.getAttribute('hasHeaderTitle') === 'true' + let hasReferenceId = xmlElement.getAttribute('hasReferenceId') === 'true' + let hasMedia = xmlElement.getAttribute('hasMedia') === 'true' + let hasAction = xmlElement.getAttribute('hasAction') === 'true' + let hasActionButton1 = xmlElement.getAttribute('hasActionButton1') === 'true' + let hasActionButton2 = xmlElement.getAttribute('hasActionButton2') === 'true' + let hasActionButton3 = xmlElement.getAttribute('hasActionButton3') === 'true' + this.updateShape(hasUser, false, hasIconInfo, hasTag, hasHeaderTitle, hasReferenceId, hasAction, hasActionButton1, hasActionButton2, hasActionButton3, hasMedia) + } + } + + javascriptGenerator.forBlock['oh_sendCloudNotification'] = function (block) { + const message = javascriptGenerator.valueToCode(block, 'message', javascriptGenerator.ORDER_ATOMIC) + const type = block.getFieldValue('type') + const icon = javascriptGenerator.valueToCode(block, 'icon', javascriptGenerator.ORDER_ATOMIC) + const tag = javascriptGenerator.valueToCode(block, 'tag', javascriptGenerator.ORDER_ATOMIC) + const headerTitle = javascriptGenerator.valueToCode(block, 'headerTitle', javascriptGenerator.ORDER_ATOMIC) + const reference = javascriptGenerator.valueToCode(block, 'reference', javascriptGenerator.ORDER_ATOMIC) + const media = javascriptGenerator.valueToCode(block, 'media', javascriptGenerator.ORDER_ATOMIC) + const action = javascriptGenerator.valueToCode(block, 'action', javascriptGenerator.ORDER_ATOMIC) + const actionType = (action !== '') ? blockGetCheckedInputType(block, 'action') : undefined + const actionButton1 = javascriptGenerator.valueToCode(block, 'actionButton1', javascriptGenerator.ORDER_ATOMIC) + const actionButton1Type = (actionButton1 !== '') ? blockGetCheckedInputType(block, 'actionButton1') : undefined + const actionButton2 = javascriptGenerator.valueToCode(block, 'actionButton2', javascriptGenerator.ORDER_ATOMIC) + const actionButton2Type = (actionButton2 !== '') ? blockGetCheckedInputType(block, 'actionButton2') : undefined + const actionButton3 = javascriptGenerator.valueToCode(block, 'actionButton3', javascriptGenerator.ORDER_ATOMIC) + const actionButton3Type = (actionButton3 !== '') ? blockGetCheckedInputType(block, 'actionButton3') : undefined + const users = javascriptGenerator.valueToCode(block, 'usersInput', javascriptGenerator.ORDER_ATOMIC) + + function getActionButtonCode (titleAndAction, buttonActionType) { + let actionButtonCode + if (buttonActionType === 'String') { + titleAndAction = titleAndAction.replace(/'/g, '').split('=') + if (titleAndAction.length === 2) { + let title = titleAndAction[0] + let buttonAction = titleAndAction[1].replace(/ /g, '') + actionButtonCode = `.addActionButton('${title}','${buttonAction}')` + } + } else { + actionButtonCode = titleAndAction + } + return actionButtonCode + } + + const iconCode = (icon === '') ? '' : `.withIcon(${icon})` + const tagCode = (tag === '') ? '' : `.withTag(${tag})` + const headerTitleCode = (headerTitle === '') ? '' : `.withTitle(${headerTitle})` + const referenceCode = (reference === '') ? '' : `.withReferenceId(${reference})` + const mediaCode = (media === '') ? '' : `.withMediaAttachmentUrl(${media})` + const actionCode = (action === '') ? '' : (actionType === 'String') ? `.withOnClickAction(${action})` : `.withOnClickAction('${action}')` + const usersCode = (users === '') ? '' : `.addUserId(...${users}.split(','))` + + const actionButton1Code = getActionButtonCode(actionButton1, actionButton1Type) + const actionButton2Code = getActionButtonCode(actionButton2, actionButton2Type) + const actionButton3Code = getActionButtonCode(actionButton3, actionButton3Type) + const logOnlyCode = (type === 'push') ? '' : '.logOnly()' + + if (isGraalJs) { + return `actions.notificationBuilder(${message})${usersCode}${headerTitleCode}${iconCode}${tagCode}${referenceCode}${mediaCode}${actionCode}${actionButton1Code}${actionButton2Code}${actionButton3Code}${logOnlyCode}.send()\n` + } else { + throw new Error(unavailMsg) + } + } + + Blockly.Blocks['oh_cloudNotificationButton'] = { + init: function () { + const actionButtonImageField = new Blockly.FieldImage(actionButtonImage, 15, 15, undefined) + this.appendValueInput('label') + .appendField(actionButtonImageField, 'actionButtonHint') + .appendField('label') + .setCheck('String') + this.appendValueInput('action') + .appendField('action') + .setCheck(['String', 'oh_notificationAction']) + this.setInputsInline(false) + this.setColour(0) + this.setTooltip('Notification Button with label and action') + this.setOutput(true, 'oh_notificationActionButton') + + this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#cloudNotificationButton') + } + } + + javascriptGenerator.forBlock['oh_cloudNotificationButton'] = function (block) { + let label = javascriptGenerator.valueToCode(block, 'label', javascriptGenerator.ORDER_ATOMIC).replace(/'/g, '') + let action = javascriptGenerator.valueToCode(block, 'action', javascriptGenerator.ORDER_ATOMIC).replace(/'/g, '') + if (isGraalJs) { + return [`.addActionButton('${label}','${action}')`, javascriptGenerator.ORDER_ATOMIC] + } else { + throw new Error(unavailMsg) + } + } + + Blockly.Blocks['oh_cloudNotification_commandAction'] = { + init: function () { + const actionImageField = new Blockly.FieldImage(actionImage, 15, 15, undefined) + this.appendValueInput('command') + .appendField(actionImageField, 'actionHint') + .appendField('send command ') + .setCheck('String') + this.appendValueInput('itemName') + .appendField('to item ') + .setCheck('oh_item') + this.setInputsInline(false) + this.setColour(0) + this.setTooltip('An action that sends a command to an item, like ON to KitchenLights') + this.setOutput(true, 'oh_notificationAction') + + this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#command-action') + } + } + + javascriptGenerator.forBlock['oh_cloudNotification_commandAction'] = function (block) { + let command = javascriptGenerator.valueToCode(block, 'command', javascriptGenerator.ORDER_ATOMIC).replace(/'/g, '') + let itemName = javascriptGenerator.valueToCode(block, 'itemName', javascriptGenerator.ORDER_ATOMIC).replace(/'/g, '') + if (isGraalJs) { + return [`command:${itemName}:${command}`, javascriptGenerator.ORDER_ATOMIC] + } else { + throw new Error(unavailMsg) + } + } + + Blockly.Blocks['oh_cloudNotification_uiAction'] = { + init: function () { + const actionImageField = new Blockly.FieldImage(actionImage, 15, 15, undefined) + this.appendDummyInput() + .appendField(actionImageField, 'actionHint') + .appendField('Call ') + .appendField(new Blockly.FieldDropdown([['Basic UI', 'ui:/basicui/app?'], ['UI absolute Path', 'ui:'], ['Main UI path', 'ui:navigate:'], ['Main UI popup', 'ui:popup:'], ['URL', '']]), 'command') + this.appendValueInput('path') + .appendField('with ') + .setCheck('String') + this.setInputsInline(false) + this.setColour(0) + + this.setTooltip(() => { + const commandType = this.getFieldValue('command') + const TIP = { + 'ui:/basicui/app?': 'Provide valid sitemap parameters in the form w=nnnn&sitemap=mysitemap', + 'ui:': 'Provide the path that follows your openhab domain name, e.g /some/absolute/path like /page/rollershutter_firstfloor', + 'ui:navigate:': '', + 'ui:popup:': 'Provide the name of a page to be shown as a popup, e.g. page_phones', + '': 'Provide a valid absolute URL like https://openhab.org' + } + return TIP[commandType] + }) + this.setOutput(true, 'oh_notificationAction') + + this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#ui-action') + } + } + + javascriptGenerator.forBlock['oh_cloudNotification_uiAction'] = function (block) { + const command = block.getFieldValue('command') + let path = javascriptGenerator.valueToCode(block, 'path', javascriptGenerator.ORDER_ATOMIC).replace(/'/g, '') + if (isGraalJs) { + return [`${command}${path}`, javascriptGenerator.ORDER_ATOMIC] + } else { + throw new Error(unavailMsg) + } + } + + Blockly.Blocks['oh_hideCloudNotification'] = { + init: function () { + this.appendValueInput('reference') + .appendField('hide notification by reference ID') + this.appendValueInput('tag') + .appendField('or tag ') + this.setPreviousStatement(true, null) + this.setNextStatement(true, null) + this.setInputsInline(false) + this.setColour(0) + this.setTooltip('Hides the notification with the given reference ID or tag') + this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-notifications.html#hide-notification') + } + } + + javascriptGenerator.forBlock['oh_hideCloudNotification'] = function (block) { + const reference = javascriptGenerator.valueToCode(block, 'reference', javascriptGenerator.ORDER_ATOMIC) + const tag = javascriptGenerator.valueToCode(block, 'tag', javascriptGenerator.ORDER_ATOMIC) + const tagCode = (tag === '') ? '' : `.withTag(${tag})` + const referenceCode = (reference === '') ? '' : `.withReferenceId(${reference})` + if (isGraalJs) { + return `actions.notificationBuilder("hide notification")${referenceCode}${tagCode}.hide().send();\n` + } else { + throw new Error(unavailMsg) + } + } } function addNotificationAction () { diff --git a/bundles/org.openhab.ui/web/src/components/config/controls/blockly-editor.vue b/bundles/org.openhab.ui/web/src/components/config/controls/blockly-editor.vue index 6a2d0e84c5..a7da4b8f82 100644 --- a/bundles/org.openhab.ui/web/src/components/config/controls/blockly-editor.vue +++ b/bundles/org.openhab.ui/web/src/components/config/controls/blockly-editor.vue @@ -853,39 +853,70 @@ helpUrl="configuration/blockly/rules-blockly-notifications.html" text="Help" callbackKey="ohBlocklyHelp" /> - - + + - test@example.org + message - + + + - message + label + + + + + action - - + + - message + label - + + + + + command + + + + + + + + + + - temperature_cold + command + + + - - + + - message + path + + + + + + + reference id - + - temperature_hot + tag