├── release
└── .gitignore
├── .gitignore
├── com.dflydev.streamdeck.restreamio.sdPlugin
├── public
│ ├── dist
│ │ └── .gitignore
│ ├── icons
│ │ └── caret.svg
│ ├── plugin.html
│ ├── property-inspector.html
│ └── sdpi.css
├── icons
│ ├── action.png
│ ├── action@2x.png
│ ├── actionimage.png
│ ├── channel-off.png
│ ├── channel-on.png
│ ├── plugin-icon.png
│ ├── category-icon.png
│ ├── channel-on@2x.png
│ ├── actionimage@2x.png
│ ├── category-icon@2x.png
│ ├── channel-off@2x.png
│ ├── channel-unknown.png
│ ├── plugin-icon@2x.png
│ ├── channel-unknown@2x.png
│ ├── channel-action-image.png
│ └── channel-action-image@2x.png
└── manifest.json
├── .github
├── FUNDING.yml
└── workflows
│ └── pre-release.yml
├── tsconfig.eslint.json
├── .eslintignore
├── .prettierignore
├── .env.example
├── .prettierrc.json
├── .editorconfig
├── src
├── settings
│ ├── global-settings.ts
│ ├── settings.ts
│ ├── managed-settings.ts
│ ├── managed-global-settings.ts
│ └── deferred-global-settings-requests.ts
├── actions
│ ├── action.ts
│ └── channel-action.ts
├── restreamio-plugin.ts
├── restreamio
│ ├── types.ts
│ └── restreamio-client.ts
└── restreamio-property-inspector.ts
├── tsconfig.json
├── LICENSE.txt
├── package.json
├── .eslintrc.json
└── README.md
/release/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 | .env
3 | package-lock.json
4 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/public/dist/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 |
--------------------------------------------------------------------------------
/.github/FUNDING.yml:
--------------------------------------------------------------------------------
1 | github: [simensen, dflydev]
2 | open_collective: dflydev
3 |
--------------------------------------------------------------------------------
/tsconfig.eslint.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "./tsconfig.json",
3 | "exclude": []
4 | }
5 |
--------------------------------------------------------------------------------
/.eslintignore:
--------------------------------------------------------------------------------
1 | /build/
2 | /com.dflydev.streamdeck.restreamio.sdPlugin/public/dist/
3 | /node_modules/
4 |
--------------------------------------------------------------------------------
/.prettierignore:
--------------------------------------------------------------------------------
1 | /build/
2 | /com.dflydev.streamdeck.restreamio.sdPlugin/public/dist/
3 | /node_modules/
4 |
--------------------------------------------------------------------------------
/.env.example:
--------------------------------------------------------------------------------
1 | STREAMDECK_RESTREAMIO_LOGIN_URL=https://example.test/login
2 | STREAMDECK_RESTREAMIO_REFRESH_URL=https://example.test/refresh
3 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/action.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/action.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/action@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/action@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/actionimage.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/actionimage.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-off.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-off.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-on.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-on.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/plugin-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/plugin-icon.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/category-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/category-icon.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-on@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-on@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/actionimage@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/actionimage@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/category-icon@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/category-icon@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-off@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-off@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-unknown.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-unknown.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/plugin-icon@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/plugin-icon@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-unknown@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-unknown@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-action-image.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-action-image.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-action-image@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dflydev/streamdeck-restreamio/main/com.dflydev.streamdeck.restreamio.sdPlugin/icons/channel-action-image@2x.png
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/public/icons/caret.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/.prettierrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "printWidth": 80,
3 | "tabWidth": 2,
4 | "useTabs": false,
5 | "semi": false,
6 | "singleQuote": true,
7 | "trailingComma": "all",
8 | "bracketSpacing": false,
9 | "arrowParens": "avoid",
10 | "parser": "typescript"
11 | }
12 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/public/plugin.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | com.dflydev.streamdeck.restreamio
5 |
6 |
7 |
8 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | root = true
2 |
3 | [*]
4 | charset = utf-8
5 | end_of_line = lf
6 | insert_final_newline = true
7 | indent_style = space
8 | indent_size = 4
9 | trim_trailing_whitespace = true
10 |
11 | [*.md]
12 | trim_trailing_whitespace = false
13 |
14 | [*.yml]
15 | indent_size = 2
16 |
17 | [*.{js,ts,json}]
18 | indent_size = 2
19 |
20 | [*.vue]
21 | indent_size = 2
22 |
--------------------------------------------------------------------------------
/src/settings/global-settings.ts:
--------------------------------------------------------------------------------
1 | import {
2 | RestreamioAccountCollection,
3 | RestreamioStreamingPlatform,
4 | } from '../restreamio/types'
5 |
6 | export function createEmptyGlobalSettings(): GlobalSettings {
7 | return {
8 | restreamioAccounts: {},
9 | restreamioStreamingPlatforms: [],
10 | }
11 | }
12 |
13 | export interface GlobalSettings {
14 | restreamioAccounts: RestreamioAccountCollection
15 | restreamioStreamingPlatforms: RestreamioStreamingPlatform[]
16 | }
17 |
--------------------------------------------------------------------------------
/src/settings/settings.ts:
--------------------------------------------------------------------------------
1 | import {RestreamioAccountId, RestreamioChannelId} from '../restreamio/types'
2 |
3 | export function createEmptySettings(): Settings {
4 | return new (class implements Settings {
5 | restreamioAccountId: null
6 | restreamioChannelId: null
7 | name: null
8 | platform: null
9 | })()
10 | }
11 |
12 | export interface Settings {
13 | restreamioAccountId: RestreamioAccountId | null
14 | restreamioChannelId: RestreamioChannelId | null
15 | name: string | null
16 | platform: string | null
17 | }
18 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "module": "CommonJS",
4 | "target": "ES6",
5 | "noImplicitAny": true,
6 | "removeComments": true,
7 | "preserveConstEnums": true,
8 | "declaration": true,
9 | "sourceMap": false,
10 | "forceConsistentCasingInFileNames": true,
11 | "strictPropertyInitialization": false,
12 | "skipLibCheck": true,
13 | "outDir": "./build",
14 | "esModuleInterop": true,
15 | "experimentalDecorators": true,
16 | "lib": [
17 | "dom",
18 | "ESNext"
19 | ]
20 | },
21 | "include": [
22 | "src/**/*"
23 | ],
24 | "exclude": [
25 | "node_modules",
26 | "**/*.spec.ts"
27 | ],
28 | "extends": "@tsconfig/recommended/tsconfig.json"
29 | }
30 |
--------------------------------------------------------------------------------
/src/actions/action.ts:
--------------------------------------------------------------------------------
1 | import {RestreamioPlugin} from '../restreamio-plugin'
2 | import {ManagedGlobalSettings} from '../settings/managed-global-settings'
3 | import {ManagedSettings} from '../settings/managed-settings'
4 | import {RestreamioClient} from '../restreamio/restreamio-client'
5 | import {StreamDeckAction} from 'streamdeck-typescript'
6 |
7 | export abstract class Action extends StreamDeckAction<
8 | RestreamioPlugin,
9 | TheAction
10 | > {
11 | protected restreamioClient(context?: string): RestreamioClient {
12 | return RestreamioClient.fromSettingsManager(
13 | this.plugin.settingsManager,
14 | context,
15 | )
16 | }
17 |
18 | protected get globalSettings(): ManagedGlobalSettings {
19 | return new ManagedGlobalSettings(this.plugin.settingsManager)
20 | }
21 |
22 | protected settings(context: string): ManagedSettings {
23 | return new ManagedSettings(this.plugin.settingsManager, context)
24 | }
25 |
26 | constructor(protected plugin: RestreamioPlugin, actionName: string) {
27 | super(plugin, actionName)
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/LICENSE.txt:
--------------------------------------------------------------------------------
1 | Copyright (c) 2021 dflydev
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining a copy
4 | of this software and associated documentation files (the "Software"), to deal
5 | in the Software without restriction, including without limitation the rights
6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7 | copies of the Software, and to permit persons to whom the Software is furnished
8 | to do so, subject to the following conditions:
9 |
10 | The above copyright notice and this permission notice shall be included in all
11 | copies or substantial portions of the Software.
12 |
13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
19 | THE SOFTWARE.
20 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "Actions": [
3 | {
4 | "Icon": "icons/channel-action-image",
5 | "Name": "Channel",
6 | "States": [
7 | {
8 | "Name": "On",
9 | "Image": "icons/channel-on",
10 | "TitleAlignment": "bottom",
11 | "FontSize": "11"
12 | },
13 | {
14 | "Name": "Off",
15 | "Image": "icons/channel-off",
16 | "TitleAlignment": "bottom",
17 | "FontSize": "11"
18 | }
19 | ],
20 | "Tooltip": "Toggle channel off/on",
21 | "UUID": "com.dflydev.streamdeck.restreamio.channel"
22 | }
23 | ],
24 | "SDKVersion": 2,
25 | "Author": "dflydev",
26 | "Category": "Restream.io",
27 | "CategoryIcon": "icons/category-icon",
28 | "CodePath": "public/plugin.html",
29 | "PropertyInspectorPath": "public/property-inspector.html",
30 | "Description": "Control and view stats from Restream.io",
31 | "Name": "Restream.io plugin",
32 | "Icon": "icons/category-icon",
33 | "URL": "https://www.elgato.com/en/gaming/stream-deck",
34 | "Version": "1.0.0",
35 | "OS": [
36 | {
37 | "Platform": "mac",
38 | "MinimumVersion" : "10.11"
39 | },
40 | {
41 | "Platform": "windows",
42 | "MinimumVersion" : "10"
43 | }
44 | ],
45 | "Software":
46 | {
47 | "MinimumVersion" : "4.1"
48 | }
49 | }
50 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/public/property-inspector.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | com.dflydev.streamdeck.restreamio.pi
5 |
6 |
7 |
8 |
9 |
10 |
30 |
31 |
32 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@dflydev/streamdeck-restreamio",
3 | "version": "0.0.0",
4 | "description": "Restream.io plugin for Stream Deck",
5 | "repository": "https://github.com/dflydev/streamdeck-restreamio",
6 | "scripts": {
7 | "browserify": "browserify -t @sethvincent/dotenvify build/restreamio-plugin.js > com.dflydev.streamdeck.restreamio.sdPlugin/public/dist/restreamio-plugin-bundle.js && browserify -t @sethvincent/dotenvify build/restreamio-property-inspector.js > com.dflydev.streamdeck.restreamio.sdPlugin/public/dist/restreamio-property-inspector-bundle.js",
8 | "build": "npm run tsc && npm run browserify",
9 | "lint": "eslint src/**/*.ts",
10 | "fix": "eslint --fix src/**/*.ts",
11 | "test": "echo \"Error: no test specified\" && exit 1",
12 | "tsc": "tsc -p tsconfig.json",
13 | "watch": "tsc-watch --onSuccess \"npm run browserify\""
14 | },
15 | "author": "Beau Simensen (https://beausimensen.com)",
16 | "license": "MIT",
17 | "devDependencies": {
18 | "@sethvincent/dotenvify": "^1.0.4",
19 | "@tsconfig/recommended": "^1.0.1",
20 | "@types/node": "^14.14.31",
21 | "@typescript-eslint/eslint-plugin": "^4.15.2",
22 | "@typescript-eslint/parser": "^4.15.2",
23 | "axios": "^0.21.1",
24 | "axios-auth-refresh": "^3.1.0",
25 | "browserify": "^17.0.0",
26 | "dotenv": "^8.2.0",
27 | "eslint": "^7.21.0",
28 | "eslint-plugin-github": "^4.1.1",
29 | "prettier": "^2.2.1",
30 | "retry-axios": "^2.4.0",
31 | "streamdeck-typescript": "^3.0.0",
32 | "tsc-watch": "^4.2.9",
33 | "typescript": "^4.2.2",
34 | "watchify": "^4.0.0"
35 | }
36 | }
37 |
--------------------------------------------------------------------------------
/src/restreamio-plugin.ts:
--------------------------------------------------------------------------------
1 | import {SDOnActionEvent, StreamDeckPluginHandler} from 'streamdeck-typescript'
2 | import {ChannelAction} from './actions/channel-action'
3 | import {RestreamioClient} from './restreamio/restreamio-client'
4 | import {ManagedGlobalSettings} from './settings/managed-global-settings'
5 | import {
6 | DeferredFunction,
7 | DeferredGlobalSettingsRequests,
8 | } from './settings/deferred-global-settings-requests'
9 |
10 | export class RestreamioPlugin extends StreamDeckPluginHandler {
11 | protected deferredGlobalSettingsRequests: DeferredGlobalSettingsRequests
12 |
13 | constructor() {
14 | super()
15 | this.deferredGlobalSettingsRequests = new DeferredGlobalSettingsRequests(
16 | this.settingsManager,
17 | )
18 | new ChannelAction(this, 'com.dflydev.streamdeck.restreamio.channel')
19 | }
20 |
21 | get managedGlobalSettings(): ManagedGlobalSettings {
22 | return new ManagedGlobalSettings(this.settingsManager)
23 | }
24 |
25 | @SDOnActionEvent('setupReady')
26 | private async onSetupReady(): Promise {
27 | const restreamioStreamingPlatforms = await RestreamioClient.fromSettingsManager(
28 | this.settingsManager,
29 | ).getStreamingPlatforms()
30 |
31 | this.managedGlobalSettings.setRestreamioStreamingPlatforms(
32 | restreamioStreamingPlatforms,
33 | )
34 |
35 | this.deferredGlobalSettingsRequests.handle()
36 | }
37 |
38 | withGlobalSettings(context: string, fnc: DeferredFunction): void {
39 | this.deferredGlobalSettingsRequests.deferOrNow(
40 | this.globalSettingsReady,
41 | fnc,
42 | context,
43 | )
44 | }
45 | }
46 |
47 | new RestreamioPlugin()
48 |
--------------------------------------------------------------------------------
/src/restreamio/types.ts:
--------------------------------------------------------------------------------
1 | export interface RestreamioAccountCollection {
2 | [key: string]: RestreamioAccount
3 | }
4 |
5 | export interface RestreamioAccount {
6 | profile: RestreamioProfile
7 | token: RestreamioToken
8 | }
9 |
10 | export interface RestreamioProfile {
11 | id: RestreamioAccountId
12 | email: RestreamioEmail
13 | username: RestreamioUsername
14 | }
15 |
16 | export type RestreamioRefreshedTokenCallback = (
17 | restreamioToken: RestreamioToken,
18 | ) => void
19 | export type RestreamioChannelId = number
20 |
21 | export type RestreamioAccountId = number
22 | export type RestreamioEmail = string
23 | export type RestreamioUsername = string
24 |
25 | export type RestreamioStreamingPlatformId = number
26 |
27 | export type RestreamioStreamingPlatformImageUrl = string
28 |
29 | export interface RestreamioStreamingPlatformImage {
30 | svg: RestreamioStreamingPlatformImageUrl
31 | png: RestreamioStreamingPlatformImageUrl
32 | }
33 |
34 | export interface RestreamioStreamingPlatform {
35 | id: RestreamioStreamingPlatformId
36 | name: string
37 | url: string
38 | image: RestreamioStreamingPlatformImage
39 | altImage: RestreamioStreamingPlatformImage
40 | }
41 |
42 | export interface RestreamioToken {
43 | tokenType: RestreamioTokenType
44 | tokenScope: RestreamioTokenScope
45 | accessToken: RestreamioAccessToken
46 | accessTokenEpochExpirationTime: RestreamioAccessTokenEpochExpirationTime
47 | refreshToken: RestreamioRefreshToken
48 | refreshTokenEpochExpirationTime: RestreamioRefreshTokenEpochExpirationTime
49 | }
50 |
51 | export type RestreamioEpochExpirationTime = bigint
52 |
53 | export type RestreamioTokenType = string
54 | export type RestreamioTokenScope = string
55 |
56 | export type RestreamioAccessToken = string
57 | export type RestreamioAccessTokenEpochExpirationTime = RestreamioEpochExpirationTime
58 |
59 | export type RestreamioRefreshToken = string
60 | export type RestreamioRefreshTokenEpochExpirationTime = RestreamioEpochExpirationTime
61 |
--------------------------------------------------------------------------------
/.github/workflows/pre-release.yml:
--------------------------------------------------------------------------------
1 | name: "pre-release"
2 |
3 | on:
4 | push:
5 | branches:
6 | - "main"
7 |
8 | jobs:
9 | pre-release:
10 | name: "Pre Release"
11 | runs-on: "macos-latest"
12 |
13 | steps:
14 | - uses: actions/checkout@v2
15 | with:
16 | fetch-depth: 0
17 |
18 | - uses: actions/setup-node@v2
19 |
20 | - name: Get npm cache directory
21 | id: npm_cache
22 | run: |
23 | echo "::set-output name=dir::$(npm config get cache)"
24 |
25 | - name: Cache npm cache
26 | uses: actions/cache@v1
27 | with:
28 | path: ${{ steps.npm_cache.outputs.dir }}
29 | key: ${{ runner.os }}-npm-cache-${{ hashFiles('**/package-lock.json') }}
30 | restore-keys: |
31 | ${{ runner.os }}-npm-cache-
32 |
33 | - name: Cache npm dependencies
34 | uses: actions/cache@v1
35 | with:
36 | path: node_modules
37 | key: ${{ runner.os }}-npm-node-modules-${{ hashFiles('**/package-lock.json') }}
38 | restore-keys: |
39 | ${{ runner.os }}-npm-node-modules-
40 |
41 | - run: npm install
42 |
43 | - run: npm run lint
44 |
45 | - name: Prepare the environment
46 | run: cp .env.example .env
47 |
48 | - run: npm run build
49 | env:
50 | STREAMDECK_RESTREAMIO_LOGIN_URL: ${{ secrets.STREAMDECK_RESTREAMIO_LOGIN_URL }}
51 | STREAMDECK_RESTREAMIO_REFRESH_URL: ${{ secrets.STREAMDECK_RESTREAMIO_REFRESH_URL }}
52 |
53 | - name: Copy additional assets
54 | run: |
55 | cp LICENSE.txt README.md com.dflydev.streamdeck.restreamio.sdPlugin
56 |
57 | - name: StreamDeck Distribution Tool
58 | uses: AdamCarballo/streamdeck-distribution-tool@v1
59 | with:
60 | input: com.dflydev.streamdeck.restreamio.sdPlugin
61 | output: release
62 |
63 | - uses: "marvinpinto/action-automatic-releases@latest"
64 | with:
65 | repo_token: "${{ secrets.GITHUB_TOKEN }}"
66 | automatic_release_tag: "latest"
67 | prerelease: true
68 | title: "Latest Build"
69 | files: |
70 | release/com.dflydev.streamdeck.restreamio.streamDeckPlugin
71 |
--------------------------------------------------------------------------------
/src/settings/managed-settings.ts:
--------------------------------------------------------------------------------
1 | import {RestreamioAccountId, RestreamioChannelId} from '../restreamio/types'
2 | import {SettingsManager} from 'streamdeck-typescript'
3 | import {createEmptySettings, Settings} from './settings'
4 |
5 | export class ManagedSettings implements Settings {
6 | private context: string
7 | private settingsManager: SettingsManager
8 |
9 | private get rawSettings(): Settings {
10 | const rawSettings = this.settingsManager.getContextSettings(
11 | this.context,
12 | )
13 |
14 | if (!rawSettings) {
15 | return createEmptySettings()
16 | }
17 |
18 | return rawSettings
19 | }
20 |
21 | get restreamioAccountId(): RestreamioAccountId | null {
22 | return this.rawSettings.restreamioAccountId
23 | }
24 |
25 | get restreamioChannelId(): RestreamioChannelId | null {
26 | return this.rawSettings.restreamioChannelId
27 | }
28 |
29 | get name(): string | null {
30 | return this.rawSettings.name
31 | }
32 |
33 | get platform(): string | null {
34 | return this.rawSettings.platform
35 | }
36 |
37 | constructor(settingsManager: SettingsManager, context: string) {
38 | this.settingsManager = settingsManager
39 | this.context = context
40 | }
41 |
42 | assignRestreamioAccountId(restreamioAccountId: RestreamioAccountId): void {
43 | this.settingsManager.setContextSettingsAttributes(this.context, {
44 | restreamioAccountId,
45 | })
46 | }
47 |
48 | assignRestreamioChannelId(
49 | restreamioChannelId: RestreamioChannelId,
50 | platform: string,
51 | name: string,
52 | ): void {
53 | this.settingsManager.setContextSettingsAttributes(this.context, {
54 | restreamioChannelId,
55 | platform,
56 | name,
57 | })
58 | }
59 |
60 | assignRestreamioChannelPlatform(platform: string): void {
61 | this.settingsManager.setContextSettingsAttributes(this.context, {
62 | platform,
63 | })
64 | }
65 |
66 | assignRestreamioChannelName(name: string): void {
67 | this.settingsManager.setContextSettingsAttributes(this.context, {
68 | name,
69 | })
70 | }
71 |
72 | unassignRestreamioAccount(): void {
73 | this.settingsManager.setContextSettingsAttributes(this.context, {
74 | restreamioAccountId: null,
75 | restreamioChannelId: null,
76 | platform: null,
77 | name: null,
78 | })
79 | }
80 | }
81 |
--------------------------------------------------------------------------------
/src/settings/managed-global-settings.ts:
--------------------------------------------------------------------------------
1 | import {
2 | RestreamioAccount,
3 | RestreamioAccountCollection,
4 | RestreamioAccountId,
5 | RestreamioStreamingPlatform,
6 | RestreamioStreamingPlatformId,
7 | } from '../restreamio/types'
8 | import {SettingsManager} from 'streamdeck-typescript'
9 | import {createEmptyGlobalSettings, GlobalSettings} from './global-settings'
10 |
11 | export class ManagedGlobalSettings implements GlobalSettings {
12 | private settingsManager: SettingsManager
13 |
14 | get rawGlobalSettings(): GlobalSettings {
15 | const rawGlobalSettings = this.settingsManager.getGlobalSettings() as GlobalSettings
16 |
17 | if (!rawGlobalSettings) {
18 | return createEmptyGlobalSettings()
19 | }
20 |
21 | return rawGlobalSettings
22 | }
23 |
24 | get restreamioAccounts(): RestreamioAccountCollection {
25 | return this.rawGlobalSettings.restreamioAccounts
26 | }
27 |
28 | get restreamioStreamingPlatforms(): RestreamioStreamingPlatform[] {
29 | return this.rawGlobalSettings.restreamioStreamingPlatforms
30 | }
31 |
32 | constructor(settingsManager: SettingsManager) {
33 | this.settingsManager = settingsManager
34 | }
35 |
36 | setRestreamioStreamingPlatforms(
37 | restreamioStreamingPlatforms: RestreamioStreamingPlatform[],
38 | ): void {
39 | this.settingsManager.setGlobalSettingsAttributes({
40 | restreamioStreamingPlatforms,
41 | })
42 | }
43 |
44 | registerRestreamioAccount(
45 | restreamioAccount: RestreamioAccount,
46 | ): RestreamioAccountId {
47 | this.settingsManager.setGlobalSettings({
48 | ...this.rawGlobalSettings,
49 | restreamioAccounts: {
50 | ...this.rawGlobalSettings.restreamioAccounts,
51 | [restreamioAccount.profile.id]: restreamioAccount,
52 | },
53 | })
54 |
55 | return restreamioAccount.profile.id
56 | }
57 |
58 | getRestreamioAccounts(): RestreamioAccount[] {
59 | if (!this.rawGlobalSettings.restreamioAccounts) {
60 | return []
61 | }
62 |
63 | return Object.keys(this.rawGlobalSettings.restreamioAccounts).map(
64 | k => this.rawGlobalSettings.restreamioAccounts[k],
65 | )
66 | }
67 |
68 | getRestreamioStreamingPlatformById(
69 | restreamStreamingPlatformId: RestreamioStreamingPlatformId,
70 | ): RestreamioStreamingPlatform | undefined {
71 | return this.rawGlobalSettings.restreamioStreamingPlatforms.find(
72 | platform => platform.id === restreamStreamingPlatformId,
73 | )
74 | }
75 | }
76 |
--------------------------------------------------------------------------------
/src/settings/deferred-global-settings-requests.ts:
--------------------------------------------------------------------------------
1 | import {ManagedGlobalSettings} from './managed-global-settings'
2 | import {ManagedSettings} from './managed-settings'
3 | import {SettingsManager} from 'streamdeck-typescript'
4 |
5 | export type DeferredFunctionWithoutContext = (
6 | managedGlobalSettings: ManagedGlobalSettings,
7 | ) => void
8 |
9 | export type DeferredFunctionWithContext = (
10 | managedGlobalSettings: ManagedGlobalSettings,
11 | managedSeetings: ManagedSettings,
12 | ) => void
13 |
14 | export type DeferredFunction =
15 | | DeferredFunctionWithoutContext
16 | | DeferredFunctionWithContext
17 |
18 | interface DeferredRequest {
19 | fnc: DeferredFunction
20 | context?: string
21 | }
22 |
23 | export class DeferredGlobalSettingsRequests {
24 | private deferredGlobalSettingsRequests: DeferredRequest[] = []
25 | private settingsManager: SettingsManager
26 |
27 | constructor(settingsManager: SettingsManager) {
28 | this.settingsManager = settingsManager
29 | }
30 |
31 | handle(): void {
32 | while (this.deferredGlobalSettingsRequests.length > 0) {
33 | const deferredRequest = this.deferredGlobalSettingsRequests.shift()
34 |
35 | if (!deferredRequest) {
36 | continue
37 | }
38 |
39 | if (deferredRequest.context) {
40 | this.withGlobalSettingsResolvedForContext(
41 | deferredRequest.fnc,
42 | deferredRequest.context,
43 | )
44 | } else {
45 | this.withGlobalSettingsResolved(
46 | deferredRequest.fnc as DeferredFunctionWithoutContext,
47 | )
48 | }
49 | }
50 | }
51 |
52 | deferOrNow(isReady: boolean, fnc: DeferredFunction, context?: string): void {
53 | if (isReady) {
54 | if (context) {
55 | this.withGlobalSettingsResolvedForContext(fnc, context)
56 | } else {
57 | this.withGlobalSettingsResolved(fnc as DeferredFunctionWithoutContext)
58 | }
59 | } else {
60 | this.defer(fnc, context)
61 | }
62 | }
63 |
64 | defer(fnc: DeferredFunction, context?: string): void {
65 | this.deferredGlobalSettingsRequests.push({fnc, context})
66 | }
67 |
68 | private withGlobalSettingsResolved(
69 | fnc: DeferredFunctionWithoutContext,
70 | ): void {
71 | fnc(new ManagedGlobalSettings(this.settingsManager))
72 | }
73 |
74 | private withGlobalSettingsResolvedForContext(
75 | fnc: DeferredFunctionWithContext,
76 | context: string,
77 | ): void {
78 | fnc(
79 | new ManagedGlobalSettings(this.settingsManager),
80 | new ManagedSettings(this.settingsManager, context ?? ''),
81 | )
82 | }
83 | }
84 |
--------------------------------------------------------------------------------
/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "plugins": ["@typescript-eslint"],
3 | "extends": [
4 | "plugin:github/browser",
5 | "plugin:@typescript-eslint/recommended",
6 | "prettier"
7 | ],
8 | "parser": "@typescript-eslint/parser",
9 | "parserOptions": {
10 | "ecmaVersion": 9,
11 | "sourceType": "module",
12 | "project": "./tsconfig.eslint.json"
13 | },
14 | "rules": {
15 | "eslint-comments/no-use": "off",
16 | "import/no-namespace": "off",
17 | "no-unused-vars": "off",
18 | "@typescript-eslint/no-unused-vars": "error",
19 | "@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}],
20 | "@typescript-eslint/no-require-imports": "error",
21 | "@typescript-eslint/array-type": "error",
22 | "@typescript-eslint/await-thenable": "error",
23 | //"@typescript-eslint/ban-ts-ignore": "error",
24 | "camelcase": "off",
25 | "@typescript-eslint/naming-convention": "error",
26 | "@typescript-eslint/explicit-function-return-type": ["error", {"allowExpressions": true}],
27 | "@typescript-eslint/func-call-spacing": ["error", "never"],
28 | "@typescript-eslint/no-array-constructor": "error",
29 | "@typescript-eslint/no-empty-interface": "error",
30 | "@typescript-eslint/no-explicit-any": "error",
31 | "@typescript-eslint/no-extraneous-class": "error",
32 | "@typescript-eslint/no-for-in-array": "error",
33 | "@typescript-eslint/no-inferrable-types": "error",
34 | "@typescript-eslint/no-misused-new": "error",
35 | "@typescript-eslint/no-namespace": "error",
36 | "@typescript-eslint/no-non-null-assertion": "warn",
37 | //"@typescript-eslint/no-object-literal-type-assertion": "error",
38 | "@typescript-eslint/no-unnecessary-qualifier": "error",
39 | "@typescript-eslint/no-unnecessary-type-assertion": "error",
40 | "@typescript-eslint/no-useless-constructor": "error",
41 | "@typescript-eslint/no-var-requires": "error",
42 | "@typescript-eslint/prefer-for-of": "warn",
43 | "@typescript-eslint/prefer-function-type": "warn",
44 | "@typescript-eslint/prefer-includes": "error",
45 | //"@typescript-eslint/prefer-interface": "error",
46 | "@typescript-eslint/prefer-string-starts-ends-with": "error",
47 | "@typescript-eslint/promise-function-async": "error",
48 | "@typescript-eslint/require-array-sort-compare": "error",
49 | "@typescript-eslint/restrict-plus-operands": "error",
50 | "semi": "off",
51 | "@typescript-eslint/semi": ["error", "never"],
52 | "@typescript-eslint/type-annotation-spacing": "error",
53 | "@typescript-eslint/unbound-method": "error"
54 | },
55 | "env": {
56 | "node": true,
57 | "es6": true
58 | }
59 | }
60 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | 🔌 Stream Deck Restream.io plugin
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 | > Integrate [Elgato Stream Deck](https://www.elgato.com/en/gaming/stream-deck) with [Restream.io](https://restream.io).
21 |
22 | ## 🚀 Quick start
23 |
24 | [Download the latest release](https://github.com/dflydev/streamdeck-restreamio/releases/download/latest/com.dflydev.streamdeck.restreamio.streamDeckPlugin) from GitHub.
25 |
26 | ## Author
27 |
28 | 👥 **dflydev**
29 |
30 | * Website: [dflydev.com](https://dflydev.com?utm_source=streamdeck-restreamio&utm_medium=readme-author)
31 | * Twitter: [@dflydev](https://twitter.com/dflydev)
32 | * Github: [@dflydev](https://github.com/dflydev)
33 |
34 | 👤 **Beau Simensen**
35 |
36 | * Website: [beausimensen.com](https://beausimensen.com?utm_source=streamdeck-restreamio&utm_medium=readme-author)
37 | * Twitter: [@dflydev](https://twitter.com/beausimensen)
38 | * Github: [@simensen](https://github.com/simensen)
39 |
40 | ## 🤝 Contributing
41 |
42 | Contributions, issues and feature requests are welcome!
Feel free to check [issues page](https://github.com/dflydev/streamdeck-restreamio/issues).
43 |
44 |
45 | ## ❤️ Support the development
46 |
47 | Give a ⭐️ if this project helped you!
48 |
49 | Did this project save you time? Did this project increase your productivity? Did this project solve a problem for you? Did this project make your life easier? Please also consider donating or buying a license!
50 |
51 | * [Sponsor Beau Simensen on GitHub](https://github.com/sponsors/simensen)
52 | * [Sponsor dflydev on GitHub](https://github.com/sponsors/dflydev)
53 | * [Sponsor dflydev on OpenCollecctive](https://opencollective.com/dflydev)
54 |
55 |
56 | ## 📝 License
57 |
58 | Copyright © 2021 [dflydev](https://github.com/dflydev).
59 | This project is [MIT](https://github.com/dflydev/streamdeck-restreamio/blob/master/LICENSE) licensed.
60 |
--------------------------------------------------------------------------------
/src/actions/channel-action.ts:
--------------------------------------------------------------------------------
1 | import {
2 | DidReceiveSettingsEvent,
3 | KeyUpEvent,
4 | SDOnActionEvent,
5 | StateType,
6 | WillAppearEvent,
7 | WillDisappearEvent,
8 | } from 'streamdeck-typescript'
9 | import {RestreamioPlugin} from '../restreamio-plugin'
10 | import {Action} from './action'
11 | import {Settings} from '../settings/settings'
12 |
13 | export class ChannelAction extends Action {
14 | private channels: Set
15 |
16 | constructor(plugin: RestreamioPlugin, actionName: string) {
17 | super(plugin, actionName)
18 | this.channels = new Set()
19 | }
20 |
21 | @SDOnActionEvent('willAppear')
22 | private onWillAppear({
23 | context,
24 | payload: {isInMultiAction},
25 | }: WillAppearEvent): void {
26 | if (!isInMultiAction) {
27 | this.channels.add(context)
28 | }
29 |
30 | this.plugin.setState(StateType.OFF, context)
31 | this.updateUi(context)
32 |
33 | this.plugin.withGlobalSettings(context, () => this.getChannelState(context))
34 | }
35 |
36 | @SDOnActionEvent('willDisappear')
37 | private onWillDisappear({
38 | context,
39 | payload: {isInMultiAction},
40 | }: WillDisappearEvent): void {
41 | if (!isInMultiAction) {
42 | this.channels.delete(context)
43 | }
44 | }
45 |
46 | private getChannelState(context: string): void {
47 | this.restreamioClient(context)
48 | .getChannel()
49 | .then(channelData => {
50 | const channel = channelData as {active: boolean}
51 | this.plugin.setState(
52 | channel.active ? StateType.ON : StateType.OFF,
53 | context,
54 | )
55 | })
56 | }
57 |
58 | @SDOnActionEvent('didReceiveSettings')
59 | private onDidReceiveSettings({context}: DidReceiveSettingsEvent): void {
60 | this.updateUi(context)
61 | }
62 |
63 | @SDOnActionEvent('keyUp')
64 | private async onKeyUp({
65 | context,
66 | payload: {settings, isInMultiAction, userDesiredState, state},
67 | }: KeyUpEvent): Promise {
68 | if (isInMultiAction) {
69 | if (userDesiredState === StateType.ON) {
70 | await this.restreamioClient(context).enableChannel()
71 | this.setAllChannelActionsTo(context, settings, StateType.ON)
72 | } else {
73 | await this.restreamioClient(context).disableChannel()
74 | this.setAllChannelActionsTo(context, settings, StateType.OFF)
75 | }
76 | } else {
77 | if (state === StateType.OFF) {
78 | try {
79 | await this.restreamioClient(context).enableChannel()
80 | this.plugin.showOk(context)
81 | this.setAllChannelActionsTo(context, settings, StateType.ON)
82 | } catch (e) {
83 | console.log('Could not enable channel', e)
84 | this.plugin.setState(StateType.OFF, context)
85 | this.plugin.showAlert(context)
86 | }
87 | }
88 | if (state === StateType.ON) {
89 | try {
90 | await this.restreamioClient(context).disableChannel()
91 | this.plugin.showOk(context)
92 | this.setAllChannelActionsTo(context, settings, StateType.OFF)
93 | } catch (e) {
94 | console.log('Could not disable channel', e)
95 | this.plugin.setState(StateType.ON, context)
96 | this.plugin.showAlert(context)
97 | }
98 | }
99 | }
100 |
101 | return Promise.resolve()
102 | }
103 |
104 | private setAllChannelActionsTo(
105 | context: string,
106 | settings: Settings,
107 | desiredState: StateType,
108 | ): void {
109 | const ourContext = context
110 | this.channels.forEach((context: string) => {
111 | if (ourContext === context) {
112 | return
113 | }
114 |
115 | const theirSettings = this.settings(context)
116 |
117 | if (theirSettings.restreamioAccountId !== settings.restreamioAccountId) {
118 | return
119 | }
120 |
121 | if (theirSettings.restreamioChannelId !== settings.restreamioChannelId) {
122 | return
123 | }
124 |
125 | this.plugin.setState(desiredState, context)
126 | })
127 | }
128 |
129 | private async updateUi(context: string): Promise {
130 | if (
131 | !this.settings(context).restreamioAccountId ||
132 | !this.settings(context).restreamioChannelId
133 | ) {
134 | return Promise.resolve()
135 | }
136 |
137 | const platform = this.settings(context).platform
138 | const name = this.settings(context).name
139 |
140 | this.plugin.setTitle(`${name}\n${platform}`, context)
141 |
142 | return Promise.resolve()
143 | }
144 | }
145 |
--------------------------------------------------------------------------------
/src/restreamio/restreamio-client.ts:
--------------------------------------------------------------------------------
1 | import axios, {AxiosInstance, AxiosRequestConfig} from 'axios'
2 | import {ManagedGlobalSettings} from '../settings/managed-global-settings'
3 | import {ManagedSettings} from '../settings/managed-settings'
4 | import {RestreamioStreamingPlatform, RestreamioToken} from './types'
5 | import {SettingsManager} from 'streamdeck-typescript'
6 | import createAuthRefreshInterceptor from 'axios-auth-refresh'
7 | import * as rax from 'retry-axios'
8 |
9 | export class RestreamioClient {
10 | private managedGlobalSettings: ManagedGlobalSettings
11 | private managedSettings?: ManagedSettings
12 | private authenticatedAxios: AxiosInstance
13 | private unAuthenticatedAxios: AxiosInstance
14 |
15 | static fromSettingsManager(
16 | settingsManager: SettingsManager,
17 | context?: string,
18 | ): RestreamioClient {
19 | return new RestreamioClient(
20 | new ManagedGlobalSettings(settingsManager),
21 | context ? new ManagedSettings(settingsManager, context) : undefined,
22 | )
23 | }
24 |
25 | constructor(
26 | managedGlobalSettings: ManagedGlobalSettings,
27 | managedSettings?: ManagedSettings,
28 | ) {
29 | this.managedGlobalSettings = managedGlobalSettings
30 | this.managedSettings = managedSettings
31 | this.authenticatedAxios = this.createAuthenticatedAxios()
32 | this.unAuthenticatedAxios = RestreamioClient.createUnauthenticatedAxios()
33 | }
34 |
35 | private static createUnauthenticatedAxios(): AxiosInstance {
36 | const unauthenticatedAxios = axios.create()
37 |
38 | unauthenticatedAxios.defaults.raxConfig = {
39 | instance: unauthenticatedAxios,
40 | }
41 |
42 | rax.attach(unauthenticatedAxios)
43 |
44 | return unauthenticatedAxios
45 | }
46 |
47 | private createAuthenticatedAxios(): AxiosInstance {
48 | const authenticatedAxios = axios.create()
49 |
50 | authenticatedAxios.interceptors.request.use(request => {
51 | const token = this.getToken()
52 |
53 | if (!token) {
54 | return request
55 | }
56 |
57 | request.headers['Authorization'] = `Bearer ${token.accessToken}`
58 |
59 | return request
60 | })
61 |
62 | createAuthRefreshInterceptor(authenticatedAxios, async failedRequest => {
63 | const formData = new FormData()
64 | formData.append('refresh_token', this.getToken()?.refreshToken ?? '')
65 | const refreshUrl = process.env.STREAMDECK_RESTREAMIO_REFRESH_URL
66 |
67 | if (!refreshUrl) {
68 | throw new Error('Refresh URL is not defined')
69 | }
70 |
71 | return axios
72 | .post(refreshUrl, formData)
73 | .then(async tokenRefreshResponse => {
74 | this.managedGlobalSettings.registerRestreamioAccount(
75 | tokenRefreshResponse.data,
76 | )
77 | failedRequest.response.config.headers[
78 | 'Authorization'
79 | ] = `Bearer ${tokenRefreshResponse.data.token.accessToken}`
80 |
81 | return Promise.resolve()
82 | })
83 | })
84 |
85 | authenticatedAxios.defaults.raxConfig = {
86 | instance: authenticatedAxios,
87 | }
88 |
89 | rax.attach(authenticatedAxios)
90 |
91 | return authenticatedAxios
92 | }
93 |
94 | async getChannel(): Promise {
95 | if (!this.managedSettings?.restreamioChannelId) {
96 | return {active: false}
97 | }
98 |
99 | return this.get(
100 | `https://api.restream.io/v2/user/channel/${this.managedSettings?.restreamioChannelId}`,
101 | )
102 | }
103 |
104 | async enableChannel(): Promise {
105 | if (!this.managedSettings?.restreamioChannelId) {
106 | return
107 | }
108 |
109 | return await this.patch(
110 | `https://api.restream.io/v2/user/channel/${this.managedSettings?.restreamioChannelId}`,
111 | {
112 | active: true,
113 | },
114 | )
115 | }
116 |
117 | async disableChannel(): Promise {
118 | if (!this.managedSettings?.restreamioChannelId) {
119 | return
120 | }
121 |
122 | return await this.patch(
123 | `https://api.restream.io/v2/user/channel/${this.managedSettings?.restreamioChannelId}`,
124 | {
125 | active: false,
126 | },
127 | )
128 | }
129 |
130 | async getChannels(): Promise {
131 | return this.get('https://api.restream.io/v2/user/channel/all')
132 | }
133 |
134 | private getToken(): RestreamioToken | null {
135 | if (!this.managedSettings?.restreamioAccountId) {
136 | return null
137 | }
138 |
139 | return this.managedGlobalSettings.restreamioAccounts[
140 | this.managedSettings.restreamioAccountId
141 | ].token
142 | }
143 |
144 | async getStreamingPlatforms(): Promise {
145 | const options = {
146 | method: 'GET',
147 | url: 'https://api.restream.io/v2/platform/all',
148 | } as AxiosRequestConfig
149 |
150 | return await axios.request(options).then(response => {
151 | return response.data
152 | })
153 | }
154 |
155 | private async get(url: string): Promise {
156 | const options = {
157 | method: 'GET',
158 | url,
159 | } as AxiosRequestConfig
160 |
161 | return await this.authenticatedAxios
162 | .request(options)
163 | .then(function (response) {
164 | return response.data
165 | })
166 | }
167 |
168 | private async patch(url: string, payload: unknown): Promise {
169 | const options = {
170 | method: 'PATCH',
171 | url,
172 | data: payload,
173 | } as AxiosRequestConfig
174 |
175 | return await this.authenticatedAxios.request(options)
176 | }
177 | }
178 |
--------------------------------------------------------------------------------
/src/restreamio-property-inspector.ts:
--------------------------------------------------------------------------------
1 | import {
2 | DidReceiveSettingsEvent,
3 | SDOnPiEvent,
4 | StreamDeckPropertyInspectorHandler,
5 | } from 'streamdeck-typescript'
6 | import {GlobalSettings} from './settings/global-settings'
7 | import {ManagedGlobalSettings} from './settings/managed-global-settings'
8 | import {ManagedSettings} from './settings/managed-settings'
9 | import {RestreamioAccount} from './restreamio/types'
10 | import {Settings} from './settings/settings'
11 | import {RestreamioClient} from './restreamio/restreamio-client'
12 | import {
13 | DeferredFunction,
14 | DeferredGlobalSettingsRequests,
15 | } from './settings/deferred-global-settings-requests'
16 | import Timeout = NodeJS.Timeout
17 |
18 | class RestreamioPropertyInspector extends StreamDeckPropertyInspectorHandler<
19 | Settings,
20 | GlobalSettings
21 | > {
22 | private deferredGlobalSettingsRequests: DeferredGlobalSettingsRequests
23 | private restreamioAccountIdElement: HTMLSelectElement
24 | private restreamioChannelIdElement: HTMLSelectElement
25 | private restreamioChannelElement: HTMLDivElement
26 | private restreamioChannelPlatformWrapperElement: HTMLDivElement
27 | private restreamioChannelNameWrapperElement: HTMLDivElement
28 | private restreamioChannelPlatformFieldElement: HTMLInputElement
29 | private restreamioChannelNameFieldElement: HTMLInputElement
30 |
31 | private authWindow: Window | null
32 |
33 | private updateUiTimeoutId: Timeout
34 |
35 | constructor() {
36 | super()
37 | this.deferredGlobalSettingsRequests = new DeferredGlobalSettingsRequests(
38 | this.settingsManager,
39 | )
40 | }
41 |
42 | @SDOnPiEvent('documentLoaded')
43 | private onDocumentLoaded(): void {
44 | window.addEventListener('message', e => {
45 | this.withGlobalSettings(
46 | (globalSettings: ManagedGlobalSettings, settings: ManagedSettings) => {
47 | const restreamioAccountId = globalSettings.registerRestreamioAccount(
48 | e.data,
49 | )
50 | settings.assignRestreamioAccountId(restreamioAccountId)
51 |
52 | if (this.authWindow) {
53 | this.authWindow.close()
54 |
55 | this.authWindow = null
56 | }
57 |
58 | this.updateUi()
59 | },
60 | )
61 | })
62 |
63 | this.restreamioAccountIdElement = document.getElementById(
64 | 'restreamio_account_id',
65 | ) as HTMLSelectElement
66 | this.restreamioAccountIdElement.onchange = () => {
67 | const selectedValue = this.restreamioAccountIdElement.value
68 |
69 | if (selectedValue === '__NEW__') {
70 | return
71 | }
72 |
73 | this.withGlobalSettings(
74 | (globalSettings: ManagedGlobalSettings, settings: ManagedSettings) => {
75 | if (selectedValue) {
76 | settings.assignRestreamioAccountId(Number(selectedValue))
77 | } else {
78 | settings.unassignRestreamioAccount()
79 | }
80 |
81 | this.updateUi()
82 | },
83 | )
84 | }
85 |
86 | this.restreamioAccountIdElement.onclick = () => {
87 | if (this.restreamioAccountIdElement.value !== '__NEW__') {
88 | return
89 | }
90 |
91 | const externalUrl = process.env.STREAMDECK_RESTREAMIO_LOGIN_URL
92 |
93 | this.authWindow = window.open(externalUrl, 'Reastream.io')
94 | }
95 |
96 | this.restreamioChannelElement = document.getElementById(
97 | 'restreamio_channel',
98 | ) as HTMLDivElement
99 | this.restreamioChannelPlatformWrapperElement = document.getElementById(
100 | 'restreamio_channel_platform_wrapper',
101 | ) as HTMLDivElement
102 | this.restreamioChannelNameWrapperElement = document.getElementById(
103 | 'restreamio_channel_name_wrapper',
104 | ) as HTMLDivElement
105 | this.restreamioChannelIdElement = document.getElementById(
106 | 'restreamio_channel_id',
107 | ) as HTMLSelectElement
108 | this.restreamioChannelPlatformFieldElement = document.getElementById(
109 | 'restreamio_channel_platform',
110 | ) as HTMLInputElement
111 | this.restreamioChannelNameFieldElement = document.getElementById(
112 | 'restreamio_channel_name',
113 | ) as HTMLInputElement
114 | this.restreamioChannelIdElement.onchange = () => {
115 | const selectedValue = this.restreamioChannelIdElement.value
116 |
117 | this.withGlobalSettings(
118 | (globalSettings: ManagedGlobalSettings, settings: ManagedSettings) => {
119 | const label =
120 | this.restreamioChannelIdElement.selectedOptions.item(0)?.text ||
121 | 'Unknown – Unknown'
122 | const splitLabel = label.split(' – ')
123 | const platform = splitLabel.shift()
124 | const name = splitLabel.shift()
125 | settings.assignRestreamioChannelId(
126 | Number(selectedValue),
127 | platform ?? 'Unknown',
128 | name ?? 'Unknown',
129 | )
130 |
131 | this.updateUi()
132 | },
133 | )
134 | }
135 |
136 | this.restreamioChannelNameFieldElement.onchange = () => {
137 | this.withGlobalSettings(
138 | (globalSettings: ManagedGlobalSettings, settings: ManagedSettings) => {
139 | settings.assignRestreamioChannelName(
140 | this.restreamioChannelNameFieldElement.value,
141 | )
142 | },
143 | )
144 | }
145 |
146 | this.restreamioChannelPlatformFieldElement.onchange = () => {
147 | this.withGlobalSettings(
148 | (globalSettings: ManagedGlobalSettings, settings: ManagedSettings) => {
149 | settings.assignRestreamioChannelPlatform(
150 | this.restreamioChannelPlatformFieldElement.value,
151 | )
152 | },
153 | )
154 | }
155 |
156 | this.updateUi()
157 | }
158 |
159 | @SDOnPiEvent('globalSettingsAvailable')
160 | private async onGlobalSettingsAvailable(): Promise {
161 | this.deferredGlobalSettingsRequests.handle()
162 | }
163 |
164 | @SDOnPiEvent('didReceiveSettings')
165 | private async onDidReceiveSettings({
166 | payload: {settings},
167 | }: DidReceiveSettingsEvent): Promise {
168 | this.updateUi()
169 | }
170 |
171 | @SDOnPiEvent('didReceiveGlobalSettings')
172 | private onDidReceiveGlobalSettings(): void {
173 | this.updateUi()
174 | }
175 |
176 | withGlobalSettings(fnc: DeferredFunction): void {
177 | this.deferredGlobalSettingsRequests.deferOrNow(
178 | this.globalSettingsReady,
179 | fnc,
180 | this.actionInfo.context,
181 | )
182 | }
183 |
184 | private updateUi(): void {
185 | this.withGlobalSettings(
186 | async (
187 | globalSettings: ManagedGlobalSettings,
188 | settings: ManagedSettings,
189 | ) => {
190 | clearTimeout(this.updateUiTimeoutId)
191 | this.updateUiTimeoutId = setTimeout(async () => {
192 | this.updateRestreamioAccountUi(globalSettings, settings)
193 | await this.updateRestreamioChannelUi(globalSettings, settings)
194 | }, 100)
195 | },
196 | )
197 | }
198 |
199 | private async updateRestreamioAccountUi(
200 | globalSettings: ManagedGlobalSettings,
201 | settings: ManagedSettings,
202 | ): Promise {
203 | while (this.restreamioAccountIdElement.options.length > 0) {
204 | this.restreamioAccountIdElement.options.remove(0)
205 | }
206 |
207 | let anAccountIsSelected = false
208 |
209 | globalSettings
210 | .getRestreamioAccounts()
211 | .forEach((restreamioAccount: RestreamioAccount) => {
212 | const option = document.createElement('option')
213 | option.value = restreamioAccount.profile.id.toString()
214 | option.text = restreamioAccount.profile.username
215 | if (restreamioAccount.profile.id === settings.restreamioAccountId) {
216 | option.selected = true
217 |
218 | anAccountIsSelected = true
219 | }
220 | this.restreamioAccountIdElement.options.add(option)
221 | })
222 |
223 | const blankOption = document.createElement('option')
224 |
225 | if (!anAccountIsSelected) {
226 | blankOption.selected = !anAccountIsSelected
227 | }
228 |
229 | this.restreamioAccountIdElement.options.add(blankOption)
230 |
231 | const newOption = document.createElement('option')
232 | newOption.value = '__NEW__'
233 | newOption.text = 'Add new account'
234 |
235 | this.restreamioAccountIdElement.options.add(newOption)
236 | }
237 |
238 | private async updateRestreamioChannelUi(
239 | globalSettings: ManagedGlobalSettings,
240 | settings: ManagedSettings,
241 | ): Promise {
242 | const restreamioClient = new RestreamioClient(globalSettings, settings)
243 |
244 | let channels: any[] = []
245 |
246 | if (settings.restreamioAccountId) {
247 | try {
248 | channels = (await restreamioClient.getChannels()) as any[]
249 | } catch (e) {
250 | channels = []
251 | }
252 | }
253 |
254 | while (this.restreamioChannelIdElement.options.length > 0) {
255 | this.restreamioChannelIdElement.options.remove(0)
256 | }
257 |
258 | let aChannelIsSelected = false
259 |
260 | channels.forEach((channel: any) => {
261 | const option = document.createElement('option')
262 | const platform = globalSettings.getRestreamioStreamingPlatformById(
263 | channel.streamingPlatformId,
264 | )
265 | option.value = channel.id
266 | option.text = `${platform ? platform.name : 'Unknown'} – ${
267 | channel.displayName
268 | }`
269 | if (channel.id === settings.restreamioChannelId) {
270 | option.selected = true
271 |
272 | aChannelIsSelected = true
273 | }
274 | this.restreamioChannelIdElement.options.add(option)
275 | })
276 |
277 | if (!aChannelIsSelected) {
278 | const blankOption = document.createElement('option')
279 | blankOption.selected = true
280 | this.restreamioChannelIdElement.options.add(blankOption)
281 | }
282 |
283 | if (this.restreamioChannelIdElement.options.length === 1) {
284 | this.restreamioChannelElement.style.display = 'none'
285 | } else {
286 | this.restreamioChannelElement.style.display = 'flex'
287 | }
288 |
289 | if (aChannelIsSelected) {
290 | this.restreamioChannelPlatformFieldElement.value = settings.platform ?? ''
291 | this.restreamioChannelNameFieldElement.value = settings.name ?? ''
292 |
293 | this.restreamioChannelPlatformWrapperElement.style.display = 'flex'
294 | this.restreamioChannelNameWrapperElement.style.display = 'flex'
295 | } else {
296 | this.restreamioChannelPlatformWrapperElement.style.display = 'none'
297 | this.restreamioChannelNameWrapperElement.style.display = 'none'
298 | }
299 | }
300 | }
301 |
302 | new RestreamioPropertyInspector()
303 |
--------------------------------------------------------------------------------
/com.dflydev.streamdeck.restreamio.sdPlugin/public/sdpi.css:
--------------------------------------------------------------------------------
1 | html {
2 | --sdpi-bgcolor: #2D2D2D;
3 | --sdpi-background: #3D3D3D;
4 | --sdpi-color: #d8d8d8;
5 | --sdpi-bordercolor: #3a3a3a;
6 | --sdpi-borderradius: 0px;
7 | --sdpi-width: 224px;
8 | --sdpi-fontweight: 600;
9 | --sdpi-letterspacing: -0.25pt;
10 | height: 100%;
11 | width: 100%;
12 | overflow: hidden;
13 | }
14 |
15 | html, body {
16 | font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
17 | font-size: 9pt;
18 | background-color: var(--sdpi-bgcolor);
19 | color: #9a9a9a;
20 | }
21 |
22 | body {
23 | height: 100%;
24 | padding: 0;
25 | overflow-x: hidden;
26 | overflow-y: auto;
27 | margin: 0;
28 | -webkit-overflow-scrolling: touch;
29 | -webkit-text-size-adjust: 100%;
30 | -webkit-font-smoothing: antialiased;
31 | }
32 |
33 | mark {
34 | background-color: var(--sdpi-bgcolor);
35 | color: var(--sdpi-color);
36 | }
37 |
38 | .hidden {
39 | display: none;
40 | }
41 |
42 | hr, hr2 {
43 | -webkit-margin-before: 1em;
44 | -webkit-margin-after: 1em;
45 | border-style: none;
46 | background: var(--sdpi-background);
47 | height: 1px;
48 | }
49 |
50 | hr2,
51 | .sdpi-heading {
52 | display: flex;
53 | flex-basis: 100%;
54 | align-items: center;
55 | color: inherit;
56 | font-size: 9pt;
57 | margin: 8px 0px;
58 | }
59 |
60 | .sdpi-heading::before,
61 | .sdpi-heading::after {
62 | content: "";
63 | flex-grow: 1;
64 | background: var(--sdpi-background);
65 | height: 1px;
66 | font-size: 0px;
67 | line-height: 0px;
68 | margin: 0px 16px;
69 | }
70 |
71 | hr2 {
72 | height: 2px;
73 | }
74 |
75 | hr, hr2 {
76 | margin-left:16px;
77 | margin-right:16px;
78 | }
79 |
80 | .sdpi-item-value,
81 | option,
82 | input,
83 | select,
84 | button {
85 | font-size: 10pt;
86 | font-weight: var(--sdpi-fontweight);
87 | letter-spacing: var(--sdpi-letterspacing);
88 | }
89 |
90 |
91 |
92 | .win .sdpi-item-value,
93 | .win option,
94 | .win input,
95 | .win select,
96 | .win button {
97 | font-size: 11px;
98 | font-style: normal;
99 | letter-spacing: inherit;
100 | font-weight: 100;
101 | }
102 |
103 | .win button {
104 | font-size: 12px;
105 | }
106 |
107 | ::-webkit-progress-value,
108 | meter::-webkit-meter-optimum-value {
109 | border-radius: 2px;
110 | /* background: linear-gradient(#ccf, #99f 20%, #77f 45%, #77f 55%, #cdf); */
111 | }
112 |
113 | ::-webkit-progress-bar,
114 | meter::-webkit-meter-bar {
115 | border-radius: 3px;
116 | background: var(--sdpi-background);
117 | }
118 |
119 | ::-webkit-progress-bar:active,
120 | meter::-webkit-meter-bar:active {
121 | border-radius: 3px;
122 | background: #222222;
123 | }
124 | ::-webkit-progress-value:active,
125 | meter::-webkit-meter-optimum-value:active {
126 | background: #99f;
127 | }
128 |
129 | progress,
130 | progress.sdpi-item-value {
131 | min-height: 5px !important;
132 | height: 5px;
133 | background-color: #303030;
134 | }
135 |
136 | progress {
137 | margin-top: 8px !important;
138 | margin-bottom: 8px !important;
139 | }
140 |
141 | .full progress,
142 | progress.full {
143 | margin-top: 3px !important;
144 | }
145 |
146 | ::-webkit-progress-inner-element {
147 | background-color: transparent;
148 | }
149 |
150 |
151 | .sdpi-item[type="progress"] {
152 | margin-top: 4px !important;
153 | margin-bottom: 12px;
154 | min-height: 15px;
155 | }
156 |
157 | .sdpi-item-child.full:last-child {
158 | margin-bottom: 4px;
159 | }
160 |
161 | .tabs {
162 | /**
163 | * Setting display to flex makes this container lay
164 | * out its children using flexbox, the exact same
165 | * as in the above "Stepper input" example.
166 | */
167 | display: flex;
168 |
169 | border-bottom: 1px solid #D7DBDD;
170 | }
171 |
172 | .tab {
173 | cursor: pointer;
174 | padding: 5px 30px;
175 | color: #16a2d7;
176 | font-size: 9pt;
177 | border-bottom: 2px solid transparent;
178 | }
179 |
180 | .tab.is-tab-selected {
181 | border-bottom-color: #4ebbe4;
182 | }
183 |
184 | select {
185 | -webkit-appearance: none;
186 | -moz-appearance: none;
187 | -o-appearance: none;
188 | appearance: none;
189 | background: url(./icons/caret.svg) no-repeat 97% center;
190 | }
191 |
192 | label.sdpi-file-label,
193 | input[type="button"],
194 | input[type="submit"],
195 | input[type="reset"],
196 | input[type="file"],
197 | input[type=file]::-webkit-file-upload-button,
198 | button,
199 | select {
200 | color: var(--sdpi-color);
201 | border: 1pt solid #303030;
202 | font-size: 8pt;
203 | background-color: var(--sdpi-background);
204 | border-radius: var(--sdpi-borderradius);
205 | }
206 |
207 | label.sdpi-file-label,
208 | input[type="button"],
209 | input[type="submit"],
210 | input[type="reset"],
211 | input[type="file"],
212 | input[type=file]::-webkit-file-upload-button,
213 | button {
214 | border: 1pt solid var(--sdpi-color);
215 | border-radius: var(--sdpi-borderradius);
216 | min-height: 23px !important;
217 | height: 23px !important;
218 | margin-right: 8px;
219 | }
220 |
221 | input[type=number]::-webkit-inner-spin-button,
222 | input[type=number]::-webkit-outer-spin-button {
223 | -webkit-appearance: none;
224 | margin: 0;
225 | }
226 |
227 | input[type="file"] {
228 | border-radius: var(--sdpi-borderradius);
229 | max-width: 220px;
230 | }
231 |
232 | option {
233 | height: 1.5em;
234 | padding: 4px;
235 | }
236 |
237 | /* SDPI */
238 |
239 | .sdpi-wrapper {
240 | overflow-x: hidden;
241 | }
242 |
243 | .sdpi-item {
244 | display: flex;
245 | flex-direction: row;
246 | min-height: 32px;
247 | align-items: center;
248 | margin-top: 2px;
249 | max-width: 344px;
250 | }
251 |
252 | .sdpi-item:first-child {
253 | margin-top:1px;
254 | }
255 |
256 | .sdpi-item:last-child {
257 | margin-bottom: 0px;
258 | }
259 |
260 | .sdpi-item > *:not(.sdpi-item-label):not(meter):not(details) {
261 | min-height: 26px;
262 | padding: 0px 4px 0px 4px;
263 | }
264 |
265 | .sdpi-item > *:not(.sdpi-item-label.empty):not(meter) {
266 | min-height: 26px;
267 | padding: 0px 4px 0px 4px;
268 | }
269 |
270 |
271 | .sdpi-item-group {
272 | padding: 0 !important;
273 | }
274 |
275 | meter.sdpi-item-value {
276 | margin-left: 6px;
277 | }
278 |
279 | .sdpi-item[type="group"] {
280 | display: block;
281 | margin-top: 12px;
282 | margin-bottom: 12px;
283 | /* border: 1px solid white; */
284 | flex-direction: unset;
285 | text-align: left;
286 | }
287 |
288 | .sdpi-item[type="group"] > .sdpi-item-label,
289 | .sdpi-item[type="group"].sdpi-item-label {
290 | width: 96%;
291 | text-align: left;
292 | font-weight: 700;
293 | margin-bottom: 4px;
294 | padding-left: 4px;
295 | }
296 |
297 | dl,
298 | ul,
299 | ol {
300 | -webkit-margin-before: 0px;
301 | -webkit-margin-after: 4px;
302 | -webkit-padding-start: 1em;
303 | max-height: 90px;
304 | overflow-y: scroll;
305 | cursor: pointer;
306 | user-select: none;
307 | }
308 |
309 | table.sdpi-item-value,
310 | dl.sdpi-item-value,
311 | ul.sdpi-item-value,
312 | ol.sdpi-item-value {
313 | -webkit-margin-before: 4px;
314 | -webkit-margin-after: 8px;
315 | -webkit-padding-start: 1em;
316 | width: var(--sdpi-width);
317 | text-align: center;
318 | }
319 |
320 | table > caption {
321 | margin: 2px;
322 | }
323 |
324 | .list,
325 | .sdpi-item[type="list"] {
326 | align-items: baseline;
327 | }
328 |
329 | .sdpi-item-label {
330 | text-align: right;
331 | flex: none;
332 | width: 94px;
333 | padding-right: 4px;
334 | font-weight: 600;
335 | -webkit-user-select: none;
336 | }
337 |
338 | .win .sdpi-item-label,
339 | .sdpi-item-label > small{
340 | font-weight: normal;
341 | }
342 |
343 | .sdpi-item-label:after {
344 | content: ": ";
345 | }
346 |
347 | .sdpi-item-label.empty:after {
348 | content: "";
349 | }
350 |
351 | .sdpi-test,
352 | .sdpi-item-value {
353 | flex: 1 0 0;
354 | /* flex-grow: 1;
355 | flex-shrink: 0; */
356 | margin-right: 14px;
357 | margin-left: 4px;
358 | justify-content: space-evenly;
359 | }
360 |
361 | canvas.sdpi-item-value {
362 | max-width: 144px;
363 | max-height: 144px;
364 | width: 144px;
365 | height: 144px;
366 | margin: 0 auto;
367 | cursor: pointer;
368 | }
369 |
370 | input.sdpi-item-value {
371 | margin-left: 5px;
372 | }
373 |
374 | .sdpi-item-value button,
375 | button.sdpi-item-value {
376 | margin-left: 7px;
377 | margin-right: 19px;
378 | }
379 |
380 | .sdpi-item-value.range {
381 | margin-left: 0px;
382 | }
383 |
384 | table,
385 | dl.sdpi-item-value,
386 | ul.sdpi-item-value,
387 | ol.sdpi-item-value,
388 | .sdpi-item-value > dl,
389 | .sdpi-item-value > ul,
390 | .sdpi-item-value > ol
391 | {
392 | list-style-type: none;
393 | list-style-position: outside;
394 | margin-left: -4px;
395 | margin-right: -4px;
396 | padding: 4px;
397 | border: 1px solid var(--sdpi-bordercolor);
398 | }
399 |
400 | dl.sdpi-item-value,
401 | ul.sdpi-item-value,
402 | ol.sdpi-item-value,
403 | .sdpi-item-value > ol {
404 | list-style-type: none;
405 | list-style-position: inside;
406 | margin-left: 5px;
407 | margin-right: 12px;
408 | padding: 4px !important;
409 | }
410 |
411 | ol.sdpi-item-value,
412 | .sdpi-item-value > ol[listtype="none"] {
413 | list-style-type: none;
414 | }
415 | ol.sdpi-item-value[type="decimal"],
416 | .sdpi-item-value > ol[type="decimal"] {
417 | list-style-type: decimal;
418 | }
419 |
420 | ol.sdpi-item-value[type="decimal-leading-zero"],
421 | .sdpi-item-value > ol[type="decimal-leading-zero"] {
422 | list-style-type: decimal-leading-zero;
423 | }
424 |
425 | ol.sdpi-item-value[type="lower-alpha"],
426 | .sdpi-item-value > ol[type="lower-alpha"] {
427 | list-style-type: lower-alpha;
428 | }
429 |
430 | ol.sdpi-item-value[type="upper-alpha"],
431 | .sdpi-item-value > ol[type="upper-alpha"] {
432 | list-style-type: upper-alpha;
433 | }
434 |
435 | ol.sdpi-item-value[type="upper-roman"],
436 | .sdpi-item-value > ol[type="upper-roman"] {
437 | list-style-type: upper-roman;
438 | }
439 |
440 | ol.sdpi-item-value[type="lower-roman"],
441 | .sdpi-item-value > ol[type="lower-roman"] {
442 | list-style-type: upper-roman;
443 | }
444 |
445 | tr:nth-child(even),
446 | .sdpi-item-value > ul > li:nth-child(even),
447 | .sdpi-item-value > ol > li:nth-child(even),
448 | li:nth-child(even) {
449 | background-color: rgba(0,0,0,.2)
450 | }
451 |
452 | td:hover,
453 | .sdpi-item-value > ul > li:hover:nth-child(even),
454 | .sdpi-item-value > ol > li:hover:nth-child(even),
455 | li:hover:nth-child(even),
456 | li:hover {
457 | background-color: rgba(255,255,255,.1);
458 | }
459 |
460 | td.selected,
461 | td.selected:hover,
462 | li.selected:hover,
463 | li.selected {
464 | color: white;
465 | background-color: #77f;
466 | }
467 |
468 | tr {
469 | border: 1px solid var(--sdpi-bordercolor);
470 | }
471 |
472 | td {
473 | border-right: 1px solid var(--sdpi-bordercolor);
474 | -webkit-user-select: none;
475 | }
476 |
477 | tr:last-child,
478 | td:last-child {
479 | border: none;
480 | }
481 |
482 | .sdpi-item-value.select,
483 | .sdpi-item-value > select {
484 | margin-right: 13px;
485 | margin-left: 4px;
486 | }
487 |
488 | .sdpi-item-child,
489 | .sdpi-item-group > .sdpi-item > input[type="color"] {
490 | margin-top: 0.4em;
491 | margin-right: 4px;
492 | }
493 |
494 | .full,
495 | .full *,
496 | .sdpi-item-value.full,
497 | .sdpi-item-child > full > *,
498 | .sdpi-item-child.full,
499 | .sdpi-item-child.full > *,
500 | .full > .sdpi-item-child,
501 | .full > .sdpi-item-child > *{
502 | display: flex;
503 | flex: 1 1 0;
504 | margin-bottom: 4px;
505 | margin-left: 0px;
506 | width: 100%;
507 |
508 | justify-content: space-evenly;
509 | }
510 |
511 | .sdpi-item-group > .sdpi-item > input[type="color"] {
512 | margin-top: 0px;
513 | }
514 |
515 | ::-webkit-calendar-picker-indicator:focus,
516 | input[type=file]::-webkit-file-upload-button:focus,
517 | button:focus,
518 | textarea:focus,
519 | input:focus,
520 | select:focus,
521 | option:focus,
522 | details:focus,
523 | summary:focus,
524 | .custom-select select {
525 | outline: none;
526 | }
527 |
528 | summary {
529 | cursor: default;
530 | -webkit-user-select: none;
531 | }
532 |
533 | .pointer,
534 | summary .pointer {
535 | cursor: pointer;
536 | }
537 |
538 | details.message {
539 | padding: 4px 18px 4px 12px;
540 | }
541 |
542 | details.message summary {
543 | font-size: 10pt;
544 | font-weight: 600;
545 | min-height: 18px;
546 | }
547 |
548 | details.message:first-child {
549 | margin-top: 4px;
550 | margin-left: 0;
551 | padding-left: 106px;
552 | }
553 |
554 | details.message h1 {
555 | text-align: left;
556 | }
557 |
558 | .message > summary::-webkit-details-marker {
559 | display: none;
560 | }
561 |
562 | .info20,
563 | .question,
564 | .caution,
565 | .info {
566 | background-repeat: no-repeat;
567 | background-position: 70px center;
568 | }
569 |
570 | .info20 {
571 | background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='20' height='20' viewBox='0 0 20 20'%3E%3Cpath fill='%23999' d='M10,20 C4.4771525,20 0,15.5228475 0,10 C0,4.4771525 4.4771525,0 10,0 C15.5228475,0 20,4.4771525 20,10 C20,15.5228475 15.5228475,20 10,20 Z M10,8 C8.8954305,8 8,8.84275812 8,9.88235294 L8,16.1176471 C8,17.1572419 8.8954305,18 10,18 C11.1045695,18 12,17.1572419 12,16.1176471 L12,9.88235294 C12,8.84275812 11.1045695,8 10,8 Z M10,3 C8.8954305,3 8,3.88165465 8,4.96923077 L8,5.03076923 C8,6.11834535 8.8954305,7 10,7 C11.1045695,7 12,6.11834535 12,5.03076923 L12,4.96923077 C12,3.88165465 11.1045695,3 10,3 Z'/%3E%3C/svg%3E%0A");
572 | }
573 |
574 | .info {
575 | background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='20' height='20' viewBox='0 0 20 20'%3E%3Cpath fill='%23999' d='M10,18 C5.581722,18 2,14.418278 2,10 C2,5.581722 5.581722,2 10,2 C14.418278,2 18,5.581722 18,10 C18,14.418278 14.418278,18 10,18 Z M10,8 C9.44771525,8 9,8.42137906 9,8.94117647 L9,14.0588235 C9,14.5786209 9.44771525,15 10,15 C10.5522847,15 11,14.5786209 11,14.0588235 L11,8.94117647 C11,8.42137906 10.5522847,8 10,8 Z M10,5 C9.44771525,5 9,5.44082732 9,5.98461538 L9,6.01538462 C9,6.55917268 9.44771525,7 10,7 C10.5522847,7 11,6.55917268 11,6.01538462 L11,5.98461538 C11,5.44082732 10.5522847,5 10,5 Z'/%3E%3C/svg%3E%0A");
576 | }
577 |
578 | .info2 {
579 | background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='15' height='15' viewBox='0 0 15 15'%3E%3Cpath fill='%23999' d='M7.5,15 C3.35786438,15 0,11.6421356 0,7.5 C0,3.35786438 3.35786438,0 7.5,0 C11.6421356,0 15,3.35786438 15,7.5 C15,11.6421356 11.6421356,15 7.5,15 Z M7.5,2 C6.67157287,2 6,2.66124098 6,3.47692307 L6,3.52307693 C6,4.33875902 6.67157287,5 7.5,5 C8.32842705,5 9,4.33875902 9,3.52307693 L9,3.47692307 C9,2.66124098 8.32842705,2 7.5,2 Z M5,6 L5,7.02155172 L6,7 L6,12 L5,12.0076778 L5,13 L10,13 L10,12 L9,12.0076778 L9,6 L5,6 Z'/%3E%3C/svg%3E%0A");
580 | }
581 |
582 | .sdpi-more-info {
583 | background-image: linear-gradient(to right, #00000000 0%,#00000040 80%), url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 16 16'%3E%3Cpolygon fill='%23999' points='4 7 8 7 8 5 12 8 8 11 8 9 4 9'/%3E%3C/svg%3E%0A");
584 | }
585 | .caution {
586 | background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='20' height='20' viewBox='0 0 20 20'%3E%3Cpath fill='%23999' fill-rule='evenodd' d='M9.03952676,0.746646542 C9.57068894,-0.245797319 10.4285735,-0.25196227 10.9630352,0.746646542 L19.7705903,17.2030214 C20.3017525,18.1954653 19.8777595,19 18.8371387,19 L1.16542323,19 C0.118729947,19 -0.302490098,18.2016302 0.231971607,17.2030214 L9.03952676,0.746646542 Z M10,2.25584053 L1.9601405,17.3478261 L18.04099,17.3478261 L10,2.25584053 Z M10,5.9375 C10.531043,5.9375 10.9615385,6.37373537 10.9615385,6.91185897 L10.9615385,11.6923077 C10.9615385,12.2304313 10.531043,12.6666667 10,12.6666667 C9.46895697,12.6666667 9.03846154,12.2304313 9.03846154,11.6923077 L9.03846154,6.91185897 C9.03846154,6.37373537 9.46895697,5.9375 10,5.9375 Z M10,13.4583333 C10.6372516,13.4583333 11.1538462,13.9818158 11.1538462,14.6275641 L11.1538462,14.6641026 C11.1538462,15.3098509 10.6372516,15.8333333 10,15.8333333 C9.36274837,15.8333333 8.84615385,15.3098509 8.84615385,14.6641026 L8.84615385,14.6275641 C8.84615385,13.9818158 9.36274837,13.4583333 10,13.4583333 Z'/%3E%3C/svg%3E%0A");
587 | }
588 |
589 | .question {
590 | background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='20' height='20' viewBox='0 0 20 20'%3E%3Cpath fill='%23999' d='M10,18 C5.581722,18 2,14.418278 2,10 C2,5.581722 5.581722,2 10,2 C14.418278,2 18,5.581722 18,10 C18,14.418278 14.418278,18 10,18 Z M6.77783203,7.65332031 C6.77783203,7.84798274 6.85929281,8.02888914 7.0222168,8.19604492 C7.18514079,8.36320071 7.38508996,8.44677734 7.62207031,8.44677734 C8.02409055,8.44677734 8.29703704,8.20768468 8.44091797,7.72949219 C8.59326248,7.27245865 8.77945854,6.92651485 8.99951172,6.69165039 C9.2195649,6.45678594 9.56233491,6.33935547 10.027832,6.33935547 C10.4256205,6.33935547 10.7006836,6.37695313 11.0021973,6.68847656 C11.652832,7.53271484 10.942627,8.472229 10.3750916,9.1321106 C9.80755615,9.79199219 8.29492188,11.9897461 10.027832,12.1347656 C10.4498423,12.1700818 10.7027991,11.9147157 10.7832031,11.4746094 C11.0021973,9.59857178 13.1254883,8.82415771 13.1254883,7.53271484 C13.1254883,7.07568131 12.9974785,6.65250846 12.7414551,6.26318359 C12.4854317,5.87385873 12.1225609,5.56600048 11.652832,5.33959961 C11.1831031,5.11319874 10.6414419,5 10.027832,5 C9.36767248,5 8.79004154,5.13541531 8.29492187,5.40625 C7.79980221,5.67708469 7.42317837,6.01879677 7.16503906,6.43139648 C6.90689975,6.8439962 6.77783203,7.25130007 6.77783203,7.65332031 Z M10.0099668,15 C10.2713191,15 10.5016601,14.9108147 10.7009967,14.7324415 C10.9003332,14.5540682 11,14.3088087 11,13.9966555 C11,13.7157177 10.9047629,13.4793767 10.7142857,13.2876254 C10.5238086,13.0958742 10.2890379,13 10.0099668,13 C9.72646591,13 9.48726565,13.0958742 9.2923588,13.2876254 C9.09745196,13.4793767 9,13.7157177 9,13.9966555 C9,14.313268 9.10077419,14.5596424 9.30232558,14.735786 C9.50387698,14.9119295 9.73975502,15 10.0099668,15 Z'/%3E%3C/svg%3E%0A");
591 | }
592 |
593 |
594 | .sdpi-more-info {
595 | position: fixed;
596 | left: 0px;
597 | right: 0px;
598 | bottom: 0px;
599 | min-height:16px;
600 | padding-right: 16px;
601 | text-align: right;
602 | -webkit-touch-callout: none;
603 | cursor: pointer;
604 | user-select: none;
605 | background-position: right center;
606 | background-repeat: no-repeat;
607 | border-radius: var(--sdpi-borderradius);
608 | text-decoration: none;
609 | color: var(--sdpi-color);
610 | }
611 |
612 | .sdpi-more-info-button {
613 | display: flex;
614 | align-self: right;
615 | margin-left: auto;
616 | position: fixed;
617 | right: 17px;
618 | bottom: 0px;
619 | }
620 |
621 | details a {
622 | background-position: right !important;
623 | min-height: 24px;
624 | display: inline-block;
625 | line-height: 24px;
626 | padding-right: 28px;
627 | }
628 | input:not([type="range"]),
629 | textarea {
630 | -webkit-appearance: none;
631 | background: var(--sdpi-background);
632 | color: var(--sdpi-color);
633 | font-weight: normal;
634 | font-size: 9pt;
635 | border: none;
636 | margin-top: 2px;
637 | margin-bottom: 2px;
638 | }
639 |
640 | textarea + label {
641 | display: flex;
642 | justify-content: flex-end
643 | }
644 | input[type="radio"],
645 | input[type="checkbox"] {
646 | display: none;
647 | }
648 | input[type="radio"] + label,
649 | input[type="checkbox"] + label {
650 | font-size: 9pt;
651 | color: var(--sdpi-color);
652 | font-weight: normal;
653 | margin-right: 8px;
654 | -webkit-user-select: none;
655 | }
656 |
657 | input[type="radio"] + label:after,
658 | input[type="checkbox"] + label:after {
659 | content: " " !important;
660 | }
661 |
662 | .sdpi-item[type="radio"] > .sdpi-item-value,
663 | .sdpi-item[type="checkbox"] > .sdpi-item-value {
664 | padding-top: 2px;
665 | }
666 |
667 | .sdpi-item[type="checkbox"] > .sdpi-item-value > * {
668 | margin-top: 4px;
669 | }
670 |
671 | .sdpi-item[type="checkbox"] .sdpi-item-child,
672 | .sdpi-item[type="radio"] .sdpi-item-child {
673 | display: inline-block;
674 | }
675 |
676 | .sdpi-item[type="range"] .sdpi-item-value,
677 | .sdpi-item[type="meter"] .sdpi-item-child,
678 | .sdpi-item[type="progress"] .sdpi-item-child {
679 | display: flex;
680 | }
681 |
682 | .sdpi-item[type="range"] .sdpi-item-value {
683 | min-height: 26px;
684 | }
685 |
686 | .sdpi-item[type="range"] .sdpi-item-value span,
687 | .sdpi-item[type="meter"] .sdpi-item-child span,
688 | .sdpi-item[type="progress"] .sdpi-item-child span {
689 | margin-top: -2px;
690 | min-width: 8px;
691 | text-align: right;
692 | user-select: none;
693 | cursor: pointer;
694 | }
695 |
696 | .sdpi-item[type="range"] .sdpi-item-value span {
697 | margin-top: 7px;
698 | text-align: right;
699 | }
700 |
701 | span + input[type="range"] {
702 | display: flex;
703 | max-width: 168px;
704 |
705 | }
706 |
707 | .sdpi-item[type="range"] .sdpi-item-value span:first-child,
708 | .sdpi-item[type="meter"] .sdpi-item-child span:first-child,
709 | .sdpi-item[type="progress"] .sdpi-item-child span:first-child {
710 | margin-right: 4px;
711 | }
712 |
713 | .sdpi-item[type="range"] .sdpi-item-value span:last-child,
714 | .sdpi-item[type="meter"] .sdpi-item-child span:last-child,
715 | .sdpi-item[type="progress"] .sdpi-item-child span:last-child {
716 | margin-left: 4px;
717 | }
718 |
719 | .reverse {
720 | transform: rotate(180deg);
721 | }
722 |
723 | .sdpi-item[type="meter"] .sdpi-item-child meter + span:last-child {
724 | margin-left: -10px;
725 | }
726 |
727 | .sdpi-item[type="progress"] .sdpi-item-child meter + span:last-child {
728 | margin-left: -14px;
729 | }
730 |
731 | .sdpi-item[type="radio"] > .sdpi-item-value > * {
732 | margin-top: 2px;
733 | }
734 |
735 | details {
736 | padding: 8px 18px 8px 12px;
737 | min-width: 86px;
738 | }
739 |
740 | details > h4 {
741 | border-bottom: 1px solid var(--sdpi-bordercolor);
742 | }
743 |
744 | legend {
745 | display: none;
746 | }
747 | .sdpi-item-value > textarea {
748 | padding: 0px;
749 | width: 227px;
750 | margin-left: 1px;
751 | }
752 |
753 | input[type="radio"] + label span,
754 | input[type="checkbox"] + label span {
755 | display: inline-block;
756 | width: 16px;
757 | height: 16px;
758 | margin: 2px 4px 2px 0;
759 | border-radius: 3px;
760 | vertical-align: middle;
761 | background: var(--sdpi-background);
762 | cursor: pointer;
763 | border: 1px solid rgb(0,0,0,.2);
764 | }
765 |
766 | input[type="radio"] + label span {
767 | border-radius: 100%;
768 | }
769 |
770 | input[type="radio"]:checked + label span,
771 | input[type="checkbox"]:checked + label span {
772 | background-color: #77f;
773 | background-image: url(check.svg);
774 | background-repeat: no-repeat;
775 | background-position: center center;
776 | border: 1px solid rgb(0,0,0,.4);
777 | }
778 |
779 | input[type="radio"]:active:checked + label span,
780 | input[type="radio"]:active + label span,
781 | input[type="checkbox"]:active:checked + label span,
782 | input[type="checkbox"]:active + label span {
783 | background-color: #303030;
784 | }
785 |
786 | input[type="radio"]:checked + label span {
787 | background-image: url(rcheck.svg);
788 | }
789 |
790 |
791 | /*
792 | input[type="radio"] + label span {
793 | background: url(buttons.png) -38px top no-repeat;
794 | }
795 |
796 | input[type="radio"]:checked + label span {
797 | background: url(buttons.png) -57px top no-repeat;
798 | }
799 | */
800 |
801 | input[type="range"] {
802 | width: var(--sdpi-width);
803 | height: 30px;
804 | overflow: hidden;
805 | cursor: pointer;
806 | background: transparent !important;
807 | }
808 |
809 | .sdpi-item > input[type="range"] {
810 | margin-left: 8px;
811 | max-width: var(--sdpi-width);
812 | width: var(--sdpi-width);
813 | padding: 0px;
814 | }
815 |
816 | /*
817 | input[type="range"],
818 | input[type="range"]::-webkit-slider-runnable-track,
819 | input[type="range"]::-webkit-slider-thumb {
820 | -webkit-appearance: none;
821 | }
822 | */
823 |
824 | input[type="range"]::-webkit-slider-runnable-track {
825 | height: 5px;
826 | background: #979797;
827 | border-radius: 3px;
828 | padding:0px !important;
829 | border: 1px solid var(--sdpi-background);
830 | }
831 |
832 | input[type="range"]::-webkit-slider-thumb {
833 | position: relative;
834 | -webkit-appearance: none;
835 | background-color: var(--sdpi-color);
836 | width: 12px;
837 | height: 12px;
838 | border-radius: 20px;
839 | margin-top: -5px;
840 | border: none;
841 |
842 | }
843 | input[type="range" i]{
844 | margin: 0;
845 | }
846 |
847 | input[type="range"]::-webkit-slider-thumb::before {
848 | position: absolute;
849 | content: "";
850 | height: 5px; /* equal to height of runnable track or 1 less */
851 | width: 500px; /* make this bigger than the widest range input element */
852 | left: -502px; /* this should be -2px - width */
853 | top: 8px; /* don't change this */
854 | background: #77f;
855 | }
856 |
857 | input[type="color"] {
858 | min-width: 32px;
859 | min-height: 32px;
860 | width: 32px;
861 | height: 32px;
862 | padding: 0;
863 | background-color: var(--sdpi-bgcolor);
864 | flex: none;
865 | }
866 |
867 | ::-webkit-color-swatch {
868 | min-width: 24px;
869 | }
870 |
871 | textarea {
872 | height: 3em;
873 | word-break: break-word;
874 | line-height: 1.5em;
875 | }
876 |
877 | .textarea {
878 | padding: 0px !important;
879 | }
880 |
881 | textarea {
882 | width: 221px; /*98%;*/
883 | height: 96%;
884 | min-height: 6em;
885 | resize: none;
886 | border-radius: var(--sdpi-borderradius);
887 | }
888 |
889 | /* CAROUSEL */
890 |
891 | .sdpi-item[type="carousel"]{
892 |
893 | }
894 |
895 | .sdpi-item.card-carousel-wrapper,
896 | .sdpi-item > .card-carousel-wrapper {
897 | padding: 0;
898 | }
899 |
900 |
901 | .card-carousel-wrapper {
902 | display: flex;
903 | align-items: center;
904 | justify-content: center;
905 | margin: 12px auto;
906 | color: #666a73;
907 | }
908 |
909 | .card-carousel {
910 | display: flex;
911 | justify-content: center;
912 | width: 278px;
913 | }
914 | .card-carousel--overflow-container {
915 | overflow: hidden;
916 | }
917 | .card-carousel--nav__left,
918 | .card-carousel--nav__right {
919 | /* display: inline-block; */
920 | width: 12px;
921 | height: 12px;
922 | border-top: 2px solid #42b883;
923 | border-right: 2px solid #42b883;
924 | cursor: pointer;
925 | margin: 0 4px;
926 | transition: transform 150ms linear;
927 | }
928 | .card-carousel--nav__left[disabled],
929 | .card-carousel--nav__right[disabled] {
930 | opacity: 0.2;
931 | border-color: black;
932 | }
933 | .card-carousel--nav__left {
934 | transform: rotate(-135deg);
935 | }
936 | .card-carousel--nav__left:active {
937 | transform: rotate(-135deg) scale(0.85);
938 | }
939 | .card-carousel--nav__right {
940 | transform: rotate(45deg);
941 | }
942 | .card-carousel--nav__right:active {
943 | transform: rotate(45deg) scale(0.85);
944 | }
945 | .card-carousel-cards {
946 | display: flex;
947 | transition: transform 150ms ease-out;
948 | transform: translatex(0px);
949 | }
950 | .card-carousel-cards .card-carousel--card {
951 | margin: 0 5px;
952 | cursor: pointer;
953 | /* box-shadow: 0 4px 15px 0 rgba(40, 44, 53, 0.06), 0 2px 2px 0 rgba(40, 44, 53, 0.08); */
954 | background-color: #fff;
955 | border-radius: 4px;
956 | z-index: 3;
957 | }
958 | .xxcard-carousel-cards .card-carousel--card:first-child {
959 | margin-left: 0;
960 | }
961 | .xxcard-carousel-cards .card-carousel--card:last-child {
962 | margin-right: 0;
963 | }
964 | .card-carousel-cards .card-carousel--card img {
965 | vertical-align: bottom;
966 | border-top-left-radius: 4px;
967 | border-top-right-radius: 4px;
968 | transition: opacity 150ms linear;
969 | width: 60px;
970 | }
971 | .card-carousel-cards .card-carousel--card img:hover {
972 | opacity: 0.5;
973 | }
974 | .card-carousel-cards .card-carousel--card--footer {
975 | border-top: 0;
976 | max-width: 80px;
977 | overflow: hidden;
978 | display: flex;
979 | height: 100%;
980 | flex-direction: column;
981 | }
982 | .card-carousel-cards .card-carousel--card--footer p {
983 | padding: 3px 0;
984 | margin: 0;
985 | margin-bottom: 2px;
986 | font-size: 15px;
987 | font-weight: 500;
988 | color: #2c3e50;
989 | }
990 | .card-carousel-cards .card-carousel--card--footer p:nth-of-type(2) {
991 | font-size: 12px;
992 | font-weight: 300;
993 | padding: 6px;
994 | color: #666a73;
995 | }
996 |
997 |
998 | h1 {
999 | font-size: 1.3em;
1000 | font-weight: 500;
1001 | text-align: center;
1002 | margin-bottom: 12px;
1003 | }
1004 |
1005 | ::-webkit-datetime-edit {
1006 | font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
1007 | background: url(elg_calendar_inv.svg) no-repeat left center;
1008 | padding-right: 1em;
1009 | padding-left: 25px;
1010 | background-position: 4px 0px;
1011 | }
1012 | ::-webkit-datetime-edit-fields-wrapper {
1013 |
1014 | }
1015 | ::-webkit-datetime-edit-text { padding: 0 0.3em; }
1016 | ::-webkit-datetime-edit-month-field { }
1017 | ::-webkit-datetime-edit-day-field {}
1018 | ::-webkit-datetime-edit-year-field {}
1019 | ::-webkit-inner-spin-button {
1020 |
1021 | /* display: none; */
1022 | }
1023 | ::-webkit-calendar-picker-indicator {
1024 | background: transparent;
1025 | font-size: 17px;
1026 | }
1027 |
1028 | ::-webkit-calendar-picker-indicator:focus {
1029 | background-color: rgba(0,0,0,0.2);
1030 | }
1031 |
1032 | input[type="date"] {
1033 | -webkit-align-items: center;
1034 | display: -webkit-inline-flex;
1035 | font-family: monospace;
1036 | overflow: hidden;
1037 | padding: 0;
1038 | -webkit-padding-start: 1px;
1039 | }
1040 |
1041 | input::-webkit-datetime-edit {
1042 | -webkit-flex: 1;
1043 | -webkit-user-modify: read-only !important;
1044 | display: inline-block;
1045 | min-width: 0;
1046 | overflow: hidden;
1047 | }
1048 |
1049 | /*
1050 | input::-webkit-datetime-edit-fields-wrapper {
1051 | -webkit-user-modify: read-only !important;
1052 | display: inline-block;
1053 | padding: 1px 0;
1054 | white-space: pre;
1055 |
1056 | }
1057 | */
1058 |
1059 | /*
1060 | input[type="date"] {
1061 | background-color: red;
1062 | outline: none;
1063 | }
1064 |
1065 | input[type="date"]::-webkit-clear-button {
1066 | font-size: 18px;
1067 | height: 30px;
1068 | position: relative;
1069 | }
1070 |
1071 | input[type="date"]::-webkit-inner-spin-button {
1072 | height: 28px;
1073 | }
1074 |
1075 | input[type="date"]::-webkit-calendar-picker-indicator {
1076 | font-size: 15px;
1077 | } */
1078 |
1079 | input[type="file"] {
1080 | opacity: 0;
1081 | display: none;
1082 | }
1083 |
1084 | .sdpi-item > input[type="file"] {
1085 | opacity: 1;
1086 | display: flex;
1087 | }
1088 |
1089 | input[type="file"] + span {
1090 | display: flex;
1091 | flex: 0 1 auto;
1092 | background-color: #0000ff50;
1093 | }
1094 |
1095 | label.sdpi-file-label {
1096 | cursor: pointer;
1097 | user-select: none;
1098 | display: inline-block;
1099 | min-height: 21px !important;
1100 | height: 21px !important;
1101 | line-height: 20px;
1102 | padding: 0px 4px;
1103 | margin: auto;
1104 | margin-right: 0px;
1105 | float:right;
1106 | }
1107 |
1108 | .sdpi-file-label > label:active,
1109 | .sdpi-file-label.file:active,
1110 | label.sdpi-file-label:active,
1111 | label.sdpi-file-info:active,
1112 | input[type="file"]::-webkit-file-upload-button:active,
1113 | button:active {
1114 | background-color: var(--sdpi-color);
1115 | color:#303030;
1116 | }
1117 |
1118 |
1119 | input:required:invalid, input:focus:invalid {
1120 | background: var(--sdpi-background) url() no-repeat 98% center;
1121 | }
1122 |
1123 | input:required:valid {
1124 | background: var(--sdpi-background) url() no-repeat 98% center;
1125 | }
1126 |
1127 | .tooltip,
1128 | :tooltip,
1129 | :title {
1130 | color: yellow;
1131 | }
1132 |
1133 | [title]:hover {
1134 | display: flex;
1135 | align-items: center;
1136 | justify-content: center;
1137 | }
1138 |
1139 | [title]:hover::after {
1140 | content: '';
1141 | position: absolute;
1142 | bottom: -1000px;
1143 | left: 8px;
1144 | display: none;
1145 | color: #fff;
1146 | border: 8px solid transparent;
1147 | border-bottom: 8px solid #000;
1148 | }
1149 | [title]:hover::before {
1150 | content: attr(title);
1151 | display: flex;
1152 | justify-content: center;
1153 | align-self: center;
1154 | padding: 6px 12px;
1155 | border-radius: 5px;
1156 | background: rgba(0,0,0,0.8);
1157 | color: var(--sdpi-color);
1158 | font-size: 9pt;
1159 | font-family: sans-serif;
1160 | opacity: 1;
1161 | position: absolute;
1162 | height: auto;
1163 | /* width: 50%;
1164 | left: 35%; */
1165 | text-align: center;
1166 | bottom: 2px;
1167 | z-index: 100;
1168 | box-shadow: 0px 3px 6px rgba(0, 0, 0, .5);
1169 | /* box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19); */
1170 | }
1171 |
1172 | .sdpi-item-group.file {
1173 | width: 232px;
1174 | display: flex;
1175 | align-items: center;
1176 | }
1177 |
1178 | .sdpi-file-info {
1179 | overflow-wrap: break-word;
1180 | word-wrap: break-word;
1181 | hyphens: auto;
1182 |
1183 | min-width: 132px;
1184 | max-width: 144px;
1185 | max-height: 32px;
1186 | margin-top: 0px;
1187 | margin-left: 5px;
1188 | display: inline-block;
1189 | overflow: hidden;
1190 | padding: 6px 4px;
1191 | background-color: var(--sdpi-background);
1192 | }
1193 |
1194 |
1195 | ::-webkit-scrollbar {
1196 | width: 8px;
1197 | }
1198 |
1199 | ::-webkit-scrollbar-track {
1200 | -webkit-box-shadow: inset 0 0 6px rgba(0,0,0,0.3);
1201 | }
1202 |
1203 | ::-webkit-scrollbar-thumb {
1204 | background-color: #999999;
1205 | outline: 1px solid slategrey;
1206 | border-radius: 8px;
1207 | }
1208 |
1209 | a {
1210 | color: #7397d2;
1211 | }
1212 |
1213 | .testcontainer {
1214 | display: flex;
1215 | background-color: #0000ff20;
1216 | max-width: 400px;
1217 | height: 200px;
1218 | align-content: space-evenly;
1219 | }
1220 |
1221 | input[type=range] {
1222 | -webkit-appearance: none;
1223 | /* background-color: green; */
1224 | height:6px;
1225 | margin-top: 12px;
1226 | z-index: 0;
1227 | overflow: visible;
1228 | }
1229 |
1230 | /*
1231 | input[type="range"]::-webkit-slider-thumb {
1232 | -webkit-appearance: none;
1233 | background-color: var(--sdpi-color);
1234 | width: 12px;
1235 | height: 12px;
1236 | border-radius: 20px;
1237 | margin-top: -6px;
1238 | border: none;
1239 | } */
1240 |
1241 | :-webkit-slider-thumb {
1242 | -webkit-appearance: none;
1243 | background-color: var(--sdpi-color);
1244 | width: 16px;
1245 | height: 16px;
1246 | border-radius: 20px;
1247 | margin-top: -6px;
1248 | border: 1px solid #999999;
1249 | }
1250 |
1251 | .sdpi-item[type="range"] .sdpi-item-group {
1252 | display: flex;
1253 | flex-direction: column;
1254 | }
1255 |
1256 | .xxsdpi-item[type="range"] .sdpi-item-group input {
1257 | max-width: 204px;
1258 | }
1259 |
1260 | .sdpi-item[type="range"] .sdpi-item-group span {
1261 | margin-left: 0px !important;
1262 | }
1263 |
1264 | .sdpi-item[type="range"] .sdpi-item-group > .sdpi-item-child {
1265 | display: flex;
1266 | flex-direction: row;
1267 | }
1268 |
1269 | :disabled {
1270 | color: #993333;
1271 | }
1272 |
1273 | select,
1274 | select option {
1275 | color: var(--sdpi-color);
1276 | }
1277 |
1278 | select.disabled,
1279 | select option:disabled {
1280 | color: #fd9494;
1281 | font-style: italic;
1282 | }
1283 |
1284 | .runningAppsContainer {
1285 | display: none;
1286 | }
1287 |
1288 | /* debug
1289 | div {
1290 | background-color: rgba(64,128,255,0.2);
1291 | }
1292 | */
1293 |
1294 | .min80 > .sdpi-item-child {
1295 | min-width: 80px;
1296 | }
1297 |
1298 | .min100 > .sdpi-item-child {
1299 | min-width: 100px;
1300 | }
1301 |
1302 | .min120 > .sdpi-item-child {
1303 | min-width: 120px;
1304 | }
1305 |
1306 | .min140 > .sdpi-item-child {
1307 | min-width: 140px;
1308 | }
1309 |
1310 | .min160 > .sdpi-item-child {
1311 | min-width: 160px;
1312 | }
1313 |
1314 | .min200 > .sdpi-item-child {
1315 | min-width: 200px;
1316 | }
1317 |
1318 | .max40 {
1319 | flex-basis: 40%;
1320 | flex-grow: 0;
1321 | }
1322 |
1323 | .max30 {
1324 | flex-basis: 30%;
1325 | flex-grow: 0;
1326 | }
1327 |
1328 | .max20 {
1329 | flex-basis: 20%;
1330 | flex-grow: 0;
1331 | }
1332 |
1333 | .up20 {
1334 | margin-top: -20px;
1335 | }
1336 |
1337 | .alignCenter {
1338 | align-items: center;
1339 | }
1340 |
1341 | .alignTop {
1342 | align-items: flex-start;
1343 | }
1344 |
1345 | .alignBaseline {
1346 | align-items: baseline;
1347 | }
1348 |
1349 | .noMargins,
1350 | .noMargins *,
1351 | .noInnerMargins * {
1352 | margin: 0;
1353 | padding: 0;
1354 | }
1355 |
1356 |
1357 | /**
1358 | input[type=range].vVertical {
1359 | -webkit-appearance: none;
1360 | background-color: green;
1361 | margin-left: -60px;
1362 | width: 100px;
1363 | height:6px;
1364 | margin-top: 0px;
1365 | transform:rotate(90deg);
1366 | z-index: 0;
1367 | overflow: visible;
1368 | }
1369 |
1370 | input[type=range].vHorizon {
1371 | -webkit-appearance: none;
1372 | background-color: pink;
1373 | height: 10px;
1374 | width:200px;
1375 |
1376 | }
1377 |
1378 | .test2 {
1379 | background-color: #00ff0020;
1380 | display: flex;
1381 | }
1382 |
1383 |
1384 | .vertical.sdpi-item[type="range"] .sdpi-item-value {
1385 | display: block;
1386 | }
1387 |
1388 |
1389 | .vertical.sdpi-item:first-child,
1390 | .vertical {
1391 | margin-top: 12px;
1392 | margin-bottom: 16px;
1393 | }
1394 | .vertical > .sdpi-item-value {
1395 | margin-right: 16px;
1396 | }
1397 |
1398 | .vertical .sdpi-item-group {
1399 | width: 100%;
1400 | display: flex;
1401 | justify-content: space-evenly;
1402 | }
1403 |
1404 | .vertical input[type=range] {
1405 | height: 100px;
1406 | width: 21px;
1407 | -webkit-appearance: slider-vertical;
1408 | display: flex;
1409 | flex-flow: column;
1410 | }
1411 |
1412 | .vertical input[type="range"]::-webkit-slider-runnable-track {
1413 | height: auto;
1414 | width: 5px;
1415 | }
1416 |
1417 | .vertical input[type="range"]::-webkit-slider-thumb {
1418 | margin-top: 0px;
1419 | margin-left: -6px;
1420 | }
1421 |
1422 | .vertical .sdpi-item-value {
1423 | flex-flow: column;
1424 | align-items: flex-start;
1425 | }
1426 |
1427 | .vertical.sdpi-item[type="range"] .sdpi-item-value {
1428 | align-items: center;
1429 | margin-right: 16px;
1430 | text-align: center;
1431 | }
1432 |
1433 | .vertical.sdpi-item[type="range"] .sdpi-item-value span,
1434 | .vertical input[type="range"] .sdpi-item-value span {
1435 | text-align: center;
1436 | margin: 4px 0px;
1437 | }
1438 | */
1439 |
1440 | /*
1441 | .file {
1442 | box-sizing: border-box;
1443 | display: block;
1444 | overflow: hidden;
1445 | padding: 10px;
1446 | position: relative;
1447 | text-indent: 100%;
1448 | white-space: nowrap;
1449 | height: 190px;
1450 | width: 160px;
1451 | }
1452 | .file::before {
1453 | content: "";
1454 | display: block;
1455 | position: absolute;
1456 | top: 10px;
1457 | left: 10px;
1458 | height: 170px;
1459 | width: 140px;
1460 | }
1461 | .file::after {
1462 | content: "";
1463 | height: 90px;
1464 | width: 90px;
1465 | position: absolute;
1466 | right: 0;
1467 | bottom: 0;
1468 | overflow: visible;
1469 | }
1470 |
1471 | .list--files {
1472 | display: flex;
1473 | flex-wrap: wrap;
1474 | justify-content: center;
1475 | margin: auto;
1476 | padding: 30px 0;
1477 | width: 630px;
1478 | }
1479 | .list--files > li {
1480 | margin: 0;
1481 | padding: 15px;
1482 | }
1483 |
1484 | .type-document::before {
1485 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSIxNDBweCIgaGVpZ2h0PSIxNzBweCIgdmlld0JveD0iMCAwIDE0MCAxNzAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPHBhdGggZmlsbD0iI0E3QTlBQyIgZD0iTTAsMHYxNzBoMTQwVjBIMHogTTEzMCwxNjBIMTBWMTBoMTIwVjE2MHogTTExMCw0MEgzMFYzMGg4MFY0MHogTTExMCw2MEgzMFY1MGg4MFY2MHogTTExMCw4MEgzMFY3MGg4MFY4MHoNCiAgIE0xMTAsMTAwSDMwVjkwaDgwVjEwMHogTTExMCwxMjBIMzB2LTEwaDgwVjEyMHogTTkwLDE0MEgzMHYtMTBoNjBWMTQweiIvPg0KPC9zdmc+);
1486 | }
1487 |
1488 | .type-image {
1489 | height: 160px;
1490 | }
1491 | .type-image::before {
1492 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSIxNDBweCIgaGVpZ2h0PSIxNDBweCIgdmlld0JveD0iMCAwIDE0MCAxNDAiIGVuYWJsZS1iYWNrZ3JvdW5kPSJuZXcgMCAwIDE0MCAxNDAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxwYXRoIGZpbGw9IiNBN0E5QUMiIGQ9Ik0wLDB2MTQwaDE0MFYwSDB6IE0xMzAsMTMwSDEwVjEwaDEyMFYxMzB6Ii8+DQogIDxwb2x5Z29uIGZpbGw9IiNFNkU3RTgiIHBvaW50cz0iOTAsMTEwIDQwLDQwIDEwLDgwIDEwLDEzMCA5MCwxMzAgICIvPg0KICA8cG9seWdvbiBmaWxsPSIjRDFEM0Q0IiBwb2ludHM9IjEwLDEzMCA1MCw5MCA2MCwxMDAgMTAwLDYwIDEzMCwxMzAgICIvPg0KPC9nPg0KPC9zdmc+);
1493 | height: 140px;
1494 | }
1495 |
1496 | .state-synced::after {
1497 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iIzAwQTY1MSIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik0yMCw0NUwyMCw0NWMtMi44LDIuOC0yLjgsNy4yLDAsMTBsMTAuMSwxMC4xYzIuNywyLjcsNy4yLDIuNyw5LjksMEw3MCwzNWMyLjgtMi44LDIuOC03LjIsMC0xMGwwLDANCiAgICBjLTIuOC0yLjgtNy4yLTIuOC0xMCwwTDM1LDUwbC01LTVDMjcuMiw0Mi4yLDIyLjgsNDIuMiwyMCw0NXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1498 | }
1499 |
1500 | .state-deleted::after {
1501 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iI0VEMUMyNCIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik02NSwyNUw2NSwyNWMtMi44LTIuOC03LjItMi44LTEwLDBMNDUsMzVMMzUsMjVjLTIuOC0yLjgtNy4yLTIuOC0xMCwwbDAsMGMtMi44LDIuOC0yLjgsNy4yLDAsMTBsMTAsMTANCiAgICBMMjUsNTVjLTIuOCwyLjgtMi44LDcuMiwwLDEwbDAsMGMyLjgsMi44LDcuMiwyLjgsMTAsMGwxMC0xMGwxMCwxMGMyLjgsMi44LDcuMiwyLjgsMTAsMGwwLDBjMi44LTIuOCwyLjgtNy4yLDAtMTBMNTUsNDVsMTAtMTANCiAgICBDNjcuOCwzMi4yLDY3LjgsMjcuOCw2NSwyNXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1502 | }
1503 | .state-deleted::before {
1504 | opacity: .25;
1505 | }
1506 |
1507 | .state-locked::after {
1508 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iIzU4NTk1QiIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxyZWN0IHg9IjIwIiB5PSI0MCIgZmlsbD0iI0ZGRkZGRiIgd2lkdGg9IjUwIiBoZWlnaHQ9IjMwIi8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik0zMi41LDQ2LjVjLTIuOCwwLTUtMi4yLTUtNVYyOWMwLTkuNiw3LjktMTcuNSwxNy41LTE3LjVTNjIuNSwxOS40LDYyLjUsMjljMCwyLjgtMi4yLDUtNSw1cy01LTIuMi01LTUNCiAgICBjMC00LjEtMy40LTcuNS03LjUtNy41cy03LjUsMy40LTcuNSw3LjV2MTIuNUMzNy41LDQ0LjMsMzUuMyw0Ni41LDMyLjUsNDYuNXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1509 | }
1510 |
1511 |
1512 |
1513 | html {
1514 | --fheight: 95px;
1515 | --fwidth: 80px;
1516 | --fspacing: 5px;
1517 | --ftotalwidth: 315px;
1518 | --bgsize: 50%;
1519 | --bgsize2: cover;
1520 | --bgsize3: contain;
1521 | }
1522 |
1523 | ul {
1524 | list-style: none;
1525 | }
1526 |
1527 |
1528 | .file {
1529 | height: var(--fheight);
1530 | width: var(--fwidth);
1531 | }
1532 | .file::before {
1533 | content: "";
1534 | display: block;
1535 | position: absolute;
1536 | top: var(--fspacing);
1537 | left: var(--fspacing);
1538 | height: calc(var(--fheight) - var(--fspacing)*2);
1539 | width: calc(var(--fwidth) - var(--fspacing)*2);
1540 | }
1541 | .file::after {
1542 | content: "";
1543 | height: calc(var(--fheight)/2);
1544 | width: calc(var(--fheight)/2);
1545 | position: absolute;
1546 | right: 0;
1547 | bottom: 0;
1548 | overflow: visible;
1549 | }
1550 |
1551 | .list--files {
1552 | display: flex;
1553 | flex-wrap: wrap;
1554 | justify-content: center;
1555 | margin: auto;
1556 | padding: calc(var(--fspacing)*3) 0;
1557 | width: var(--ftotalwidth);
1558 | }
1559 | .list--files > li {
1560 | margin: 0;
1561 | padding: var(--fspacing);
1562 | }
1563 |
1564 | .type-document::before {
1565 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSIxNDBweCIgaGVpZ2h0PSIxNzBweCIgdmlld0JveD0iMCAwIDE0MCAxNzAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPHBhdGggZmlsbD0iI0E3QTlBQyIgZD0iTTAsMHYxNzBoMTQwVjBIMHogTTEzMCwxNjBIMTBWMTBoMTIwVjE2MHogTTExMCw0MEgzMFYzMGg4MFY0MHogTTExMCw2MEgzMFY1MGg4MFY2MHogTTExMCw4MEgzMFY3MGg4MFY4MHoNCiAgIE0xMTAsMTAwSDMwVjkwaDgwVjEwMHogTTExMCwxMjBIMzB2LTEwaDgwVjEyMHogTTkwLDE0MEgzMHYtMTBoNjBWMTQweiIvPg0KPC9zdmc+);
1566 | height: calc(var(--fheight) - var(--fspacing)*2);
1567 | background-size: var(--bgsize2);
1568 | background-repeat: no-repeat;
1569 | }
1570 |
1571 | .type-image {
1572 | height: var(--fwidth);
1573 | height: calc(var(--fheight) - var(--fspacing)*2);
1574 | }
1575 | .type-image::before {
1576 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSIxNDBweCIgaGVpZ2h0PSIxNDBweCIgdmlld0JveD0iMCAwIDE0MCAxNDAiIGVuYWJsZS1iYWNrZ3JvdW5kPSJuZXcgMCAwIDE0MCAxNDAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxwYXRoIGZpbGw9IiNBN0E5QUMiIGQ9Ik0wLDB2MTQwaDE0MFYwSDB6IE0xMzAsMTMwSDEwVjEwaDEyMFYxMzB6Ii8+DQogIDxwb2x5Z29uIGZpbGw9IiNFNkU3RTgiIHBvaW50cz0iOTAsMTEwIDQwLDQwIDEwLDgwIDEwLDEzMCA5MCwxMzAgICIvPg0KICA8cG9seWdvbiBmaWxsPSIjRDFEM0Q0IiBwb2ludHM9IjEwLDEzMCA1MCw5MCA2MCwxMDAgMTAwLDYwIDEzMCwxMzAgICIvPg0KPC9nPg0KPC9zdmc+);
1577 | height: calc(var(--fheight) - var(--fspacing)*2);
1578 | background-size: var(--bgsize3);
1579 | background-repeat: no-repeat;
1580 | }
1581 |
1582 | .state-synced::after {
1583 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iIzAwQTY1MSIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik0yMCw0NUwyMCw0NWMtMi44LDIuOC0yLjgsNy4yLDAsMTBsMTAuMSwxMC4xYzIuNywyLjcsNy4yLDIuNyw5LjksMEw3MCwzNWMyLjgtMi44LDIuOC03LjIsMC0xMGwwLDANCiAgICBjLTIuOC0yLjgtNy4yLTIuOC0xMCwwTDM1LDUwbC01LTVDMjcuMiw0Mi4yLDIyLjgsNDIuMiwyMCw0NXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1584 | background-size: var(--bgsize);
1585 | background-repeat: no-repeat;
1586 | background-position: bottom right;
1587 | }
1588 |
1589 | .state-deleted::after {
1590 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iI0VEMUMyNCIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik02NSwyNUw2NSwyNWMtMi44LTIuOC03LjItMi44LTEwLDBMNDUsMzVMMzUsMjVjLTIuOC0yLjgtNy4yLTIuOC0xMCwwbDAsMGMtMi44LDIuOC0yLjgsNy4yLDAsMTBsMTAsMTANCiAgICBMMjUsNTVjLTIuOCwyLjgtMi44LDcuMiwwLDEwbDAsMGMyLjgsMi44LDcuMiwyLjgsMTAsMGwxMC0xMGwxMCwxMGMyLjgsMi44LDcuMiwyLjgsMTAsMGwwLDBjMi44LTIuOCwyLjgtNy4yLDAtMTBMNTUsNDVsMTAtMTANCiAgICBDNjcuOCwzMi4yLDY3LjgsMjcuOCw2NSwyNXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1591 | background-size: var(--bgsize);
1592 | background-repeat: no-repeat;
1593 | background-position: bottom right;
1594 | }
1595 | .state-deleted::before {
1596 | opacity: .25;
1597 | }
1598 |
1599 | .state-locked::after {
1600 | background-image: url(data:image/svg+xml;charset=utf-8;base64,PHN2ZyB2ZXJzaW9uPSIxLjEiDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4bWxuczphPSJodHRwOi8vbnMuYWRvYmUuY29tL0Fkb2JlU1ZHVmlld2VyRXh0ZW5zaW9ucy8zLjAvIg0KICAgeD0iMHB4IiB5PSIwcHgiIHdpZHRoPSI5MHB4IiBoZWlnaHQ9IjkwcHgiIHZpZXdCb3g9IjAgMCA5MCA5MCIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgOTAgOTAiIHhtbDpzcGFjZT0icHJlc2VydmUiPg0KPGc+DQogIDxjaXJjbGUgZmlsbD0iIzU4NTk1QiIgY3g9IjQ1IiBjeT0iNDUiIHI9IjQ1Ii8+DQogIDxyZWN0IHg9IjIwIiB5PSI0MCIgZmlsbD0iI0ZGRkZGRiIgd2lkdGg9IjUwIiBoZWlnaHQ9IjMwIi8+DQogIDxwYXRoIGZpbGw9IiNGRkZGRkYiIGQ9Ik0zMi41LDQ2LjVjLTIuOCwwLTUtMi4yLTUtNVYyOWMwLTkuNiw3LjktMTcuNSwxNy41LTE3LjVTNjIuNSwxOS40LDYyLjUsMjljMCwyLjgtMi4yLDUtNSw1cy01LTIuMi01LTUNCiAgICBjMC00LjEtMy40LTcuNS03LjUtNy41cy03LjUsMy40LTcuNSw3LjV2MTIuNUMzNy41LDQ0LjMsMzUuMyw0Ni41LDMyLjUsNDYuNXoiLz4NCjwvZz4NCjwvc3ZnPg==);
1601 | background-size: var(--bgsize);
1602 | background-repeat: no-repeat;
1603 | background-position: bottom right;
1604 | }
1605 | */
--------------------------------------------------------------------------------