diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 51c2f2aa6..a4e6c0c5b 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -453,6 +453,15 @@ export default [ creatorAlias: "gaimerI17", note: "Extension thumbnail made by Dillon." }, + { + name: "Input Validator", + description: "A set of blocks to verify conditions, check data types, and validate inputs easily.", + code: "logise/inputvalidator.js", + banner: "logise/verificator.png", + creator: "Logise1123", + isGitHub: true, + creatorAlias: "Logise", + }, /* these extensions are completely dead as of now { name: "Online Captcha", diff --git a/static/extensions/logise/inputvalidator.js b/static/extensions/logise/inputvalidator.js new file mode 100644 index 000000000..3cf8de238 --- /dev/null +++ b/static/extensions/logise/inputvalidator.js @@ -0,0 +1,421 @@ +// Name: Input Validator +// ID: inputValidator +// Description: Blocks to easily validate user inputs like emails, numbers, and text patterns. +// By: Logise +// License: MIT + +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('Input Validator extension must run unsandboxed'); + } + + const menuIconURI = ""; + + class InputValidator { + getInfo() { + return { + id: 'inputValidator', + name: 'Input Validator', + color1: '#4CBF57', + color2: '#3A9944', + menuIconURI: menuIconURI, + blocks: [{ + opcode: 'isValidEmail', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [TEXT] a valid email?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'test@example.com', + }, + }, + }, + { + opcode: 'isNumberInRange', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [NUMBER] in range min [MIN] max [MAX]?', + arguments: { + NUMBER: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 50, + }, + MIN: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0, + }, + MAX: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + }, + }, + { + opcode: 'isTextType', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [TEXT] [TYPE] only?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'HelloWorld', + }, + TYPE: { + type: Scratch.ArgumentType.STRING, + menu: 'textType', + defaultValue: 'letters', + }, + }, + }, + { + opcode: 'isTextLengthBetween', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is length of [TEXT] between [MIN] and [MAX]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'penguin', + }, + MIN: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 3, + }, + MAX: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 10, + }, + }, + }, + { + opcode: 'matchesRegex', + blockType: Scratch.BlockType.BOOLEAN, + text: 'does [TEXT] match regex pattern [PATTERN]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'abc-123', + }, + PATTERN: { + type: Scratch.ArgumentType.STRING, + defaultValue: '^[a-z]{3}-[0-9]{3}$', + }, + }, + }, + { + opcode: 'isCase', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [TEXT] [CASE]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'HELLO', + }, + CASE: { + type: Scratch.ArgumentType.STRING, + menu: 'caseType', + defaultValue: 'UPPERCASE', + }, + }, + }, + { + opcode: 'startsWith', + blockType: Scratch.BlockType.BOOLEAN, + text: 'does [TEXT] start with [PREFIX]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'PenguinMod', + }, + PREFIX: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'Penguin', + }, + }, + }, + { + opcode: 'endsWith', + blockType: Scratch.BlockType.BOOLEAN, + text: 'does [TEXT] end with [SUFFIX]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'PenguinMod', + }, + SUFFIX: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'Mod', + }, + }, + }, + { + opcode: 'containsText', + blockType: Scratch.BlockType.BOOLEAN, + text: 'does [TEXT] contain [SUBSTRING]?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'PenguinMod is cool', + }, + SUBSTRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'Mod', + }, + }, + }, + { + opcode: 'isNumberType', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [NUMBER] an [NUM_TYPE]?', + arguments: { + NUMBER: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 42, + }, + NUM_TYPE: { + type: Scratch.ArgumentType.STRING, + menu: 'numberType', + defaultValue: 'integer', + }, + }, + }, + { + opcode: 'isNumberSign', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [NUMBER] [SIGN]?', + arguments: { + NUMBER: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 10, + }, + SIGN: { + type: Scratch.ArgumentType.STRING, + menu: 'signType', + defaultValue: 'positive', + }, + }, + }, + { + opcode: 'isValidURL', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [TEXT] a valid URL?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'https://penguinmod.com', + }, + }, + }, + { + opcode: 'isValidHexColor', + blockType: Scratch.BlockType.BOOLEAN, + text: 'is [TEXT] a valid hex color code?', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: '#4CBF57', + }, + }, + }, + { + opcode: 'countOccurrences', + blockType: Scratch.BlockType.REPORTER, + text: 'count occurrences of [SUBSTRING] in [TEXT]', + arguments: { + SUBSTRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'p', + }, + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'apple pineapple', + }, + }, + }, + { + opcode: 'sanitizeText', + blockType: Scratch.BlockType.REPORTER, + text: 'sanitize [TEXT] by removing [SANITIZE_ACTION]', + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: ' Hello World! 123 ', + }, + SANITIZE_ACTION: { + type: Scratch.ArgumentType.STRING, + menu: 'sanitizeAction', + defaultValue: 'leading/trailing spaces', + }, + }, + }, + ], + menus: { + textType: { + acceptReporters: true, + items: ['letters', 'numbers', 'alphanumeric'], + }, + caseType: { + acceptReporters: true, + items: ['UPPERCASE', 'lowercase'], + }, + numberType: { + acceptReporters: true, + items: ['integer', 'decimal'], + }, + signType: { + acceptReporters: true, + items: ['positive', 'negative'], + }, + sanitizeAction: { + acceptReporters: true, + items: ['leading/trailing spaces', 'all spaces', 'letters', 'numbers', 'special characters'], + }, + }, + }; + } + + isValidEmail({ TEXT }) { + const text = String(TEXT); + const pattern = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; + return pattern.test(text); + } + + isNumberInRange({ NUMBER, MIN, MAX }) { + const num = Number(NUMBER); + const min = Number(MIN); + const max = Number(MAX); + if (isNaN(num) || isNaN(min) || isNaN(max)) { + return false; + } + return num >= min && num <= max; + } + + isTextType({ TEXT, TYPE }) { + const text = String(TEXT); + let pattern; + switch (TYPE) { + case 'letters': + pattern = /^[a-zA-Z]+$/; + break; + case 'numbers': + pattern = /^[0-9]+$/; + break; + case 'alphanumeric': + pattern = /^[a-zA-Z0-9]+$/; + break; + default: + return false; + } + return pattern.test(text); + } + + isTextLengthBetween({ TEXT, MIN, MAX }) { + const text = String(TEXT); + const min = Number(MIN); + const max = Number(MAX); + if (isNaN(min) || isNaN(max)) { + return false; + } + const length = text.length; + return length >= min && length <= max; + } + + matchesRegex({ TEXT, PATTERN }) { + const text = String(TEXT); + const pattern = String(PATTERN); + try { + const regex = new RegExp(pattern); + return regex.test(text); + } catch (e) { + console.error("Invalid Regex Pattern:", e); + return false; + } + } + + isCase({ TEXT, CASE }) { + const text = String(TEXT); + if (CASE === 'UPPERCASE') { + return text === text.toUpperCase(); + } else if (CASE === 'lowercase') { + return text === text.toLowerCase(); + } + return false; + } + + startsWith({ TEXT, PREFIX }) { + return String(TEXT).startsWith(String(PREFIX)); + } + + endsWith({ TEXT, SUFFIX }) { + return String(TEXT).endsWith(String(SUFFIX)); + } + + containsText({ TEXT, SUBSTRING }) { + return String(TEXT).includes(String(SUBSTRING)); + } + + isNumberType({ NUMBER, NUM_TYPE }) { + const num = Number(NUMBER); + if (isNaN(num)) return false; + if (NUM_TYPE === 'integer') { + return Number.isInteger(num); + } else if (NUM_TYPE === 'decimal') { + return !Number.isInteger(num); + } + return false; + } + + isNumberSign({ NUMBER, SIGN }) { + const num = Number(NUMBER); + if (isNaN(num)) return false; + if (SIGN === 'positive') { + return num > 0; + } else if (SIGN === 'negative') { + return num < 0; + } + return false; + } + + isValidURL({ TEXT }) { + const text = String(TEXT); + try { + new URL(text); + return true; + } catch (_) { + return false; + } + } + + isValidHexColor({ TEXT }) { + const text = String(TEXT); + const pattern = /^#([0-9A-Fa-f]{3}){1,2}$/; + return pattern.test(text); + } + + countOccurrences({ SUBSTRING, TEXT }) { + const text = String(TEXT); + const substring = String(SUBSTRING); + if (substring.length === 0) return 0; + return text.split(substring).length - 1; + } + + sanitizeText({ TEXT, SANITIZE_ACTION }) { + const text = String(TEXT); + switch (SANITIZE_ACTION) { + case 'leading/trailing spaces': + return text.trim(); + case 'all spaces': + return text.replace(/\s/g, ''); + case 'letters': + return text.replace(/[a-zA-Z]/g, ''); + case 'numbers': + return text.replace(/[0-9]/g, ''); + case 'special characters': + return text.replace(/[^a-zA-Z0-9\s]/g, ''); + default: + return text; + } + } + } + + Scratch.extensions.register(new InputValidator()); +})(Scratch); diff --git a/static/images/logise/verificator.png b/static/images/logise/verificator.png new file mode 100644 index 000000000..ffcfcc10f Binary files /dev/null and b/static/images/logise/verificator.png differ