├── .eslintrc.js ├── .gitattributes ├── .github └── workflows │ └── pages.yml ├── .gitignore ├── .prettierrc ├── README.md ├── package.json ├── public ├── android-chrome-192x192.png ├── android-chrome-512x512.png ├── apple-touch-icon.png ├── browserconfig.xml ├── favicon-16x16.png ├── favicon-32x32.png ├── favicon.ico ├── index.html ├── manifest.json ├── mstile-150x150.png ├── safari-pinned-tab.svg └── site.webmanifest ├── src ├── App.css ├── App.tsx ├── CreateEditShortcut.css ├── CreateEditShortcut.tsx ├── DownloadButton.tsx ├── DownloadModal.tsx ├── DownloadStatusSwitcher.tsx ├── ErrorBoundary.tsx ├── FileManager.ts ├── FilePane.css ├── FilePane.tsx ├── Key.ts ├── Modal.tsx ├── ModalContainer.tsx ├── SearchActions.css ├── SearchActions.tsx ├── UploadShortcutModal.css ├── UploadShortcutModal.tsx ├── Uploader.ts ├── ace │ ├── mode-scpl.ts │ └── scpl.iro ├── data │ ├── ActionCategories.ts │ └── ShortcutMeta.ts ├── img │ ├── back.svg │ ├── bg.jpg │ ├── close.svg │ ├── delete.svg │ ├── delete_w.svg │ ├── down.svg │ ├── edit.svg │ ├── edit_w.svg │ ├── error.svg │ ├── error_grey.svg │ ├── error_w.svg │ ├── error_x.svg │ ├── file.png │ ├── file_active.png │ ├── file_error.png │ ├── file_warning.png │ ├── folder.png │ ├── folder_error.png │ ├── folder_warning.png │ ├── glyph_sheet.png │ ├── glyph_sheet_white.png │ ├── icon.png │ ├── import.png │ ├── magic.svg │ ├── magic_w.svg │ ├── menu.svg │ ├── new_file.png │ ├── new_folder.png │ ├── qr.svg │ ├── right.svg │ ├── search.svg │ ├── settings.svg │ ├── settings_w.svg │ ├── shortcut-file.png │ ├── shortcuts_grey.png │ ├── success.svg │ ├── upload.png │ └── var-sheet.png ├── index.css ├── index.tsx ├── logo.svg ├── react-app-env.d.ts ├── serviceWorker.ts └── testshortcut.json └── tsconfig.json /.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | // http://eslint.org/docs/rules/ 3 | //"parser": "esprima", 4 | // "parser": "espree", 5 | parser: "@typescript-eslint/parser", 6 | /*"ecmaFeatures": { 7 | "binaryLiterals": true, // enable binary literals 8 | "blockBindings": true, // enable let and const (aka block bindings) 9 | "defaultParams": true, // enable default function parameters 10 | "forOf": true, // enable for-of loops 11 | "generators": true, // enable generators 12 | "objectLiteralComputedProperties": true, // enable computed object literal property names 13 | "objectLiteralDuplicateProperties": true, // enable duplicate object literal properties in strict mode 14 | "objectLiteralShorthandMethods": true, // enable object literal shorthand methods 15 | "objectLiteralShorthandProperties": true, // enable object literal shorthand properties 16 | "experimentalObjectRestSpread": true, // enable {...a} 17 | "octalLiterals": true, // enable octal literals 18 | "regexUFlag": true, // enable the regular expression u flag 19 | "regexYFlag": true, // enable the regular expression y flag 20 | "templateStrings": true, // enable template strings 21 | "unicodeCodePointEscapes": true, // enable code point escapes 22 | "jsx": true, // enable JSX 23 | "es6": true, 24 | "ecmascript6": true 25 | },*/ 26 | parserOptions: { 27 | ecmaVersion: 2018, 28 | sourceType: "module", 29 | ecmaFeatures: { 30 | jsx: true, 31 | modules: true 32 | }, 33 | jsx: true 34 | }, 35 | 36 | env: { 37 | browser: true, // browser global variables. 38 | node: true, // Node.js global variables and Node.js-specific rules. 39 | amd: false, // defines require() and define() as global variables as per the amd spec. 40 | mocha: false, // adds all of the Mocha testing global variables. 41 | jasmine: false, // adds all of the Jasmine testing global variables for version 1.3 and 2.0. 42 | phantomjs: false, // phantomjs global variables. 43 | jquery: false, // jquery global variables. 44 | prototypejs: false, // prototypejs global variables. 45 | shelljs: false, // shelljs global variables. 46 | es6: true 47 | }, 48 | 49 | globals: { 50 | // e.g. "angular": true 51 | }, 52 | 53 | plugins: [ 54 | // e.g. "react" (must run `npm install eslint-plugin-react` first) 55 | "@typescript-eslint" 56 | ], 57 | extends: ["eslint:recommended"], 58 | 59 | rules: { 60 | ////////// Possible Errors ////////// 61 | 62 | //"comma-dangle": 2, // disallow trailing commas in object literals 63 | "no-cond-assign": 0, // disallow assignment in conditional expressions 64 | "no-console": 1, // disallow use of console (off by default in the node environment) 65 | "no-constant-condition": 0, // disallow use of constant expressions in conditions 66 | "no-control-regex": 0, // disallow control characters in regular expressions 67 | "no-debugger": 0, // disallow use of debugger 68 | "no-dupe-keys": 0, // disallow duplicate keys when creating object literals 69 | "no-empty": 0, // disallow empty statements 70 | "no-empty-class": 0, // disallow the use of empty character classes in regular expressions 71 | "no-ex-assign": 0, // disallow assigning to the exception in a catch block 72 | "no-extra-boolean-cast": 0, // disallow double-negation boolean casts in a boolean context 73 | "no-extra-parens": 0, // disallow unnecessary parentheses (off by default) 74 | "no-extra-semi": 1, // disallow unnecessary semicolons 75 | "no-func-assign": 2, // disallow overwriting functions written as function declarations 76 | "no-inner-declarations": 0, // disallow function or variable declarations in nested blocks 77 | "no-invalid-regexp": 1, // disallow invalid regular expression strings in the RegExp constructor 78 | "no-irregular-whitespace": 1, // disallow irregular whitespace outside of strings and comments 79 | "no-negated-in-lhs": 2, // disallow negation of the left operand of an in expression 80 | "no-obj-calls": 0, // disallow the use of object properties of the global object (Math and JSON) as functions 81 | "no-regex-spaces": 0, // disallow multiple spaces in a regular expression literal 82 | "no-sparse-arrays": 0, // disallow sparse arrays 83 | "no-unreachable": 2, // disallow unreachable statements after a return, throw, continue, or break statement 84 | "use-isnan": 0, // disallow comparisons with the value NaN 85 | "valid-jsdoc": 0, // Ensure JSDoc comments are valid (off by default) 86 | "valid-typeof": 0, // Ensure that the results of typeof are compared against a valid string 87 | 88 | ////////// Best Practices ////////// 89 | 90 | "block-scoped-var": 0, // treat var statements as if they were block scoped (off by default) 91 | complexity: 0, // specify the maximum cyclomatic complexity allowed in a program (off by default) 92 | "consistent-return": 0, // require return statements to either always or never specify values 93 | curly: [1], // specify curly brace conventions for all control statements 94 | "default-case": 1, // require default case in switch statements (off by default) 95 | "dot-notation": [1, { allowPattern: "(^[A-Z]|^[a-z]+_[a-z_]+$)" }], 96 | eqeqeq: ["error", "always", { null: "never" }], // require the use of === and !== 97 | "guard-for-in": 0, // make sure for-in loops have an if statement (off by default) 98 | "no-alert": 0, // disallow the use of alert, confirm, and prompt 99 | "no-caller": 0, // disallow use of arguments.caller or arguments.callee 100 | "no-div-regex": 0, // disallow division operators explicitly at beginning of regular expression (off by default) 101 | "no-else-return": 2, // disallow else after a return in an if (off by default) 102 | "no-empty-label": 0, // disallow use of labels for anything other then loops and switches 103 | "no-eq-null": 2, // disallow comparisons to null without a type-checking operator (off by default) 104 | "no-eval": 2, // disallow use of eval() 105 | "no-extend-native": 0, // disallow adding to native types 106 | "no-extra-bind": 0, // disallow unnecessary function binding 107 | "no-fallthrough": 2, // disallow fallthrough of case statements 108 | "no-floating-decimal": 0, // disallow the use of leading or trailing decimal points in numeric literals (off by default) 109 | "no-implied-eval": 2, // disallow use of eval()-like methods 110 | "no-iterator": 0, // disallow usage of __iterator__ property 111 | "no-labels": 0, // disallow use of labeled statements 112 | "no-lone-blocks": 2, // disallow unnecessary nested blocks 113 | "no-loop-func": 0, // disallow creation of functions within loops 114 | "no-multi-spaces": 0, // disallow use of multiple spaces 115 | "no-multi-str": 0, // disallow use of multiline strings 116 | "no-native-reassign": 2, // disallow reassignments of native objects 117 | "no-new": 2, // disallow use of new operator when not part of the assignment or comparison 118 | "no-new-func": 2, // disallow use of new operator for Function object 119 | "no-new-wrappers": 2, // disallows creating new instances of String, Number, and Boolean 120 | "no-octal": 0, // disallow use of octal literals 121 | "no-octal-escape": 0, // disallow use of octal escape sequences in string literals, such as var foo = "Copyright \251"; 122 | "no-process-env": 0, // disallow use of process.env (off by default) 123 | "no-proto": 0, // disallow usage of __proto__ property 124 | "no-redeclare": 2, // disallow declaring the same variable more then once 125 | "no-return-assign": 0, // disallow use of assignment in return statement 126 | "no-script-url": 0, // disallow use of javascript: urls. 127 | "no-self-compare": 0, // disallow comparisons where both sides are exactly the same (off by default) 128 | "no-sequences": 0, // disallow use of comma operator 129 | "no-unused-expressions": 0, // disallow usage of expressions in statement position 130 | "no-void": 0, // disallow use of void operator (off by default) 131 | "no-warning-comments": 0, // disallow usage of configurable warning terms in comments, e.g. TODO or FIXME (off by default) 132 | "no-with": 0, // disallow use of the with statement 133 | radix: 2, // require use of the second argument for parseInt() (off by default) 134 | "vars-on-top": 0, // requires to declare all vars on top of their containing scope (off by default) 135 | "wrap-iife": 2, // require immediate function invocation to be wrapped in parentheses (off by default) 136 | yoda: 1, // require or disallow Yoda conditions 137 | 138 | // "max-len": 1, // there is no fixer for this 139 | 140 | ////////// Strict Mode ////////// 141 | 142 | "global-strict": 0, // (deprecated) require or disallow the "use strict" pragma in the global scope (off by default in the node environment) 143 | "no-extra-strict": 0, // (deprecated) disallow unnecessary use of "use strict"; when already in strict mode 144 | strict: 0, // controls location of Use Strict Directives 145 | 146 | ////////// Variables ////////// 147 | 148 | "no-catch-shadow": 2, // disallow the catch clause parameter name being the same as a variable in the outer scope (off by default in the node environment) 149 | "no-delete-var": 2, // disallow deletion of variables 150 | "no-label-var": 1, // disallow labels that share a name with a variable 151 | "no-shadow": 0, // disallow declaration of variables already declared in the outer scope 152 | "no-shadow-restricted-names": 2, // disallow shadowing of names such as arguments 153 | "no-undef": 2, // disallow use of undeclared variables unless mentioned in a /*global */ block 154 | "no-undef-init": 1, // disallow use of undefined when initializing variables 155 | "no-undefined": 0, // disallow use of undefined variable (off by default) 156 | "no-unused-vars": 0, // disallow declaration of variables that are not used in the code 157 | "no-use-before-define": 0, // disallow use of variables before they are defined 158 | 159 | ////////// Node.js ////////// 160 | 161 | "handle-callback-err": 1, // enforces error handling in callbacks (off by default) (on by default in the node environment) 162 | "no-mixed-requires": [1, { grouping: true, allowCall: true }], // disallow mixing regular variable and require declarations (off by default) (on by default in the node environment) 163 | "no-new-require": 0, // disallow use of new operator with the require function (off by default) (on by default in the node environment) 164 | "no-path-concat": 2, // disallow string concatenation with __dirname and __filename (off by default) (on by default in the node environment) 165 | "no-process-exit": 0, // disallow process.exit() (on by default in the node environment) 166 | "no-restricted-modules": 0, // restrict usage of specified node modules (off by default) 167 | "no-sync": 0, // disallow use of synchronous methods (off by default) 168 | 169 | ////////// Stylistic Issues ////////// 170 | 171 | //"brace-style": [1, "1tbs", {"allowSingleLine": true}], // enforce one true brace style (off by default) 172 | //"camelcase": 1, // require camel case names 173 | "comma-spacing": [1, { before: false, after: true }], // enforce spacing before and after comma 174 | "comma-style": 0, // enforce one true comma style (off by default) 175 | "consistent-this": 0, // enforces consistent naming when capturing the current execution context (off by default) 176 | "eol-last": 0, // enforce newline at the end of file, with no multiple empty lines 177 | "func-names": [1, "as-needed"], // require function expressions to have a name (off by default) 178 | "func-style": 0, // enforces use of function declarations or expressions (off by default) 179 | "key-spacing": [ 180 | 1, 181 | { beforeColon: false, afterColon: true, mode: "strict" } 182 | ], // enforces spacing between keys and values in object literal properties 183 | "max-nested-callbacks": 0, // specify the maximum depth callbacks can be nested (off by default) 184 | // "new-cap": 1, // require a capital letter for constructors // Disabled because it applies to modules as well -.- 185 | "new-cap": 0, 186 | // "new-parens": 2, 187 | "new-parens": 0, // disallow the omission of parentheses when invoking a constructor with no arguments 188 | "no-array-constructor": 2, // disallow use of the Array constructor 189 | "no-inline-comments": 0, // disallow comments inline after code (off by default) 190 | "no-lonely-if": 1, // disallow if as the only statement in an else block (off by default) 191 | "no-multiple-empty-lines": 0, // disallow multiple empty lines (off by default) 192 | "no-nested-ternary": 0, // disallow nested ternary expressions (off by default) 193 | "no-new-object": 1, // disallow use of the Object constructor 194 | "semi-spacing": [1, { before: false, after: true }], // disallow space before semicolon 195 | "no-spaced-func": 1, // disallow space between function identifier and application 196 | "no-ternary": 0, // disallow the use of ternary operators (off by default) 197 | "no-trailing-spaces": 0, // disallow trailing whitespace at the end of lines 198 | "no-underscore-dangle": 0, // disallow dangling underscores in identifiers 199 | "one-var": 0, // allow just one var statement per function (off by default) 200 | "operator-assignment": 2, // require assignment operator shorthand where possible or prohibit it entirely (off by default) 201 | "padded-blocks": [1, "never"], // enforce padding within blocks (off by default) 202 | "quote-props": [1, "as-needed"], // require quotes around object literal property names (off by default) 203 | quotes: [ 204 | 1, 205 | "double", 206 | { avoidEscape: true, allowTemplateLiterals: true } 207 | ], // specify whether double or single quotes should be used 208 | "prefer-template": 1, // prefer template strings 209 | semi: [1, "always"], // require or disallow use of semicolons instead of ASI 210 | "sort-vars": 0, // sort variables within the same declaration block (off by default) 211 | "space-before-function-paren": [1, "never"], // require a space after certain keywords (off by default) 212 | "space-before-blocks": [1, "always"], // require or disallow space before blocks (off by default) 213 | "space-in-parens": 0, // require or disallow spaces inside parentheses (off by default) 214 | "space-unary-ops": 0, // Require or disallow spaces before/after unary operators (words on by default, nonwords off by default) 215 | "prefer-const": 1, 216 | complexity: 1, 217 | "no-mixed-spaces-and-tabs": 0, 218 | 219 | ////////// ECMAScript 6 ////////// 220 | 221 | "no-var": 2, // require let or const instead of var (off by default) 222 | "generator-star-spacing": ["error", { before: true, after: false }] // enforce the position of the * in generator functions (off by default) 223 | } 224 | }; 225 | -------------------------------------------------------------------------------- /.gitattributes: -------------------------------------------------------------------------------- 1 | docs/**/* linguist-generated=true 2 | -------------------------------------------------------------------------------- /.github/workflows/pages.yml: -------------------------------------------------------------------------------- 1 | name: Build and Deploy 2 | on: 3 | push: 4 | branches: 5 | - no-files 6 | jobs: 7 | build-and-deploy: 8 | runs-on: ubuntu-latest 9 | steps: 10 | - name: Checkout 🛎️ 11 | uses: actions/checkout@v2.3.1 12 | with: 13 | fetch-depth: 1 14 | 15 | - name: Install and Build 🔧 16 | run: | 17 | yarn install 18 | yarn add sass 19 | env node_env=PRODUCTION yarn build 20 | - name: Deploy 🚀 21 | uses: JamesIves/github-pages-deploy-action@4.0.0 22 | with: 23 | branch: gh-pages # The branch the action should deploy to. 24 | folder: docs # The folder the action should deploy. 25 | clean: true # Automatically remove deleted files from the deploy branch 26 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # production 12 | /build 13 | /docs 14 | 15 | # misc 16 | .DS_Store 17 | .env.local 18 | .env.development.local 19 | .env.test.local 20 | .env.production.local 21 | 22 | npm-debug.log* 23 | yarn-debug.log* 24 | yarn-error.log* 25 | 26 | # junk folder 27 | /junk 28 | package-lock.json 29 | 30 | 31 | yarn.lock -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "printWidth": 80, 3 | "tabWidth": 4, 4 | "useTabs": true, 5 | "semi": true, 6 | "singleQuote": false, 7 | "trailingComma": "none", 8 | "bracketSpacing": true, 9 | "jsxBracketSameLine": false, 10 | "arrowParens": "avoid", 11 | "proseWrap": "always", 12 | "overrides": [ 13 | { 14 | "files": ".prettierrc", 15 | "options": { "parser": "json" } 16 | } 17 | ] 18 | } 19 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # scpl editor 2 | 3 | ![screenshot](https://i.imgur.com/hOXtMNh.png) 4 | 5 | Shortcuts are previewed using [shortcut-preview](https://github.com/xalien95/shortcut-preview). 6 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "scpl-editor", 3 | "version": "0.1.0", 4 | "private": true, 5 | "homepage": "https://scpl.dev/", 6 | "dependencies": { 7 | "browserfs": "^1.4.3", 8 | "pretty-bytes": "^5.1.0", 9 | "qrcode": "^1.3.3", 10 | "react": "^16.8.4", 11 | "react-ace": "^6.4.0", 12 | "react-dom": "^16.8.4", 13 | "react-dropzone": "^10.0.5", 14 | "react-helmet": "^5.2.0", 15 | "react-markdown": "^4.0.8", 16 | "react-portal": "^4.2.0", 17 | "react-scripts": "2.1.8", 18 | "scpl": "^1.9.3", 19 | "shortcut-preview": "pfgithub/shortcut-preview", 20 | "typescript": "3.3.3333", 21 | "yarn": "^1.17.3" 22 | }, 23 | "scripts": { 24 | "start": "react-scripts start", 25 | "build": "react-scripts build && rm -rf docs && mv build docs && echo \"editor.scpl.dev\" > docs/CNAME", 26 | "test": "react-scripts test", 27 | "eject": "react-scripts eject", 28 | "format": "prettier --write src/***.{css,tsx,ts,json} .prettierrc .eslintrc.js package.json" 29 | }, 30 | "eslintConfig": { 31 | "extends": "react-app" 32 | }, 33 | "browserslist": [ 34 | ">0.2%", 35 | "not dead", 36 | "not ie <= 11", 37 | "not op_mini all" 38 | ], 39 | "devDependencies": { 40 | "@types/jest": "24.0.11", 41 | "@types/node": "11.11.0", 42 | "@types/pretty-bytes": "^5.1.0", 43 | "@types/qrcode": "^1.3.3", 44 | "@types/react": "16.8.7", 45 | "@types/react-dom": "16.8.2", 46 | "@types/react-helmet": "^5.0.8", 47 | "@types/react-portal": "^4.0.2", 48 | "@typescript-eslint/eslint-plugin": "^1.4.2", 49 | "@typescript-eslint/parser": "^1.4.2", 50 | "onchange": "^5.2.0", 51 | "prettier": "^1.17.0" 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /public/android-chrome-192x192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/android-chrome-192x192.png -------------------------------------------------------------------------------- /public/android-chrome-512x512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/android-chrome-512x512.png -------------------------------------------------------------------------------- /public/apple-touch-icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/apple-touch-icon.png -------------------------------------------------------------------------------- /public/browserconfig.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | #1e1f57 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /public/favicon-16x16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/favicon-16x16.png -------------------------------------------------------------------------------- /public/favicon-32x32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/favicon-32x32.png -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/favicon.ico -------------------------------------------------------------------------------- /public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 20 | 21 | 30 | ScPL Editor 31 | 32 | 33 | 34 | 35 |
36 | 46 | 47 | 48 | 49 | -------------------------------------------------------------------------------- /public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "React App", 3 | "name": "Create React App Sample", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | } 10 | ], 11 | "start_url": ".", 12 | "display": "standalone", 13 | "theme_color": "#000000", 14 | "background_color": "#ffffff" 15 | } 16 | -------------------------------------------------------------------------------- /public/mstile-150x150.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/public/mstile-150x150.png -------------------------------------------------------------------------------- /public/safari-pinned-tab.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | 7 | 8 | Created by potrace 1.11, written by Peter Selinger 2001-2013 9 | 10 | 12 | 22 | 31 | 55 | 72 | 78 | 79 | 80 | -------------------------------------------------------------------------------- /public/site.webmanifest: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ScPL Editor", 3 | "short_name": "ScPL Editor", 4 | "icons": [ 5 | { 6 | "src": "/android-chrome-192x192.png", 7 | "sizes": "192x192", 8 | "type": "image/png" 9 | }, 10 | { 11 | "src": "/android-chrome-512x512.png", 12 | "sizes": "512x512", 13 | "type": "image/png" 14 | } 15 | ], 16 | "theme_color": "#1e1f57", 17 | "background_color": "#1e1f57", 18 | "display": "standalone" 19 | } 20 | -------------------------------------------------------------------------------- /src/App.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | import { parse, PositionedError } from "scpl"; 3 | import "./App.css"; 4 | // import {Helmet} from "react-helmet"; 5 | 6 | import testshortcut from "./testshortcut.json"; 7 | 8 | import ace from "brace"; 9 | import "./ace/mode-scpl"; 10 | import "brace/theme/chrome"; 11 | import "brace/ext/language_tools"; 12 | import "brace/ext/searchbox"; 13 | import AceEditor from "react-ace"; 14 | 15 | import { FilePane } from "./FilePane"; 16 | import { SearchActions } from "./SearchActions"; 17 | import { DownloadModal } from "./DownloadModal"; 18 | import { keys } from "./Key"; 19 | import { UploadShortcutModal } from "./UploadShortcutModal"; 20 | import { ErrorBoundary } from "./ErrorBoundary"; 21 | 22 | import ShortcutPreview from "shortcut-preview"; 23 | 24 | let timeout: NodeJS.Timeout; 25 | 26 | const Range = ace.acequire("ace/range").Range; 27 | 28 | const langTools = ace.acequire("ace/ext/language_tools"); 29 | const findAndReplace = ace.acequire("ace/ext/searchbox"); 30 | 31 | const rhymeCompleter = { 32 | getCompletions: ( 33 | editor: ace.Editor, 34 | session: ace.IEditSession, 35 | pos: ace.Position, 36 | prefix: string, 37 | callback: ( 38 | thing: null, 39 | completions: { 40 | name: string; 41 | value: string; 42 | score: number; 43 | meta: string; 44 | }[] 45 | ) => void 46 | ) => { 47 | if (prefix.length === 0) { 48 | callback(null, []); 49 | return; 50 | } 51 | const variables = session.getValue().match(/(v|mv):([A-Za-z0-9]+)/g); 52 | console.log(variables); 53 | if (!variables) { 54 | callback(null, []); 55 | return; 56 | } 57 | callback( 58 | null, 59 | variables.map(v => ({ 60 | name: v, 61 | value: v, 62 | score: 25, 63 | meta: v 64 | })) 65 | ); 66 | } 67 | }; 68 | langTools.addCompleter(rhymeCompleter); 69 | 70 | const hotkey = 71 | window.navigator.platform === "MacIntel" || 72 | window.navigator.platform.includes("Win") 73 | ? "⌘" 74 | : "^"; 75 | 76 | class MaybeUpdate extends Component<{ shouldUpdate: boolean }, {}> { 77 | shouldComponentUpdate(nextProps: { shouldUpdate: boolean }) { 78 | return nextProps.shouldUpdate; 79 | } 80 | render() { 81 | return this.props.children; 82 | } 83 | } 84 | 85 | class App extends Component< 86 | {}, 87 | { 88 | fileValue: string; 89 | shortcutData: any; 90 | shortcutDownload: Buffer | undefined; 91 | annotations: Array; 92 | markers: Array<{ 93 | startRow: number; 94 | endRow: number; 95 | startCol: number; 96 | endCol: number; 97 | className: string; 98 | type: string; 99 | }>; 100 | errors: Array<{ 101 | startRow: number; 102 | endRow: number; 103 | startCol: number; 104 | endCol: number; 105 | message: string; 106 | }>; 107 | loading: boolean; 108 | took: { waitedFor: number; convertedIn: number }; 109 | mobileFilemenu: boolean; 110 | openDownload: boolean; 111 | showPreview: boolean; 112 | showPreviewFullscreen: boolean; 113 | showUploadShortcutModal: boolean; 114 | tabs: { filename: string; active: boolean }[]; 115 | files: { 116 | [filename: string]: string; 117 | }; 118 | } 119 | > { 120 | reactAceComponentRef: React.RefObject; 121 | constructor(props: Readonly<{}>) { 122 | super(props); 123 | this.state = { 124 | fileValue: "...", 125 | shortcutData: testshortcut, 126 | shortcutDownload: undefined, 127 | annotations: [], 128 | markers: [], 129 | loading: false, 130 | took: { waitedFor: 0, convertedIn: 0 }, 131 | mobileFilemenu: false, 132 | openDownload: false, 133 | showPreview: false, 134 | showPreviewFullscreen: false, 135 | showUploadShortcutModal: false, 136 | errors: [], 137 | tabs: [{ filename: "download.scpl", active: true }], 138 | files: { 139 | "download.scpl": `ShowResult "Hello ScPL" 140 | ChooseFromMenu "ScPL Editor" items=["Getting Started", "View Documentation"] 141 | Case "Getting Started" 142 | URL "https://docs.scpl.dev/gettingstarted" 143 | Case "View Documentation" 144 | URL "https://docs.scpl.dev/" 145 | End Menu 146 | OpenURLs`, 147 | "other.scpl": `ShowResult "ScPL"`, 148 | urlquery: 149 | new URLSearchParams(window.location.search).get("scpl") || 150 | "" 151 | } 152 | }; 153 | this.reactAceComponentRef = React.createRef(); 154 | } 155 | componentDidMount() {} 156 | componentWillMount() { 157 | const urlParams = new URLSearchParams(window.location.search); 158 | this.onChange( 159 | new URLSearchParams(window.location.search).get("scpl") || 160 | `ShowResult "Hello ScPL" 161 | ChooseFromMenu "ScPL Editor" items=["Getting Started", "View Documentation"] 162 | Case "Getting Started" 163 | URL "https://docs.scpl.dev/gettingstarted" 164 | Case "View Documentation" 165 | URL "https://docs.scpl.dev/" 166 | End Menu 167 | OpenURLs` 168 | ); 169 | } 170 | render() { 171 | return ( 172 |
{ 174 | if (keys.save(e)) { 175 | alert("save"); 176 | } 177 | if (keys.export(e)) { 178 | this.setState({ openDownload: true }); 179 | } 180 | }} 181 | > 182 |
183 |
Drop file anywhere to upload
184 |
185 | {this.state.openDownload ? ( 186 | { 188 | this.setState({ openDownload: false }); 189 | }} 190 | filename={ 191 | this.state.shortcutData._filename || 192 | "download.shortcut" 193 | } 194 | file={this.state.shortcutDownload} 195 | /> 196 | ) : null} 197 | {this.state.showUploadShortcutModal ? ( 198 | 200 | this.setState({ showUploadShortcutModal: false }) 201 | } 202 | onResult={result => { 203 | this.setState({ showUploadShortcutModal: false }); 204 | this.onChange(result); 205 | }} 206 | /> 207 | ) : null} 208 | 209 |
210 |
216 | this.setState({ 217 | mobileFilemenu: !this.state.mobileFilemenu 218 | }) 219 | } 220 | /> 221 |
222 |
ScPL Editor
223 | 257 |
258 | 357 |
358 |
359 | 390 |
391 |
392 |
393 | { 395 | console.log("insert", text); 396 | const reactAceComponent = this 397 | .reactAceComponentRef.current; 398 | const editor = (reactAceComponent as any) 399 | .editor as ace.Editor; 400 | editor.session.insert( 401 | editor.getCursorPosition(), 402 | text 403 | ); 404 | console.log( 405 | editor.session.getValue(), 406 | text, 407 | editor.getCursorPosition() 408 | ); 409 | this.setState({ 410 | fileValue: editor.session.getValue() 411 | }); 412 | }} 413 | /> 414 |
415 |
416 |
417 |
418 | {this.state.shortcutData && 419 | this.state.shortcutData[0].WFWorkflowActions 420 | .length}{" "} 421 | action 422 | {this.state.shortcutData && 423 | this.state.shortcutData[0].WFWorkflowActions 424 | .length === 1 425 | ? "" 426 | : "s"} 427 |
428 |
429 | 430 | 445 |
449 | this.setState({ 450 | openDownload: true 451 | }) 452 | } 453 | > 454 | Export 455 |
456 |
457 |
458 |
459 |
464 | this.onChange(file)} 508 | /> 509 |
510 |
511 |
512 | {this.state.errors.map(err => ( 513 |
514 |
515 | {err.message} 516 |
517 |
518 | 525 |
526 |
527 | ))} 528 |
529 |
530 | {this.state.tabs.length > 1 ? ( 531 |
532 | {this.state.tabs.map(tab => ( 533 |
538 |
539 | × 540 |
541 |
542 | {tab.filename} 543 |
544 |
545 | ))} 546 |
547 | ) : null} 548 |
549 | 563 |
564 |
578 |
579 | Converted in {this.state.took.convertedIn} ms. 580 |
581 | {this.state.showPreview && this.state.shortcutData ? ( 582 | ( 584 |
585 |

586 | A fatal error occured in 587 | shortcut-preview. The error is{" "} 588 | {err.toString()} 589 |

590 |
591 | )} 592 | > 593 | 595 | this.onActionSelect(data) 596 | } 597 | data={this.state.shortcutData} 598 | /> 599 |
600 | ) : null} 601 |
602 |
603 |
604 |
605 |
606 |
607 |
608 |
609 |
610 |
611 |
612 |
613 |
614 |
615 |
616 |
617 |
618 |
619 | {!this.state.showPreview ? ( 620 |
621 |
622 |

623 | There are too many actions to render a 624 | preview automatically. 625 |

626 | 636 |
637 |
638 | ) : null} 639 | {!this.state.shortcutData ? ( 640 |
641 |
642 |

643 | Failed to render because of code error. 644 |

645 |
646 |
647 | ) : null} 648 |
649 |
650 |
651 | ); 652 | } 653 | onActionSelect(data: { type: "action" | "parameter"; actionData: any }) { 654 | if (data.actionData.SCPLData) { 655 | const scpldata = data.actionData.SCPLData; 656 | 657 | const reactAceComponent = this.reactAceComponentRef.current; 658 | if (!reactAceComponent) { 659 | return; 660 | } 661 | const editor = (reactAceComponent as any).editor as ace.Editor; 662 | // .Position.start|end 663 | const line = scpldata.Position.start[0]; 664 | const col = scpldata.Position.start[1] - 1; 665 | editor.gotoLine(line, col, true); 666 | editor.selection.setRange( 667 | new Range( 668 | scpldata.Position.start[0] - 1, 669 | scpldata.Position.start[1] - 1, 670 | scpldata.Position.end[0] - 1, 671 | scpldata.Position.end[1] - 1 672 | ), 673 | true 674 | ); 675 | } 676 | } 677 | getAce(): ace.Editor { 678 | const reactAceComponent = this.reactAceComponentRef.current; 679 | if (!reactAceComponent) { 680 | throw new Error("No reactacecomponent"); 681 | } 682 | const editor = (reactAceComponent as any).editor as ace.Editor; 683 | return editor; 684 | } 685 | jumpToError(d: { 686 | startCol: number; 687 | startRow: number; 688 | endCol: number; 689 | endRow: number; 690 | }) { 691 | const reactAceComponent = this.reactAceComponentRef.current; 692 | if (!reactAceComponent) { 693 | return; 694 | } 695 | const editor = (reactAceComponent as any).editor as ace.Editor; 696 | const line = d.startRow; 697 | const col = d.startCol - 1; 698 | editor.gotoLine(line, col, true); 699 | editor.selection.setRange( 700 | new Range( 701 | d.startRow - 1, 702 | d.startCol - 1, 703 | d.endRow - 1, 704 | d.endCol - 1 705 | ), 706 | true 707 | ); 708 | } 709 | onChange(text: string) { 710 | // Enable navigation prompt 711 | window.onbeforeunload = () => { 712 | return true; 713 | }; 714 | const willWaitFor = Math.max(this.state.took.convertedIn * 4, 100); 715 | if (!this.state.loading) { 716 | this.setState({ 717 | showPreview: this.state.took.convertedIn < 50, 718 | fileValue: text, 719 | loading: true 720 | }); 721 | } 722 | if (timeout) { 723 | clearTimeout(timeout); 724 | } 725 | timeout = setTimeout( 726 | () => this.onChangeLimited(text, willWaitFor), 727 | willWaitFor 728 | ); 729 | } 730 | onChangeLimited(text: string, waitedFor: number) { 731 | // parse 732 | let output: { 733 | shortcutjson: any; 734 | shortcutplist: Buffer; 735 | warnings: PositionedError[]; 736 | }; 737 | const annotationFromError = ( 738 | err: PositionedError, 739 | isWarning: boolean 740 | ) => { 741 | return { 742 | row: err.start[0] - 1, 743 | column: err.start[1] - 1, 744 | text: err.message, // Or the Json reply from the parser 745 | type: isWarning ? "warning" : "error" 746 | }; 747 | }; 748 | const markerFromError = (er: PositionedError, isWarning: boolean) => { 749 | return { 750 | startRow: er.start[0] - 1, 751 | endRow: er.end[0] - 1, 752 | startCol: er.start[1] - 1, 753 | endCol: er.end[1] - 1, 754 | className: `ace_active-line ${isWarning ? "warning" : "error"}`, 755 | type: "background" 756 | }; 757 | }; 758 | const errorBannerFromError = ( 759 | er: PositionedError, 760 | isWarning: boolean 761 | ) => { 762 | return { 763 | startRow: er.start[0], 764 | endRow: er.end[0], 765 | startCol: er.start[1], 766 | endCol: er.end[1], 767 | message: er.message, 768 | type: isWarning ? "warning" : "error" 769 | }; 770 | }; 771 | const startTime = new Date().getTime(); 772 | try { 773 | output = parse(text, { 774 | make: ["shortcutjson", "shortcutplist"], 775 | useWarnings: true 776 | }); 777 | } catch (er) { 778 | console.log(er.message); //eslint-disable-line no-console 779 | if (!(er instanceof PositionedError)) { 780 | throw er; 781 | } 782 | this.setState({ 783 | fileValue: text, 784 | loading: false, 785 | shortcutDownload: undefined, 786 | shortcutData: undefined, 787 | took: { 788 | waitedFor: waitedFor, 789 | convertedIn: new Date().getTime() - startTime 790 | }, 791 | annotations: [annotationFromError(er, false)], 792 | markers: [markerFromError(er, false)], 793 | errors: [errorBannerFromError(er, false)] 794 | }); 795 | return; 796 | } 797 | const { shortcutjson, shortcutplist } = output; 798 | this.setState({ 799 | fileValue: text, 800 | took: { 801 | waitedFor: waitedFor, 802 | convertedIn: new Date().getTime() - startTime 803 | }, 804 | loading: false, 805 | shortcutData: shortcutjson, 806 | annotations: [ 807 | ...output.warnings.map(warn => annotationFromError(warn, false)) 808 | ], 809 | markers: [ 810 | ...output.warnings.map(warn => markerFromError(warn, false)) 811 | ], 812 | errors: [ 813 | ...output.warnings.map(warn => 814 | errorBannerFromError(warn, false) 815 | ) 816 | ], 817 | shortcutDownload: shortcutplist 818 | }); 819 | } 820 | } 821 | 822 | export default App; 823 | -------------------------------------------------------------------------------- /src/CreateEditShortcut.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | 3 | import { 4 | GlyphName, 5 | objectGlyphs, 6 | peopleGlyphs, 7 | symbolGlyphs, 8 | ColorName, 9 | colors 10 | } from "./data/ShortcutMeta"; 11 | import { ModalContainer } from "./ModalContainer"; 12 | 13 | import "./CreateEditShortcut.css"; 14 | 15 | type CreateEditShortcutProps = { 16 | onCancel: () => void; 17 | onResult: (name: string, color: undefined, glyph: undefined) => void; 18 | }; 19 | 20 | export class CreateEditShortcut extends Component< 21 | CreateEditShortcutProps, 22 | { chosenGlyph: GlyphName; chosenColor: ColorName; mode: "color" | "glyph" } 23 | > { 24 | constructor(props: Readonly) { 25 | super(props); 26 | this.state = { 27 | chosenGlyph: "magicwand", 28 | chosenColor: "purple", 29 | mode: "color" 30 | }; 31 | } 32 | render() { 33 | return ( 34 | this.props.onCancel()}> 35 |
40 |
this.props.onCancel()} 44 | > 45 | Cancel 46 |
47 |

New ScPL File

48 |
49 |
50 |
54 |
58 |
59 |
60 |
61 | 67 |
72 | Name is required. 73 |
74 |
75 |
76 |
77 |
78 | 87 | 96 |
97 | 98 |
99 |
108 |
109 |
Objects
110 | 111 | {objectGlyphs.map(id => ( 112 |
113 | 122 | this.setState({ 123 | chosenGlyph: id 124 | }) 125 | } 126 | /> 127 |
129 | ))} 130 |
131 | 132 |
133 |
People
134 | 135 | {peopleGlyphs.map(id => ( 136 |
137 | 146 | this.setState({ 147 | chosenGlyph: id 148 | }) 149 | } 150 | /> 151 |
153 | ))} 154 |
155 | 156 |
157 |
Symbols
158 | 159 | {symbolGlyphs.map(id => ( 160 |
161 | 170 | this.setState({ 171 | chosenGlyph: id 172 | }) 173 | } 174 | /> 175 |
177 | ))} 178 |
179 |
180 |
181 | 182 |
190 |
191 | {colors.map(color => ( 192 |
193 | 201 | this.setState({ 202 | chosenColor: color 203 | }) 204 | } 205 | /> 206 |
208 | ))} 209 |
210 |
211 | 212 |
213 | Create 214 |
215 |
216 | 217 | ); 218 | } 219 | } 220 | -------------------------------------------------------------------------------- /src/DownloadButton.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | 3 | export class DownloadButton extends Component< 4 | { filename: string; file: Buffer | undefined }, 5 | {} 6 | > { 7 | // from https://github.com/axetroy/react-download 8 | render() { 9 | return ( 10 | this.onClick(e)} 14 | > 15 | {this.props.children} 16 | 17 | ); 18 | } 19 | fakeClick(obj: HTMLAnchorElement) { 20 | const ev = document.createEvent("MouseEvents"); 21 | ev.initMouseEvent( 22 | "click", 23 | true, 24 | false, 25 | window, 26 | 0, 27 | 0, 28 | 0, 29 | 0, 30 | 0, 31 | false, 32 | false, 33 | false, 34 | false, 35 | 0, 36 | null 37 | ); 38 | obj.dispatchEvent(ev); 39 | } 40 | exportRaw(name: string, data: Buffer) { 41 | const urlObject = window.URL || (window as any).webkitURL || window; 42 | const export_blob = new Blob([data]); 43 | 44 | if ("msSaveBlob" in navigator) { 45 | // Prefer msSaveBlob if available - Edge supports a[download] but 46 | // ignores the filename provided, using the blob UUID instead. 47 | // msSaveBlob will respect the provided filename 48 | navigator.msSaveBlob(export_blob, name); 49 | } else if ("download" in HTMLAnchorElement.prototype) { 50 | const save_link = document.createElementNS( 51 | "http://www.w3.org/1999/xhtml", 52 | "a" 53 | ) as HTMLAnchorElement; 54 | save_link.href = urlObject.createObjectURL(export_blob); 55 | save_link.download = name; 56 | this.fakeClick(save_link); 57 | } else { 58 | alert( 59 | "Downloading shortcuts is not available on this browser yet :(\nIt should be implemented within a few days from a few days from a while from now." 60 | ); 61 | } 62 | } 63 | onClick(_e: React.MouseEvent) { 64 | if (!this.props.file) { 65 | return; 66 | } 67 | this.exportRaw(this.props.filename, this.props.file); 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /src/DownloadModal.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | 3 | import * as QRCode from "qrcode"; 4 | 5 | import { ModalContainer } from "./ModalContainer"; 6 | import { DownloadButton } from "./DownloadButton"; 7 | 8 | import prettyBytes from "pretty-bytes"; 9 | 10 | import shortcutDownloadPreviewIcon from "./img/shortcut-file.png"; 11 | import { uploadShortcut } from "./Uploader"; 12 | import { DownloadStatusSwitcher, UploadStatus } from "./DownloadStatusSwitcher"; 13 | 14 | import "./CreateEditShortcut.css"; 15 | 16 | export class ShortcutDownloadPreviewButton extends Component< 17 | { filename: string; filesize: number }, 18 | {} 19 | > { 20 | render() { 21 | return ( 22 |
23 | 24 |
{this.props.filename}
25 |
26 | {prettyBytes(this.props.filesize)} 27 |
28 |
29 | ); 30 | } 31 | } 32 | 33 | type DownloadModalProps = { 34 | onCancel: () => void; 35 | filename: string; 36 | file: Buffer | undefined; 37 | }; 38 | 39 | export class DownloadModal extends Component { 40 | constructor(props: Readonly) { 41 | super(props); 42 | this.state = { uploadStatus: "None" }; 43 | } 44 | componentWillReceiveProps(nextProps: Readonly) { 45 | if (nextProps.file !== this.props.file) { 46 | this.setState({ uploadStatus: "None" }); 47 | } 48 | } 49 | async uploadFile() { 50 | if (!this.props.file) { 51 | this.setState({ 52 | uploadStatus: "Error", 53 | uploadError: "No file. Make sure your shortcut has no errors." 54 | }); 55 | return; 56 | } 57 | this.setState({ 58 | uploadStatus: "Uploading" 59 | }); 60 | const uploadResult = await uploadShortcut( 61 | this.props.file, 62 | this.props.filename 63 | ); 64 | if (uploadResult.result === "error") { 65 | this.setState({ 66 | uploadStatus: "Error", 67 | uploadError: uploadResult.message 68 | }); 69 | return; 70 | } 71 | const qrcode = await QRCode.toDataURL(uploadResult.url); 72 | this.setState({ 73 | uploadStatus: "URL", 74 | uploadedURL: uploadResult.url, 75 | qrcode: qrcode 76 | }); 77 | } 78 | render() { 79 | return ( 80 | this.props.onCancel()}> 81 |
e.stopPropagation()} 88 | > 89 |

Export Shortcut File

90 | {this.props.file ? ( 91 |
92 |
93 | {"download" in HTMLAnchorElement.prototype ? ( 94 | 98 | 104 | 105 | ) : ( 106 | this.uploadFile()} 109 | detailsMsg={ 110 |
111 | To download on this browser, 112 | ScPL editor will upload your 113 | shortcut to{" "} 114 | 119 | file.io 120 | 121 | . 122 |
123 | } 124 | uploadAction="Create Download Link" 125 | > 126 | 133 | 139 | 140 |
141 | )} 142 |
143 |
or
144 |
145 | this.uploadFile()} 148 | detailsMsg={ 149 |
150 | QR codes must be 151 | scanned using{" "} 152 | 157 | This Shortcut 158 | 159 | {" "}due to new security 160 | restrictions in iOS 13.{" "} 161 | To generate a QR code, ScPL editor 162 | will upload your shortcut to{" "} 163 | 168 | file.io 169 | 170 | . 171 |
172 | } 173 | uploadAction="Generate QR Code" 174 | additionalButtonClasses="qr-btn" 175 | > 176 |
177 |

178 | Add to your library via QR Code: 179 |

180 |
181 |
190 |
191 |

192 | Open your Camera app and point it 193 | steady for 2-3 seconds at this QR 194 | Code. 195 |

196 |
197 | 198 |
199 |
200 | ) : ( 201 |
202 |

Your code seems to have some errors.

203 |
204 | )} 205 |
206 |
this.props.onCancel()} 210 | > 211 | Done 212 |
213 |
214 | 215 | ); 216 | } 217 | } 218 | -------------------------------------------------------------------------------- /src/DownloadStatusSwitcher.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | 3 | export type UploadStatus = 4 | | { 5 | uploadStatus: "None" | "Uploading"; 6 | } 7 | | { 8 | uploadStatus: "URL"; 9 | uploadedURL: string; 10 | qrcode: string; 11 | } 12 | | { 13 | uploadStatus: "Error"; 14 | uploadError: string; 15 | }; 16 | 17 | export class DownloadStatusSwitcher extends Component<{ 18 | status: UploadStatus; 19 | requestUpload: () => void; 20 | detailsMsg: React.ReactNode; 21 | uploadAction: string; 22 | additionalButtonClasses?: string; 23 | }> { 24 | render() { 25 | switch (this.props.status.uploadStatus) { 26 | case "None": 27 | return ( 28 |
29 |

30 | 37 |

{this.props.detailsMsg}

38 |
39 | ); 40 | case "URL": 41 | return this.props.children; 42 | case "Uploading": 43 | return ( 44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |

Uploading...

61 |
62 |
63 | ); 64 | case "Error": 65 | return ( 66 |
67 |

Error uploading: {this.props.status.uploadError}

68 |
69 | ); 70 | default: 71 | return ( 72 |
73 |

Something bad happened.

74 |
75 | ); 76 | } 77 | } 78 | } 79 | -------------------------------------------------------------------------------- /src/ErrorBoundary.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | 3 | type ErrorBoundaryProps = { 4 | errorDisplay: (error: Error) => React.ReactNode; 5 | }; 6 | export class ErrorBoundary extends Component< 7 | ErrorBoundaryProps, 8 | { error: Error | undefined } 9 | > { 10 | constructor(props: Readonly) { 11 | super(props); 12 | this.state = { error: undefined }; 13 | } 14 | componentDidCatch(error: Error) { 15 | this.setState({ error: error }); 16 | } 17 | render() { 18 | if (this.state.error) { 19 | console.log(this.state.error); 20 | return this.props.errorDisplay(this.state.error); 21 | } 22 | return this.props.children; 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /src/FileManager.ts: -------------------------------------------------------------------------------- 1 | export class FileManager { 2 | constructor() {} 3 | listItems() { 4 | // returns a list of files and folders 5 | } 6 | createFolder() {} 7 | saveFile(path: string, data: string) { 8 | // saves the file at the path 9 | } 10 | loadFile(path: string, data: string) {} 11 | renameFile(path: string, newPath: string) {} 12 | } 13 | -------------------------------------------------------------------------------- /src/FilePane.css: -------------------------------------------------------------------------------- 1 | /* File pane */ 2 | .file-pane { 3 | width: 300px; 4 | background-color: #fff; 5 | border-right: 0.5px solid #c7c7cc; 6 | transition: 0.1s left ease-out; 7 | z-index: 100; 8 | padding-bottom: 20px; 9 | overflow-x: hidden; 10 | overflow-y: auto; 11 | height: stretch; 12 | } 13 | .file-pane input { 14 | width: 90%; 15 | margin-left: 15px; 16 | } 17 | .file-pane h2 { 18 | margin: 20px 15px; 19 | } 20 | /* Files Navigation */ 21 | .files-header { 22 | position: sticky; 23 | top: 0; 24 | background-color: #fafafa; 25 | height: 55px; 26 | border-bottom: 1px solid #ddd; 27 | margin-bottom: 10px; 28 | padding: 15px 0; 29 | box-shadow: 0px -5px 20px rgba(0, 0, 0, 0.15); 30 | } 31 | .file-btns { 32 | display: grid; 33 | grid-template-columns: 1fr 1fr 1fr; 34 | padding: 0 10px; 35 | } 36 | .file-btn { 37 | font-size: 15px; 38 | padding: 13px 0; 39 | width: 100%; 40 | width: stretch; 41 | padding-left: 30px; 42 | font-weight: normal; 43 | outline: none; 44 | } 45 | .upload-btn { 46 | background: #eee url(img/upload.png) center / 25px no-repeat; 47 | } 48 | .large-upload-btn { 49 | background: #eee url(img/upload.png) 22% center / 25px no-repeat; 50 | padding-left: 10%; 51 | } 52 | .new-btn { 53 | background: #eee url(img/new_file.png) center / 25px no-repeat; 54 | } 55 | .newf-btn { 56 | background: #eee url(img/new_folder.png) center / 25px no-repeat; 57 | } 58 | /* File list */ 59 | .file-list li { 60 | padding: 8px 10px; 61 | padding-left: 25px; 62 | cursor: pointer; 63 | user-select: none; 64 | transition: 0.3s background-color; 65 | } 66 | .file-list li:hover:not(.open-folder) { 67 | background-color: #e8f6fe !important; 68 | } 69 | .file-list li:active:not(.open-folder) { 70 | background-color: #daeffc !important; 71 | } 72 | .file-list li.open-folder > ul > li { 73 | width: 94%; 74 | } 75 | .file-list ul { 76 | list-style-type: none; 77 | padding: 0; 78 | margin: 0; 79 | } 80 | .file-list ul ul { 81 | border-left: 1px solid #ddd; 82 | margin: 5px 10px; 83 | } 84 | .list-item-file > div, 85 | .list-item-folder > div { 86 | padding-left: 27px; 87 | font-size: 14px; 88 | } 89 | .item-name { 90 | width: 70%; 91 | height: 18px; 92 | white-space: nowrap; 93 | overflow: hidden; 94 | text-overflow: ellipsis; 95 | display: inline-block; 96 | margin-top: 3px; 97 | } 98 | .item-name .load { 99 | display: none; 100 | } 101 | .file-list ul li.open-folder ul li > div > .item-name { 102 | width: 60%; 103 | } 104 | .file-list ul li.open-folder ul li.open-folder ul > li > div > .item-name { 105 | width: 50%; 106 | } 107 | /* File items */ 108 | .list-item-file > div { 109 | background: url(img/file.png) 2px center / 18px no-repeat; 110 | } 111 | .file-list li.list-item-file.active { 112 | background-color: #007aff20; 113 | font-weight: 500; 114 | } 115 | .file-list li.list-item-file.active:hover { 116 | background-color: #007aff10 !important; 117 | } 118 | .file-list li.list-item-file.active > div { 119 | background-image: url(img/file_active.png); 120 | } 121 | /* uploading state */ 122 | .list-item-file.uploading { 123 | background: none; 124 | pointer-events: none; 125 | user-select: none; 126 | color: #777; 127 | } 128 | .list-item-file.uploading .action-btns { 129 | opacity: 0.5; 130 | } 131 | .list-item-file.uploading > div { 132 | background: none; 133 | } 134 | .list-item-file.uploading .item-name .load { 135 | display: inline-block; 136 | position: absolute; 137 | margin-left: -25px; 138 | margin-top: 2px; 139 | width: 10px; 140 | height: 10px; 141 | } 142 | /* Folder items */ 143 | .list-item-folder > div { 144 | background: url(img/folder.png) left center / 20px no-repeat; 145 | } 146 | .file-list li.list-item-folder > ul { 147 | display: none; 148 | transform-origin: top center; 149 | transition: 0.1s left ease-out; 150 | transform: scale(1, 0); 151 | } 152 | .file-list li.list-item-folder { 153 | background: url(img/right.svg) 10px 15px / 10px no-repeat; 154 | } 155 | .file-list li.list-item-folder.open-folder { 156 | background: url(img/down.svg) 10px 15px / 10px no-repeat; 157 | padding-bottom: 0; 158 | } 159 | .file-list li.list-item-folder.open-folder > ul { 160 | display: block; 161 | animation: scaleIn 0.1s; 162 | transform: scale(1, 1); 163 | } 164 | .list-item-folder.open-folder > div { 165 | font-weight: 500; 166 | } 167 | @keyframes scaleIn { 168 | from { 169 | transform: scale(1, 0); 170 | } 171 | to { 172 | transform: scale(1, 1); 173 | } 174 | } 175 | /* Rename and delete btns */ 176 | .action-btns { 177 | float: right; 178 | margin-top: -3px; 179 | } 180 | .file-list ul ul ul .action-btns { 181 | margin-right: 3px; 182 | } 183 | .edit-btn, 184 | .delete-btn { 185 | background: url() center / 20px no-repeat; 186 | padding: 15px; 187 | float: right; 188 | border-radius: 3px; 189 | cursor: pointer; 190 | } 191 | .action-btns div:active { 192 | background-color: #8c99b3; 193 | } 194 | .edit-btn { 195 | background-image: url(img/edit.svg); 196 | } 197 | .edit-btn:active { 198 | background-image: url(img/edit_w.svg); 199 | } 200 | .delete-btn { 201 | background-image: url(img/delete.svg); 202 | margin-left: 3px; 203 | } 204 | .delete-btn:active { 205 | background-image: url(img/delete_w.svg); 206 | } 207 | -------------------------------------------------------------------------------- /src/FilePane.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | import Dropzone, { DropEvent } from "react-dropzone"; 3 | import { inverse } from "scpl"; 4 | 5 | import { CreateEditShortcut } from "./CreateEditShortcut"; 6 | 7 | import "./FilePane.css"; 8 | 9 | type FileData = { type: "file"; name: string }; 10 | type FolderData = { 11 | type: "folder"; 12 | name: string; 13 | files: (FileData | FolderData)[]; 14 | }; 15 | 16 | class ActionButtons extends Component<{}> { 17 | render() { 18 | return ( 19 |
20 |
{ 23 | e.stopPropagation(); 24 | }} 25 | /> 26 |
{ 29 | e.stopPropagation(); 30 | }} 31 | /> 32 |
33 | ); 34 | } 35 | } 36 | 37 | class FileComponent extends Component<{ 38 | data: FileData; 39 | }> { 40 | render() { 41 | return ( 42 |
  • { 45 | e.stopPropagation(); 46 | }} 47 | title={this.props.data.name} 48 | > 49 |
    50 |
    51 |
    52 | {this.props.data.name} 53 |
    54 | 55 |
    56 |
  • 57 | ); 58 | } 59 | } 60 | 61 | class FolderComponent extends Component< 62 | { 63 | data: FolderData; 64 | searchTerm: string; 65 | }, 66 | { 67 | expanded: boolean; 68 | } 69 | > { 70 | constructor( 71 | props: Readonly<{ 72 | data: FolderData; 73 | searchTerm: string; 74 | }> 75 | ) { 76 | super(props); 77 | this.state = { expanded: false }; 78 | } 79 | render() { 80 | return ( 81 |
  • { 86 | this.setState({ expanded: !this.state.expanded }); 87 | e.stopPropagation(); 88 | }} 89 | title={this.props.data.name} 90 | > 91 |
    92 |
    {this.props.data.name}
    93 | 94 |
    95 | 99 |
  • 100 | ); 101 | } 102 | } 103 | 104 | // sort: 105 | // a.order - b.order != 0 ? a.order - b.order 106 | // : a.name.localeCompare(b.name, undefined, {numeric: true}) 107 | 108 | class FileList extends Component<{ 109 | files: (FileData | FolderData)[]; 110 | searchTerm: string; 111 | }> { 112 | render() { 113 | return ( 114 |
      115 | {this.props.files.map(file => { 116 | if (file.type === "file") { 117 | if ( 118 | file.name 119 | .toLowerCase() 120 | .indexOf(this.props.searchTerm.toLowerCase()) > 121 | -1 122 | ) { 123 | return ( 124 | 125 | ); 126 | } 127 | return null; 128 | } 129 | return ( 130 | 135 | ); 136 | })} 137 |
    138 | ); 139 | } 140 | } 141 | 142 | export class FilePane extends Component< 143 | { 144 | onActiveFileChanged: (fileContents: string) => void; 145 | files: (FileData | FolderData)[]; 146 | }, 147 | { 148 | showFileModal: boolean; 149 | searchTerm: string; 150 | } 151 | > { 152 | constructor( 153 | props: Readonly<{ 154 | onActiveFileChanged: (fileContents: string) => void; 155 | files: (FileData | FolderData)[]; 156 | }> 157 | ) { 158 | super(props); 159 | this.state = { 160 | showFileModal: false, 161 | searchTerm: "" 162 | }; 163 | } 164 | onDrop(acceptedFiles: File[], _rejectedFiles: File[], _event: DropEvent) { 165 | const reader = new FileReader(); 166 | 167 | reader.onabort = () => alert("file reading was aborted"); 168 | reader.onerror = () => alert("file reading has failed"); 169 | reader.onload = () => { 170 | // Do whatever you want with the file contents 171 | const binaryStr = new Buffer(reader.result as ArrayBuffer); 172 | let inverted: string; 173 | try { 174 | inverted = inverse(binaryStr); 175 | } catch (e) { 176 | alert( 177 | `An unhandled error occured while trying to convert shortcut -> scpl. The error is: ${e.toString()}` 178 | ); 179 | throw new Error(e); 180 | } 181 | this.props.onActiveFileChanged(inverted); 182 | }; 183 | 184 | acceptedFiles.forEach(file => reader.readAsArrayBuffer(file)); 185 | } 186 | render() { 187 | return ( 188 |
    189 | 190 | {({ getRootProps, getInputProps }) => ( 191 |
    192 |
    193 |
    194 |
    199 | Upload Shortcut 200 | 201 | 202 |
    203 |
    204 |
    205 |
    206 | )} 207 |
    208 | 209 | {this.state.showFileModal ? ( 210 | this.setState({ showFileModal: false })} 212 | onResult={(name, color, glyph) => { 213 | this.setState({ showFileModal: false }); 214 | }} 215 | /> 216 | ) : null} 217 |
    218 | ); 219 | } 220 | } 221 | -------------------------------------------------------------------------------- /src/Key.ts: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | type KeyNames = "save" | "closePanel" | "export"; 4 | 5 | export const keys: { 6 | [key in KeyNames]: (e: React.KeyboardEvent) => boolean 7 | } = { 8 | save: (e: React.KeyboardEvent) => { 9 | if ( 10 | e.nativeEvent.code === "KeyS" && 11 | (e.ctrlKey || e.metaKey) && 12 | !e.shiftKey && 13 | !e.altKey 14 | ) { 15 | e.preventDefault(); 16 | e.stopPropagation(); 17 | return true; 18 | } 19 | return false; 20 | }, 21 | closePanel: (e: React.KeyboardEvent) => { 22 | if ( 23 | e.nativeEvent.code === "Escape" && 24 | !(e.ctrlKey || e.metaKey) && 25 | !e.shiftKey && 26 | !e.altKey 27 | ) { 28 | e.preventDefault(); 29 | e.stopPropagation(); 30 | return true; 31 | } 32 | return false; 33 | }, 34 | export: (e: React.KeyboardEvent) => { 35 | if ( 36 | e.nativeEvent.code === "KeyS" && 37 | (e.ctrlKey || e.metaKey) && 38 | e.shiftKey && 39 | !e.altKey 40 | ) { 41 | e.preventDefault(); 42 | e.stopPropagation(); 43 | return true; 44 | } 45 | return false; 46 | } 47 | }; 48 | -------------------------------------------------------------------------------- /src/Modal.tsx: -------------------------------------------------------------------------------- 1 | export const _ = class {}; 2 | 3 | /* 4 |
    5 |
    9 | Cancel 10 |
    11 |

    Import from iCloud

    12 |
    iCloud URL
    13 | 19 |
    20 | Import Shortcut 21 |
    22 |
    23 | 24 |
    25 |
    29 | Cancel 30 |
    31 |

    Create New Folder

    32 |
    New Folder Name
    33 | 39 |
    40 | Create Folder 41 |
    42 |
    43 | 44 |
    45 |
    49 | Cancel 50 |
    51 |

    Rename File

    52 | 58 |
    59 | Change Name 60 |
    61 |
    62 | 63 |
    64 |
    68 | Cancel 69 |
    70 |

    Rename Folder

    71 | 77 |
    78 | Change Name 79 |
    80 |
    81 | 82 | 83 | */ 84 | -------------------------------------------------------------------------------- /src/ModalContainer.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | import { Portal } from "react-portal"; 3 | 4 | import { keys } from "./Key"; 5 | 6 | export class ModalContainer extends Component<{ onCancel: () => void }> { 7 | render() { 8 | return ( 9 | 10 |
    this.props.onCancel()} 16 | onKeyDown={e => { 17 | if (keys.closePanel(e)) { 18 | this.props.onCancel(); 19 | } 20 | }} 21 | > 22 |
    e.stopPropagation()}> 23 | {this.props.children} 24 |
    25 |
    26 |
    27 | ); 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /src/SearchActions.css: -------------------------------------------------------------------------------- 1 | /* Search actions */ 2 | .search-container { 3 | padding-top: 5px; 4 | } 5 | .search-container input { 6 | width: 75%; 7 | } 8 | .close-search-btn { 9 | padding: 5px; 10 | font-size: 15px; 11 | text-align: center; 12 | display: inline-block; 13 | margin-left: 10px; 14 | color: #007aff; 15 | cursor: pointer; 16 | } 17 | .close-mobile-search-btn { 18 | float: right; 19 | position: sticky; 20 | top: 0; 21 | padding: 10px 20px; 22 | font-size: 30px; 23 | text-align: center; 24 | display: inline-block; 25 | background-color: #fafafa; 26 | color: #007aff; 27 | cursor: pointer; 28 | display: none; 29 | border-bottom-left-radius: 10px; 30 | border-left: 0.5px solid #c7c7cc50; 31 | border-bottom: 0.5px solid #c7c7cc50; 32 | } 33 | .close-search-btn a { 34 | text-decoration: none !important; 35 | } 36 | .search-action-results { 37 | position: absolute; 38 | opacity: 0; 39 | pointer-events: none; 40 | cursor: pointer; 41 | z-index: 2000; 42 | margin: 0 auto; 43 | top: 50px; 44 | right: 0; 45 | left: 0; 46 | width: 500px; 47 | max-height: 500px; 48 | overflow-y: scroll; 49 | background-color: #fff; 50 | border-radius: 8px; 51 | box-shadow: 0 0 10px rgba(0, 0, 0, 0.3); 52 | transition: 0.3s opacity; 53 | } 54 | .search-action-results.search-action-results-visible { 55 | opacity: 1; 56 | pointer-events: all; 57 | } 58 | .action-item { 59 | padding: 15px 20px; 60 | padding-top: 18px; 61 | transition: 0.3s background-color; 62 | } 63 | .action-item-unknown .action-item-title:before { 64 | padding: 10px; 65 | width: auto; 66 | height: auto; 67 | background: url(img/shortcuts_grey.png) center / 30px no-repeat; 68 | } 69 | .action-item-title:before { 70 | content: ' '; 71 | padding: 3px 11px; 72 | margin-right: 8px; 73 | background: center / contain no-repeat; 74 | } 75 | .action-item:hover { 76 | background-color: #fafafa; 77 | } 78 | .action-item:active { 79 | background-color: #007aff20; 80 | } 81 | .search-action-results .action-item:not(:last-child) { 82 | border-bottom: 0.5px solid #eee; 83 | } 84 | .action-item-title { 85 | font-weight: bold; 86 | display: inline-block; 87 | } 88 | .action-item-code { 89 | display: inline-block; 90 | margin: 0 10px; 91 | background-color: rgba(150, 150, 150, 0.2); 92 | border: 1px solid #ddd; 93 | color: #444; 94 | padding: 3px 8px; 95 | border-radius: 3px; 96 | font-size: 13px; 97 | font-family: monospace; 98 | font-weight: normal; 99 | } 100 | .action-item-description { 101 | font-size: 14px; 102 | color: #555; 103 | margin: 8px 0; 104 | } 105 | .action-item-url { 106 | font-size: 14px; 107 | } 108 | .action-item-usage { 109 | margin: 10px 0; 110 | padding: 10px; 111 | background: rgba(235, 235, 235, 0.67); 112 | border: 1px solid #ddd; 113 | border-radius: 5px; 114 | color: #1000a2; 115 | box-shadow: inset 0 0 5px rgba(0, 0, 0, 0.15); 116 | font-family: monospace; 117 | } 118 | /* Icons */ 119 | .action-item-photos div:before { 120 | background-image: url(); 121 | } 122 | .action-item-gear div:before { 123 | background-image: url(); 124 | } 125 | .action-item-calendar div:before { 126 | background: url(img/var-sheet.png) 0px / cover no-repeat; 127 | background-position: -23px 0px !important; 128 | } 129 | .action-item-files div:before { 130 | background-image: url(); 131 | } 132 | .action-item-music div:before { 133 | background-image: url(); 134 | } 135 | .action-item-sharing div:before { 136 | background-image: url(); 137 | } 138 | .action-item-text div:before { 139 | background-image: url(); 140 | } 141 | .action-item-web div:before { 142 | background-image: url(); 143 | } 144 | .action-item-contacts div:before { 145 | background-image: url(); 146 | } 147 | .action-item-location div:before { 148 | background-image: url(); 149 | } 150 | -------------------------------------------------------------------------------- /src/SearchActions.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | import ReactMarkdown from "react-markdown"; 3 | 4 | import { allActions, getActionFromID } from "scpl"; 5 | 6 | import { keys } from "./Key"; 7 | import { categories } from "./data/ActionCategories"; 8 | 9 | import "./SearchActions.css"; 10 | 11 | const actionsByName = allActions.sort((a, b) => { 12 | if (a.name > b.name) { 13 | return 1; 14 | } 15 | if (a.name < b.name) { 16 | return -1; 17 | } 18 | return 0; 19 | }); 20 | 21 | class ActionData extends Component<{ 22 | onSelect: (text: string) => void; 23 | actionID: string; 24 | canShowDetail: boolean; 25 | }> { 26 | render() { 27 | const action = getActionFromID(this.props.actionID); 28 | if (!action) { 29 | return null; 30 | } 31 | const usage = action 32 | .genDocsUsage() 33 | .replace(/^```\s+(.+?)\s+```$/, "$1"); 34 | return ( 35 |
    { 42 | e.stopPropagation(); 43 | this.props.onSelect(usage); 44 | }} 45 | > 46 |
    47 | {action.name} 48 |
    {action.shortName}
    49 |
    50 |
    51 | {this.props.canShowDetail ? ( 52 | 53 | ) : ( 54 | ( 55 | action._data.Description || { 56 | DescriptionSummary: "No description" 57 | } 58 | ).DescriptionSummary 59 | )} 60 |
    61 |
    {usage}
    62 | 70 | See futher documentation 71 | 72 |
    73 | ); 74 | } 75 | } 76 | 77 | type SearchActionsProps = { insertText: (text: string) => void }; 78 | export class SearchActions extends Component< 79 | SearchActionsProps, 80 | { searchTerm: string | undefined } 81 | > { 82 | input: HTMLInputElement | null; 83 | constructor(props: Readonly) { 84 | super(props); 85 | this.state = { searchTerm: undefined }; 86 | this.input = null; 87 | } 88 | searchChanged(value: string | undefined) { 89 | const searchTerm = value; 90 | this.setState({ searchTerm: searchTerm }); 91 | } 92 | render() { 93 | return ( 94 |
    e.stopPropagation()}> 95 | (this.input = c)} 99 | onKeyDown={e => { 100 | if (keys.closePanel(e)) { 101 | if (this.input) { 102 | this.input.blur(); 103 | } 104 | this.searchChanged(undefined); 105 | } 106 | }} 107 | onKeyUp={e => this.searchChanged(e.currentTarget.value)} 108 | onFocus={e => this.searchChanged(e.currentTarget.value)} 109 | onBlur={e => 110 | setTimeout(() => this.searchChanged(undefined), 300) 111 | } 112 | /> 113 | {this.state.searchTerm !== undefined ? ( 114 |
    this.searchChanged(undefined)} 117 | > 118 | Cancel 119 |
    120 | ) : null} 121 |
    128 |
    this.searchChanged(undefined)} 131 | > 132 | × 133 |
    134 | {actionsByName 135 | .filter( 136 | action => 137 | action.name 138 | .toLowerCase() 139 | .replace(/[^A-Za-z0-9]/g, "") 140 | .indexOf( 141 | ( 142 | this.state.searchTerm || "kjhgvb" 143 | ).replace(/[^A-Za-z0-9]/g, "") 144 | ) > -1 145 | ) 146 | .map((action, i, v) => ( 147 | { 150 | this.props.insertText(`\n${text}`); 151 | this.searchChanged(undefined); 152 | }} 153 | canShowDetail={v.length === 1} 154 | key={action.id} 155 | /> 156 | ))} 157 |
    158 |
    159 | ); 160 | } 161 | } 162 | -------------------------------------------------------------------------------- /src/UploadShortcutModal.css: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/UploadShortcutModal.css -------------------------------------------------------------------------------- /src/UploadShortcutModal.tsx: -------------------------------------------------------------------------------- 1 | import React, { Component } from "react"; 2 | import { inverse } from "scpl"; 3 | 4 | import { ModalContainer } from "./ModalContainer"; 5 | 6 | import { DownloadStatusSwitcher, UploadStatus } from "./DownloadStatusSwitcher"; 7 | import { downloadShortcut } from "./Uploader"; 8 | 9 | import "./UploadShortcutModal.css"; 10 | 11 | type UploadShortcutProps = { 12 | onCancel: () => void; 13 | onResult: (filecont: string) => void; 14 | }; 15 | 16 | export class UploadShortcutModal extends Component< 17 | UploadShortcutProps, 18 | { uploadStatus: UploadStatus; shortcutID: string } 19 | > { 20 | constructor(props: Readonly) { 21 | super(props); 22 | this.state = { 23 | uploadStatus: { uploadStatus: "None" }, 24 | shortcutID: "" 25 | }; 26 | } 27 | async convertShortcut(data: ArrayBuffer) { 28 | const binaryStr = new Buffer(data); 29 | let inverted: string; 30 | try { 31 | inverted = inverse(binaryStr); 32 | } catch (e) { 33 | this.setState({ 34 | uploadStatus: { 35 | uploadStatus: "Error", 36 | uploadError: `An unhandled error occured while trying to convert shortcut -> scpl. The error is: ${e.toString()}` 37 | } 38 | }); 39 | throw new Error(e); 40 | } 41 | this.props.onResult(inverted); 42 | } 43 | async downloadShortcut() { 44 | if (!this.state.shortcutID) { 45 | this.setState({ 46 | uploadStatus: { 47 | uploadStatus: "Error", 48 | uploadError: "No ID." 49 | } 50 | }); 51 | return; 52 | } 53 | this.setState({ 54 | uploadStatus: { uploadStatus: "Uploading" } 55 | }); 56 | const uploadResult = await downloadShortcut(this.state.shortcutID); 57 | if (uploadResult.result === "error") { 58 | this.setState({ 59 | uploadStatus: { 60 | uploadStatus: "Error", 61 | uploadError: uploadResult.message 62 | } 63 | }); 64 | return; 65 | } 66 | // done 67 | await this.convertShortcut(uploadResult.shortcut); 68 | } 69 | render() { 70 | return ( 71 | this.props.onCancel()}> 72 |
    77 |
    this.props.onCancel()} 81 | > 82 | Cancel 83 |
    84 |

    Import Shortcut

    85 | { 89 | const shortcutID = e.currentTarget.value.match( 90 | /[a-z0-9]{32}/ 91 | ); 92 | if (!shortcutID) { 93 | this.setState({ shortcutID: "" }); 94 | return; 95 | } 96 | this.setState({ shortcutID: shortcutID[0] }); 97 | }} 98 | /> 99 |

    {this.state.shortcutID}

    100 | this.downloadShortcut()} 103 | detailsMsg={ 104 |
    105 | To convert the imported shortcut, ScPL Editor will upload your shortcut 106 | to{" "} 107 | 112 | shortcutsweb.app 113 | 114 | . 115 |
    116 | } 117 | uploadAction="Import" 118 | additionalButtonClasses="large-btn stretch-btn" 119 | /> 120 | 121 |
    122 |
    123 | ); 124 | } 125 | } 126 | -------------------------------------------------------------------------------- /src/Uploader.ts: -------------------------------------------------------------------------------- 1 | type UploadResult = 2 | | { 3 | success: true; 4 | key: string; 5 | link: string; 6 | expiry: string; 7 | } 8 | | { success: false; error: number; message: string }; 9 | 10 | export function uploadShortcut( 11 | shortcut: Buffer, 12 | filename: string, 13 | onProgress?: (percent: number) => void 14 | ): Promise< 15 | { result: "success"; url: string } | { result: "error"; message: string } 16 | > { 17 | return new Promise(resolve => { 18 | const formdata = new FormData(); 19 | formdata.append( 20 | "file", 21 | new Blob([shortcut], { type: "application/x-octet-stream" }), 22 | filename 23 | ); 24 | 25 | const xhttp = new XMLHttpRequest(); 26 | xhttp.open("POST", "https://file.io", true); 27 | 28 | xhttp.onprogress = e => { 29 | if (e.lengthComputable) { 30 | onProgress && onProgress(e.loaded / e.total); 31 | } else { 32 | onProgress && onProgress(0.1); 33 | } 34 | }; 35 | xhttp.onerror = () => { 36 | resolve({ 37 | result: "error", 38 | message: 39 | "An error occured. This may be because you do not have internet access or because the file.io servers were unreachable." 40 | }); 41 | }; 42 | xhttp.onreadystatechange = () => { 43 | if (xhttp.readyState === 4) { 44 | onProgress && onProgress(1); 45 | // Typical action to be performed when the document is ready: 46 | if (xhttp.response) { 47 | const result: UploadResult = JSON.parse(xhttp.response); 48 | if (!result.success) { 49 | resolve({ 50 | result: "error", 51 | message: `The file could not be uploaded. Error: ${ 52 | result.message 53 | }` 54 | }); 55 | } else { 56 | resolve({ result: "success", url: result.link }); 57 | } 58 | } 59 | } 60 | }; 61 | 62 | xhttp.send(formdata); 63 | }); 64 | } 65 | 66 | export function downloadShortcut( 67 | id: string, 68 | onProgress?: (percent: number) => void 69 | ): Promise< 70 | | { result: "success"; shortcut: ArrayBuffer } 71 | | { result: "error"; message: string } 72 | > { 73 | return new Promise(resolve => { 74 | const xhttp = new XMLHttpRequest(); 75 | xhttp.open( 76 | "GET", 77 | `https://shortcutsweb.app/inspectshortcut?id=${encodeURIComponent( 78 | id 79 | )}`, 80 | true 81 | ); 82 | xhttp.responseType = "arraybuffer"; 83 | 84 | xhttp.onprogress = e => { 85 | if (e.lengthComputable) { 86 | onProgress && onProgress(e.loaded / e.total); 87 | } else { 88 | onProgress && onProgress(0.1); 89 | } 90 | }; 91 | xhttp.onerror = () => { 92 | resolve({ 93 | result: "error", 94 | message: 95 | "An error occured. This may be because you do not have internet access or because the shortcutsweb.app servers are unreachable." 96 | }); 97 | }; 98 | xhttp.onreadystatechange = () => { 99 | if (xhttp.readyState === 4) { 100 | onProgress && onProgress(1); 101 | // Typical action to be performed when the document is ready: 102 | if (xhttp.response) { 103 | const result: ArrayBuffer = xhttp.response; 104 | if (xhttp.status === 200) { 105 | return resolve({ result: "success", shortcut: result }); 106 | } 107 | return resolve({ 108 | result: "error", 109 | message: `There was an error getting your shortcut data. Maybe you entered the wrong URL? Additional information: Error ${ 110 | xhttp.status 111 | }: ${xhttp.statusText}: ${Buffer.from(result).toString( 112 | "utf-8" 113 | )}` 114 | }); 115 | } 116 | } 117 | }; 118 | 119 | xhttp.send(); 120 | }); 121 | } 122 | -------------------------------------------------------------------------------- /src/ace/mode-scpl.ts: -------------------------------------------------------------------------------- 1 | /*eslint-disable func-names*/ 2 | 3 | import ace from "brace"; 4 | 5 | // @ts-ignore 6 | ace.define("ace/mode/scpl_highlight_rules", function( 7 | require: any, 8 | exports: any, 9 | _module: any 10 | ) { 11 | "use strict"; 12 | const oop = require("../lib/oop"); 13 | const TextHighlightRules = require("./text_highlight_rules") 14 | .TextHighlightRules; 15 | /* --------------------- START ----------------------------- */ 16 | const ScplHighlightRules = function() { 17 | // @ts-ignore 18 | this.$rules = { 19 | start: [ 20 | { 21 | token: "punctuation", 22 | regex: '(\\")', 23 | push: "string__1" 24 | }, 25 | { 26 | token: "punctuation", 27 | regex: "(\\')", 28 | push: "string__2" 29 | }, 30 | { 31 | token: "punctuation", 32 | regex: "(^\\s*\\|\\s*)", 33 | push: "barlist__1" 34 | }, 35 | { 36 | token: "variable", 37 | regex: "(([A-Za-z0-9@._]+):([A-Za-z0-9@._]+))" 38 | }, 39 | { 40 | token: "support.type", 41 | regex: "(([A-Za-z0-9@._]+)=)" 42 | }, 43 | { 44 | token: "support.constant", 45 | regex: "(@(([A-Za-z0-9@._]+))?)" 46 | }, 47 | { 48 | token: "keyword", 49 | regex: 50 | "([Ii][Ff]|[Rr][Ee][Pp][Ee][Aa][Tt][Ww][Ii][Tt][Hh][Ee][Aa][Cc][Hh]|[Cc][Hh][Oo][Oo][Ss][Ee][Ff][Rr][Oo][Mm][Mm][Ee][Nn][Uu]|[Rr][Ee][Pp][Ee][Aa][Tt]|[Oo][Tt][Hh][Ee][Rr][Ww][Ii][Ss][Ee]|[Ee][Nn][Dd]|[Ff][Ll][Oo][Ww]|[Ee][Ll][Ss][Ee]|[Cc][Aa][Ss][Ee])" 51 | }, 52 | { 53 | token: "entity.name.function", 54 | regex: "([A-Za-z0-9@._]+)" 55 | }, 56 | { 57 | token: "comment", 58 | regex: "(/\\*)", 59 | push: "multi_line_comment__1" 60 | }, 61 | { 62 | token: "comment", 63 | regex: "(--\\[)", 64 | push: "multi_line_comment__2" 65 | }, 66 | { 67 | token: "comment", 68 | regex: "(--.*)" 69 | }, 70 | { 71 | token: "comment", 72 | regex: "(#.*)" 73 | }, 74 | { 75 | token: "comment", 76 | regex: "(//.*)" 77 | }, 78 | { 79 | token: "punctuation", 80 | regex: "(\\()", 81 | push: "action__1" 82 | }, 83 | { 84 | token: "punctuation", 85 | regex: "(\\[)", 86 | push: "action__2" 87 | }, 88 | { 89 | token: "punctuation", 90 | regex: "({)", 91 | push: "action__3" 92 | }, 93 | { 94 | token: "punctuation", 95 | regex: '(["\\:,;^\\->=])' 96 | }, 97 | { 98 | token: "invalid", 99 | regex: "(\\?\\?.+?\\?\\?)" 100 | }, 101 | { 102 | token: "invalid", 103 | regex: "([^\\s])" 104 | }, 105 | { 106 | defaultToken: "text" 107 | } 108 | ], 109 | action__1: [ 110 | { 111 | token: "punctuation", 112 | regex: "(\\))", 113 | next: "pop" 114 | }, 115 | { 116 | token: "punctuation", 117 | regex: '(\\")', 118 | push: "string__1" 119 | }, 120 | { 121 | token: "punctuation", 122 | regex: "(\\')", 123 | push: "string__2" 124 | }, 125 | { 126 | token: "punctuation", 127 | regex: "(^\\s*\\|\\s*)", 128 | push: "barlist__1" 129 | }, 130 | { 131 | token: "variable", 132 | regex: "(([A-Za-z0-9@._]+):([A-Za-z0-9@._]+))" 133 | }, 134 | { 135 | token: "support.type", 136 | regex: "(([A-Za-z0-9@._]+)=)" 137 | }, 138 | { 139 | token: "support.constant", 140 | regex: "(@(([A-Za-z0-9@._]+))?)" 141 | }, 142 | { 143 | token: "keyword", 144 | regex: 145 | "([Ii][Ff]|[Rr][Ee][Pp][Ee][Aa][Tt][Ww][Ii][Tt][Hh][Ee][Aa][Cc][Hh]|[Cc][Hh][Oo][Oo][Ss][Ee][Ff][Rr][Oo][Mm][Mm][Ee][Nn][Uu]|[Rr][Ee][Pp][Ee][Aa][Tt]|[Oo][Tt][Hh][Ee][Rr][Ww][Ii][Ss][Ee]|[Ee][Nn][Dd]|[Ff][Ll][Oo][Ww]|[Ee][Ll][Ss][Ee]|[Cc][Aa][Ss][Ee])" 146 | }, 147 | { 148 | token: "entity.name.function", 149 | regex: "([A-Za-z0-9@._]+)" 150 | }, 151 | { 152 | token: "comment", 153 | regex: "(/\\*)", 154 | push: "multi_line_comment__1" 155 | }, 156 | { 157 | token: "comment", 158 | regex: "(--\\[)", 159 | push: "multi_line_comment__2" 160 | }, 161 | { 162 | token: "comment", 163 | regex: "(--.*)" 164 | }, 165 | { 166 | token: "comment", 167 | regex: "(#.*)" 168 | }, 169 | { 170 | token: "comment", 171 | regex: "(//.*)" 172 | }, 173 | { 174 | token: "punctuation", 175 | regex: "(\\()", 176 | push: "action__1" 177 | }, 178 | { 179 | token: "punctuation", 180 | regex: "(\\[)", 181 | push: "action__2" 182 | }, 183 | { 184 | token: "punctuation", 185 | regex: "({)", 186 | push: "action__3" 187 | }, 188 | { 189 | token: "punctuation", 190 | regex: '(["\\:,;^\\->=])' 191 | }, 192 | { 193 | token: "invalid", 194 | regex: "(\\?\\?.+?\\?\\?)" 195 | }, 196 | { 197 | token: "invalid", 198 | regex: "([^\\s])" 199 | }, 200 | { 201 | defaultToken: "text" 202 | } 203 | ], 204 | action__2: [ 205 | { 206 | token: "punctuation", 207 | regex: "(\\])", 208 | next: "pop" 209 | }, 210 | { 211 | token: "punctuation", 212 | regex: '(\\")', 213 | push: "string__1" 214 | }, 215 | { 216 | token: "punctuation", 217 | regex: "(\\')", 218 | push: "string__2" 219 | }, 220 | { 221 | token: "punctuation", 222 | regex: "(^\\s*\\|\\s*)", 223 | push: "barlist__1" 224 | }, 225 | { 226 | token: "variable", 227 | regex: "(([A-Za-z0-9@._]+):([A-Za-z0-9@._]+))" 228 | }, 229 | { 230 | token: "support.type", 231 | regex: "(([A-Za-z0-9@._]+)=)" 232 | }, 233 | { 234 | token: "support.constant", 235 | regex: "(@(([A-Za-z0-9@._]+))?)" 236 | }, 237 | { 238 | token: "keyword", 239 | regex: 240 | "([Ii][Ff]|[Rr][Ee][Pp][Ee][Aa][Tt][Ww][Ii][Tt][Hh][Ee][Aa][Cc][Hh]|[Cc][Hh][Oo][Oo][Ss][Ee][Ff][Rr][Oo][Mm][Mm][Ee][Nn][Uu]|[Rr][Ee][Pp][Ee][Aa][Tt]|[Oo][Tt][Hh][Ee][Rr][Ww][Ii][Ss][Ee]|[Ee][Nn][Dd]|[Ff][Ll][Oo][Ww]|[Ee][Ll][Ss][Ee]|[Cc][Aa][Ss][Ee])" 241 | }, 242 | { 243 | token: "entity.name.function", 244 | regex: "([A-Za-z0-9@._]+)" 245 | }, 246 | { 247 | token: "comment", 248 | regex: "(/\\*)", 249 | push: "multi_line_comment__1" 250 | }, 251 | { 252 | token: "comment", 253 | regex: "(--\\[)", 254 | push: "multi_line_comment__2" 255 | }, 256 | { 257 | token: "comment", 258 | regex: "(--.*)" 259 | }, 260 | { 261 | token: "comment", 262 | regex: "(#.*)" 263 | }, 264 | { 265 | token: "comment", 266 | regex: "(//.*)" 267 | }, 268 | { 269 | token: "punctuation", 270 | regex: "(\\()", 271 | push: "action__1" 272 | }, 273 | { 274 | token: "punctuation", 275 | regex: "(\\[)", 276 | push: "action__2" 277 | }, 278 | { 279 | token: "punctuation", 280 | regex: "({)", 281 | push: "action__3" 282 | }, 283 | { 284 | token: "punctuation", 285 | regex: '(["\\:,;^\\->=])' 286 | }, 287 | { 288 | token: "invalid", 289 | regex: "(\\?\\?.+?\\?\\?)" 290 | }, 291 | { 292 | token: "invalid", 293 | regex: "([^\\s])" 294 | }, 295 | { 296 | defaultToken: "text" 297 | } 298 | ], 299 | action__3: [ 300 | { 301 | token: "punctuation", 302 | regex: "(})", 303 | next: "pop" 304 | }, 305 | { 306 | token: "punctuation", 307 | regex: '(\\")', 308 | push: "string__1" 309 | }, 310 | { 311 | token: "punctuation", 312 | regex: "(\\')", 313 | push: "string__2" 314 | }, 315 | { 316 | token: "punctuation", 317 | regex: "(^\\s*\\|\\s*)", 318 | push: "barlist__1" 319 | }, 320 | { 321 | token: "variable", 322 | regex: "(([A-Za-z0-9@._]+):([A-Za-z0-9@._]+))" 323 | }, 324 | { 325 | token: "support.type", 326 | regex: "(([A-Za-z0-9@._]+)=)" 327 | }, 328 | { 329 | token: "support.constant", 330 | regex: "(@(([A-Za-z0-9@._]+))?)" 331 | }, 332 | { 333 | token: "keyword", 334 | regex: 335 | "([Ii][Ff]|[Rr][Ee][Pp][Ee][Aa][Tt][Ww][Ii][Tt][Hh][Ee][Aa][Cc][Hh]|[Cc][Hh][Oo][Oo][Ss][Ee][Ff][Rr][Oo][Mm][Mm][Ee][Nn][Uu]|[Rr][Ee][Pp][Ee][Aa][Tt]|[Oo][Tt][Hh][Ee][Rr][Ww][Ii][Ss][Ee]|[Ee][Nn][Dd]|[Ff][Ll][Oo][Ww]|[Ee][Ll][Ss][Ee]|[Cc][Aa][Ss][Ee])" 336 | }, 337 | { 338 | token: "entity.name.function", 339 | regex: "([A-Za-z0-9@._]+)" 340 | }, 341 | { 342 | token: "comment", 343 | regex: "(/\\*)", 344 | push: "multi_line_comment__1" 345 | }, 346 | { 347 | token: "comment", 348 | regex: "(--\\[)", 349 | push: "multi_line_comment__2" 350 | }, 351 | { 352 | token: "comment", 353 | regex: "(--.*)" 354 | }, 355 | { 356 | token: "comment", 357 | regex: "(#.*)" 358 | }, 359 | { 360 | token: "comment", 361 | regex: "(//.*)" 362 | }, 363 | { 364 | token: "punctuation", 365 | regex: "(\\()", 366 | push: "action__1" 367 | }, 368 | { 369 | token: "punctuation", 370 | regex: "(\\[)", 371 | push: "action__2" 372 | }, 373 | { 374 | token: "punctuation", 375 | regex: "({)", 376 | push: "action__3" 377 | }, 378 | { 379 | token: "punctuation", 380 | regex: '(["\\:,;^\\->=])' 381 | }, 382 | { 383 | token: "invalid", 384 | regex: "(\\?\\?.+?\\?\\?)" 385 | }, 386 | { 387 | token: "invalid", 388 | regex: "([^\\s])" 389 | }, 390 | { 391 | defaultToken: "text" 392 | } 393 | ], 394 | barlist__1: [ 395 | { 396 | token: "punctuation", 397 | regex: "($)", 398 | next: "pop" 399 | }, 400 | { 401 | token: "escape", 402 | regex: "(\\\\[\"'\\\\n])" 403 | }, 404 | { 405 | token: "escape", 406 | regex: "(\\\\\\()", 407 | push: "escape__1" 408 | }, 409 | { 410 | token: "support.constant", 411 | regex: "([^\\\\]+)" 412 | }, 413 | { 414 | defaultToken: "text" 415 | } 416 | ], 417 | escape__1: [ 418 | { 419 | token: "escape", 420 | regex: "(\\))", 421 | next: "pop" 422 | }, 423 | { 424 | token: "punctuation", 425 | regex: '(\\")', 426 | push: "string__1" 427 | }, 428 | { 429 | token: "punctuation", 430 | regex: "(\\')", 431 | push: "string__2" 432 | }, 433 | { 434 | token: "punctuation", 435 | regex: "(^\\s*\\|\\s*)", 436 | push: "barlist__1" 437 | }, 438 | { 439 | token: "variable", 440 | regex: "(([A-Za-z0-9@._]+):([A-Za-z0-9@._]+))" 441 | }, 442 | { 443 | token: "support.type", 444 | regex: "(([A-Za-z0-9@._]+)=)" 445 | }, 446 | { 447 | token: "support.constant", 448 | regex: "(@(([A-Za-z0-9@._]+))?)" 449 | }, 450 | { 451 | token: "keyword", 452 | regex: 453 | "([Ii][Ff]|[Rr][Ee][Pp][Ee][Aa][Tt][Ww][Ii][Tt][Hh][Ee][Aa][Cc][Hh]|[Cc][Hh][Oo][Oo][Ss][Ee][Ff][Rr][Oo][Mm][Mm][Ee][Nn][Uu]|[Rr][Ee][Pp][Ee][Aa][Tt]|[Oo][Tt][Hh][Ee][Rr][Ww][Ii][Ss][Ee]|[Ee][Nn][Dd]|[Ff][Ll][Oo][Ww]|[Ee][Ll][Ss][Ee]|[Cc][Aa][Ss][Ee])" 454 | }, 455 | { 456 | token: "entity.name.function", 457 | regex: "([A-Za-z0-9@._]+)" 458 | }, 459 | { 460 | token: "comment", 461 | regex: "(/\\*)", 462 | push: "multi_line_comment__1" 463 | }, 464 | { 465 | token: "comment", 466 | regex: "(--\\[)", 467 | push: "multi_line_comment__2" 468 | }, 469 | { 470 | token: "comment", 471 | regex: "(--.*)" 472 | }, 473 | { 474 | token: "comment", 475 | regex: "(#.*)" 476 | }, 477 | { 478 | token: "comment", 479 | regex: "(//.*)" 480 | }, 481 | { 482 | token: "punctuation", 483 | regex: "(\\()", 484 | push: "action__1" 485 | }, 486 | { 487 | token: "punctuation", 488 | regex: "(\\[)", 489 | push: "action__2" 490 | }, 491 | { 492 | token: "punctuation", 493 | regex: "({)", 494 | push: "action__3" 495 | }, 496 | { 497 | token: "punctuation", 498 | regex: '(["\\:,;^\\->=])' 499 | }, 500 | { 501 | token: "invalid", 502 | regex: "(\\?\\?.+?\\?\\?)" 503 | }, 504 | { 505 | token: "invalid", 506 | regex: "([^\\s])" 507 | }, 508 | { 509 | defaultToken: "text" 510 | } 511 | ], 512 | multi_line_comment__1: [ 513 | { 514 | token: "comment", 515 | regex: "(\\*/)", 516 | next: "pop" 517 | }, 518 | { 519 | defaultToken: "comment" 520 | } 521 | ], 522 | multi_line_comment__2: [ 523 | { 524 | token: "comment", 525 | regex: "(--\\])", 526 | next: "pop" 527 | }, 528 | { 529 | defaultToken: "comment" 530 | } 531 | ], 532 | string__1: [ 533 | { 534 | token: "punctuation", 535 | regex: '(\\")', 536 | next: "pop" 537 | }, 538 | { 539 | token: "escape", 540 | regex: "(\\\\[\"'\\\\n])" 541 | }, 542 | { 543 | token: "escape", 544 | regex: "(\\\\\\()", 545 | push: "escape__1" 546 | }, 547 | { 548 | token: "support.constant", 549 | regex: '([^"\\\\]+)' 550 | }, 551 | { 552 | defaultToken: "text" 553 | } 554 | ], 555 | string__2: [ 556 | { 557 | token: "punctuation", 558 | regex: "(\\')", 559 | next: "pop" 560 | }, 561 | { 562 | token: "escape", 563 | regex: "(\\\\[\"'\\\\n])" 564 | }, 565 | { 566 | token: "escape", 567 | regex: "(\\\\\\()", 568 | push: "escape__1" 569 | }, 570 | { 571 | token: "support.constant", 572 | regex: "([^'\\\\]+)" 573 | }, 574 | { 575 | defaultToken: "text" 576 | } 577 | ] 578 | }; 579 | // @ts-ignore 580 | this.normalizeRules(); 581 | }; 582 | /* ------------------------ END ------------------------------ */ 583 | oop.inherits(ScplHighlightRules, TextHighlightRules); 584 | exports.ScplHighlightRules = ScplHighlightRules; 585 | }); 586 | 587 | // @ts-ignore 588 | ace.define("ace/mode/scpl", ["ace/mode/scpl_highlight_rules"], function( 589 | // @ts-ignore 590 | require, 591 | // @ts-ignore 592 | exports, 593 | // @ts-ignore 594 | _module 595 | ) { 596 | const oop = require("ace/lib/oop"); 597 | const TextMode = require("ace/mode/text").Mode; 598 | const ExampleHighlightRules = require("ace/mode/scpl_highlight_rules") 599 | .ScplHighlightRules; 600 | 601 | const Mode = function() { 602 | // @ts-ignore 603 | this.HighlightRules = ExampleHighlightRules; 604 | }; 605 | oop.inherits(Mode, TextMode); 606 | 607 | (function() { 608 | // Extra logic goes here. (see below) 609 | }.call(Mode.prototype)); 610 | 611 | exports.Mode = Mode; 612 | }); 613 | // (function() { 614 | // // @ts-ignore 615 | // ace.require(["ace/mode/scpl"], function(m) { 616 | // if (typeof module === "object" && typeof exports === "object" && module) { 617 | // module.exports = m; 618 | // } 619 | // }); 620 | // }()); 621 | -------------------------------------------------------------------------------- /src/ace/scpl.iro: -------------------------------------------------------------------------------- 1 | ################################################################# 2 | ## Iro 3 | ################################################################ 4 | ## 5 | ## * Press Ctrl + '+'/'-' To Zoom in 6 | ## * Press Ctrl + S to save and recalculate... 7 | ## * Documents are saved to web storage. 8 | ## * Only one save slot supported. 9 | ## * Matches cannot span lines. 10 | ## * Unicode chars must be defined in \u0000 to \uffff format. 11 | ## * All matches must be contained by a single group ( ... ) 12 | ## * Look behinds not permitted, (?<= or (?=]) 181 | styles [] = .punctuation; 182 | } 183 | : pattern { 184 | regex \= (\?\?.+?\?\?) 185 | styles [] = .illegal; 186 | } 187 | : pattern { 188 | regex \= ([^\s]) 189 | styles [] = .illegal; 190 | } 191 | } 192 | string : context { 193 | : inline_push { 194 | regex \= (\") 195 | styles [] = .punctuation; 196 | : pop { 197 | regex \= (\") 198 | styles [] = .punctuation; 199 | } 200 | : include "escape"; 201 | : pattern{ 202 | regex \= ([^"\\]+) 203 | styles [] = .text; 204 | } 205 | } 206 | : inline_push { 207 | regex \= (\') 208 | styles [] = .punctuation; 209 | : pop { 210 | regex \= (\') 211 | styles [] = .punctuation; 212 | } 213 | : include "escape"; 214 | : pattern{ 215 | regex \= ([^'\\]+) 216 | styles [] = .text; 217 | } 218 | } 219 | } 220 | barlist : context { 221 | : inline_push { 222 | regex \= (^\s*\|\s*) 223 | styles [] = .punctuation; 224 | : pop { 225 | regex \= ($) 226 | styles [] = .punctuation; 227 | } 228 | : include "escape"; 229 | : pattern{ 230 | regex \= ([^\\]+) 231 | styles [] = .text; 232 | } 233 | } 234 | } 235 | escape : context { 236 | : pattern { 237 | regex \= (\\["'\\n]) 238 | styles [] = .escape; 239 | } 240 | : inline_push { 241 | regex \= (\\\() 242 | styles [] = .escape; 243 | : pop { 244 | regex \= (\)) 245 | styles [] = .escape; 246 | } 247 | : include "action"; 248 | } 249 | } 250 | 251 | 252 | multi_line_comment : context { 253 | description = multiline 254 | : inline_push { 255 | regex \= (/\*) 256 | styles [] = .comment; 257 | default_style = .comment 258 | : pop { 259 | regex \= (\*/) 260 | styles [] = .comment; 261 | } 262 | } 263 | : inline_push { 264 | regex \= (--\[) 265 | styles [] = .comment; 266 | default_style = .comment 267 | : pop { 268 | regex \= (--\]) 269 | styles [] = .comment; 270 | } 271 | } 272 | } 273 | 274 | inline_comment : context { 275 | : pattern { 276 | regex \= (--.*) 277 | styles [] = .comment; 278 | } 279 | : pattern { 280 | regex \= (#.*) 281 | styles [] = .comment; 282 | } 283 | : pattern { 284 | regex \= (//.*) 285 | styles [] = .comment; 286 | } 287 | } 288 | 289 | } 290 | -------------------------------------------------------------------------------- /src/data/ActionCategories.ts: -------------------------------------------------------------------------------- 1 | export const categories: {[key: string]: string | undefined} = { 2 | "Photos & Video": "action-item-photos", 3 | "Scripting": "action-item-gear", 4 | "Calendar": "action-item-calendar", 5 | "Documents": "action-item-files", 6 | "Music": "action-item-music", 7 | "Sharing": "action-item-sharing", 8 | "Text": "action-item-text", 9 | "Web": "action-item-web", 10 | "Contacts": "action-item-contacts", 11 | "Location": "action-item-location" 12 | }; 13 | -------------------------------------------------------------------------------- /src/data/ShortcutMeta.ts: -------------------------------------------------------------------------------- 1 | export type GlyphName = 2 | | "car" 3 | | "amb" 4 | | "house" 5 | | "cart" 6 | | "forkknife" 7 | | "sun" 8 | | "cloud" 9 | | "tree" 10 | | "footprints" 11 | | "compass" 12 | | "photo" 13 | | "bus" 14 | | "plane" 15 | | "hospital" 16 | | "purse" 17 | | "gaspump" 18 | | "moon" 19 | | "rain" 20 | | "flower" 21 | | "signs" 22 | | "earth" 23 | | "film" 24 | | "motorcycle" 25 | | "boat" 26 | | "city" 27 | | "stand" 28 | | "temp" 29 | | "snow" 30 | | "umbrella" 31 | | "fire" 32 | | "binoculars" 33 | | "mountain" 34 | | "filmfull" 35 | | "camera" 36 | | "videomarker" 37 | | "calendar" 38 | | "comment" 39 | | "paperairplane" 40 | | "creditcard" 41 | | "smartphone" 42 | | "emptykeyboard" 43 | | "printer" 44 | | "database" 45 | | "cube" 46 | | "videocamera" 47 | | "playbutton-one" 48 | | "message" 49 | | "letter" 50 | | "suitcase" 51 | | "watch" 52 | | "laptop" 53 | | "calculator" 54 | | "harddrive" 55 | | "servers" 56 | | "television" 57 | | "microphone" 58 | | "clipboard" 59 | | "messages" 60 | | "openletter" 61 | | "folder" 62 | | "phone" 63 | | "keyboard" 64 | | "stats" 65 | | "serverset" 66 | | "inbox" 67 | | "controller" 68 | | "puzzle" 69 | | "speaker" 70 | | "bookmark" 71 | | "mask" 72 | | "dice" 73 | | "soccer" 74 | | "lifesaver" 75 | | "chess" 76 | | "stopwatch" 77 | | "platter" 78 | | "trophy" 79 | | "headphones" 80 | | "books" 81 | | "emptyglasses" 82 | | "ticket" 83 | | "baseball" 84 | | "tennisball" 85 | | "telescope" 86 | | "clock" 87 | | "volume" 88 | | "heart" 89 | | "lightbulb" 90 | | "musicnote" 91 | | "book" 92 | | "glasses" 93 | | "masks" 94 | | "basketball" 95 | | "football" 96 | | "microscope" 97 | | "alarmclock" 98 | | "bell" 99 | | "star" 100 | | "lightning" 101 | | "flag" 102 | | "hourglass" 103 | | "battery" 104 | | "paintbrush" 105 | | "scissors" 106 | | "colorpicker" 107 | | "hammerwrench" 108 | | "screwdriver" 109 | | "trashcan" 110 | | "soupbowl" 111 | | "fish" 112 | | "tag" 113 | | "locked" 114 | | "magicwand" 115 | | "pencil" 116 | | "magnify" 117 | | "tool" 118 | | "gears" 119 | | "hand" 120 | | "teardrop" 121 | | "apple" 122 | | "cake" 123 | | "key" 124 | | "unlocked" 125 | | "magicstar" 126 | | "paperclip" 127 | | "link" 128 | | "wrench" 129 | | "hammer" 130 | | "privacy" 131 | | "cup" 132 | | "carrot" 133 | | "bottle" 134 | | "wineglass" 135 | | "oven" 136 | | "showerhead" 137 | | "pillbottle" 138 | | "scope" 139 | | "beaker" 140 | | "pawprint" 141 | | "gift" 142 | | "stairs" 143 | | "hanger" 144 | | "shirt" 145 | | "pill" 146 | | "bandaid" 147 | | "needle" 148 | | "cat" 149 | | "like" 150 | | "alien" 151 | | "rocket" 152 | | "laundry" 153 | | "bath" 154 | | "pills" 155 | | "inhaler" 156 | | "atom" 157 | | "dog" 158 | | "cap" 159 | | "bed" 160 | | "girlbaby" 161 | | "mansymbol" 162 | | "user" 163 | | "accessibility" 164 | | "dance" 165 | | "snowboard" 166 | | "activity" 167 | | "boybaby" 168 | | "womansymbol" 169 | | "users" 170 | | "podium" 171 | | "gym" 172 | | "swim" 173 | | "sprint" 174 | | "person" 175 | | "handicap" 176 | | "group" 177 | | "handraised" 178 | | "hike" 179 | | "hiking" 180 | | "cane" 181 | | "alert" 182 | | "bookmarkthis" 183 | | "stopfilled" 184 | | "left" 185 | | "up" 186 | | "play" 187 | | "stop" 188 | | "checked" 189 | | "moneysign" 190 | | "yensign" 191 | | "info" 192 | | "shareleft" 193 | | "barcode" 194 | | "frame" 195 | | "right" 196 | | "down" 197 | | "prev" 198 | | "next" 199 | | "plus" 200 | | "eurosign" 201 | | "bitcoinsign" 202 | | "smile" 203 | | "shareright" 204 | | "qrcode" 205 | | "sizes" 206 | | "download" 207 | | "upload" 208 | | "power" 209 | | "help" 210 | | "xfilled" 211 | | "pounds" 212 | | "pi" 213 | | "cssfile" 214 | | "money" 215 | | "yen" 216 | | "filefilled" 217 | | "list" 218 | | "more" 219 | | "share" 220 | | "spinner" 221 | | "target" 222 | | "location" 223 | | "crop" 224 | | "move" 225 | | "euro" 226 | | "bitcoin" 227 | | "file" 228 | | "document" 229 | | "listitems" 230 | | "infinite" 231 | | "loading" 232 | | "podcasts" 233 | | "mapmarker" 234 | | "exit" 235 | | "repeat" 236 | | "poundsign" 237 | | "asterisk" 238 | | "filedoc" 239 | | "fourgrid" 240 | | "sixgrid" 241 | | "recycle" 242 | | "playvideo" 243 | | "bigtarget" 244 | | "squarep" 245 | | "resize" 246 | | "sync" 247 | | "playsolo" 248 | | "rss" 249 | | "quotes" 250 | | "text" 251 | | "shuffle" 252 | | "signal" 253 | | "peace" 254 | | "cloudservice" 255 | | "settings" 256 | | "wifi" 257 | | "nuclear"; 258 | 259 | export const objectGlyphs: GlyphName[] = [ 260 | "car", 261 | "amb", 262 | "house", 263 | "cart", 264 | "forkknife", 265 | "sun", 266 | "cloud", 267 | "tree", 268 | "footprints", 269 | "compass", 270 | "photo", 271 | "bus", 272 | "plane", 273 | "hospital", 274 | "purse", 275 | "gaspump", 276 | "moon", 277 | "rain", 278 | "flower", 279 | "signs", 280 | "earth", 281 | "film", 282 | "motorcycle", 283 | "boat", 284 | "city", 285 | "stand", 286 | "temp", 287 | "snow", 288 | "umbrella", 289 | "fire", 290 | "binoculars", 291 | "mountain", 292 | "filmfull", 293 | "camera", 294 | "videomarker", 295 | "calendar", 296 | "comment", 297 | "paperairplane", 298 | "creditcard", 299 | "smartphone", 300 | "emptykeyboard", 301 | "printer", 302 | "database", 303 | "cube", 304 | "videocamera", 305 | "playbutton-one", 306 | "message", 307 | "letter", 308 | "suitcase", 309 | "watch", 310 | "laptop", 311 | "calculator", 312 | "harddrive", 313 | "servers", 314 | "television", 315 | "microphone", 316 | "clipboard", 317 | "messages", 318 | "openletter", 319 | "folder", 320 | "phone", 321 | "keyboard", 322 | "stats", 323 | "serverset", 324 | "inbox", 325 | "controller", 326 | "puzzle", 327 | "speaker", 328 | "bookmark", 329 | "mask", 330 | "dice", 331 | "soccer", 332 | "lifesaver", 333 | "chess", 334 | "stopwatch", 335 | "platter", 336 | "trophy", 337 | "headphones", 338 | "books", 339 | "emptyglasses", 340 | "ticket", 341 | "baseball", 342 | "tennisball", 343 | "telescope", 344 | "clock", 345 | "volume", 346 | "heart", 347 | "lightbulb", 348 | "musicnote", 349 | "book", 350 | "glasses", 351 | "masks", 352 | "basketball", 353 | "football", 354 | "microscope", 355 | "alarmclock", 356 | "bell", 357 | "star", 358 | "lightning", 359 | "flag", 360 | "hourglass", 361 | "battery", 362 | "paintbrush", 363 | "scissors", 364 | "colorpicker", 365 | "hammerwrench", 366 | "screwdriver", 367 | "trashcan", 368 | "soupbowl", 369 | "fish", 370 | "tag", 371 | "locked", 372 | "magicwand", 373 | "pencil", 374 | "magnify", 375 | "tool", 376 | "gears", 377 | "hand", 378 | "teardrop", 379 | "apple", 380 | "cake", 381 | "key", 382 | "unlocked", 383 | "magicstar", 384 | "paperclip", 385 | "link", 386 | "wrench", 387 | "hammer", 388 | "privacy", 389 | "cup", 390 | "carrot", 391 | "bottle", 392 | "wineglass", 393 | "oven", 394 | "showerhead", 395 | "pillbottle", 396 | "scope", 397 | "beaker", 398 | "pawprint", 399 | "gift", 400 | "stairs", 401 | "hanger", 402 | "shirt", 403 | "pill", 404 | "bandaid", 405 | "needle", 406 | "cat", 407 | "like", 408 | "alien", 409 | "rocket", 410 | "laundry", 411 | "bath", 412 | "pills", 413 | "inhaler", 414 | "atom", 415 | "dog", 416 | "cap", 417 | "bed" 418 | ]; 419 | 420 | export const peopleGlyphs: GlyphName[] = [ 421 | "girlbaby", 422 | "mansymbol", 423 | "user", 424 | "accessibility", 425 | "dance", 426 | "snowboard", 427 | "activity", 428 | "boybaby", 429 | "womansymbol", 430 | "users", 431 | "podium", 432 | "gym", 433 | "swim", 434 | "sprint", 435 | "person", 436 | "handicap", 437 | "group", 438 | "handraised", 439 | "hike", 440 | "hiking", 441 | "cane" 442 | ]; 443 | 444 | export const symbolGlyphs: GlyphName[] = [ 445 | "alert", 446 | "bookmarkthis", 447 | "stopfilled", 448 | "left", 449 | "up", 450 | "play", 451 | "stop", 452 | "checked", 453 | "moneysign", 454 | "yensign", 455 | "info", 456 | "shareleft", 457 | "barcode", 458 | "frame", 459 | "right", 460 | "down", 461 | "prev", 462 | "next", 463 | "plus", 464 | "eurosign", 465 | "bitcoinsign", 466 | "smile", 467 | "shareright", 468 | "qrcode", 469 | "sizes", 470 | "download", 471 | "upload", 472 | "power", 473 | "help", 474 | "xfilled", 475 | "pounds", 476 | "pi", 477 | "cssfile", 478 | "money", 479 | "yen", 480 | "filefilled", 481 | "list", 482 | "more", 483 | "share", 484 | "spinner", 485 | "target", 486 | "location", 487 | "crop", 488 | "move", 489 | "euro", 490 | "bitcoin", 491 | "file", 492 | "document", 493 | "listitems", 494 | "infinite", 495 | "loading", 496 | "podcasts", 497 | "mapmarker", 498 | "exit", 499 | "repeat", 500 | "poundsign", 501 | "asterisk", 502 | "filedoc", 503 | "fourgrid", 504 | "sixgrid", 505 | "recycle", 506 | "playvideo", 507 | "bigtarget", 508 | "squarep", 509 | "resize", 510 | "sync", 511 | "playsolo", 512 | "rss", 513 | "quotes", 514 | "text", 515 | "shuffle", 516 | "signal", 517 | "peace", 518 | "cloudservice", 519 | "settings", 520 | "wifi", 521 | "nuclear" 522 | ]; 523 | 524 | export type ColorName = 525 | | "red" 526 | | "darkorange" 527 | | "orange" 528 | | "yellow" 529 | | "green" 530 | | "seagreen" 531 | | "lightblue" 532 | | "blue" 533 | | "darkblue" 534 | | "darkpurple" 535 | | "purple" 536 | | "pink" 537 | | "black" 538 | | "brown" 539 | | "grey"; 540 | 541 | export const colors: ColorName[] = [ 542 | "red", 543 | "darkorange", 544 | "orange", 545 | "yellow", 546 | "green", 547 | "seagreen", 548 | "lightblue", 549 | "blue", 550 | "darkblue", 551 | "darkpurple", 552 | "purple", 553 | "pink", 554 | "black", 555 | "brown", 556 | "grey" 557 | ]; 558 | -------------------------------------------------------------------------------- /src/img/back.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 9 | 10 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | -------------------------------------------------------------------------------- /src/img/bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/bg.jpg -------------------------------------------------------------------------------- /src/img/close.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /src/img/delete.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /src/img/delete_w.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /src/img/down.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /src/img/edit.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /src/img/edit_w.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /src/img/error.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 11 | 12 | -------------------------------------------------------------------------------- /src/img/error_grey.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 11 | 12 | -------------------------------------------------------------------------------- /src/img/error_w.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 11 | 12 | -------------------------------------------------------------------------------- /src/img/error_x.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 10 | 11 | -------------------------------------------------------------------------------- /src/img/file.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/file.png -------------------------------------------------------------------------------- /src/img/file_active.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/file_active.png -------------------------------------------------------------------------------- /src/img/file_error.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/file_error.png -------------------------------------------------------------------------------- /src/img/file_warning.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/file_warning.png -------------------------------------------------------------------------------- /src/img/folder.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/folder.png -------------------------------------------------------------------------------- /src/img/folder_error.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/folder_error.png -------------------------------------------------------------------------------- /src/img/folder_warning.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/folder_warning.png -------------------------------------------------------------------------------- /src/img/glyph_sheet.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/glyph_sheet.png -------------------------------------------------------------------------------- /src/img/glyph_sheet_white.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/glyph_sheet_white.png -------------------------------------------------------------------------------- /src/img/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/icon.png -------------------------------------------------------------------------------- /src/img/import.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/import.png -------------------------------------------------------------------------------- /src/img/magic.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | 7 | 8 | Created by potrace 1.11, written by Peter Selinger 2001-2013 9 | 10 | 12 | 22 | 31 | 55 | 72 | 78 | 79 | 80 | -------------------------------------------------------------------------------- /src/img/magic_w.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | 7 | 8 | Created by potrace 1.11, written by Peter Selinger 2001-2013 9 | 10 | 12 | 22 | 31 | 55 | 72 | 78 | 79 | 80 | -------------------------------------------------------------------------------- /src/img/menu.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /src/img/new_file.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/new_file.png -------------------------------------------------------------------------------- /src/img/new_folder.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/new_folder.png -------------------------------------------------------------------------------- /src/img/right.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /src/img/search.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /src/img/settings.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 32 | 33 | 34 | 35 | 36 | 37 | -------------------------------------------------------------------------------- /src/img/settings_w.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 32 | 33 | 34 | 35 | 36 | 37 | -------------------------------------------------------------------------------- /src/img/shortcut-file.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/shortcut-file.png -------------------------------------------------------------------------------- /src/img/shortcuts_grey.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/shortcuts_grey.png -------------------------------------------------------------------------------- /src/img/success.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /src/img/upload.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/upload.png -------------------------------------------------------------------------------- /src/img/var-sheet.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/pfgithub/scpl-editor/9e3d5e13a65d723362e1f76c300ea4b4bf621617/src/img/var-sheet.png -------------------------------------------------------------------------------- /src/index.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 0; 3 | margin: 0; 4 | } 5 | -------------------------------------------------------------------------------- /src/index.tsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import ReactDOM from "react-dom"; 3 | import "./index.css"; 4 | import App from "./App"; 5 | import * as serviceWorker from "./serviceWorker"; 6 | 7 | ReactDOM.render(, document.getElementById("root")); 8 | 9 | // If you want your app to work offline and load faster, you can change 10 | // unregister() to register() below. Note this comes with some pitfalls. 11 | // Learn more about service workers: https://bit.ly/CRA-PWA 12 | serviceWorker.register(); 13 | -------------------------------------------------------------------------------- /src/logo.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /src/react-app-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | -------------------------------------------------------------------------------- /src/serviceWorker.ts: -------------------------------------------------------------------------------- 1 | // This optional code is used to register a service worker. 2 | // register() is not called by default. 3 | 4 | // This lets the app load faster on subsequent visits in production, and gives 5 | // it offline capabilities. However, it also means that developers (and users) 6 | // will only see deployed updates on subsequent visits to a page, after all the 7 | // existing tabs open on the page have been closed, since previously cached 8 | // resources are updated in the background. 9 | 10 | // To learn more about the benefits of this model and instructions on how to 11 | // opt-in, read https://bit.ly/CRA-PWA 12 | 13 | const isLocalhost = Boolean( 14 | window.location.hostname === "localhost" || 15 | // [::1] is the IPv6 localhost address. 16 | window.location.hostname === "[::1]" || 17 | // 127.0.0.1/8 is considered localhost for IPv4. 18 | window.location.hostname.match( 19 | /^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/ 20 | ) 21 | ); 22 | 23 | type Config = { 24 | onSuccess?: (registration: ServiceWorkerRegistration) => void; 25 | onUpdate?: (registration: ServiceWorkerRegistration) => void; 26 | }; 27 | 28 | export function register(config?: Config) { 29 | if (process.env.NODE_ENV === "production" && "serviceWorker" in navigator) { 30 | // The URL constructor is available in all browsers that support SW. 31 | const publicUrl = new URL( 32 | (process as { env: { [key: string]: string } }).env.PUBLIC_URL, 33 | window.location.href 34 | ); 35 | if (publicUrl.origin !== window.location.origin) { 36 | // Our service worker won't work if PUBLIC_URL is on a different origin 37 | // from what our page is served on. This might happen if a CDN is used to 38 | // serve assets; see https://github.com/facebook/create-react-app/issues/2374 39 | return; 40 | } 41 | 42 | window.addEventListener("load", () => { 43 | const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`; 44 | 45 | if (isLocalhost) { 46 | // This is running on localhost. Let's check if a service worker still exists or not. 47 | checkValidServiceWorker(swUrl, config); 48 | 49 | // Add some additional logging to localhost, pointing developers to the 50 | // service worker/PWA documentation. 51 | navigator.serviceWorker.ready.then(() => { 52 | console.log( 53 | "This web app is being served cache-first by a service " + 54 | "worker. To learn more, visit https://bit.ly/CRA-PWA" 55 | ); 56 | }); 57 | } else { 58 | // Is not localhost. Just register service worker 59 | registerValidSW(swUrl, config); 60 | } 61 | }); 62 | } 63 | } 64 | 65 | function registerValidSW(swUrl: string, config?: Config) { 66 | navigator.serviceWorker 67 | .register(swUrl) 68 | .then(registration => { 69 | registration.onupdatefound = () => { 70 | const installingWorker = registration.installing; 71 | if (installingWorker == null) { 72 | return; 73 | } 74 | installingWorker.onstatechange = () => { 75 | if (installingWorker.state === "installed") { 76 | if (navigator.serviceWorker.controller) { 77 | // At this point, the updated precached content has been fetched, 78 | // but the previous service worker will still serve the older 79 | // content until all client tabs are closed. 80 | console.log( 81 | "New content is available and will be used when all " + 82 | "tabs for this page are closed. See https://bit.ly/CRA-PWA." 83 | ); 84 | 85 | // Execute callback 86 | if (config && config.onUpdate) { 87 | config.onUpdate(registration); 88 | } 89 | } else { 90 | // At this point, everything has been precached. 91 | // It's the perfect time to display a 92 | // "Content is cached for offline use." message. 93 | console.log("Content is cached for offline use."); 94 | 95 | // Execute callback 96 | if (config && config.onSuccess) { 97 | config.onSuccess(registration); 98 | } 99 | } 100 | } 101 | }; 102 | }; 103 | }) 104 | .catch(error => { 105 | console.error("Error during service worker registration:", error); 106 | }); 107 | } 108 | 109 | function checkValidServiceWorker(swUrl: string, config?: Config) { 110 | // Check if the service worker can be found. If it can't reload the page. 111 | fetch(swUrl) 112 | .then(response => { 113 | // Ensure service worker exists, and that we really are getting a JS file. 114 | const contentType = response.headers.get("content-type"); 115 | if ( 116 | response.status === 404 || 117 | (contentType != null && 118 | contentType.indexOf("javascript") === -1) 119 | ) { 120 | // No service worker found. Probably a different app. Reload the page. 121 | navigator.serviceWorker.ready.then(registration => { 122 | registration.unregister().then(() => { 123 | window.location.reload(); 124 | }); 125 | }); 126 | } else { 127 | // Service worker found. Proceed as normal. 128 | registerValidSW(swUrl, config); 129 | } 130 | }) 131 | .catch(() => { 132 | console.log( 133 | "No internet connection found. App is running in offline mode." 134 | ); 135 | }); 136 | } 137 | 138 | export function unregister() { 139 | if ("serviceWorker" in navigator) { 140 | navigator.serviceWorker.ready.then(registration => { 141 | registration.unregister(); 142 | }); 143 | } 144 | } 145 | -------------------------------------------------------------------------------- /src/testshortcut.json: -------------------------------------------------------------------------------- 1 | [ 2 | { 3 | "WFWorkflowClientVersion": "754", 4 | "WFWorkflowClientRelese": "2.1.2", 5 | "WFWorkflowMinimumClientVersion": 411, 6 | "WFWorkflowIcon": { 7 | "WFWorkflowIconStartColor": 2071128575, 8 | "WFWorkflowIconImageData": { 9 | "type": "Buffer", 10 | "data": [] 11 | }, 12 | "WFWorkflowIconGlyphNumber": 59511 13 | }, 14 | "WFWorkflowTypes": ["NCWidget", "WatchKit"], 15 | "WFWorkflowInputContentItemClasses": [ 16 | "WFAppStoreAppContentItem", 17 | "WFArticleContentItem", 18 | "WFContactContentItem", 19 | "WFDateContentItem", 20 | "WFEmailAddressContentItem", 21 | "WFGenericFileContentItem", 22 | "WFImageContentItem", 23 | "WFiTunesProductContentItem", 24 | "WFLocationContentItem", 25 | "WFDCMapsLinkContentItem", 26 | "WFAVAssetContentItem", 27 | "WFPDFContentItem", 28 | "WFPhoneNumberContentItem", 29 | "WFRichTextContentItem", 30 | "WFSafariWebPageContentItem", 31 | "WFStringContentItem", 32 | "WFURLContentItem" 33 | ], 34 | "WFWorkflowActions": [] 35 | } 36 | ] 37 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es5", 4 | "lib": [ 5 | "dom", 6 | "dom.iterable", 7 | "esnext" 8 | ], 9 | "allowJs": true, 10 | "skipLibCheck": true, 11 | "esModuleInterop": true, 12 | "allowSyntheticDefaultImports": true, 13 | "strict": true, 14 | "forceConsistentCasingInFileNames": true, 15 | "module": "esnext", 16 | "moduleResolution": "node", 17 | "resolveJsonModule": true, 18 | "isolatedModules": true, 19 | "noEmit": true, 20 | "jsx": "preserve" 21 | }, 22 | "include": [ 23 | "src" 24 | ] 25 | } 26 | --------------------------------------------------------------------------------