diff --git a/.circleci/config.yml b/.circleci/config.yml index edbf7f419401..4183c21bbf13 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -44,7 +44,7 @@ jobs: command: sudo apt-get install zip - run: name: Build Maskbook - command: npm run build-ci + command: npx build-ci - store_artifacts: path: Maskbook.base.zip destination: /Maskbook.base.zip diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index fcc6bcdaf30e..09b7be4434c8 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -24,7 +24,7 @@ jobs: ${{ runner.os }}-node- - run: sudo npm i -g pnpm - run: npm run ci - - run: npm run build-ci + - run: npx build-ci - name: Upload `Maskbook.base.zip` uses: actions/upload-artifact@v2 with: diff --git a/.github/workflows/linters.yml b/.github/workflows/linters.yml index cc535dba607d..d0822eacbcac 100644 --- a/.github/workflows/linters.yml +++ b/.github/workflows/linters.yml @@ -21,18 +21,26 @@ jobs: key: ${{ runner.os }}-node-${{ hashFiles('**/pnpm-lock.yaml') }} restore-keys: | ${{ runner.os }}-node- - - run: sudo npm i -g pnpm - - run: npm run ci - - run: npx prettier --list-different . + - run: sudo npm install --global pnpm + - run: pnpm ci - run: npx build -- echo "Check tsc" + - run: npx locale-kit - run: npm run lint:typecoverage - - run: npm run locale-kit + working-directory: packages/maskbook - run: npm run lint:report - name: Upload ESLint Report uses: actions/upload-artifact@v2 with: name: eslint path: ./reports/junit + prettier: + runs-on: ubuntu-20.04 + steps: + - uses: actions/checkout@v2 + with: + ref: ${{ github.event.pull_request.head.sha }} + - uses: actions/setup-node@v2-beta + - run: npx prettier --list-different . storybook: runs-on: ubuntu-20.04 continue-on-error: false diff --git a/.husky/.gitignore b/.husky/.gitignore index c9cdc63b0701..31354ec13899 100644 --- a/.husky/.gitignore +++ b/.husky/.gitignore @@ -1 +1 @@ -_ \ No newline at end of file +_ diff --git a/.husky/pre-commit b/.husky/pre-commit index 60003c35330d..8c404449e3ae 100755 --- a/.husky/pre-commit +++ b/.husky/pre-commit @@ -2,4 +2,3 @@ . "$(dirname $0)/_/husky.sh" npx lint-staged -npx locale-kit --sync-keys diff --git a/.vscode/settings.json b/.vscode/settings.json index 5314e6a3b198..b09315befb4a 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -23,5 +23,23 @@ "cSpell.words": ["Reificated"], "markdownlint.config": { "no-inline-html": false + }, + "[html]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[javascript]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[typescript]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[typescriptreact]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[json]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[jsonc]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" } } diff --git a/netlify.toml b/netlify.toml index 885bccd61998..22f2a524f310 100644 --- a/netlify.toml +++ b/netlify.toml @@ -1,5 +1,5 @@ [build] - command = "npm run netlify-install && cd packages/cli-dev-build && npx gulp buildNetlify" + command = "./scripts/netlify.sh" publish = "packages/netlify" ignore = "git diff --quiet HEAD^ HEAD packages/dashboard packages/storybook-shared packages/theme packages/netlify packages/shared packages/icons" [build.environment] diff --git a/package.json b/package.json index ce9311c1a813..68b294db61aa 100644 --- a/package.json +++ b/package.json @@ -6,30 +6,27 @@ "scripts": { "start": "dev", "start-e2e": "env target=E2E dev", - "ts": "dev -- echo Starting TypeScript compiler...", + "ts": "dev --daemon", "go": "dev --help", "start:storybook": "dev -- start-storybook -p 9009 -s public --quiet", "build": "build", "build-e2e": "env target=E2E build", - "build-ci": "ts-node scripts/ci-build.ts", "build-storybook": "build -- build-storybook -s public --quiet", "build-ios": "run-s build:ios:webpack build:ios:ext:prebuilt", - "build:ios:ext:prebuilt": "ts-node scripts/webextensiom-shim-prebuilt.ts", + "build:ios:ext:prebuilt": "webextensiom-shim-prebuilt", "build:ios:webpack": "build iOS", - "lint": "eslint -c ./packages/.eslintrc.json packages/ --ext .ts,.tsx --fix --cache", - "lint:report": "eslint -c ./packages/.eslintrc.json packages/ --ext .ts,.tsx --format junit -o reports/junit/eslint-results.xml --cache", - "lint:typecoverage": "type-coverage --strict -p packages/maskbook --cache --ignore-catch --ignore-files packages/maskbook/src/plugins/Wallet/contracts/** --ignore-files **/__tests__/** --ignore-files **\\__tests__\\** --detail --at-least 98", - "locale-kit": "locale-kit", + "lint": "eslint -c packages/.eslintrc.json packages --ext .ts,.tsx --fix --cache", + "lint:report": "eslint -c packages/.eslintrc.json packages --ext .ts,.tsx --format junit -o reports/junit/eslint-results.xml --cache", "test": "jest --verbose --watch", "test:ci": "jest --verbose --ci --collectCoverage=true --reporters=default --reporters=jest-junit -w 1 --forceExit", "test:e2e": "jest --verbose --runInBand --config=jest-e2e.config.js", "preinstall": "npx only-allow pnpm", - "ci": "pnpm install --frozen-lockfile --store-dir ~/.pnpm/", - "netlify-install": "test \"$CI\" = true && npx pnpm install -r --store=node_modules/.pnpm-store || echo skiping pnpm install" + "ci": "pnpm install --frozen-lockfile --store-dir ~/.pnpm/" }, "lint-staged": { - "*": "prettier --check", - "*.{ts,tsx,js,jsx}": "eslint --cache --fix" + "*": "prettier --write --ignore-unknown", + "*.{ts,tsx,js,jsx}": "eslint -c packages/.eslintrc.json --cache --fix", + "packages/maskbook/**/*": "locale-kit --sync-keys --remove-unused-keys" }, "dependencies": { "@emotion/cache": "^11", @@ -49,8 +46,7 @@ "devDependencies": { "@commitlint/cli": "^11.0.0", "@commitlint/config-conventional": "^11.0.0", - "@dimensiondev/locale-kit": "workspace:*", - "@dimensiondev/maskbook-cli-dev-or-build": "workspace:*", + "@dimensiondev/cli": "workspace:*", "@magic-works/commonjs-import.meta": "^1.0.0", "@magic-works/i18n-codegen": "^0.0.5", "@material-ui/codemod": "^5.0.0-alpha.24", @@ -70,17 +66,16 @@ "prettier": "^2.1.2", "ts-node": "^9.1.1", "ttypescript": "^1.5.12", - "type-coverage": "^2.14.8", "typescript": "4.1.3" }, + "engines": { + "node": ">=14.4.0", + "pnpm": ">=5" + }, "pnpm": { "overrides": { "react": "=0.0.0-experimental-dc27b5aaa", "react-dom": "=0.0.0-experimental-dc27b5aaa" } - }, - "engines": { - "node": ">=14.4.0", - "pnpm": ">=5" } } diff --git a/packages/README.md b/packages/README.md index d236d02e7a1e..81bf809aab4d 100644 --- a/packages/README.md +++ b/packages/README.md @@ -20,7 +20,7 @@ ## Tools -- [cli-dev-build](cli-dev-build) +- [cli](cli) Because our project is using TypeScript project reference and code generation, to make the build process work normally, those process must run before any other commands. diff --git a/packages/cli-dev-build/build.js b/packages/cli-dev-build/build.js deleted file mode 100755 index 43ba2a68466f..000000000000 --- a/packages/cli-dev-build/build.js +++ /dev/null @@ -1,14 +0,0 @@ -#!/usr/bin/env node - -const { spawn } = require('child_process') -async function main() { - await require('./ts').build() - if (process.argv[2] === '--') { - spawn(process.argv.slice(3).join(' '), { shell: true, stdio: 'inherit' }) - } else { - require('./main.js')('build') - } -} -main().catch((e) => { - throw e -}) diff --git a/packages/cli-dev-build/dev.js b/packages/cli-dev-build/dev.js deleted file mode 100755 index c7503eb80a51..000000000000 --- a/packages/cli-dev-build/dev.js +++ /dev/null @@ -1,17 +0,0 @@ -#!/usr/bin/env node - -const { spawn } = require('child_process') -async function main() { - await require('./ts') - .build() - .catch(() => {}) - require('./ts').dev() - if (process.argv[2] === '--') { - spawn(process.argv.slice(3).join(' '), { shell: true, stdio: 'inherit' }) - } else { - require('./main.js')('dev') - } -} -main().catch((e) => { - throw e -}) diff --git a/packages/cli-dev-build/gulpfile.js b/packages/cli-dev-build/gulpfile.js deleted file mode 100644 index 51395a1148e3..000000000000 --- a/packages/cli-dev-build/gulpfile.js +++ /dev/null @@ -1,2 +0,0 @@ -Object.assign(module.exports, require('./netlify')) -Object.assign(module.exports, require('./ts')) diff --git a/packages/cli-dev-build/main.js b/packages/cli-dev-build/main.js deleted file mode 100644 index ec9eb72f6e22..000000000000 --- a/packages/cli-dev-build/main.js +++ /dev/null @@ -1,43 +0,0 @@ -const { spawn } = require('child_process') -const { resolve } = require('path') - -const presets = ['chromium', 'E2E', 'firefox', 'android', 'iOS', 'base'] -const otherFlags = ['beta', 'insider', 'reproducible', 'profile', 'manifest-v3'] -const knownTargets = ['-h', '--help', ...presets, ...otherFlags] -/** @param {'dev' | 'build'} mode */ -async function main(mode) { - let args = process.argv.slice(2) - - if (args.includes('-h') || args.includes('--help')) { - const inquirer = require('inquirer') - const { preset } = await inquirer.prompt({ - type: 'list', - name: 'preset', - message: 'Choose preset', - choices: presets, - }) - const { flags } = await inquirer.prompt({ - type: 'checkbox', - name: 'flags', - choices: otherFlags, - }) - args = [...flags, preset] - const { confirm } = await inquirer.prompt({ - type: 'confirm', - name: 'confirm', - message: `Command is: "npx ${mode === 'dev' ? 'dev' : 'build'} ${args.join(' ')}". Is that OK?`, - }) - if (!confirm) return - } - - const command = ['--mode', mode === 'dev' ? 'development' : 'production'] - if (mode === 'dev') command.unshift('serve') - args.filter((x) => !x.startsWith('-')).forEach((target) => { - command.push('--env', target) - if (!knownTargets.includes(target)) { - throw new TypeError('Unknown target ' + target + '. Known targets: ' + knownTargets.join(',')) - } - }) - spawn('npx', ['webpack', ...command], { stdio: 'inherit', shell: true, cwd: resolve(__dirname, '../maskbook/') }) -} -module.exports = main diff --git a/packages/cli-dev-build/netlify.js b/packages/cli-dev-build/netlify.js deleted file mode 100644 index 37ce7b71403c..000000000000 --- a/packages/cli-dev-build/netlify.js +++ /dev/null @@ -1,45 +0,0 @@ -const { series, parallel } = require('gulp') -const { spawn } = require('child_process') - -const { join, relative } = require('path') -const netlify = join(__dirname, '../netlify') - -const createBuildStorybook6 = (basePath, output, name) => { - const f = () => { - const r = relative(basePath, output) - return spawn(`npx`, ['build-storybook', '-o', r, '-s', r, '--quiet'], { - cwd: basePath, - shell: true, - stdio: 'inherit', - }) - } - f.displayName = name + '-storybook' - f.description = `Build storybook of ${name} to ${output}` - return f -} - -const createBuildSnowpack = (basePath, output, name) => { - const f = () => { - const r = relative(basePath, output) - return spawn(`npx`, ['snowpack', 'build', '--buildOptions.out', r, '--quiet'], { - cwd: basePath, - shell: true, - stdio: 'inherit', - }) - } - f.displayName = name + '-snowpack' - f.description = `Build snowpack of ${name} to ${output}` - return f -} - -const { build } = require('./ts') -const iconsSnowpack = createBuildSnowpack(join(__dirname, '../icons'), join(netlify, 'snowpack/icons'), 'icons') -const dashboard = createBuildSnowpack(join(__dirname, '../dashboard'), join(netlify, 'snowpack/dashboard'), 'dashboard') -const dashboardSB = createBuildStorybook6( - join(__dirname, '../dashboard'), - join(netlify, 'storybook-static/dashboard'), - 'dashboard-snowpack', -) -const themeSB = createBuildStorybook6(join(__dirname, '../theme'), join(netlify, 'storybook-static/theme'), 'theme') - -exports.buildNetlify = parallel(iconsSnowpack, series(build, parallel(dashboardSB, themeSB, dashboard))) diff --git a/packages/cli-dev-build/package.json b/packages/cli-dev-build/package.json deleted file mode 100644 index 097081d4dc67..000000000000 --- a/packages/cli-dev-build/package.json +++ /dev/null @@ -1,15 +0,0 @@ -{ - "name": "@dimensiondev/maskbook-cli-dev-or-build", - "private": true, - "bin": { - "build": "./build.js", - "dev": "./dev.js" - }, - "dependencies": { - "inquirer": "^7.3.3", - "proper-lockfile": "^4.1.1" - }, - "devDependencies": { - "gulp": "^4.0.2" - } -} diff --git a/packages/cli-dev-build/process-lock.js b/packages/cli-dev-build/process-lock.js deleted file mode 100644 index 6c66cebbee67..000000000000 --- a/packages/cli-dev-build/process-lock.js +++ /dev/null @@ -1,25 +0,0 @@ -const properLock = require('proper-lockfile') -const path = require('path') -const lockfilePath = path.join(__dirname, './lockfile.log') - -function delay(ms) { - return new Promise((r) => setTimeout(r, ms)) -} -module.exports = async function* () { - while (true) { - if (!(await properLock.check(__filename, { lockfilePath }))) yield lock - await delay(2000) - } - - /** @returns {Promise} */ - function lock() { - return new Promise((resolve) => { - properLock - .lock(__filename, { - onCompromised: () => resolve(false), - lockfilePath, - }) - .then(() => resolve(true)) - }) - } -} diff --git a/packages/cli-dev-build/ts.js b/packages/cli-dev-build/ts.js deleted file mode 100644 index e69b49593ed3..000000000000 --- a/packages/cli-dev-build/ts.js +++ /dev/null @@ -1,30 +0,0 @@ -// start a one-time "tsc -b" or parallel "tsc -b -w" - -const { spawn } = require('child_process') -const { join, resolve } = require('path') -const Lock = require('./process-lock') -const { runCli } = require('@magic-works/i18n-codegen') - -const configFile = resolve(__dirname, '../../.i18n-codegen.json') -const args = { stdio: 'inherit', shell: true, cwd: join(__dirname, '../..') } - -module.exports.dev = async function dev() { - lock: for await (const lock of Lock()) { - if (await lock()) break lock - } - runCli({ config: configFile, watch: true }) - spawn('npx', ['tsc', '-b', '-w'], args) -} -module.exports.dev.displayName = 'ts' -module.exports.dev.description = 'Start to watch TypeScript project reference' - -module.exports.build = () => { - return new Promise((resolve, reject) => { - runCli({ config: configFile }) - const p = spawn('npx', ['tsc', '-b'], args) - p.on('close', resolve) - p.on('error', reject) - }) -} -module.exports.build.displayName = 'build-ts' -module.exports.build.description = 'Build TypeScript project reference' diff --git a/packages/cli/.gitignore b/packages/cli/.gitignore new file mode 100644 index 000000000000..36f6455180d4 --- /dev/null +++ b/packages/cli/.gitignore @@ -0,0 +1 @@ +!/build diff --git a/packages/cli-dev-build/README.md b/packages/cli/README.md similarity index 99% rename from packages/cli-dev-build/README.md rename to packages/cli/README.md index c81c86525299..94290590bdd2 100644 --- a/packages/cli-dev-build/README.md +++ b/packages/cli/README.md @@ -1,4 +1,4 @@ -# cli-dev-build +# cli This package provides two commands: `build` and `dev`. diff --git a/scripts/ci-build.ts b/packages/cli/build/build-ci.ts old mode 100644 new mode 100755 similarity index 92% rename from scripts/ci-build.ts rename to packages/cli/build/build-ci.ts index 39ddc681e4df..7e070b3026ef --- a/scripts/ci-build.ts +++ b/packages/cli/build/build-ci.ts @@ -1,5 +1,6 @@ +#!/usr/bin/env ts-node import git from '@nice-labs/git-rev' -import { BUILD_PATH, run } from './utils' +import { BUILD_PATH, run } from '../utils' const branch = git.branchName() const types = { diff --git a/packages/cli/build/build.ts b/packages/cli/build/build.ts new file mode 100755 index 000000000000..353f50befadf --- /dev/null +++ b/packages/cli/build/build.ts @@ -0,0 +1,21 @@ +#!/usr/bin/env ts-node +import { spawn } from 'child_process' +import { awaitChildProcess, ROOT_PATH } from '../utils' +import onMain from './main' +import { build } from './typescript' + +async function main() { + await build() + if (process.argv[2] === '--') { + return spawn(process.argv[3], process.argv.slice(4), { + cwd: ROOT_PATH, + stdio: 'inherit', + shell: true, + }) + } + return onMain('build') +} + +main().then(async (child) => { + process.exit(await awaitChildProcess(child)) +}) diff --git a/packages/cli/build/dev.ts b/packages/cli/build/dev.ts new file mode 100755 index 000000000000..e993cd6cdfc5 --- /dev/null +++ b/packages/cli/build/dev.ts @@ -0,0 +1,28 @@ +#!/usr/bin/env ts-node +import { spawn } from 'child_process' +import { awaitChildProcess, ROOT_PATH } from '../utils' +import onMain from './main' +import { build, dev } from './typescript' +import { noop } from 'lodash' + +async function main() { + await build() + dev() + if (process.argv[2] === '--daemon') { + console.log('Starting TypeScript compiler...') + // Never ends + return new Promise(noop) + } + if (process.argv[2] === '--') { + return spawn(process.argv[3], process.argv.slice(4), { + cwd: ROOT_PATH, + stdio: 'inherit', + shell: true, + }) + } + return onMain('dev') +} + +main().then(async (child) => { + process.exit(await awaitChildProcess(child)) +}) diff --git a/packages/cli/build/gulpfile.ts b/packages/cli/build/gulpfile.ts new file mode 100644 index 000000000000..7fc3b8018ebb --- /dev/null +++ b/packages/cli/build/gulpfile.ts @@ -0,0 +1,2 @@ +export * from './netlify' +export * from './typescript' diff --git a/packages/cli/build/main.ts b/packages/cli/build/main.ts new file mode 100644 index 000000000000..2dca9c88a3e4 --- /dev/null +++ b/packages/cli/build/main.ts @@ -0,0 +1,59 @@ +import { spawn } from 'child_process' +import { resolve } from 'path' +import { PKG_PATH } from '../utils' +import { compact } from 'lodash' + +const presets = ['chromium', 'E2E', 'firefox', 'android', 'iOS', 'base'] +const otherFlags = ['beta', 'insider', 'reproducible', 'profile', 'manifest-v3'] +const knownTargets = ['-h', '--help', ...presets, ...otherFlags] + +async function main(mode: 'dev' | 'build') { + let targets = process.argv.slice(2) + + if (targets.includes('-h') || targets.includes('--help')) { + const inquirer = require('inquirer') + const { preset } = await inquirer.prompt({ + type: 'list', + name: 'preset', + message: 'Choose preset', + choices: presets, + }) + const { flags } = await inquirer.prompt({ + type: 'checkbox', + name: 'flags', + choices: otherFlags, + }) + targets = [...flags, preset] + + const command = ['npx', mode === 'dev' ? 'dev' : 'build', ...targets] + const { confirm } = await inquirer.prompt({ + type: 'confirm', + name: 'confirm', + message: `Command is: "${command.join(' ')}". Is that OK?`, + }) + if (!confirm) return process.exit(0) + } + + // prettier-ignore + const command = [ + 'webpack', + mode === 'dev' ? 'serve' : undefined, + '--mode', + mode === 'dev' ? 'development' : 'production', + ] + for (const target of targets) { + if (target.startsWith('-')) { + continue + } else if (!knownTargets.includes(target)) { + throw new TypeError(`Unknown target ${target}. Known targets: ${knownTargets}`) + } + command.push('--env', target) + } + return spawn('npx', compact(command), { + cwd: resolve(PKG_PATH, 'maskbook'), + stdio: 'inherit', + shell: true, + }) +} + +export default main diff --git a/packages/cli/build/netlify.ts b/packages/cli/build/netlify.ts new file mode 100644 index 000000000000..2a959291f07c --- /dev/null +++ b/packages/cli/build/netlify.ts @@ -0,0 +1,63 @@ +import { spawn } from 'child_process' +import { parallel, series } from 'gulp' +import { relative, resolve } from 'path' +import { build } from './typescript' +import { NETLIFY_PATH, PKG_PATH } from '../utils' + +const createBuildStorybook6 = (basePath: string, output: string, name: string) => { + const fn = () => { + const r = relative(basePath, output) + return spawn('npx', ['build-storybook', '-o', r, '-s', r, '--quiet'], { + cwd: basePath, + shell: true, + stdio: 'inherit', + }) + } + fn.displayName = `${name}-storybook` + fn.description = `Build storybook of ${name} to ${output}` + return fn +} + +const createBuildSnowpack = (basePath: string, output: string, name: string) => { + const fn = () => { + const r = relative(basePath, output) + return spawn('npx', ['snowpack', 'build', '--buildOptions.out', r, '--quiet'], { + cwd: basePath, + shell: true, + stdio: 'inherit', + }) + } + fn.displayName = `${name}-snowpack` + fn.description = `Build snowpack of ${name} to ${output}` + return fn +} + +const SNOWPACK_PATH = relative(NETLIFY_PATH, 'snowpack') +const STATIC_PATH = resolve(NETLIFY_PATH, 'storybook-static') + +// prettier-ignore +const iconsSnowpack = createBuildSnowpack( + resolve(PKG_PATH, 'icons'), + resolve(SNOWPACK_PATH, 'icons'), + 'icons', +) +// prettier-ignore +const dashboard = createBuildSnowpack( + resolve(PKG_PATH, 'dashboard'), + resolve(SNOWPACK_PATH, 'dashboard'), + 'dashboard', +) +// prettier-ignore +const dashboardSB = createBuildStorybook6( + resolve(PKG_PATH, 'dashboard'), + resolve(STATIC_PATH, 'dashboard'), + 'dashboard-snowpack', +) +// prettier-ignore +const themeSB = createBuildStorybook6( + resolve(PKG_PATH, 'theme'), + resolve(STATIC_PATH, 'theme'), + 'theme', +) + +export const buildNetlify = parallel(iconsSnowpack, series(build, parallel(dashboardSB, themeSB, dashboard))) diff --git a/packages/cli/build/process-lock.ts b/packages/cli/build/process-lock.ts new file mode 100644 index 000000000000..ada75900938d --- /dev/null +++ b/packages/cli/build/process-lock.ts @@ -0,0 +1,24 @@ +import path from 'path' +import Lock from 'proper-lockfile' +import { delay } from '../utils' + +const lockfilePath = path.join(__dirname, 'lockfile.log') + +export default async function* () { + while (true) { + if (!(await Lock.check(__filename, { lockfilePath }))) { + yield locker + } + await delay(2000) + } +} + +function locker() { + return new Promise(async (resolve) => { + await Lock.lock(__filename, { + lockfilePath, + onCompromised: () => resolve(false), + }) + resolve(true) + }) +} diff --git a/packages/cli/build/typescript.ts b/packages/cli/build/typescript.ts new file mode 100644 index 000000000000..4753699405b1 --- /dev/null +++ b/packages/cli/build/typescript.ts @@ -0,0 +1,34 @@ +import { runCli } from '@magic-works/i18n-codegen' +import { spawn, SpawnOptions } from 'child_process' +import { resolve } from 'path' +import { ROOT_PATH } from '../utils' +import makeLockSequence from './process-lock' + +const config = resolve(ROOT_PATH, '.i18n-codegen.json') +const options: SpawnOptions = { + cwd: ROOT_PATH, + stdio: 'inherit', + shell: true, +} + +export async function dev() { + lock: for await (const lock of makeLockSequence()) { + if (await lock()) break lock + } + runCli({ config, watch: true }, console.error) + // start a one-time "tsc -b" or parallel "tsc -b -w" + return spawn('npx', ['tsc', '-b', '-w'], options) +} +dev.displayName = 'ts' +dev.description = 'Start to watch TypeScript project reference' + +export const build = () => { + return new Promise((resolve, reject) => { + runCli({ config }, console.error) + const process = spawn('npx', ['tsc', '-b'], options) + process.on('close', resolve) + process.on('error', reject) + }) +} +build.displayName = 'build-ts' +build.description = 'Build TypeScript project reference' diff --git a/scripts/webextensiom-shim-prebuilt.ts b/packages/cli/build/webextensiom-shim-prebuilt.ts old mode 100644 new mode 100755 similarity index 90% rename from scripts/webextensiom-shim-prebuilt.ts rename to packages/cli/build/webextensiom-shim-prebuilt.ts index e03a43ff1c43..2e81d6602361 --- a/scripts/webextensiom-shim-prebuilt.ts +++ b/packages/cli/build/webextensiom-shim-prebuilt.ts @@ -1,6 +1,7 @@ +#!/usr/bin/env ts-node import fs from 'fs' import path from 'path' -import { BUILD_PATH, run } from './utils' +import { BUILD_PATH, run } from '../utils' const BUILD_JS_PATH = path.join(BUILD_PATH, 'js') diff --git a/packages/locale-kit/ast.ts b/packages/cli/locale-kit/ast.ts similarity index 100% rename from packages/locale-kit/ast.ts rename to packages/cli/locale-kit/ast.ts diff --git a/packages/locale-kit/locale-kit.ts b/packages/cli/locale-kit/locale-kit.ts old mode 100644 new mode 100755 similarity index 81% rename from packages/locale-kit/locale-kit.ts rename to packages/cli/locale-kit/locale-kit.ts index 7929a01c9f82..7992e63c4b1a --- a/packages/locale-kit/locale-kit.ts +++ b/packages/cli/locale-kit/locale-kit.ts @@ -1,13 +1,12 @@ +#!/usr/bin/env ts-node /* eslint-disable no-restricted-imports */ import { difference, isEmpty, isNil, keys, omit, pick, toPairs, without } from 'lodash' -import { run } from '../../scripts/utils' import { findAllUnusedKeys, findAllUsedKeys, getLocaleRelativePath, getMessagePath, LOCALE_NAMES, - LOCALE_PATH, readMessages, writeMessages, } from './utils' @@ -60,11 +59,10 @@ async function syncKeys(locales = without(LOCALE_NAMES, 'en')) { async function diagnosis() { const unusedKeys = await findAllUnusedKeys() if (unusedKeys.length) { + const message = 'Run `npx locale-kit --remove-unused-keys` to solve this problem' for (const locale of LOCALE_NAMES) { const filePath = getLocaleRelativePath(getMessagePath(locale)) - console.log( - `::warning file=${filePath}::Run \`npm run locale-kit -- --remove-unused-keys\` to solve this problem`, - ) + console.log(`::warning file=${filePath}::${message}`) const messages = keys(await readMessages(locale)) for (const key of unusedKeys) { const index = messages.indexOf(key) @@ -76,9 +74,10 @@ async function diagnosis() { } const unsyncedLocales = await findAllUnsyncedLocales() if (!isEmpty(unsyncedLocales)) { + const message = 'Run `npx locale-kit --sync-keys` to solve this problem' for (const [locale, names] of toPairs(unsyncedLocales)) { const filePath = getLocaleRelativePath(getMessagePath(locale)) - console.log(`::warning file=${filePath}::Run \`npm run locale-kit -- --sync-keys\` to solve this problem`) + console.log(`::warning file=${filePath}::${message}`) for (const name of names) { console.log(`::warning file=${filePath}::The ${JSON.stringify(name)} is unsynced`) } @@ -87,15 +86,8 @@ async function diagnosis() { } async function main() { - const unusedKeys = await findAllUnusedKeys() - console.error( - 'Scanned', - unusedKeys.length, - 'unused keys, run `npm run locale-kit -- --remove-unused-keys` to remove them.', - ) - console.error('Unsynced', keys(await findAllUnsyncedLocales()), 'locales') if (process.argv.includes('--remove-unused-keys')) { - await removeAllUnusedKeys(unusedKeys) + await removeAllUnusedKeys(await findAllUnusedKeys()) console.log('Unused keys removed') } if (process.argv.includes('--set-missing-keys')) { @@ -111,7 +103,5 @@ async function main() { if (process.env.CI) { diagnosis() } else { - main().then(() => { - run(LOCALE_PATH, 'git', 'add', '.') - }) + main() } diff --git a/packages/locale-kit/utils.ts b/packages/cli/locale-kit/utils.ts similarity index 82% rename from packages/locale-kit/utils.ts rename to packages/cli/locale-kit/utils.ts index 5c1702f4b3bc..2e6279370143 100644 --- a/packages/locale-kit/utils.ts +++ b/packages/cli/locale-kit/utils.ts @@ -1,19 +1,20 @@ /* eslint-disable no-restricted-imports */ import { promises as fs, readdirSync } from 'fs' import { difference, keys, uniq, without } from 'lodash' -import * as path from 'path' +import { resolve, relative } from 'path' import { getUsedKeys } from './ast' +import { ROOT_PATH, PKG_PATH } from '../utils' -const SOURCE_PATH = path.join(__dirname, '..', 'maskbook', 'src') -export const LOCALE_PATH = path.join(SOURCE_PATH, '_locales') +const SOURCE_PATH = resolve(PKG_PATH, 'maskbook', 'src') +export const LOCALE_PATH = resolve(SOURCE_PATH, '_locales') export const LOCALE_NAMES = readdirSync(LOCALE_PATH) export function getMessagePath(name: string) { - return path.join(LOCALE_PATH, name, 'messages.json') + return resolve(LOCALE_PATH, name, 'messages.json') } export function getLocaleRelativePath(...paths: string[]) { - return path.relative(path.join(__dirname, '..', '..'), path.join(LOCALE_PATH, ...paths)) + return relative(ROOT_PATH, resolve(LOCALE_PATH, ...paths)) } export async function readMessages(name: string) { @@ -51,7 +52,7 @@ export async function findAllUnsyncedLocales(locales = without(LOCALE_NAMES, 'en async function* walk(dir: string): AsyncIterableIterator { for await (const dirent of await fs.opendir(dir)) { - const entry = path.join(dir, dirent.name) + const entry = resolve(dir, dirent.name) if (dirent.isDirectory()) { yield* walk(entry) } else if (dirent.isFile() && /\.(tsx?)$/.test(entry)) { diff --git a/packages/cli/package.json b/packages/cli/package.json new file mode 100644 index 000000000000..af932ad70d9c --- /dev/null +++ b/packages/cli/package.json @@ -0,0 +1,24 @@ +{ + "name": "@dimensiondev/cli", + "private": true, + "bin": { + "build": "./build/build.ts", + "build-ci": "./build/build-ci.ts", + "dev": "./build/dev.ts", + "locale-kit": "./locale-kit/locale-kit.ts", + "webextensiom-shim-prebuilt": "./build/webextensiom-shim-prebuilt.ts" + }, + "dependencies": { + "inquirer": "^7.3.3", + "lodash": "^4.17.21", + "proper-lockfile": "^4.1.1" + }, + "devDependencies": { + "@types/gulp": "^4.0.8", + "@types/inquirer": "^7.3.1", + "@types/lodash": "^4.14.168", + "@types/proper-lockfile": "^4.1.1", + "gulp": "^4.0.2", + "ts-node": "^9.1.1" + } +} diff --git a/packages/cli/tsconfig.json b/packages/cli/tsconfig.json new file mode 100644 index 000000000000..91e8238c324f --- /dev/null +++ b/packages/cli/tsconfig.json @@ -0,0 +1,9 @@ +{ + "extends": "../../tsconfig.json", + "ts-node": { + "compilerOptions": { + "target": "ES2018", + "module": "CommonJS" + } + } +} diff --git a/packages/cli/utils.ts b/packages/cli/utils.ts new file mode 100644 index 000000000000..a274461d9d28 --- /dev/null +++ b/packages/cli/utils.ts @@ -0,0 +1,26 @@ +import { ChildProcess, spawnSync } from 'child_process' +import os from 'os' +import { relative, resolve } from 'path' + +export const ROOT_PATH = resolve(__dirname, '..', '..') +export const PKG_PATH = resolve(ROOT_PATH, 'packages') +export const BUILD_PATH = resolve(ROOT_PATH, 'build') +export const NETLIFY_PATH = resolve(PKG_PATH, 'netlify') + +export function run(cwd = ROOT_PATH, cmd: string, ...args: string[]) { + console.log('$', cmd, args.join(' '), '# cwd:', relative(ROOT_PATH, cwd)) + return spawnSync(cmd, args, { + cwd, + stdio: 'inherit', + shell: os.platform() === 'win32', + }) +} + +export const delay = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms)) + +export const awaitChildProcess = (child: ChildProcess) => { + return new Promise((resolve) => { + child.on('error', () => resolve(child.exitCode)) + child.on('exit', (code) => resolve(code)) + }) +} diff --git a/packages/contracts/CHANGELOG.md b/packages/contracts/CHANGELOG.md new file mode 100644 index 000000000000..afe987f456bb --- /dev/null +++ b/packages/contracts/CHANGELOG.md @@ -0,0 +1,45 @@ +# Changelog + +## AirDrop + +> Feb-26-2021 + +```plain +commit 9ad87a1 +Mainnet 0x7adcfcafb69aae283f4268e59209ba64a5db4d91 +Ropsten 0xf1276B659FF28EDf92accFb013B1E6c9F782bFfF +Rinkeby 0x2189D088927bfc1563d0b3E447F6472402FED6DD +``` + +## ITO + +> Jan-16-2021 + +```plain +commit 668d573 +Mainnet 0x7aEa34bE68171c6898164f3986Db03964CCa49B7 +Ropsten 0xDF0e910DcC73bdC8f4c332A4C12545928683221f +Rinkeby 0x7751b8c715d1Df74D181C86aE01801330211f370 +``` + +## MaskITO + +> Feb-21-2021 + +```plain +commit 3ea218f +Mainnet 0x86812da3A623ab9606976078588b80C315E55FA3 +Ropsten 0x677f7BbA13108649ECFF068E8B3d55631327B83a +Rinkeby 0x0aC41A27bA9F132D5687CAC986f5302Da6f5F9f6 +``` + +## HappyRedPacket + +> Jan-21-2020 + +```plain +commit cf787df +Mainnet 0x26760783c12181efa3c435aee4ae686c53bdddbb +Ropsten 0x6d84e4863c0530bc0bb4291ef0ff454a40660ca3 +Rinkeby 0x575f906db24154977c7361c2319e2b25e897e3b6 +``` diff --git a/packages/contracts/README.md b/packages/contracts/README.md index 9c319243620a..50b3327428b0 100644 --- a/packages/contracts/README.md +++ b/packages/contracts/README.md @@ -31,3 +31,7 @@ $ git add . # cwd: packages/contracts/types $ # or compile a specific contract $ npm start -- --abi ITO ``` + +## About CHANGELOG.md + +The date is the date of creating contract on the mainnet, commit is the commit of relative smart contract repo which generate the abi. diff --git a/packages/contracts/abis/ITO.json b/packages/contracts/abis/ITO.json index 2f4cc87d804b..d1570ac04aba 100644 --- a/packages/contracts/abis/ITO.json +++ b/packages/contracts/abis/ITO.json @@ -4,31 +4,6 @@ "stateMutability": "nonpayable", "type": "constructor" }, - { - "anonymous": false, - "inputs": [ - { - "indexed": false, - "internalType": "address", - "name": "claimer", - "type": "address" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "timestamp", - "type": "uint256" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "to_value", - "type": "uint256" - } - ], - "name": "ClaimSuccess", - "type": "event" - }, { "anonymous": false, "inputs": [ @@ -221,32 +196,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [], - "name": "check_claimable", - "outputs": [ - { - "internalType": "uint256", - "name": "claimable_amount", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "claim", - "outputs": [ - { - "internalType": "uint256", - "name": "claimed_amount", - "type": "uint256" - } - ], - "stateMutability": "nonpayable", - "type": "function" - }, { "inputs": [], "name": "contract_creator", @@ -336,97 +285,6 @@ "stateMutability": "payable", "type": "function" }, - { - "inputs": [], - "name": "getUnlockTime", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "get_ito_list", - "outputs": [ - { - "internalType": "bytes32[]", - "name": "", - "type": "bytes32[]" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "mask_address", - "outputs": [ - { - "internalType": "address", - "name": "", - "type": "address" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "future_admin", - "type": "address" - } - ], - "name": "setAdmin", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "_unlock_time", - "type": "uint256" - } - ], - "name": "setUnlockTime", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "bb", - "type": "address" - } - ], - "name": "set_bb_address", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "mask", - "type": "address" - } - ], - "name": "set_mask_address", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, { "inputs": [ { @@ -440,9 +298,9 @@ "type": "bytes32" }, { - "internalType": "bytes32", - "name": "verification2", - "type": "bytes32" + "internalType": "address", + "name": "_recipient", + "type": "address" }, { "internalType": "bytes32", @@ -488,31 +346,5 @@ "outputs": [], "stateMutability": "nonpayable", "type": "function" - }, - { - "inputs": [ - { - "internalType": "address[]", - "name": "addrs", - "type": "address[]" - } - ], - "name": "withdrawBatchCreator", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "addr", - "type": "address" - } - ], - "name": "withdrawCreator", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" } ] diff --git a/packages/contracts/types/ITO.d.ts b/packages/contracts/types/ITO.d.ts index fa2f719d97c2..ece4a8e532a9 100644 --- a/packages/contracts/types/ITO.d.ts +++ b/packages/contracts/types/ITO.d.ts @@ -1,5 +1,6 @@ -/* Generated by ts-generator ver. 0.0.8 */ +/* Autogenerated file. Do not edit manually. */ /* tslint:disable */ +/* eslint-disable */ import BN from 'bn.js' import { Contract, ContractOptions } from 'web3-eth-contract' @@ -34,10 +35,6 @@ export class ITO extends Contract { 5: string[] }> - check_claimable(): TransactionObject - - claim(): TransactionObject - contract_creator(): TransactionObject destruct(id: string | number[]): TransactionObject @@ -56,44 +53,18 @@ export class ITO extends Contract { _qualification: string, ): TransactionObject - getUnlockTime(): TransactionObject - - get_ito_list(): TransactionObject - - mask_address(): TransactionObject - - setAdmin(future_admin: string): TransactionObject - - setUnlockTime(_unlock_time: number | string): TransactionObject - - set_bb_address(bb: string): TransactionObject - - set_mask_address(mask: string): TransactionObject - swap( id: string | number[], verification: string | number[], - verification2: string | number[], + _recipient: string, validation: string | number[], exchange_addr_i: number | string, input_total: number | string, ): TransactionObject withdraw(id: string | number[], addr_i: number | string): TransactionObject - - withdrawBatchCreator(addrs: string[]): TransactionObject - - withdrawCreator(addr: string): TransactionObject } events: { - ClaimSuccess: ContractEvent<{ - claimer: string - timestamp: string - to_value: string - 0: string - 1: string - 2: string - }> DestructSuccess: ContractEvent<{ id: string token_address: string diff --git a/packages/dashboard/src/components/Balance/index.tsx b/packages/dashboard/src/components/Balance/index.tsx new file mode 100644 index 000000000000..3adfe9f4c7c4 --- /dev/null +++ b/packages/dashboard/src/components/Balance/index.tsx @@ -0,0 +1,108 @@ +import { memo } from 'react' +import { experimentalStyled as styled, Typography, Box, Button, buttonClasses } from '@material-ui/core' +import { useDashboardI18N } from '../../locales' +import { MaskColorVar } from '@dimensiondev/maskbook-theme' +import { MaskWalletIcon, SendIcon, CardIcon, SwapIcon, DownloadIcon } from '@dimensiondev/icons' + +export interface BalanceCardProps { + balance: number + onSend(): void + onBuy(): void + onSwap(): void + onReceive(): void +} + +const BalanceContainer = styled('div')( + ({ theme }) => ` + display: flex; + justify-content: space-between; + border-radius: 16px; + align-items: center; + padding: ${theme.spacing(2.5)}; + background: ${MaskColorVar.primaryBackground}; +`, +) + +const IconContainer = styled('div')` + // TODO: mobile + font-size: 48px; + display: flex; + justify-content: center; + align-items: center; + background: ${MaskColorVar.infoBackground}; + border-radius: 50%; +` + +const BalanceDisplayContainer = styled('div')( + ({ theme }) => ` + display: flex; + flex-direction: column; + justify-content: space-between; + margin-left: ${theme.spacing(1)}; +`, +) + +const BalanceTitle = styled(Typography)( + ({ theme }) => ` + font-size: ${theme.typography.subtitle2.fontSize}; + color: ${MaskColorVar.iconLight}; +`, +) + +const BalanceContent = styled(Typography)( + ({ theme }) => ` + font-size: ${theme.typography.h6.fontSize}; + color: ${MaskColorVar.textPrimary}; + line-height: ${theme.typography.h2.lineHeight}; +`, +) + +const ButtonGroup = styled('div')` + display: inline-grid; + gap: 10px; + grid-template-columns: repeat(4, 1fr); + & > * { + font-size: 12px; + & .${buttonClasses.endIcon} > *:nth-of-type(1) { + font-size: 16px; + fill: none; + } + } +` + +export const Balance = memo(({ balance, onSend, onBuy, onSwap, onReceive }: BalanceCardProps) => { + const t = useDashboardI18N() + + return ( + + + + + + + {t.wallets_balance()} + + {balance.toLocaleString('en', { + style: 'currency', + currency: 'USD', + })} + + + + + + + + + + + ) +}) diff --git a/packages/dashboard/src/locales/en.json b/packages/dashboard/src/locales/en.json index 6920c84334b1..0ca70124dec8 100644 --- a/packages/dashboard/src/locales/en.json +++ b/packages/dashboard/src/locales/en.json @@ -35,5 +35,10 @@ "wallets_create_wallet_tabs": "Create Wallet Tabs", "wallets_create_wallet_refresh": "Refresh", "wallets_create_wallet_remember_later": "Remember that later", - "wallets_create_wallet_verification": "Verification" + "wallets_create_wallet_verification": "Verification", + "wallets_balance": "balance", + "wallets_balance_Send": "Send", + "wallets_balance_Buy": "Buy", + "wallets_balance_Swap": "Swap", + "wallets_balance_Receive": "Receive" } diff --git a/packages/dashboard/src/locales/zh.json b/packages/dashboard/src/locales/zh.json index 9bf4d0f86c10..a3224fafd2ee 100644 --- a/packages/dashboard/src/locales/zh.json +++ b/packages/dashboard/src/locales/zh.json @@ -35,5 +35,10 @@ "wallets_create_wallet_tabs": "創建錢包標籤", "wallets_create_wallet_refresh": "刷新", "wallets_create_wallet_remember_later": "記得以後", - "wallets_create_wallet_verification": "确认" + "wallets_balance": "結餘", + "wallets_create_wallet_verification": "确认", + "wallets_balance_Send": "發送", + "wallets_balance_Buy": "購買", + "wallets_balance_Swap": "交換", + "wallets_balance_Receive": "接收" } diff --git a/packages/dashboard/stories/Balance.tsx b/packages/dashboard/stories/Balance.tsx new file mode 100644 index 000000000000..ff28c0597bf6 --- /dev/null +++ b/packages/dashboard/stories/Balance.tsx @@ -0,0 +1,16 @@ +import { story } from '@dimensiondev/maskbook-storybook-shared' +import { Balance as C } from '../src/components/Balance' +import { action } from '@storybook/addon-actions' +const { meta, of } = story(C) + +export default meta({ title: '/Components/Balance' }) + +export const Balance = of({ + args: { + balance: 9000000, + onBuy: action('onBuy'), + onSend: action('onSend'), + onSwap: action('onSwap'), + onReceive: action('onReceive'), + }, +}) diff --git a/packages/locale-kit/index.js b/packages/locale-kit/index.js deleted file mode 100755 index 271526b6bbbe..000000000000 --- a/packages/locale-kit/index.js +++ /dev/null @@ -1,3 +0,0 @@ -#!/usr/bin/env node -require('ts-node').register() -require('./locale-kit.ts') diff --git a/packages/locale-kit/package.json b/packages/locale-kit/package.json deleted file mode 100644 index f2f7fee33372..000000000000 --- a/packages/locale-kit/package.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "name": "@dimensiondev/locale-kit", - "private": true, - "bin": { - "locale-kit": "./index.js" - }, - "devDependencies": { - "lodash": "^4.17.21" - } -} diff --git a/packages/locale-kit/tsconfig.json b/packages/locale-kit/tsconfig.json deleted file mode 100644 index 4082f16a5d91..000000000000 --- a/packages/locale-kit/tsconfig.json +++ /dev/null @@ -1,3 +0,0 @@ -{ - "extends": "../../tsconfig.json" -} diff --git a/packages/maskbook/package.json b/packages/maskbook/package.json index 29e015397dbe..c463009952d1 100644 --- a/packages/maskbook/package.json +++ b/packages/maskbook/package.json @@ -2,7 +2,9 @@ "name": "@dimensiondev/maskbook", "private": true, "license": "AGPL-3.0-or-later", - "scripts": {}, + "scripts": { + "lint:typecoverage": "type-coverage" + }, "dependencies": { "@balancer-labs/sor": "^1.0.0", "@dimensiondev/common-protocols": "1.6.0-20201027083702-d0ae6e2", @@ -160,6 +162,7 @@ "storybook-addon-i18n": "^5.1.13", "ts-jest": "26.5.0", "ts-loader": "^8.0.13", + "type-coverage": "^2.17.0", "web-ext": "^5.4.1", "webpack": "^5.19.0", "webpack-bundle-analyzer": "^4.3.0", @@ -171,5 +174,17 @@ }, "peerDependencies": { "jss": "*" + }, + "typeCoverage": { + "cache": true, + "strict": true, + "detail": true, + "atLeast": 98, + "ignoreCatch": true, + "ignoreFiles": [ + "**/__tests__/**", + "**\\__tests__\\**" + ], + "showRelativePath": true } } diff --git a/packages/maskbook/src/_locales/en/messages.json b/packages/maskbook/src/_locales/en/messages.json index 7b345374a1cc..3a8e1cf89754 100644 --- a/packages/maskbook/src/_locales/en/messages.json +++ b/packages/maskbook/src/_locales/en/messages.json @@ -47,6 +47,7 @@ "dashboard": "Dashboard", "dashboard_tab_token": "Token", "dashboard_tab_collectibles": "Collectibles", + "dashboard_tab_transactions": "Transactions", "dashboard_backup_database_confirmation": "Ok, Back it up", "dashboard_backup_database_hint": "Create a database backup file. Do it regularly.", "dashboard_backup_persona_hint": "You can backup your persona in the following options.", @@ -161,7 +162,7 @@ "post_substitute_label": "Mask", "privacy_policy": "Privacy Policy", "privacy_policy_link": "https://legal.mask.io/maskbook/", - "private_key": "Private key", + "private_key": "Private Key", "persona_rename": "Rename Persona", "persona_name": "Persona Name", "qr_code": "QR Code", @@ -264,7 +265,6 @@ "wallet_price": "Price", "wallet_balance": "Balance", "wallet_value": "Value", - "wallet_chain": "on {{chain}}", "wallet_confirmation_hint": "I confirm that I need to create a wallet.", "wallet_new": "New Wallet", "wallet_notification": "This will create a new wallet other than the default wallet. You are responsible for keeping the private key to the wallet safe. Please also note that Mask won't be able to pay for your transactions on Ethereum.", diff --git a/packages/maskbook/src/_locales/ja/messages.json b/packages/maskbook/src/_locales/ja/messages.json index 054fe327987a..e39e305f084c 100644 --- a/packages/maskbook/src/_locales/ja/messages.json +++ b/packages/maskbook/src/_locales/ja/messages.json @@ -23,7 +23,7 @@ "new_avatar_url_placeholder": "空白のままであれば変更されません。", "backup": "バックアップする", "backup_database": "データベースをバックアップする", - "backup_persona": "人格をバックアップする", + "backup_persona": "ペルソナをバックアップする", "backup_wallet": "ウォレットをバックアップする", "backup_wallet_hint": "12の単語型パスフレーズ または以下の 秘密鍵大切に保管します。これがあればウォレットを復元できます。", "backup_qrcode_error": "不明なエラー: 再度 試してください。", @@ -41,15 +41,16 @@ "contacts": "連絡先", "groups": "グループ", "create": "作成", - "create_a_persona": "人格を作成", - "create_persona": "人格を作成", + "create_a_persona": "ペルソナを作成", + "create_persona": "ペルソナを作成", "creating": "作成中…", "dashboard": "ダッシュボード", "dashboard_tab_token": "トークン", "dashboard_tab_collectibles": "収集物", + "dashboard_tab_transactions": "", "dashboard_backup_database_confirmation": "バックアップしますか?", "dashboard_backup_database_hint": "データベースのバックアップを作成します。日常的に行いましょう。", - "dashboard_backup_persona_hint": "次の方法のうちから人格のバックアップ方法を選びましょう。", + "dashboard_backup_persona_hint": "次の方法のうちからペルソナのバックアップ方法を選びましょう。", "dashboard_delete_persona_confirm_hint": "{{name}}を削除しますか? この操作は取り消すことができません。", "dashboard_disconnect_profile_hint": "{{domain}}の@{{profile}}から{{persona}}を消し去りますか?この操作は取り消すことができません。", "dashboard_import_database_hint": "前回のデータベースバックアップから", @@ -58,7 +59,7 @@ "dashboard_mobile_test_link": "https://mask.io/download-links/#mobile", "dashboard_paste_database_base64_hint": "ここに Base64 を貼り付ける…", "dashboard_paste_database_backup_hint": "データベースのバックアップをテキストとしてここに貼り付けてくだい。", - "dashboard_persona_import_dialog_hint": "以下の方法で人格インポートできます", + "dashboard_persona_import_dialog_hint": "以下の方法でペルソナインポートできます", "dashboard_restoration_successful_hint": "{{time}}に取得されたバックアップから復元しました。", "dashboard_source_code": "ソースコード", "dashboard_source_code_link": "https://github.com/DimensionDev/Maskbook", @@ -78,7 +79,7 @@ "delete": "削除", "delete_contact": "連絡先を削除する", "delete_contact_confirmation": "{{contact}} を削除しますか? この操作は取り消せません。", - "delete_persona": "人格を削除する", + "delete_persona": "ペルソナを削除する", "delete_wallet": "ウォレットを削除する", "delete_wallet_hint": "本気ですか?
もしバックアップしなければ ここにある全てのお金を失う ことになります。", "hide_token": "トークンを隠す", @@ -101,7 +102,7 @@ "buy_now": "今すぐ購入", "setup_guide_optional": "[オプション]", "setup_guide_find_username_title": "ユーザー名を探す", - "setup_guide_find_username_text": "Mask の人格と接続するためにユーザー名が必要です。
正しいものを入力してください。", + "setup_guide_find_username_text": "Mask のペルソナと接続するためにユーザー名が必要です。
正しいものを入力してください。", "setup_guide_connect_auto": "接続する", "setup_guide_connect_failed": "失敗しました… もう一度お試しください。", "setup_guide_create_post_auto": "今すぐ作成", @@ -113,14 +114,14 @@ "import": "インポート", "import_failed": "インポートに失敗", "import_wallet": "ウォレットをインポート", - "import_your_persona": "人格をインポート", + "import_your_persona": "ペルソナをインポート", "internal_id": "内部 ID", "keywords": "キーワード", "language_en": "English", "language_zh": "中文", "language_ja": "日本語", "mnemonic_words": "パスフレーズ", - "my_personas": "私の人格アカウント", + "my_personas": "私のペルソナアカウント", "my_wallets": "私のウォレット", "name": "名前", "nickname": "ニックネーム", @@ -138,13 +139,13 @@ "person_or_group_in_list_0": "空のグループです", "person_or_group_in_list_many": "{{people}}… 全部で{{count}}", "person_or_group_in_list_many_no_preview": "全部で{{count}}", - "personas": "人格", + "personas": "ペルソナ", "popup_missing_permission": "Mask は権限がないため動作しません。", "popup_request_permission": "修正します", - "popup_current_persona": "現在の人格アカウント", + "popup_current_persona": "現在のペルソナアカウント", "popup_enter_dashboard": "ダッシュボードに入る", "popup_connect_wallet": "現在のウォレット", - "popup_setup_first_persona": "最初の人格アカウントを作る", + "popup_setup_first_persona": "最初のペルソナアカウントを作る", "profiles": "プロフィール", "posts": "投稿", "post_dialog__button": "終了", @@ -162,8 +163,8 @@ "privacy_policy": "プライバシー・ポリシー(個人情報に関する方針)", "privacy_policy_link": "https://legal.mask.io/maskbook/", "private_key": "秘密鍵", - "persona_rename": "人格の名前を変更", - "persona_name": "人格アカウントの名前", + "persona_rename": "ペルソナの名前を変更", + "persona_name": "ペルソナアカウントの名前", "qr_code": "QR コード", "reload": "再読み込み", "hide": "隠す", @@ -211,14 +212,14 @@ "set_up_consent_data_collection_privacy_policy_2": "プライバシーポリシー", "set_up_consent_data_collection_privacy_policy_3": "。", "set_up_create_persona": "はじめましょう", - "set_up_create_persona_hint": "次のステップでは、ソーシャルネットワークのアカウントを人格アカウントに接続することができます。", + "set_up_create_persona_hint": "次のステップでは、ソーシャルネットワークのアカウントをペルソナアカウントに接続することができます。", "set_up_connect": "「{{name}}」に接続する", "set_up_connect_hint": "Facebook と Twitter が Mask に対応しています。", "set_up_restore": "データベースの復元", "set_up_restore_hint": "以前のデータベースバックアップから復元します。", "set_up_restore_fail": "復元に失敗しました。", "set_up_advance_restore": "高度な復元オプション", - "set_up_advance_restore_hint": "以下の方法で人格のバックアップをインポートすることができます。", + "set_up_advance_restore_hint": "以下の方法でペルソナのバックアップをインポートすることができます。", "set_up_advance_restore_fail": "復元に失敗しました。", "set_up_restore_confirmation": "データベースの復元", "set_up_restore_confirmation_hint": "のように以下のデータがインポートされます。", @@ -264,7 +265,6 @@ "wallet_price": "価格", "wallet_balance": "残高", "wallet_value": "価値", - "wallet_chain": "", "wallet_confirmation_hint": "ウォレットを作成する必要があることを確認します。", "wallet_new": "新規ウォレット名", "wallet_notification": "デフォルトのウォレット以外の新しい通貨ウォレットを作成します。ウォレットの秘密鍵を安全に保管する責任はあなたにあります。また、Mask が Ethereum 上でのあなたの取引の支払いをすることはありませんのでご注意ください。", diff --git a/packages/maskbook/src/_locales/zh/messages.json b/packages/maskbook/src/_locales/zh/messages.json index 70f6f53cf30b..f0c007e37b18 100644 --- a/packages/maskbook/src/_locales/zh/messages.json +++ b/packages/maskbook/src/_locales/zh/messages.json @@ -47,6 +47,7 @@ "dashboard": "儀表板", "dashboard_tab_token": "Token", "dashboard_tab_collectibles": "收藏品", + "dashboard_tab_transactions": "", "dashboard_backup_database_confirmation": "好,現在備份", "dashboard_backup_database_hint": "創建資料庫備份文件。要經常做。", "dashboard_backup_persona_hint": "您可以通過以下任一方式備份角色。", @@ -264,7 +265,6 @@ "wallet_price": "價格", "wallet_balance": "餘額", "wallet_value": "價值", - "wallet_chain": "", "wallet_confirmation_hint": "我確認需要創建一個錢包。", "wallet_new": "", "wallet_notification": "這將創建一個默認錢包以外的新錢包。您有責任確保錢包的私鑰安全。另請注意,Mask 將無法為您在以太坊上的交易付款。", diff --git a/packages/maskbook/src/background-service.ts b/packages/maskbook/src/background-service.ts index 3d9cf5d37466..f05e0099eeda 100644 --- a/packages/maskbook/src/background-service.ts +++ b/packages/maskbook/src/background-service.ts @@ -8,7 +8,7 @@ Object.defineProperty(globalThis, 'crypto', { configurable: true, enumerable: tr import './extension/service' // setup Services.* import './utils/native-rpc' // setup Android and iOS API server -import './provider.worker' // TODO: HMR setup social network providers +import './social-network-adaptor' // setup social network providers import './extension/background-script/Jobs' // start jobs import './plugins/PluginSerivce' // setup plugins import './utils/debug/general' diff --git a/packages/maskbook/src/components/DataSource/useActivatedUI.ts b/packages/maskbook/src/components/DataSource/useActivatedUI.ts index f3c7d9168201..5f45e25cebaf 100644 --- a/packages/maskbook/src/components/DataSource/useActivatedUI.ts +++ b/packages/maskbook/src/components/DataSource/useActivatedUI.ts @@ -1,21 +1,28 @@ import type { Profile } from '../../database' import { useValueRef } from '../../utils/hooks/useValueRef' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI, globalUIState } from '../../social-network' import { currentSelectedIdentity } from '../../settings/settings' import { useMemo } from 'react' +import { ValueRef } from '@dimensiondev/holoflows-kit' +import type { SocialNetworkUI } from '../../social-network' +import { ProfileIdentifier } from '@dimensiondev/maskbook-shared' export function useFriendsList() { - const ref = useValueRef(getActivatedUI().friendsRef) + const ref = useValueRef(globalUIState.friends) return useMemo(() => [...ref.values()], [ref]) } + +const default_ = new ValueRef({ identifier: ProfileIdentifier.unknown }) export function useLastRecognizedIdentity() { - return useValueRef(getActivatedUI().lastRecognizedIdentity) + return useValueRef( + activatedSocialNetworkUI.collecting.identityProvider?.lastRecognized || default_, + ) } export function useMyIdentities() { - return useValueRef(getActivatedUI().myIdentitiesRef) + return useValueRef(globalUIState.profiles) } export function useCurrentIdentity(noDefault?: boolean): Profile | null { const all = useMyIdentities() - const current = useValueRef(currentSelectedIdentity[getActivatedUI().networkIdentifier]) + const current = useValueRef(currentSelectedIdentity[activatedSocialNetworkUI.networkIdentifier]) return all.find((i) => i.identifier.toText() === current) || (noDefault ? null : all[0]) } diff --git a/packages/maskbook/src/components/DataSource/usePostInfo.ts b/packages/maskbook/src/components/DataSource/usePostInfo.ts index a6b8164f77f5..a009d9bffb9a 100644 --- a/packages/maskbook/src/components/DataSource/usePostInfo.ts +++ b/packages/maskbook/src/components/DataSource/usePostInfo.ts @@ -4,7 +4,7 @@ import { ValueRef } from '@dimensiondev/holoflows-kit' import { useValueRef } from '../../utils/hooks/useValueRef' import { ObservableMap, ObservableSet } from '../../utils/ObservableMapSet' import { useObservableValues } from '../../utils/hooks/useObservableMapSet' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' export const PostInfoContext = createContext(emptyPostInfo) export function usePostInfo() { @@ -17,11 +17,10 @@ type ValidKeys = { }[keyof PostInfo] export function usePostLink() { - const ui = getActivatedUI() const postID = usePostInfoDetails('postID') const postIdentifier = usePostInfoDetails('postIdentifier') if (!postID || !postIdentifier) return '' - return ui.name === 'twitter' ? `https://twitter.com/${postIdentifier.identifier.userId}/status/${postID}` : '' + return activatedSocialNetworkUI.utils.getPostURL?.(postIdentifier) ?? '' } export function usePostInfoDetails( diff --git a/packages/maskbook/src/components/InjectedComponents/DecryptedPost/DecryptedPostDebug.tsx b/packages/maskbook/src/components/InjectedComponents/DecryptedPost/DecryptedPostDebug.tsx index 10ff7947cb32..568a3014f7f8 100644 --- a/packages/maskbook/src/components/InjectedComponents/DecryptedPost/DecryptedPostDebug.tsx +++ b/packages/maskbook/src/components/InjectedComponents/DecryptedPost/DecryptedPostDebug.tsx @@ -10,7 +10,6 @@ import { useValueRef } from '../../../utils/hooks/useValueRef' import { debugModeSetting } from '../../../settings/settings' import { isEnvironment, Environment } from '@dimensiondev/holoflows-kit' import { usePostInfoDetails } from '../../DataSource/usePostInfo' -import { getActivatedUI } from '../../../social-network/ui' import { deconstructPayload, PayloadAlpha38 } from '../../../utils/type-transform/Payload' interface DebugDisplayProps { @@ -21,7 +20,7 @@ interface DebugDisplayProps { export function DecryptedPostDebug(props: Partial) { const postBy = usePostInfoDetails('postBy') const postContent = usePostInfoDetails('postContent') - const payloadResult = useMemo(() => deconstructPayload(postContent, getActivatedUI().payloadDecoder), [postContent]) + const payloadResult = useMemo(() => deconstructPayload(postContent), [postContent]) const setting = useValueRef(debugModeSetting) const isDebugging = isEnvironment(Environment.ManifestOptions) ? true : setting diff --git a/packages/maskbook/src/components/InjectedComponents/DialogDismissIcon.tsx b/packages/maskbook/src/components/InjectedComponents/DialogDismissIcon.tsx index 70d2d12116ee..00fe368d0df8 100644 --- a/packages/maskbook/src/components/InjectedComponents/DialogDismissIcon.tsx +++ b/packages/maskbook/src/components/InjectedComponents/DialogDismissIcon.tsx @@ -1,7 +1,8 @@ import { useTheme, useMediaQuery } from '@material-ui/core' import CloseIcon from '@material-ui/icons/Close' import ArrowBackRoundedIcon from '@material-ui/icons/ArrowBackRounded' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' +import { isTwitter } from '../../social-network-adaptor/twitter.com/base' export interface DialogDismissIconUIProps { disableArrowBack?: boolean @@ -9,7 +10,7 @@ export interface DialogDismissIconUIProps { export function DialogDismissIconUI(props: DialogDismissIconUIProps) { return useMediaQuery(`(min-width: ${useTheme().breakpoints.width('sm')}px)`) || props.disableArrowBack ? ( - + ) : ( ) diff --git a/packages/maskbook/src/components/InjectedComponents/PostDialog.tsx b/packages/maskbook/src/components/InjectedComponents/PostDialog.tsx index 33353c56ac83..14bef6f25b0b 100644 --- a/packages/maskbook/src/components/InjectedComponents/PostDialog.tsx +++ b/packages/maskbook/src/components/InjectedComponents/PostDialog.tsx @@ -21,7 +21,7 @@ import type { Profile, Group } from '../../database' import { useFriendsList, useCurrentIdentity, useMyIdentities } from '../DataSource/useActivatedUI' import { currentImagePayloadStatus, debugModeSetting } from '../../settings/settings' import { useValueRef } from '../../utils/hooks/useValueRef' -import { editActivatedPostMetadata, getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' import Services from '../../extension/service' import { SelectRecipientsUI, SelectRecipientsUIProps } from '../shared/SelectRecipients/SelectRecipients' import { ClickableChip } from '../shared/SelectRecipients/ClickableChip' @@ -36,7 +36,6 @@ import { EthereumTokenType } from '../../web3/types' import { isDAI, isOKB } from '../../web3/helpers' import { PluginRedPacketTheme } from '../../plugins/RedPacket/theme' import { useI18N } from '../../utils/i18n-next-ui' -import { twitterUrl } from '../../social-network-provider/twitter.com/utils/url' import { RedPacketMetadataReader } from '../../plugins/RedPacket/helpers' import { PluginUI } from '../../plugins/PluginUI' import { Result } from 'ts-results' @@ -44,9 +43,10 @@ import { ErrorBoundary } from '../shared/ErrorBoundary' import { InjectedDialog } from '../shared/InjectedDialog' import { DebugMetadataInspector } from '../shared/DebugMetadataInspector' import { PluginStage } from '../../plugins/types' -import { Election2020MetadataReader } from '../../plugins/Election2020/helpers' -import { COTM_MetadataReader } from '../../plugins/COTM/helpers' import { Flags } from '../../utils/flags' +import { editActivatedPostMetadata, globalTypedMessageMetadata } from '../../protocols/typed-message/global-state' +import { isTwitter } from '../../social-network-adaptor/twitter.com/base' +import { SteganographyTextPayload } from './SteganographyTextPayload' const defaultTheme = {} @@ -231,7 +231,7 @@ export function PostDialogUI(props: PostDialogUIProps) { )} {showPostMetadata && ( (getActivatedUI().typedMessageMetadata.value = meta)} + onNewMetadata={(meta) => (globalTypedMessageMetadata.value = meta)} onExit={() => setShowPostMetadata(false)} meta={props.postContent.meta || new Map()} /> @@ -270,7 +270,7 @@ export function PostDialog({ reason: props_reason = 'timeline', ...props }: Post const onlyMyself = props.onlyMyself ?? onlyMyselfLocal const [shareToEveryoneLocal, setShareToEveryone] = useState(true) const shareToEveryone = props.shareToEveryone ?? shareToEveryoneLocal - const typedMessageMetadata = or(props.typedMessageMetadata, useValueRef(getActivatedUI().typedMessageMetadata)) + const typedMessageMetadata = or(props.typedMessageMetadata, useValueRef(globalTypedMessageMetadata)) const [open, setOpen] = or(props.open, useState(false)) as NonNullable //#region TypedMessage @@ -287,12 +287,12 @@ export function PostDialog({ reason: props_reason = 'timeline', ...props }: Post const [currentShareTarget, setCurrentShareTarget] = useState<(Profile | Group)[]>(() => []) //#endregion //#region Image Based Payload Switch - const imagePayloadStatus = useValueRef(currentImagePayloadStatus[getActivatedUI().networkIdentifier]) + const imagePayloadStatus = useValueRef(currentImagePayloadStatus[activatedSocialNetworkUI.networkIdentifier]) const imagePayloadEnabled = imagePayloadStatus === 'true' const onImagePayloadSwitchChanged = or( props.onImagePayloadSwitchChanged, useCallback((checked) => { - currentImagePayloadStatus[getActivatedUI().networkIdentifier].value = String(checked) + currentImagePayloadStatus[activatedSocialNetworkUI.networkIdentifier].value = String(checked) }, []), ) //#endregion @@ -307,15 +307,11 @@ export function PostDialog({ reason: props_reason = 'timeline', ...props }: Post currentIdentity!.identifier, !!shareToEveryone, ) - const activeUI = getActivatedUI() + const activeUI = activatedSocialNetworkUI // TODO: move into the plugin system const redPacketMetadata = RedPacketMetadataReader(typedMessageMetadata) - const election2020Metadata = Election2020MetadataReader(typedMessageMetadata) - const COTM_Metadata = COTM_MetadataReader(typedMessageMetadata) if (imagePayloadEnabled) { const isRedPacket = redPacketMetadata.ok - const isElection2020 = election2020Metadata.ok - const isCOTM = COTM_Metadata.ok const isErc20 = redPacketMetadata.ok && redPacketMetadata.val && @@ -327,44 +323,32 @@ export function PostDialog({ reason: props_reason = 'timeline', ...props }: Post const relatedText = t('additional_post_box__steganography_post_pre', { random: new Date().toLocaleString(), }) - activeUI.taskPasteIntoPostBox(relatedText, { - autoPasteFailedRecover: false, + activeUI.automation.nativeCompositionDialog?.appendText?.(relatedText, { + recover: false, }) - activeUI.taskUploadToPostBox(encrypted, { - template: isRedPacket - ? isDai - ? 'dai' - : isOkb - ? 'okb' - : 'eth' - : isElection2020 - ? 'v3' - : isCOTM - ? 'v4' - : 'v2', - autoPasteFailedRecover: true, - relatedText, + const img = await SteganographyTextPayload( + isRedPacket ? (isDai ? 'dai' : isOkb ? 'okb' : 'eth') : 'v2', + encrypted, + ) + activeUI.automation.nativeCompositionDialog?.attachImage?.(img, { + recover: true, + relatedTextPayload: relatedText, }) } else { let text = t('additional_post_box__encrypted_post_pre', { encrypted }) if (redPacketMetadata.ok) { if (i18n.language?.includes('zh')) { - text = - activeUI.networkIdentifier === twitterUrl.hostIdentifier - ? `用 #mask_io @realMaskbook 開啟紅包 ${encrypted}` - : `用 #mask_io 開啟紅包 ${encrypted}` + text = isTwitter(activeUI) + ? `用 #mask_io @realMaskbook 開啟紅包 ${encrypted}` + : `用 #mask_io 開啟紅包 ${encrypted}` } else { - text = - activeUI.networkIdentifier === twitterUrl.hostIdentifier - ? `Claim this Red Packet with #mask_io @realMaskbook ${encrypted}` - : `Claim this Red Packet with #mask_io ${encrypted}` + text = isTwitter(activeUI) + ? `Claim this Red Packet with #mask_io @realMaskbook ${encrypted}` + : `Claim this Red Packet with #mask_io ${encrypted}` } } - if (election2020Metadata.ok) { - text = `Claim the election special NFT with @realMaskbook (mask.io) #mask_io #twitternft ${encrypted}` - } - activeUI.taskPasteIntoPostBox(text, { - autoPasteFailedRecover: true, + activeUI.automation.nativeCompositionDialog?.appendText?.(text, { + recover: true, }) } // This step write data on gun. @@ -382,7 +366,7 @@ export function PostDialog({ reason: props_reason = 'timeline', ...props }: Post setShareToEveryone(true) setPostBoxContent(makeTypedMessageText('')) setCurrentShareTarget([]) - getActivatedUI().typedMessageMetadata.value = new Map() + globalTypedMessageMetadata.value = new Map() }, [setOpen]), ) const onFinishButtonClicked = useCallback(() => { diff --git a/packages/maskbook/src/components/InjectedComponents/PostDialogHint.tsx b/packages/maskbook/src/components/InjectedComponents/PostDialogHint.tsx index 5bc3dd1c2e4b..12dfe98da9f9 100644 --- a/packages/maskbook/src/components/InjectedComponents/PostDialogHint.tsx +++ b/packages/maskbook/src/components/InjectedComponents/PostDialogHint.tsx @@ -5,8 +5,8 @@ import { useMyIdentities } from '../DataSource/useActivatedUI' import type { BannerProps } from '../Welcomes/Banner' import { useValueRef } from '../../utils/hooks/useValueRef' import { currentSetupGuideStatus } from '../../settings/settings' -import { getActivatedUI } from '../../social-network/ui' -import { isMobileFacebook } from '../../social-network-provider/facebook.com/utils/isMobile' +import { activatedSocialNetworkUI } from '../../social-network' +import { isMobileFacebook } from '../../social-network-adaptor/facebook.com/utils/isMobile' import { MaskbookSharpIcon } from '../../resources/MaskbookIcon' import { makeStyles } from '@material-ui/core/styles' @@ -61,7 +61,7 @@ export interface PostDialogHintProps extends Partial { } export function PostDialogHint(props: PostDialogHintProps) { const identities = useMyIdentities() - const connecting = useValueRef(currentSetupGuideStatus[getActivatedUI().networkIdentifier]) + const connecting = useValueRef(currentSetupGuideStatus[activatedSocialNetworkUI.networkIdentifier]) if (connecting || identities.length === 0) return null return {}} {...props} /> } diff --git a/packages/maskbook/src/components/InjectedComponents/PostInspector.tsx b/packages/maskbook/src/components/InjectedComponents/PostInspector.tsx index 2c0923ba0137..b3db15657eaf 100644 --- a/packages/maskbook/src/components/InjectedComponents/PostInspector.tsx +++ b/packages/maskbook/src/components/InjectedComponents/PostInspector.tsx @@ -6,7 +6,6 @@ import Services from '../../extension/service' import { ProfileIdentifier } from '../../database/type' import type { Profile } from '../../database' import { useCurrentIdentity, useFriendsList } from '../DataSource/useActivatedUI' -import { getActivatedUI } from '../../social-network/ui' import { useValueRef } from '../../utils/hooks/useValueRef' import { debugModeSetting } from '../../settings/settings' import { DebugList } from '../DebugModeUI/DebugList' @@ -16,6 +15,7 @@ import { PluginUI } from '../../plugins/PluginUI' import { usePostInfoDetails, usePostInfo } from '../DataSource/usePostInfo' import { ErrorBoundary } from '../shared/ErrorBoundary' import type { PayloadAlpha40_Or_Alpha39, PayloadAlpha38 } from '../../utils/type-transform/Payload' +import { decodePublicKeyUI } from '../../social-network/utils/text-payload-ui' export interface PostInspectorProps { onDecrypted(post: TypedMessage, raw: string): void @@ -36,7 +36,7 @@ export function PostInspector(props: PostInspectorProps) { const whoAmI = useCurrentIdentity() const friends = useFriendsList() const [alreadySelectedPreviously, setAlreadySelectedPreviously] = useState([]) - const provePost = useMemo(() => getActivatedUI().publicKeyDecoder(postContent), [postContent]) + const provePost = useMemo(() => decodePublicKeyUI(postContent), [postContent]) const { value: sharedListOfPost } = useAsync(async () => { if (!whoAmI || !whoAmI.identifier.equals(postBy) || !encryptedPost.ok) return [] diff --git a/packages/maskbook/src/components/InjectedComponents/SetupGuide.tsx b/packages/maskbook/src/components/InjectedComponents/SetupGuide.tsx index e759c5e8db08..5574b124f0ac 100644 --- a/packages/maskbook/src/components/InjectedComponents/SetupGuide.tsx +++ b/packages/maskbook/src/components/InjectedComponents/SetupGuide.tsx @@ -24,7 +24,7 @@ import stringify from 'json-stable-stringify' import ActionButton, { ActionButtonPromise } from '../../extension/options-page/DashboardComponents/ActionButton' import { noop } from 'lodash-es' import { useI18N } from '../../utils/i18n-next-ui' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' import { currentSetupGuideStatus } from '../../settings/settings' import type { SetupGuideCrossContextStatus } from '../../settings/types' import { MaskMessage } from '../../utils/messages' @@ -36,6 +36,8 @@ import { WalletRPC } from '../../plugins/Wallet/messages' import { useMatchXS } from '../../utils/hooks/useMatchXS' import { extendsTheme } from '../../utils/theme' +import { useLastRecognizedIdentity } from '../DataSource/useActivatedUI' +import { makeTypedMessageText } from '../../protocols/typed-message' export enum SetupGuideStep { FindUsername = 'find-username', @@ -364,7 +366,8 @@ interface FindUsernameProps extends Partial { function FindUsername({ username, onConnect, onDone, onClose, onUsernameChange = noop }: FindUsernameProps) { const { t } = useI18N() - const ui = getActivatedUI() + const ui = activatedSocialNetworkUI + const gotoProfilePageImpl = ui.automation.redirect?.profilePage const classes = useWizardDialogStyles() const findUsernameClasses = useFindUsernameStyles() @@ -378,9 +381,9 @@ function FindUsername({ username, onConnect, onDone, onClose, onUsernameChange = const onJump = useCallback( (ev: React.MouseEvent) => { ev.preventDefault() - ui.taskGotoProfilePage(new ProfileIdentifier(ui.networkIdentifier, username)) + gotoProfilePageImpl?.(new ProfileIdentifier(ui.networkIdentifier, username)) }, - [ui, username], + [gotoProfilePageImpl, ui.networkIdentifier, username], ) return ( onUsernameChange(e.target.value)} onKeyDown={onKeyDown} inputProps={{ 'data-testid': 'username_input' }}> - - - - - + {gotoProfilePageImpl ? ( + + + + + + ) : null} } @@ -539,7 +544,7 @@ function SetupGuideUI(props: SetupGuideUIProps) { const { t } = useI18N() const { persona } = props const [step, setStep] = useState(SetupGuideStep.FindUsername) - const ui = getActivatedUI() + const ui = activatedSocialNetworkUI //#region parse setup status const lastStateRef = currentSetupGuideStatus[ui.networkIdentifier] @@ -559,13 +564,13 @@ function SetupGuideUI(props: SetupGuideUIProps) { //#endregion //#region setup username - const lastRecognized = useValueRef(getActivatedUI().lastRecognizedIdentity) + const lastRecognized = useLastRecognizedIdentity() const getUsername = () => lastState.username || (lastRecognized.identifier.isUnknown ? '' : lastRecognized.identifier.userId) const [username, setUsername] = useState(getUsername) useEffect( () => - getActivatedUI().lastRecognizedIdentity.addListener((val) => { + activatedSocialNetworkUI.collecting.identityProvider?.lastRecognized.addListener((val) => { if (username === '' && !val.identifier.isUnknown) setUsername(val.identifier.userId) }), [username], @@ -586,7 +591,7 @@ function SetupGuideUI(props: SetupGuideUIProps) { username, persona: persona.toText(), } as SetupGuideCrossContextStatus) - ui.taskGotoNewsFeedPage() + ui.automation.redirect?.newsFeed?.() setStep(SetupGuideStep.SayHelloWorld) break case SetupGuideStep.SayHelloWorld: @@ -634,7 +639,7 @@ function SetupGuideUI(props: SetupGuideUIProps) { const onCreate = async () => { const content = t('setup_guide_say_hello_content') copyToClipboard(content) - ui.taskOpenComposeBox(content, { + ui.automation.maskCompositionDialog?.open?.(makeTypedMessageText(content), { shareToEveryOne: true, }) } diff --git a/packages/maskbook/src/components/InjectedComponents/SteganographyTextPayload.ts b/packages/maskbook/src/components/InjectedComponents/SteganographyTextPayload.ts new file mode 100644 index 000000000000..2cebc1925843 --- /dev/null +++ b/packages/maskbook/src/components/InjectedComponents/SteganographyTextPayload.ts @@ -0,0 +1,22 @@ +import { decodeArrayBuffer } from '@dimensiondev/kit' +import Services from '../../extension/service' +import { ImageTemplateTypes, ImagePayloadURLs } from '../../resources/image-payload' +import { activatedSocialNetworkUI } from '../../social-network' +import { downloadUrl } from '../../utils/utils' + +export async function SteganographyTextPayload(template: ImageTemplateTypes, text: string) { + const pass = activatedSocialNetworkUI.configuration.steganography?.password?.() || 'mask' + const blankImage = await downloadUrl(ImagePayloadURLs[template]).then((x) => x.arrayBuffer()) + const secretImage = new Uint8Array( + decodeArrayBuffer( + await Services.Steganography.encodeImage(new Uint8Array(blankImage), { + text, + pass, + template, + grayscaleAlgorithm: activatedSocialNetworkUI.configuration.steganography?.grayscaleAlgorithm, + }), + ), + ) + const blob = new Blob([secretImage], { type: 'image/png' }) + return blob +} diff --git a/packages/maskbook/src/components/Welcomes/Banner.tsx b/packages/maskbook/src/components/Welcomes/Banner.tsx index c94780abce1e..19b85ef0c832 100644 --- a/packages/maskbook/src/components/Welcomes/Banner.tsx +++ b/packages/maskbook/src/components/Welcomes/Banner.tsx @@ -3,7 +3,7 @@ import { makeStyles } from '@material-ui/core/styles' import { Theme, IconButton } from '@material-ui/core' import { useLastRecognizedIdentity, useMyIdentities } from '../DataSource/useActivatedUI' import Services from '../../extension/service' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' import { setStorage } from '../../utils/browser.storage' import { useStylesExtends } from '../custom-ui-helper' import { DashboardRoute } from '../../extension/options-page/Route' @@ -49,7 +49,7 @@ export interface BannerProps extends Partial {} export function Banner(props: BannerProps) { const lastRecognizedIdentity = useLastRecognizedIdentity() const { nextStep } = props - const networkIdentifier = getActivatedUI()?.networkIdentifier + const networkIdentifier = activatedSocialNetworkUI?.networkIdentifier const identities = useMyIdentities() const [value, onChange] = useState('') const defaultNextStep = useCallback(() => { @@ -67,7 +67,7 @@ export function Banner(props: BannerProps) { defaultValue: lastRecognizedIdentity.identifier.isUnknown ? '' : lastRecognizedIdentity.identifier.userId, value, onChange, - isValid: getActivatedUI().isValidUsername, + isValid: activatedSocialNetworkUI.utils.isValidUsername || (() => true), } : ('hidden' as const) diff --git a/packages/maskbook/src/components/custom-ui-helper.tsx b/packages/maskbook/src/components/custom-ui-helper.tsx index f3d9ecc75c02..17a90f9a9ffd 100644 --- a/packages/maskbook/src/components/custom-ui-helper.tsx +++ b/packages/maskbook/src/components/custom-ui-helper.tsx @@ -1,6 +1,4 @@ /// -import { safeGetActiveUI } from '../utils/safeRequire' - // Priority: classes from props > configHooks > defaultStyles export function useStylesExtends( defaultStyles: Record, @@ -30,10 +28,6 @@ export function mergeClasses( return result } -export function getCustomUIOverwrite() { - return safeGetActiveUI().componentOverwrite || {} -} - /** * In case of y is a react hooks call, we should always call the hooks. So we can't use x ?? y directly. */ diff --git a/packages/maskbook/src/components/shared/ChooseIdentity.tsx b/packages/maskbook/src/components/shared/ChooseIdentity.tsx index 9cf23938f7cd..bd6c74f9fbfe 100644 --- a/packages/maskbook/src/components/shared/ChooseIdentity.tsx +++ b/packages/maskbook/src/components/shared/ChooseIdentity.tsx @@ -4,7 +4,7 @@ import ExpandMoreIcon from '@material-ui/icons/ExpandMore' import type { Profile } from '../../database' import { List, Accordion, AccordionSummary } from '@material-ui/core' import { ProfileOrGroupInList, ProfileOrGroupInListProps } from './SelectPeopleAndGroups' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' import { useCurrentIdentity, useMyIdentities } from '../DataSource/useActivatedUI' import { ProfileIdentifier } from '../../database/type' import { currentSelectedIdentity } from '../../settings/settings' @@ -80,7 +80,7 @@ export function ChooseIdentity(props: ChooseIdentityProps) { const expansionPanelSummaryClasses = useStylesExtends(useAccordionSummaryStyle(), props) const [expanded, setExpanded] = useState(false) - const ui = getActivatedUI() + const ui = activatedSocialNetworkUI const current = useCurrentIdentity() || ({ identifier: ProfileIdentifier.unknown, nickname: 'Nothing' } as Profile) const onChange = useCallback(() => { diff --git a/packages/maskbook/src/components/shared/InjectedDialog.tsx b/packages/maskbook/src/components/shared/InjectedDialog.tsx index d05d4f94f5ea..c373f22b160f 100644 --- a/packages/maskbook/src/components/shared/InjectedDialog.tsx +++ b/packages/maskbook/src/components/shared/InjectedDialog.tsx @@ -16,9 +16,10 @@ import { import { Children, cloneElement } from 'react' import { useI18N } from '../../utils/i18n-next-ui' import { usePortalShadowRoot } from '../../utils/shadow-root/usePortalShadowRoot' -import { getCustomUIOverwrite, mergeClasses, useStylesExtends } from '../custom-ui-helper' +import { mergeClasses, useStylesExtends } from '../custom-ui-helper' import { DialogDismissIconUI } from '../InjectedComponents/DialogDismissIcon' import { ErrorBoundary } from '@dimensiondev/maskbook-theme' +import { activatedSocialNetworkUI } from '../../social-network' const useStyles = makeStyles((theme) => createStyles({ @@ -51,7 +52,7 @@ export interface InjectedDialogProps extends withClasses } export function InjectedDialog(props: InjectedDialogProps) { const classes = useStyles() - const overwrite = getCustomUIOverwrite() + const overwrite = activatedSocialNetworkUI.customization.componentOverwrite || {} props = overwrite.InjectedDialog?.props?.(props) ?? props const { dialogActions, diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/addPerson.ts b/packages/maskbook/src/extension/background-script/CryptoServices/addPerson.ts index 5582a2dbb68e..94ef8572eec4 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/addPerson.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/addPerson.ts @@ -1,6 +1,6 @@ import { GunAPI as Gun2 } from '../../../network/gun' import { ProfileIdentifier, PostIdentifier } from '../../../database/type' -import getCurrentNetworkWorker from '../../../social-network/utils/getCurrentNetworkWorker' +import { getNetworkWorker } from '../../../social-network' import { verifyOthersProve } from './verifyOthersProve' import { memoizePromise } from '../../../utils/memoize' import { queryPersonaRecord } from '../../../database' @@ -8,7 +8,7 @@ import type { PersonaRecord } from '../../../database/Persona/Persona.db' import { i18n } from '../../../utils/i18n-next' async function getUserPublicKeyFromBio(user: ProfileIdentifier) { - const profile = await getCurrentNetworkWorker(user).unwrap().fetchProfile(user) + const profile = await (await getNetworkWorker(user)).tasks.fetchProfile!(user) if (!(await verifyOthersProve(profile.bioContent, user))) { throw new Error('Not in bio!') } @@ -20,7 +20,7 @@ async function getUserPublicKeyFromProvePost(user: ProfileIdentifier) { throw new Error('Not in gun!') } const postId = new PostIdentifier(user, proverPostID) - const post = await getCurrentNetworkWorker(postId).unwrap().fetchPostContent(postId) + const post = await (await getNetworkWorker(user)).tasks.fetchPostContent!(postId) if ((await verifyOthersProve(post, user)) === false) throw new Error('Not in prove post!') } async function getUserPublicKeyFromNetwork(user: ProfileIdentifier) { diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/appendShareTarget.ts b/packages/maskbook/src/extension/background-script/CryptoServices/appendShareTarget.ts index 9bedb136fddb..1ebd0d2f4b27 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/appendShareTarget.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/appendShareTarget.ts @@ -4,7 +4,7 @@ import { ProfileIdentifier, PostIVIdentifier } from '../../../database/type' import { prepareRecipientDetail } from './prepareRecipientDetail' import { cryptoProviderTable } from './cryptoProviderTable' import { updatePostDB, RecipientDetail, RecipientReason } from '../../../database/post' -import { getNetworkWorker } from '../../../social-network/worker' +import { getNetworkWorkerUninitialized } from '../../../social-network/worker' import { queryPrivateKey, queryLocalKey } from '../../../database' import { IdentifierMap } from '../../../database/IdentifierMap' import type { AESJsonWebKey, EC_Private_JsonWebKey } from '../../../modules/CryptoAlgorithm/interfaces/utils' @@ -35,12 +35,8 @@ export async function appendShareTarget( myPrivateKey, Array.from(toKey.values()), ) - Gun2.publishPostAESKeyOnGun2( - version, - iv, - getNetworkWorker(whoAmI).unwrap().gunNetworkHint, - othersAESKeyEncrypted, - ) + const gunHint = getNetworkWorkerUninitialized(whoAmI)?.gunNetworkHint + gunHint && Gun2.publishPostAESKeyOnGun2(version, iv, gunHint, othersAESKeyEncrypted) updatePostDB( { identifier: new PostIVIdentifier(whoAmI.network, iv), diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/debugShowAllPossibleHashForPost.ts b/packages/maskbook/src/extension/background-script/CryptoServices/debugShowAllPossibleHashForPost.ts index 738bb1d13b7c..4f1d42a69448 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/debugShowAllPossibleHashForPost.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/debugShowAllPossibleHashForPost.ts @@ -1,10 +1,11 @@ import type { PostIVIdentifier } from '../../../database/type' import { hashPostSalt, hashCryptoKey, hashCryptoKeyUnstable } from '../../../network/gun/version.2/hash' import { queryProfilesWithQuery, queryPublicKey } from '../../../database' -import { getNetworkWorker } from '../../../social-network/worker' +import { getNetworkWorkerUninitialized } from '../../../social-network/worker' export async function debugShowAllPossibleHashForPost(post: PostIVIdentifier, payloadVersion: -38 | -39 | -40) { const friends = await queryProfilesWithQuery((x) => x.identifier.network === post.network) + const gunHint = getNetworkWorkerUninitialized(post.network)?.gunNetworkHint return Promise.all( friends .filter((x) => x.linkedPersona) @@ -12,7 +13,7 @@ export async function debugShowAllPossibleHashForPost(post: PostIVIdentifier, pa async (x) => [ x.identifier.toText(), - (await hashPostSalt(post.postIV, getNetworkWorker(post).unwrap().gunNetworkHint)) + + (await hashPostSalt(post.postIV, gunHint ?? '')) + '-' + (await (payloadVersion <= -39 ? hashCryptoKeyUnstable : hashCryptoKey)( (await queryPublicKey(x.identifier))!, diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/decryptFrom.ts b/packages/maskbook/src/extension/background-script/CryptoServices/decryptFrom.ts index bf80d23b272e..067fd9e575ff 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/decryptFrom.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/decryptFrom.ts @@ -8,7 +8,7 @@ import { queryPersonaRecord, queryLocalKey } from '../../../database' import { ProfileIdentifier, PostIVIdentifier } from '../../../database/type' import { queryPostDB, updatePostDB } from '../../../database/post' import { addPerson } from './addPerson' -import { getNetworkWorker } from '../../../social-network/worker' +import { getNetworkWorker, getNetworkWorkerUninitialized } from '../../../social-network/worker' import { cryptoProviderTable } from './cryptoProviderTable' import type { PersonaRecord } from '../../../database/Persona/Persona.db' import { verifyOthersProve } from './verifyOthersProve' @@ -24,6 +24,8 @@ import type { SharedAESKeyGun2 } from '../../../network/gun/version.2' import { MaskMessage } from '../../../utils/messages' import { GunAPI } from '../../../network/gun' import { calculatePostKeyPartition } from '../../../network/gun/version.2/hash' +import { Err, Ok, Result } from 'ts-results' +import { decodeTextPayloadWorker } from '../../../social-network/utils/text-payload-worker' type Progress = ( | { progress: 'finding_person_public_key' | 'finding_post_key' | 'init' | 'decode_post' } @@ -131,8 +133,10 @@ async function* decryptFromPayloadWithProgress_raw( if (successDecryptionCache.has(cacheKey)) return successDecryptionCache.get(cacheKey)! yield makeProgress('init') - const authorNetworkWorker = getNetworkWorker(author.network) - if (authorNetworkWorker.err) return makeError(authorNetworkWorker.val) + const authorNetworkWorker = Result.wrap(() => getNetworkWorkerUninitialized(author.network)).andThen((x) => + x ? Ok(x) : Err(new Error('Worker not found')), + ) + if (authorNetworkWorker.err) return makeError(authorNetworkWorker.val as Error) const data = post const { version } = data @@ -177,10 +181,10 @@ async function* decryptFromPayloadWithProgress_raw( if (!mine?.privateKey) return makeError(DecryptFailedReason.MyCryptoKeyNotFound) const { publicKey: minePublic, privateKey: minePrivate } = mine - const networkWorker = getNetworkWorker(whoAmI) + const networkWorker = getNetworkWorkerUninitialized(whoAmI) try { if (version === -40) throw '' - const gunNetworkHint = networkWorker.unwrap().gunNetworkHint + const gunNetworkHint = networkWorker!.gunNetworkHint const { keyHash, postHash } = await calculatePostKeyPartition(version, iv, minePublic, gunNetworkHint) yield { type: 'debug', debug: 'debug_finding_hash', hash: [postHash, keyHash] } } catch {} @@ -312,9 +316,9 @@ async function* decryptFromImageUrlWithProgress_raw( }) if (post.indexOf('🎼') !== 0 && !/https:\/\/.+\..+\/(\?PostData_v\d=)?%20(.+)%40/.test(post)) return makeError(i18n.t('service_decode_image_payload_failed'), true) - const worker = getNetworkWorker(author) - if (worker.err) return makeError(worker.val) - const payload = deconstructPayload(post, worker.val.payloadDecoder) + const worker = await Result.wrapAsync(() => getNetworkWorker(author)) + if (worker.err) return makeError(worker.val as Error) + const payload = deconstructPayload(post, await decodeTextPayloadWorker(author)) if (payload.err) return makeError(payload.val) return yield* decryptFromText(payload.val, author, whoAmI, publicShared) } diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/encryptTo.ts b/packages/maskbook/src/extension/background-script/CryptoServices/encryptTo.ts index 887880afa570..9ecc1e0f8a98 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/encryptTo.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/encryptTo.ts @@ -11,6 +11,7 @@ import { queryPersonaByProfileDB } from '../../../database/Persona/Persona.db' import { compressSecp256k1Key } from '../../../utils/type-transform/SECP256k1-Compression' import { i18n } from '../../../utils/i18n-next' import type { TypedMessage } from '../../../protocols/typed-message' +import { encodeTextPayloadWorker } from '../../../social-network/utils/text-payload-worker' type EncryptedText = string type OthersAESKeyEncryptedToken = string @@ -86,9 +87,9 @@ export async function encryptTo( }) const postAESKeyToken = encodeArrayBuffer(iv) - const worker = getNetworkWorker(whoAmI).unwrap() + const worker = await getNetworkWorker(whoAmI)! OthersAESKeyEncryptedMap.set(postAESKeyToken, [worker.gunNetworkHint, othersAESKeyEncrypted]) - return [constructAlpha38(payload, worker.payloadEncoder), postAESKeyToken] + return [constructAlpha38(payload, await encodeTextPayloadWorker(whoAmI)), postAESKeyToken] } /** diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/getMyProveBio.ts b/packages/maskbook/src/extension/background-script/CryptoServices/getMyProveBio.ts index a558070a04f3..b4a809f314ba 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/getMyProveBio.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/getMyProveBio.ts @@ -1,9 +1,8 @@ import { compressSecp256k1Key } from '../../../utils/type-transform/SECP256k1-Compression' import { ProfileIdentifier, PersonaIdentifier } from '../../../database/type' -import { getNetworkWorker } from '../../../social-network/worker' +import { encodePublicKeyWorker } from '../../../social-network/utils/text-payload-worker' import { queryPublicKey } from '../../../database' -//#endregion -//#region ProvePost, create & verify + export async function getMyProveBio( whoAmI: ProfileIdentifier | PersonaIdentifier, networkIdentifier?: string, @@ -11,10 +10,9 @@ export async function getMyProveBio( const myIdentity = await queryPublicKey(whoAmI) if (!myIdentity) return null const compressed = compressSecp256k1Key(myIdentity, 'public') - // FIXME: wait for #191 return whoAmI instanceof ProfileIdentifier - ? getNetworkWorker(whoAmI.network).unwrap().publicKeyEncoder(compressed) + ? (await encodePublicKeyWorker(whoAmI))(compressed) : networkIdentifier - ? getNetworkWorker(networkIdentifier).unwrap().publicKeyEncoder(compressed) + ? (await encodePublicKeyWorker(networkIdentifier))(compressed) : compressed } diff --git a/packages/maskbook/src/extension/background-script/CryptoServices/verifyOthersProve.ts b/packages/maskbook/src/extension/background-script/CryptoServices/verifyOthersProve.ts index 88a846b0cf18..33a631711684 100644 --- a/packages/maskbook/src/extension/background-script/CryptoServices/verifyOthersProve.ts +++ b/packages/maskbook/src/extension/background-script/CryptoServices/verifyOthersProve.ts @@ -1,11 +1,10 @@ import { decompressSecp256k1Key } from '../../../utils/type-transform/SECP256k1-Compression' import { ECKeyIdentifierFromJsonWebKey, ProfileIdentifier } from '../../../database/type' -import { getNetworkWorker } from '../../../social-network/worker' +import { decodePublicKeyWorker } from '../../../social-network/utils/text-payload-worker' import { createProfileWithPersona, queryPersonaRecord } from '../../../database' export async function verifyOthersProve(bio: string | { raw: string }, others: ProfileIdentifier): Promise { - const compressedX = - typeof bio === 'string' ? getNetworkWorker(others.network).unwrap().publicKeyDecoder(bio) : [bio.raw] + const compressedX = typeof bio === 'string' ? (await decodePublicKeyWorker(others.network))(bio) : [bio.raw] if (!compressedX) return false const publicKey = compressedX .map((x) => { diff --git a/packages/maskbook/src/extension/background-script/IdentityService.ts b/packages/maskbook/src/extension/background-script/IdentityService.ts index 2fd819e5bc52..9f97b101bb40 100644 --- a/packages/maskbook/src/extension/background-script/IdentityService.ts +++ b/packages/maskbook/src/extension/background-script/IdentityService.ts @@ -32,7 +32,7 @@ export { queryProfile, queryProfilePaged } from '../../database' export function queryProfiles(network?: string): Promise { return queryProfilesWithQuery(network) } -export async function queryMyProfiles(network?: string) { +export async function queryMyProfiles(network?: string): Promise { const myPersonas = (await queryMyPersonas(network)).filter((x) => !x.uninitialized) return Promise.all( myPersonas diff --git a/packages/maskbook/src/extension/background-script/SteganographyService.ts b/packages/maskbook/src/extension/background-script/SteganographyService.ts index 463ea2fc4b92..9b64e41ad0ae 100644 --- a/packages/maskbook/src/extension/background-script/SteganographyService.ts +++ b/packages/maskbook/src/extension/background-script/SteganographyService.ts @@ -8,7 +8,7 @@ import { getDimension } from '../../utils/image' import { decodeArrayBuffer, encodeArrayBuffer } from '../../utils/type-transform/String-ArrayBuffer' import { assertEnvironment, Environment } from '@dimensiondev/holoflows-kit' -import type { ImageTemplateTypes } from '../../social-network/ui' +import type { ImageTemplateTypes } from '../../resources/image-payload' assertEnvironment(Environment.ManifestBackground) type Mask = 'v1' | 'v2' | 'v4' | 'transparent' diff --git a/packages/maskbook/src/extension/content-script/tasks.ts b/packages/maskbook/src/extension/content-script/tasks.ts index 5f0b030bcbb3..40b80cbd037f 100644 --- a/packages/maskbook/src/extension/content-script/tasks.ts +++ b/packages/maskbook/src/extension/content-script/tasks.ts @@ -1,38 +1,27 @@ -import { - AutomatedTabTask, - isEnvironment, - AutomatedTabTaskRuntimeOptions, - Environment, -} from '@dimensiondev/holoflows-kit' -import { ProfileIdentifier, ECKeyIdentifier, Identifier } from '../../database/type' -import { disableOpenNewTabInBackgroundSettings, currentSetupGuideStatus } from '../../settings/settings' -import type { SetupGuideCrossContextStatus } from '../../settings/types' -import type { SocialNetworkUI } from '../../social-network/ui' +import { AutomatedTabTask, AutomatedTabTaskRuntimeOptions } from '@dimensiondev/holoflows-kit' +import { disableOpenNewTabInBackgroundSettings } from '../../settings/settings' +import type { SocialNetworkUI } from '../../social-network' import { memoizePromise } from '../../utils/memoize' -import { safeGetActiveUI } from '../../utils/safeRequire' import Serialization from '../../utils/type-transform/Serialization' -import { sideEffect } from '../../utils/side-effects' -import { untilDocumentReady } from '../../utils/dom' import { delay } from '../../utils/utils' import { Flags } from '../../utils/flags' - -function getActivatedUI() { - return safeGetActiveUI() -} +import { activatedSocialNetworkUI } from '../../social-network' const _tasks = { - getPostContent: () => getActivatedUI().taskGetPostContent(), + getPostContent: async () => (await activatedSocialNetworkUI.collecting.getPostContent?.()) || '', /** * Access profile page * Get Profile */ - getProfile: (identifier: ProfileIdentifier) => getActivatedUI().taskGetProfile(identifier), + getProfile: async () => (await activatedSocialNetworkUI.collecting.getProfile?.()) || { bioContent: '' }, /** * Access main page * Paste text into PostBox */ - pasteIntoPostBox: async (text: string, options: Parameters[1]) => - getActivatedUI().taskPasteIntoPostBox(text, options), + pasteIntoPostBox: async ( + text: string, + options: SocialNetworkUI.AutomationCapabilities.NativeCompositionAttachTextOptions, + ) => activatedSocialNetworkUI.automation.nativeCompositionDialog?.appendText?.(text, options), /** * Fetch a url in the current context */ @@ -45,9 +34,6 @@ const _tasks = { }, (x) => x, ), - async SetupGuide(for_: ECKeyIdentifier) { - getActivatedUI().taskStartSetupGuide(for_) - }, async noop() {}, } const realTasks = AutomatedTabTask(_tasks, { @@ -121,19 +107,3 @@ export function exclusiveTasks(...args: Parameters) { }, }) } - -sideEffect.then(untilDocumentReady).then(() => { - if (!isEnvironment(Environment.ContentScript)) return - - //#region setup guide - const network = getActivatedUI().networkIdentifier - const id = currentSetupGuideStatus[network].value - const onStatusUpdate = (id: string) => { - const { persona, status }: SetupGuideCrossContextStatus = JSON.parse(id || '{}') - if (persona && status) _tasks.SetupGuide(Identifier.fromString(persona, ECKeyIdentifier).unwrap()) - } - currentSetupGuideStatus[network].addListener(onStatusUpdate) - currentSetupGuideStatus[network].readyPromise.then(onStatusUpdate) - onStatusUpdate(id) - //#endregion -}) diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/Card.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/Card.tsx new file mode 100644 index 000000000000..718cb30ba11f --- /dev/null +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/Card.tsx @@ -0,0 +1,57 @@ +import Tilt from 'react-tilt' +import { Card, createStyles, Link, makeStyles } from '@material-ui/core' +import { Image } from '../../../../components/shared/Image' +import { MaskbookIconOutlined } from '../../../../resources/MaskbookIcon' + +const useStyles = makeStyles((theme) => + createStyles({ + root: { + display: 'flex', + alignItems: 'center', + justifyContent: 'center', + borderRadius: 4, + position: 'relative', + backgroundColor: theme.palette.background.paper, + }, + placeholder: { + width: 64, + height: 64, + opacity: 0.1, + }, + }), +) + +export interface CollectibleCardProps { + url: string + link: string + canViewOnEtherscan?: boolean +} + +export function CollectibleCard(props: CollectibleCardProps) { + const classes = useStyles(props) + + return ( + + + + {props.url ? ( + + ) : ( + + )} + + + + ) +} diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/index.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/index.tsx new file mode 100644 index 000000000000..71360d3b6b40 --- /dev/null +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/CollectibleList/index.tsx @@ -0,0 +1,86 @@ +import { Box, Button, makeStyles, Skeleton, Typography } from '@material-ui/core' +import { CollectibleCard } from './Card' +import { useCollectibles } from '../../../../plugins/Wallet/hooks/useCollectibles' +import { AssetProvider } from '../../../../plugins/Wallet/types' +import { useAccount } from '../../../../web3/hooks/useAccount' + +const useStyles = makeStyles((theme) => ({ + root: { + display: 'flex', + flexWrap: 'wrap', + padding: theme.spacing(1), + }, + card: { + padding: theme.spacing(1), + }, + description: { + textAlign: 'center', + marginTop: theme.spacing(0.5), + maxWidth: 160, + }, +})) + +export function CollectibleList() { + const account = useAccount() + const classes = useStyles() + const { + value: collectibles = [], + loading: collectiblesLoading, + error: collectiblesError, + retry: collectiblesRetry, + } = useCollectibles(account, AssetProvider.OPENSEAN) + + if (collectiblesLoading) + return ( + + {new Array(3).fill(0).map((_, i) => ( + + + + + ))} + + ) + + if (collectiblesError || collectibles.length === 0) + return ( + + No collectible found. + + + ) + + return ( + + {collectibles.map((x) => ( +
+ +
+ + {x.name ?? x.collection.slug} + +
+
+ ))} +
+ ) +} diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/ProfileBox.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/ProfileBox.tsx index f92c3f4cf61e..d6cedade6dd8 100644 --- a/packages/maskbook/src/extension/options-page/DashboardComponents/ProfileBox.tsx +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/ProfileBox.tsx @@ -1,8 +1,7 @@ import type { Persona } from '../../../database' -import { definedSocialNetworkWorkers } from '../../../social-network/worker' +import { definedSocialNetworkUIs, loadSocialNetworkUI } from '../../../social-network' import ProviderLine, { ProviderLineProps } from './ProviderLine' -import getCurrentNetworkUI from '../../../social-network/utils/getCurrentNetworkUI' import { currentSetupGuideStatus } from '../../../settings/settings' import type { SetupGuideCrossContextStatus } from '../../../settings/types' import { exclusiveTasks } from '../../content-script/tasks' @@ -11,6 +10,7 @@ import { useModal } from '../DashboardDialogs/Base' import { DashboardPersonaUnlinkConfirmDialog } from '../DashboardDialogs/Persona' import { delay } from '../../../utils/utils' import { SetupGuideStep } from '../../../components/InjectedComponents/SetupGuide' +import { Flags } from '../../../utils/flags' interface ProfileBoxProps { persona: Persona | null @@ -20,21 +20,30 @@ interface ProfileBoxProps { export default function ProfileBox({ persona, ProviderLineProps }: ProfileBoxProps) { const profiles = persona ? [...persona.linkedProfiles] : [] - const providers = [...definedSocialNetworkWorkers].map((i) => { - const profile = profiles.find(([key, value]) => key.network === i.networkIdentifier) - return { - internalName: i.name, - network: i.networkIdentifier, - connected: !!profile, - userId: profile?.[0].userId, - identifier: profile?.[0], - } - }) + const providers = [...definedSocialNetworkUIs.values()] + .map((i) => { + const profile = profiles.find(([key, value]) => key.network === i.networkIdentifier) + if (i.networkIdentifier === 'localhost') return null! + return { + internalName: i.networkIdentifier, + network: i.networkIdentifier, + connected: !!profile, + userId: profile?.[0].userId, + identifier: profile?.[0], + } + }) + .filter((x) => x) const [detachProfile, , setDetachProfile] = useModal(DashboardPersonaUnlinkConfirmDialog) const onConnect = async (provider: typeof providers[0]) => { + const ui = await loadSocialNetworkUI(provider.internalName) + // TODO: what if it does not have a (single?) home page? (e.g. mastdon) + // TODO: maybe add a new action "onConnect"? + const home = ui.utils.getHomePage?.() if (!persona) return - if (!(await getCurrentNetworkUI(provider.network).requestPermission())) return + if (!Flags.no_web_extension_dynamic_permission_request) { + if (!(await ui.permission.request())) return + } // FIXME: // setting storage race condition here @@ -43,12 +52,7 @@ export default function ProfileBox({ persona, ProviderLineProps }: ProfileBoxPro persona: persona.identifier.toText(), } as SetupGuideCrossContextStatus) await delay(100) - exclusiveTasks(getCurrentNetworkUI(provider.network).getHomePage(), { - active: true, - autoClose: false, - important: true, - memorable: false, - }).SetupGuide(persona.identifier) + home && browser.tabs.create({ active: true, url: home }) } const onDisconnect = (provider: typeof providers[0]) => { setDetachProfile({ nickname: persona?.nickname, identifier: provider.identifier }) diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/ProviderLine.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/ProviderLine.tsx index 06fa71c76ebe..531efe27410a 100644 --- a/packages/maskbook/src/extension/options-page/DashboardComponents/ProviderLine.tsx +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/ProviderLine.tsx @@ -7,8 +7,8 @@ import { useI18N } from '../../../utils/i18n-next-ui' import LinkOffIcon from '@material-ui/icons/LinkOff' import ArrowForwardIcon from '@material-ui/icons/ArrowForward' import { useStylesExtends } from '../../../components/custom-ui-helper' -import { facebookDomain } from '../../../social-network-provider/facebook.com/utils/isMobile' -import { twitterDomain } from '../../../social-network-provider/twitter.com/utils/isMobile' +import { facebookDomain } from '../../../social-network-adaptor/facebook.com/utils/isMobile' +import { twitterDomain } from '../../../social-network-adaptor/twitter.com/utils/isMobile' import { Flags } from '../../../utils/flags' const useStyles = makeStyles((theme) => @@ -54,6 +54,7 @@ export interface ProviderLineProps extends withClasses { export default function ProviderLine(props: ProviderLineProps) { const { t } = useI18N() + // TODO: internal name should not be used to display const { internalName, network, connected, userId, onAction } = props const classes = useStylesExtends(useStyles(), props) return ( diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/TokenActionsBar.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/TokenActionsBar.tsx index a40d5433d6c9..11d9d3605906 100644 --- a/packages/maskbook/src/extension/options-page/DashboardComponents/TokenActionsBar.tsx +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/TokenActionsBar.tsx @@ -13,6 +13,7 @@ import { PluginTransakMessages } from '../../../plugins/Transak/messages' import { useAccount } from '../../../web3/hooks/useAccount' import { DashboardWalletTransferDialog } from './TransferDialog' import { useChainIdValid } from '../../../web3/hooks/useChainState' +import { forwardRef } from 'react' //#region token actions menu export interface TokenActionsMenuProps extends withClasses { @@ -35,7 +36,7 @@ export interface TokenActionsMenuProps extends withClasses { ) => void } -export function TokenActionsMenu(props: TokenActionsMenuProps) { +export const TokenActionsMenu = forwardRef((props: TokenActionsMenuProps) => { const { chain, wallet, token, onTransferDialogOpen, onHideTokenConfirmDialogOpen } = props const account = useAccount() const { t } = useI18N() @@ -70,7 +71,7 @@ export function TokenActionsMenu(props: TokenActionsMenuProps) { ) -} +}) //#endregion //#region ERC20 token actions bar diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Record.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Record.tsx new file mode 100644 index 000000000000..dae8b3825dd4 --- /dev/null +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Record.tsx @@ -0,0 +1,31 @@ +import { createStyles, makeStyles, Typography } from '@material-ui/core' +import type { FC } from 'react' +import { TokenIcon } from '../TokenIcon' +import classNames from 'classnames' +import type { Transaction } from '../../../../plugins/Wallet/services' + +const useStyles = makeStyles((theme) => + createStyles({ + root: { display: 'flex', alignItems: 'center' }, + direction: { marginLeft: theme.spacing(0.5), marginRight: theme.spacing(0.5) }, + amount: { marginRight: theme.spacing(0.5) }, + symbol: { marginRight: theme.spacing(0.5) }, + receive: { color: '#00c087' }, + }), +) + +export const Record: FC<{ pair: Transaction['pairs'][number] }> = ({ pair }) => { + const styles = useStyles() + return ( + + + {pair.direction === 'send' ? '-' : '+'} + {pair.amount.toFixed(4)} + {pair.symbol} + + ) +} diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Row.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Row.tsx new file mode 100644 index 000000000000..d2594a4a6bd8 --- /dev/null +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/Row.tsx @@ -0,0 +1,71 @@ +import type { FC } from 'react' +import classNames from 'classnames' +import { isNil } from 'lodash-es' +import { createStyles, Link, makeStyles, TableCell, TableRow, Typography } from '@material-ui/core' +import { Record } from './Record' +import type { Transaction } from '../../../../plugins/Wallet/services' + +interface Props { + transaction: Transaction +} + +const useStyles = makeStyles(() => + createStyles({ + failed: { opacity: 0.3 }, + hidden: { visibility: 'hidden' }, + overflow: { + overflow: 'hidden', + textOverflow: 'ellipsis', + whiteSpace: 'nowrap', + }, + }), +) + +export const Row: FC = ({ transaction }) => { + const styles = useStyles() + return ( + + + + {transaction.timeAt.toLocaleString()} + +
+ + + + {transaction.type} + +
+ + + {transaction.pairs.map((pair, index) => ( + + ))} + + + + Gas fee + + + {transaction.gasFee?.eth.toFixed(4)} ETH + + + {transaction.gasFee?.usd.toFixed(2)} USD + + + + ) +} + +const Address: FC<{ id: string | undefined }> = ({ id }) => ( + + {id?.slice(0, 5)} + ... + {id?.slice(id.length - 5)} + +) diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/index.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/index.tsx new file mode 100644 index 000000000000..7a8c1849ebd1 --- /dev/null +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/TransactionList/index.tsx @@ -0,0 +1,80 @@ +import { + Box, + Button, + createStyles, + makeStyles, + Skeleton, + Table, + TableBody, + TableCell, + TableRow, + Typography, +} from '@material-ui/core' +import { useTransactions } from '../../../../plugins/Wallet/hooks/useTransactions' +import { TransactionProvider } from '../../../../plugins/Wallet/types' +import { useAccount } from '../../../../web3/hooks/useAccount' +import { Row } from './Row' + +const useStyles = makeStyles(() => + createStyles({ + fixed: { tableLayout: 'fixed' }, + }), +) + +export function TransactionList() { + const styles = useStyles() + const account = useAccount() + const { + value: transactions = [], + loading: transactionsLoading, + error: transactionsError, + retry: transactionsRetry, + } = useTransactions(account, TransactionProvider.DEBANK) + + if (transactionsLoading) + return ( + + + {new Array(3).fill(0).map((_, i) => ( + + + + + + ))} + +
+ ) + + if (transactionsError || transactions.length === 0) + return ( + + No transaction found. + + + ) + + return ( + + + {transactions.map((transaction) => ( + + ))} + +
+ ) +} diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/WalletAssetsTable.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/WalletAssetsTable.tsx index 76a0b221d31b..9078d38ab595 100644 --- a/packages/maskbook/src/extension/options-page/DashboardComponents/WalletAssetsTable.tsx +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/WalletAssetsTable.tsx @@ -4,6 +4,7 @@ import { IconButton, createStyles, makeStyles, + Skeleton, Table, TableBody, TableCell, @@ -174,8 +175,10 @@ function ViewDetailed(props: ViewDetailedProps) { ))} -
- {menu} + + + {menu} + {hideTokenConfirmDialog} {transeferDialog} @@ -231,8 +234,6 @@ export function WalletAssetsTable(props: WalletAssetsTableProps) { const [viewLength, setViewLength] = useState(MAX_TOKENS_LENGTH) const [price, setPrice] = useState(MIN_VALUE) - if (detailedTokensLoading) return null - if (detailedTokensError) return ( - {detailedTokens - .filter((x) => - Number(price) !== 0 - ? new BigNumber(x.value?.[CurrencyType.USD] || '0').isGreaterThan(price) || - x.token.type === EthereumTokenType.Ether - : true, - ) - .map((y, idx) => - idx < viewLength ? ( - - ) : null, - )} + {detailedTokensLoading + ? new Array(3).fill(0).map((_, i) => ( + + + + + + + + + + + + + + + + + + )) + : detailedTokens + .filter((x) => + Number(price) !== 0 + ? new BigNumber(x.value?.[CurrencyType.USD] || '0').isGreaterThan(price) || + x.token.type === EthereumTokenType.Ether + : true, + ) + .map((y, idx) => + idx < viewLength ? ( + + ) : null, + )} diff --git a/packages/maskbook/src/extension/options-page/DashboardComponents/WalletContent.tsx b/packages/maskbook/src/extension/options-page/DashboardComponents/WalletContent.tsx index 543d3c9e2901..e9b6b1e9e8b2 100644 --- a/packages/maskbook/src/extension/options-page/DashboardComponents/WalletContent.tsx +++ b/packages/maskbook/src/extension/options-page/DashboardComponents/WalletContent.tsx @@ -20,9 +20,9 @@ import { WalletAssetsTable } from './WalletAssetsTable' import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' import { PluginTransakMessages } from '../../../plugins/Transak/messages' import { Flags } from '../../../utils/flags' -import { ElectionTokenAlbum } from '../../../plugins/Election2020/UI/ElectionTokenAlbum' -import { TokenAlbum as COTM_TokenAlbum } from '../../../plugins/COTM/UI/TokenAlbum' import { useChainIdValid } from '../../../web3/hooks/useChainState' +import { TransactionList } from './TransactionList' +import { CollectibleList } from './CollectibleList' const useStyles = makeStyles((theme) => createStyles({ @@ -132,6 +132,7 @@ export const WalletContent = forwardRef(({ w onChange={onTabChange}> + @@ -178,8 +179,8 @@ export const WalletContent = forwardRef(({ w {tabIndex === 0 ? ( ) : null} - {Flags.COTM_enabled && tabIndex === 1 ? : null} - {Flags.election2020_enabled && tabIndex === 1 ? : null} + {tabIndex === 1 ? : null} + {tabIndex === 2 ? : null} {addToken} {walletBackup} diff --git a/packages/maskbook/src/extension/options-page/index.tsx b/packages/maskbook/src/extension/options-page/index.tsx index fed4b1a70a0d..08527046fd6c 100644 --- a/packages/maskbook/src/extension/options-page/index.tsx +++ b/packages/maskbook/src/extension/options-page/index.tsx @@ -1,4 +1,4 @@ -import '../../provider.worker' +import '../../social-network-adaptor' import '../../setup.ui' import { useState } from 'react' diff --git a/packages/maskbook/src/extension/popup-page/UI.tsx b/packages/maskbook/src/extension/popup-page/UI.tsx index 4ecf1eb6623a..68121ac7869e 100644 --- a/packages/maskbook/src/extension/popup-page/UI.tsx +++ b/packages/maskbook/src/extension/popup-page/UI.tsx @@ -1,4 +1,4 @@ -import '../../social-network-provider/popup-page/index' +import '../../social-network-adaptor/popup-page/index' import '../../setup.ui' import { useCallback, memo } from 'react' @@ -7,9 +7,8 @@ import { ThemeProvider, makeStyles, Theme, withStyles, StylesProvider, jssPreset import { Button, Paper, Divider, Typography, Box } from '@material-ui/core' import { useMaskbookTheme } from '../../utils/theme' import { ChooseIdentity } from '../../components/shared/ChooseIdentity' -import { getActivatedUI } from '../../social-network/ui' +import { activatedSocialNetworkUI } from '../../social-network' import { useI18N } from '../../utils/i18n-next-ui' -import { useValueRef } from '../../utils/hooks/useValueRef' import { delay } from '../../utils/utils' import { WalletMessages } from '../../plugins/Wallet/messages' import { useRemoteControlledDialog } from '../../utils/hooks/useRemoteControlledDialog' @@ -17,6 +16,8 @@ import { Alert } from '@material-ui/core' import { useAsyncRetry } from 'react-use' import { MaskbookUIRoot } from '../../UIRoot' import { create } from 'jss' +import { useMyIdentities } from '../../components/DataSource/useActivatedUI' +import { Flags } from '../../utils/flags' const GlobalCss = withStyles({ '@global': { @@ -74,10 +75,10 @@ function PopupUI() { const { t } = useI18N() const classes = useStyles() - const ui = getActivatedUI() - const identities = useValueRef(ui.myIdentitiesRef) + const ui = activatedSocialNetworkUI + const identities = useMyIdentities() - const { value: hasPermission = true, retry: checkPermission } = useAsyncRetry(ui.hasPermission) + const { value: hasPermission = true, retry: checkPermission } = useAsyncRetry(ui.permission.has) const onEnter = useCallback((event: React.MouseEvent) => { if (event.shiftKey) { @@ -122,7 +123,10 @@ function PopupUI() { color="primary" variant="contained" size="small" - onClick={() => ui.requestPermission().then(checkPermission)}> + onClick={() => { + if (Flags.no_web_extension_dynamic_permission_request) return + ui.permission.request().then(checkPermission) + }}> {t('popup_request_permission')} diff --git a/packages/maskbook/src/extension/popup-page/index.tsx b/packages/maskbook/src/extension/popup-page/index.tsx index 4e2a5789bce1..20fb61ec32d0 100644 --- a/packages/maskbook/src/extension/popup-page/index.tsx +++ b/packages/maskbook/src/extension/popup-page/index.tsx @@ -1,7 +1,6 @@ -import '../../social-network-provider/popup-page/index' import '../../setup.ui' - +import { activateSocialNetworkUI } from '../../social-network' import { SSRRenderer } from '../../utils/SSRRenderer' import { Popup } from './UI' -export default SSRRenderer() +activateSocialNetworkUI().then(() => SSRRenderer()) diff --git a/packages/maskbook/src/manifest.json b/packages/maskbook/src/manifest.json index c8e2b96318c5..71933cdaad0a 100644 --- a/packages/maskbook/src/manifest.json +++ b/packages/maskbook/src/manifest.json @@ -1,6 +1,6 @@ { "name": "Mask Network", - "version": "1.29.9", + "version": "1.29.12", "manifest_version": 2, "permissions": ["storage", "downloads", "webNavigation", "activeTab"], "optional_permissions": ["", "notifications"], diff --git a/packages/maskbook/src/plugins/Airdrop/UI/AirdropClaimCard.tsx b/packages/maskbook/src/plugins/Airdrop/UI/AirdropClaimCard.tsx index 503d8cd718dd..720416a34536 100644 --- a/packages/maskbook/src/plugins/Airdrop/UI/AirdropClaimCard.tsx +++ b/packages/maskbook/src/plugins/Airdrop/UI/AirdropClaimCard.tsx @@ -5,9 +5,8 @@ import { useCallback, useEffect, useState } from 'react' import { useStylesExtends } from '../../../components/custom-ui-helper' import { usePostLink } from '../../../components/DataSource/usePostInfo' import { AirdropIcon } from '../../../resources/AirdropIcon' -import { getActivatedUI } from '../../../social-network/ui' +import { activatedSocialNetworkUI } from '../../../social-network' import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { useAccount } from '../../../web3/hooks/useAccount' import { TransactionStateType } from '../../../web3/hooks/useTransactionState' import type { ERC20TokenDetailed } from '../../../web3/types' @@ -19,6 +18,7 @@ import { CheckStateType, useCheckCallback } from '../hooks/useCheckCallback' import { ClaimDialog } from './ClaimDialog' import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' import { useChainId } from '../../../web3/hooks/useChainState' +import { isTwitter } from '../../../social-network-adaptor/twitter.com/base' const useStyles = makeStyles((theme) => createStyles({ @@ -108,19 +108,21 @@ export function AirdropClaimCard(props: AirdropClaimCardProps) { //#endregion //#region transaction dialog - const cashTag = getActivatedUI()?.networkIdentifier === 'twitter.com' ? '$' : '' + const cashTag = isTwitter(activatedSocialNetworkUI) ? '$' : '' const postLink = usePostLink() - const shareLink = useShareLink( - [ - `I just claimed ${cashTag}${token?.symbol} with ${ - new BigNumber(packet?.amount ?? '0') - .multipliedBy(checkState.type === CheckStateType.YEP ? checkState.ratio : 1) - .dp(0) - .toFixed() + '.00' - }. Follow @realMaskbook (mask.io) to claim airdrop.`, - postLink, - ].join('\n'), - ) + const shareLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + [ + `I just claimed ${cashTag}${token?.symbol} with ${ + new BigNumber(packet?.amount ?? '0') + .multipliedBy(checkState.type === CheckStateType.YEP ? checkState.ratio : 1) + .dp(0) + .toFixed() + '.00' + }. Follow @realMaskbook (mask.io) to claim airdrop.`, + postLink, + ].join('\n'), + ) + .toString() // close the transaction dialog const [_, setTransactionDialogOpen] = useRemoteControlledDialog( diff --git a/packages/maskbook/src/plugins/COTM/README.md b/packages/maskbook/src/plugins/COTM/README.md deleted file mode 100644 index 6143636a7bc5..000000000000 --- a/packages/maskbook/src/plugins/COTM/README.md +++ /dev/null @@ -1,9 +0,0 @@ -# Plugin: CreativityOnTheMove - -## Feature Set - -- [x] View COTM packet - -## Prototype - -- diff --git a/packages/maskbook/src/plugins/COTM/UI/CompositionDialog.tsx b/packages/maskbook/src/plugins/COTM/UI/CompositionDialog.tsx deleted file mode 100644 index c20220b8fa96..000000000000 --- a/packages/maskbook/src/plugins/COTM/UI/CompositionDialog.tsx +++ /dev/null @@ -1,100 +0,0 @@ -import { useCallback, useState } from 'react' -import { Box, createStyles, DialogContent, DialogProps, FormControl, makeStyles, TextField } from '@material-ui/core' -import { pick } from 'lodash-es' -import { InjectedDialog } from '../../../components/shared/InjectedDialog' -import type { COTM_JSONPayload } from '../types' -import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' -import { useChainId } from '../../../web3/hooks/useChainState' -import type { EthereumNetwork } from '../../../web3/types' -import { useAccount } from '../../../web3/hooks/useAccount' -import { resolveChainName } from '../../../web3/pipes' -import { useConstant } from '../../../web3/hooks/useConstant' -import { COTM_MetaKey, COTM_CONSTANTS } from '../constants' -import { getActivatedUI } from '../../../social-network/ui' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { useI18N } from '../../../utils/i18n-next-ui' - -const useStyles = makeStyles((theme) => - createStyles({ - control: { - width: '100%', - paddingBottom: theme.spacing(2), - }, - }), -) - -export interface COTM_CompositionDialogProps { - open: boolean - onClose: () => void - DialogProps?: Partial -} - -export function COTM_CompositionDialog(props: COTM_CompositionDialogProps) { - const { t } = useI18N() - const classes = useStyles() - - const account = useAccount() - const chainId = useChainId() - - // fetch the NTF token - const COTM_TOKEN_ADDRESS = useConstant(COTM_CONSTANTS, 'COTM_TOKEN_ADDRESS') - const { value: nftToken } = useERC721TokenDetailed(COTM_TOKEN_ADDRESS) - - // payload settings - const [name, setName] = useState('') - const [message, setMessage] = useState('') - - const onConfirm = useCallback(() => { - if (!nftToken) return - - // compose payload - const payload: COTM_JSONPayload = { - sender: { - address: account, - name, - message, - }, - creation_time: new Date().getTime(), - network: resolveChainName(chainId) as EthereumNetwork, - ntf_token: pick(nftToken, ['address', 'name', 'symbol']), - } - - // update the composition dialog - const ref = getActivatedUI().typedMessageMetadata - const next = new Map(ref.value.entries()) - payload ? next.set(COTM_MetaKey, payload) : next.delete(COTM_MetaKey) - ref.value = next - - // close the dialog - props.onClose() - }, [account, chainId, name, message, nftToken, props.onClose]) - - return ( - - - - setName(e.target.value)} variant="outlined" /> - - - setMessage(e.target.value)} - variant="outlined" - /> - - - - - {t('confirm')} - - - - - - ) -} diff --git a/packages/maskbook/src/plugins/COTM/UI/TokenAlbum.tsx b/packages/maskbook/src/plugins/COTM/UI/TokenAlbum.tsx deleted file mode 100644 index 1f1a90737a6c..000000000000 --- a/packages/maskbook/src/plugins/COTM/UI/TokenAlbum.tsx +++ /dev/null @@ -1,52 +0,0 @@ -import { createStyles, makeStyles } from '@material-ui/core' -import { useConstant } from '../../../web3/hooks/useConstant' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { COTM_CONSTANTS } from '../constants' -import { useAllTokensOfOwner } from '../hooks/useAllTokensOfOwner' -import { TokenCard } from './TokenCard' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - padding: theme.spacing(0, 3), - }, - content: { - margin: '0 auto', - display: 'flex', - flexFlow: 'row wrap', - justifyContent: 'flex-start', - scrollSnapAlign: 'center', - '&::after': { - content: '""', - flex: 'auto', - }, - }, - tile: { - padding: theme.spacing(1), - }, - }), -) - -export interface TokenAlbumProps {} - -export function TokenAlbum(props: TokenAlbumProps) { - const classes = useStyles(props) - - // fetch the NFT token - const COTM_TOKEN_ADDRESS = useConstant(COTM_CONSTANTS, 'COTM_TOKEN_ADDRESS') - const { value: COTM_Token } = useERC721TokenDetailed(COTM_TOKEN_ADDRESS) - const tokens = useAllTokensOfOwner(COTM_Token) - - if (!tokens.value.length) return null - return ( -
-
- {tokens.value.map((token) => ( -
- -
- ))} -
-
- ) -} diff --git a/packages/maskbook/src/plugins/COTM/UI/TokenCard.tsx b/packages/maskbook/src/plugins/COTM/UI/TokenCard.tsx deleted file mode 100644 index 07d4498a324b..000000000000 --- a/packages/maskbook/src/plugins/COTM/UI/TokenCard.tsx +++ /dev/null @@ -1,79 +0,0 @@ -import Tilt from 'react-tilt' -import { Card, createStyles, Link, makeStyles } from '@material-ui/core' -import type { COTM_Token } from '../types' -import { resolveLinkOnEtherscan } from '../../../web3/pipes' -import { useChainId } from '../../../web3/hooks/useChainState' -import { useConstant } from '../../../web3/hooks/useConstant' -import { COTM_CONSTANTS } from '../constants' -import { Video } from '../../../components/shared/Video' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - borderRadius: 4, - position: 'relative', - backgroundColor: theme.palette.background.paper, - }, - content: {}, - identifier: { - color: theme.palette.common.white, - fontSize: 12, - borderRadius: 4, - display: 'block', - width: '100%', - textAlign: 'center', - backgroundColor: 'rgba(0, 0, 0, 0.3)', - left: 0, - bottom: 0, - position: 'absolute', - }, - }), -) - -export interface TokenCardProps { - token: COTM_Token - canViewOnEtherscan?: boolean -} - -export function TokenCard(props: TokenCardProps) { - const classes = useStyles(props) - const chainId = useChainId() - const COTM_TOKEN_ADDRESS = useConstant(COTM_CONSTANTS, 'COTM_TOKEN_ADDRESS') - - const CardComponent = ( - - - - - ) - return props.canViewOnEtherscan && props.token.tokenId ? ( - - {CardComponent} - - ) : ( - CardComponent - ) -} diff --git a/packages/maskbook/src/plugins/COTM/UI/TokenPacket.tsx b/packages/maskbook/src/plugins/COTM/UI/TokenPacket.tsx deleted file mode 100644 index fd35fb275b86..000000000000 --- a/packages/maskbook/src/plugins/COTM/UI/TokenPacket.tsx +++ /dev/null @@ -1,254 +0,0 @@ -import { useCallback, useEffect } from 'react' -import { Box, Card, CardContent, CardHeader, createStyles, Link, makeStyles, Typography } from '@material-ui/core' -import OpenInNewIcon from '@material-ui/icons/OpenInNew' -import classNames from 'classnames' -import LisaImage from '../assets/Lisa' -import { TokenCard } from './TokenCard' -import type { COTM_JSONPayload } from '../types' -import { useConstant } from '../../../web3/hooks/useConstant' -import { COTM_CONSTANTS } from '../constants' -import { useAccount } from '../../../web3/hooks/useAccount' -import { resolveChainId, resolveChainName, resolveTokenLinkOnEtherscan } from '../../../web3/pipes' -import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' -import { TransactionStateType } from '../../../web3/hooks/useTransactionState' -import { WalletMessages } from '../../Wallet/messages' -import { useTokens } from '../hooks/useTokens' -import { useTokensOfOwner } from '../hooks/useTokensOfOwner' -import { useShareLink } from '../../../utils/hooks/useShareLink' -import { useAvailability } from '../hooks/useAvailability' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { useI18N } from '../../../utils/i18n-next-ui' -import { usePostLink } from '../../../components/DataSource/usePostInfo' -import { useChainId, useChainIdValid } from '../../../web3/hooks/useChainState' -import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' -import { first } from 'lodash-es' -import { useMintFromServerCallback } from '../hooks/useMintFromServerCallback' -import { EthereumMessages } from '../../Ethereum/messages' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - userSelect: 'none', - backgroundImage: `url(${LisaImage})`, - position: 'relative', - }, - header: { - zIndex: 0, - position: 'relative', - padding: 0, - }, - content: { - zIndex: 1, - position: 'relative', - paddingBottom: `${theme.spacing(2)} !important`, - }, - footer: { - paddingTop: theme.spacing(2), - display: 'flex', - justifyContent: 'center', - }, - cards: { - scrollSnapAlign: 'center', - overflow: 'auto', - display: 'flex', - flexDirection: 'row', - marginTop: theme.spacing(3), - scrollbarWidth: 'none', - '&::-webkit-scrollbar': { - display: 'none', - }, - }, - notes: { - fontSize: 12, - fontWeight: 500, - width: '100%', - display: 'flex', - flexDirection: 'row', - justifyContent: 'space-between', - }, - title: { - fontSize: '24px !important', - textAlign: 'center', - }, - card: { - padding: theme.spacing(2, 1), - '&:first-child': { - paddingLeft: 0, - }, - '&:last-child': { - paddingRight: 0, - }, - }, - note: { - color: theme.palette.common.white, - fontSize: 'inherit', - textShadow: [ - '-1px 0 0 #c2130f', - '1px 0 0 #c2130f', - '0 -1px 0 #c2130f', - '0 1px 0 #c2130f', - '-1px -1px 0 #c2130f', - '1px -1px 0 #c2130f', - '-1px 1px 0 #c2130f', - '1px 1px 0 #c2130f', - ].join(','), - }, - link: { - color: theme.palette.common.white, - display: 'flex', - alignItems: 'center', - }, - }), -) - -export interface TokenPacketProps { - payload: COTM_JSONPayload -} - -export function TokenPacket(props: TokenPacketProps) { - const { payload } = props - - const { t } = useI18N() - const classes = useStyles() - - // fetch the NTF token - const COTM_TOKEN_ADDRESS = useConstant(COTM_CONSTANTS, 'COTM_TOKEN_ADDRESS') - const { value: COTM_Token } = useERC721TokenDetailed(COTM_TOKEN_ADDRESS) - - const { value: remaining, loading: loadingRemaining, retry: revalidateAvailability } = useAvailability() - const tokens = useTokens(COTM_Token) - const { value: tokensOfOwner, loading: loadingTokensOfOwner, retry: revalidateTokensOfOwner } = useTokensOfOwner( - COTM_Token, - ) - const prefaceToken = first(tokensOfOwner.length ? tokensOfOwner : tokens) - - // context - const account = useAccount() - const chainId = useChainId() - const chainIdValid = useChainIdValid() - - //#region mint - const [mintState, mintCallback, resetMintCallback] = useMintFromServerCallback(account) - //#endregion - - //#region remote controlled transaction dialog - const postLink = usePostLink() - const shareLink = useShareLink( - [ - `I just received a special NFT from @${payload.sender.name}. Follow @realmaskbook (mask.io) to get your first NFT on Twitter.`, - '#mask_io #twitternft', - postLink, - ].join('\n'), - ) - - // close the transaction dialog - const [_, setTransactionDialogOpen] = useRemoteControlledDialog( - EthereumMessages.events.transactionDialogUpdated, - (ev) => { - if (ev.open) return - - // reset state - resetMintCallback() - - if (mintState.type !== TransactionStateType.CONFIRMED) return - - // revalidate my tokens - revalidateAvailability() - revalidateTokensOfOwner() - }, - ) - - // open the transaction dialog - useEffect(() => { - if (mintState.type === TransactionStateType.UNKNOWN) return - setTransactionDialogOpen({ - open: true, - shareLink, - state: mintState, - summary: `Getting #CreativityOnTheMove NFT token.`, - }) - }, [mintState /* update tx dialog only if state changed */]) - //#endregion - - //#region remote controlled select provider dialog - const [, setSelectProviderDialogOpen] = useRemoteControlledDialog(WalletMessages.events.selectProviderDialogUpdated) - const onConnect = useCallback(() => { - setSelectProviderDialogOpen({ - open: true, - }) - }, [setSelectProviderDialogOpen]) - //#endregion - - // TODO: - // loading UI - if (!COTM_Token || !tokens.length) return null - if (loadingRemaining || loadingTokensOfOwner) return Loading the packet… - if (resolveChainId(payload.network) !== chainId) - return Not available on {resolveChainName(chainId)}. - return ( - <> - - - - - #CreativityOnTheMove - -
- {tokensOfOwner.length ? ( - - You've collected {tokensOfOwner.length} {COTM_Token.symbol}. - - ) : null} - {tokensOfOwner.length === 0 && tokens.length > 0 ? ( - - {remaining > 0 ? `Has been collected ${remaining}/${tokens.length}` : 'None Left'} - - ) : null} - - - - {COTM_Token.name}({COTM_Token.symbol}) - - - - -
-
- {prefaceToken ? ( -
- -
- ) : null} -
-
- From @{payload.sender.name || 'Unknown'} -
-
-
- - {!account ? ( - - {t('plugin_wallet_connect_a_wallet')} - - ) : remaining > 0 && tokensOfOwner.length === 0 ? ( - - {`Get #CreativityOnTheMove NFT`} - - ) : null} - - - ) -} diff --git a/packages/maskbook/src/plugins/COTM/assets/Lisa.ts b/packages/maskbook/src/plugins/COTM/assets/Lisa.ts deleted file mode 100644 index 72be9729ad8f..000000000000 --- a/packages/maskbook/src/plugins/COTM/assets/Lisa.ts +++ /dev/null @@ -1 +0,0 @@ -export default `` diff --git a/packages/maskbook/src/plugins/COTM/constants.ts b/packages/maskbook/src/plugins/COTM/constants.ts deleted file mode 100644 index bfef4258a10f..000000000000 --- a/packages/maskbook/src/plugins/COTM/constants.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { ChainId } from '../../web3/types' - -export const COTM_MetaKey = 'com.maskbook.COTM:1' -export const COTM_PluginID = 'com.maskbook.COTM' - -export const COTM_CONSTANTS = { - COTM_TOKEN_ADDRESS: { - [ChainId.Mainnet]: '0x15fd3b2b6d61dd72df3c68639f6159ea2ebee41c', - [ChainId.Ropsten]: '', - [ChainId.Rinkeby]: '0x73945b47a02a77a6b2ae86a023d1106ce3f01fa0', - [ChainId.Kovan]: '', - [ChainId.Gorli]: '', - }, -} - -// the total NFT token count for #CreativityOnTheMove season -export const MAX_TOKEN_COUNT = 20 diff --git a/packages/maskbook/src/plugins/COTM/contracts/useCOTM_TokenContract.ts b/packages/maskbook/src/plugins/COTM/contracts/useCOTM_TokenContract.ts deleted file mode 100644 index 7438beb0bfca..000000000000 --- a/packages/maskbook/src/plugins/COTM/contracts/useCOTM_TokenContract.ts +++ /dev/null @@ -1,11 +0,0 @@ -import type { AbiItem } from 'web3-utils' -import COTM_TokenABI from '@dimensiondev/contracts/abis/COTM_Token.json' -import type { COTMToken as COTM_Token } from '@dimensiondev/contracts/types/COTM_Token' -import { useConstant } from '../../../web3/hooks/useConstant' -import { useContract } from '../../../web3/hooks/useContract' -import { COTM_CONSTANTS } from '../constants' - -export function useCOTM_TokenContract() { - const address = useConstant(COTM_CONSTANTS, 'COTM_TOKEN_ADDRESS') - return useContract(address, COTM_TokenABI as AbiItem[]) -} diff --git a/packages/maskbook/src/plugins/COTM/define.tsx b/packages/maskbook/src/plugins/COTM/define.tsx deleted file mode 100644 index 32090a421599..000000000000 --- a/packages/maskbook/src/plugins/COTM/define.tsx +++ /dev/null @@ -1,43 +0,0 @@ -import { SnackbarContent } from '@material-ui/core' -import { Suspense } from 'react' -import MaskbookPluginWrapper from '../MaskbookPluginWrapper' -import { COTM_MetaKey, COTM_PluginID } from './constants' -import { COTM_MetadataReader } from './helpers' -import type { COTM_JSONPayload } from './types' -import { createCompositionDialog } from '../utils/createCompositionDialog' -import { TokenPacket } from './UI/TokenPacket' -import { COTM_CompositionDialog } from './UI/CompositionDialog' -import { Flags } from '../../utils/flags' -import { PluginConfig, PluginScope, PluginStage } from '../types' - -const [COTM_CompositionEntry, COTM_CompositionUI] = createCompositionDialog('🇳🇱 #CreativityOnTheMove', (props) => ( - -)) - -export const COTM_PluginDefine: PluginConfig = { - pluginName: 'COTM', - identifier: COTM_PluginID, - stage: PluginStage.Production, - scope: PluginScope.Public, - successDecryptionInspector: function Comp(props) { - const payload = COTM_MetadataReader(props.message.meta) - if (!payload.ok) return null - return ( - - }> - - - - ) - }, - postDialogMetadataBadge: new Map([ - [ - COTM_MetaKey, - (payload: COTM_JSONPayload) => { - return '#CreativityOnTheMove - Strange Design' - }, - ], - ]), - PageComponent: Flags.COTM_composition_dialog_enabled ? COTM_CompositionUI : undefined, - postDialogEntries: Flags.COTM_composition_dialog_enabled ? [COTM_CompositionEntry] : undefined, -} diff --git a/packages/maskbook/src/plugins/COTM/helpers.ts b/packages/maskbook/src/plugins/COTM/helpers.ts deleted file mode 100644 index d28e6a5d9dbd..000000000000 --- a/packages/maskbook/src/plugins/COTM/helpers.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { createTypedMessageMetadataReader, createRenderWithMetadata } from '../../protocols/typed-message/metadata' -import { COTM_MetaKey } from './constants' -import type { COTM_JSONPayload } from './types' -import schema from './schema.json' - -export const COTM_MetadataReader = createTypedMessageMetadataReader(COTM_MetaKey, schema) -export const renderWithCOTM_Metadata = createRenderWithMetadata(COTM_MetadataReader) diff --git a/packages/maskbook/src/plugins/COTM/hooks/useAllTokensOfOwner.ts b/packages/maskbook/src/plugins/COTM/hooks/useAllTokensOfOwner.ts deleted file mode 100644 index aa8132f0310c..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useAllTokensOfOwner.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { useERC721TokenIdsOfOwner } from '../../../web3/hooks/useERC721TokensOfOwner' -import type { ERC721TokenDetailed } from '../../../web3/types' - -export function useAllTokensOfOwner(token?: ERC721TokenDetailed) { - const { value: tokenIds, ...result } = useERC721TokenIdsOfOwner(token) - return { - ...result, - value: tokenIds.filter(Boolean).map((tokenId) => ({ - tokenId: tokenId ?? '', - tokenImageURL: tokenId ? `${token?.baseURI}${tokenId}.mp4` : '', - })), - } -} diff --git a/packages/maskbook/src/plugins/COTM/hooks/useAvailability.ts b/packages/maskbook/src/plugins/COTM/hooks/useAvailability.ts deleted file mode 100644 index b70ccd6e108f..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useAvailability.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { useAsyncRetry } from 'react-use' -import { useCOTM_TokenContract } from '../contracts/useCOTM_TokenContract' - -export function useAvailability() { - const COTM_TokenContract = useCOTM_TokenContract() - const { value, ...result } = useAsyncRetry(async () => { - if (!COTM_TokenContract) return null - return COTM_TokenContract.methods.check_availability().call() - }, []) - return { - value: Number.parseInt(value ?? '0', 10), - ...result, - } -} diff --git a/packages/maskbook/src/plugins/COTM/hooks/useMintCallback.ts b/packages/maskbook/src/plugins/COTM/hooks/useMintCallback.ts deleted file mode 100644 index d43d75b5cd6c..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useMintCallback.ts +++ /dev/null @@ -1,87 +0,0 @@ -import BigNumber from 'bignumber.js' -import { useCallback } from 'react' -import type { Tx } from '@dimensiondev/contracts/types/types' -import { addGasMargin } from '../../../web3/helpers' -import { TransactionStateType, useTransactionState } from '../../../web3/hooks/useTransactionState' -import { useCOTM_TokenContract } from '../contracts/useCOTM_TokenContract' - -export function useMintCallback(from: string) { - const [mintState, setMintState] = useTransactionState() - const COTM_TokenContract = useCOTM_TokenContract() - - const mintCallback = useCallback(async () => { - if (!COTM_TokenContract) { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - return - } - - // pre-step: start waiting for provider to confirm tx - setMintState({ - type: TransactionStateType.WAIT_FOR_CONFIRMING, - }) - - // step 1: check remaining - const remaining = await COTM_TokenContract.methods.check_availability().call() - if (Number.parseInt(remaining || '0', 10) <= 0) { - setMintState({ - type: TransactionStateType.FAILED, - error: new Error('There is none NTF token left.'), - }) - return - } - - // step 2-1: estimatedGas - const config: Tx = { - from, - to: COTM_TokenContract.options.address, - } - const estimatedGas = await COTM_TokenContract.methods - .mintToken(from) - .estimateGas(config) - .catch((error) => { - setMintState({ - type: TransactionStateType.FAILED, - error, - }) - throw error - }) - - // step 2-2: blocking - return new Promise((resolve, reject) => { - const onSucceed = (hash: string) => { - setMintState({ - type: TransactionStateType.HASH, - hash, - }) - resolve(hash) - } - const onFailed = (error: Error) => { - setMintState({ - type: TransactionStateType.FAILED, - error, - }) - reject(error) - } - COTM_TokenContract.methods.mintToken(from).send( - { - gas: addGasMargin(new BigNumber(estimatedGas)).toFixed(), - ...config, - }, - async (error, hash) => { - if (hash) onSucceed(hash) - else if (error) onFailed(error) - }, - ) - }) - }, [from, COTM_TokenContract]) - - const resetCallback = useCallback(() => { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - }, []) - - return [mintState, mintCallback, resetCallback] as const -} diff --git a/packages/maskbook/src/plugins/COTM/hooks/useMintFromServerCallback.ts b/packages/maskbook/src/plugins/COTM/hooks/useMintFromServerCallback.ts deleted file mode 100644 index c12aa90ff487..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useMintFromServerCallback.ts +++ /dev/null @@ -1,73 +0,0 @@ -import { useCallback, useEffect, useState } from 'react' -import { useTransactionReceipt } from '../../../web3/hooks/useTransaction' -import { TransactionStateType, useTransactionState } from '../../../web3/hooks/useTransactionState' -import { useCOTM_TokenContract } from '../contracts/useCOTM_TokenContract' -import { PluginCOTM } from '../messages' - -export function useMintFromServerCallback(from: string) { - const [txHash, setTxHash] = useState('') - const [mintState, setMintState] = useTransactionState() - const COTM_TokenContract = useCOTM_TokenContract() - - const mintCallback = useCallback(async () => { - if (!COTM_TokenContract) { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - return - } - - // pre-step: start waiting for provider to confirm tx - setMintState({ - type: TransactionStateType.WAIT_FOR_CONFIRMING, - }) - - // step 1: check remaining - const remaining = await COTM_TokenContract.methods.check_availability().call() - if (Number.parseInt(remaining || '0', 10) <= 0) { - setMintState({ - type: TransactionStateType.FAILED, - error: new Error('There is none NTF token left.'), - }) - return - } - - // step 2: mint by server - try { - const { mint_transaction_hash } = await PluginCOTM.mintCOTM_Token(from) - setTxHash(mint_transaction_hash) - } catch (error) { - setMintState({ - type: TransactionStateType.FAILED, - error, - }) - } - }, [from, COTM_TokenContract]) - - const resetCallback = useCallback(() => { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - }, []) - - //#region tracking receipt - const receipt = useTransactionReceipt(txHash) - - useEffect(() => { - if (!receipt) return - if (receipt.status) - setMintState({ - type: TransactionStateType.CONFIRMED, - no: 0, - receipt, - }) - else - setMintState({ - type: TransactionStateType.FAILED, - error: new Error('The contract execution was not successful, check your transaction.'), - }) - }, [receipt]) - //#endregion - - return [mintState, mintCallback, resetCallback] as const -} diff --git a/packages/maskbook/src/plugins/COTM/hooks/useTokens.ts b/packages/maskbook/src/plugins/COTM/hooks/useTokens.ts deleted file mode 100644 index 923f748edda3..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useTokens.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { hexToNumberString, soliditySha3 } from 'web3-utils' -import type { COTM_Token } from '../types' -import type { ERC721TokenDetailed } from '../../../web3/types' -import { MAX_TOKEN_COUNT } from '../constants' - -export function useTokens(token?: ERC721TokenDetailed): COTM_Token[] { - if (!token) return [] - return new Array(MAX_TOKEN_COUNT).fill(0).map((_, idx) => { - const tokenId_ = soliditySha3({ t: 'uint8', v: idx }) - const tokenId = tokenId_ ? hexToNumberString(tokenId_) : '' - return { - tokenId, - tokenImageURL: tokenId ? `${token.baseURI}${tokenId}.mp4` : '', - } - }) -} diff --git a/packages/maskbook/src/plugins/COTM/hooks/useTokensOfOwner.ts b/packages/maskbook/src/plugins/COTM/hooks/useTokensOfOwner.ts deleted file mode 100644 index 0988e563e9ba..000000000000 --- a/packages/maskbook/src/plugins/COTM/hooks/useTokensOfOwner.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { useERC721TokenIdsOfOwner } from '../../../web3/hooks/useERC721TokensOfOwner' -import type { ERC721TokenDetailed } from '../../../web3/types' -import { useTokens } from './useTokens' - -export function useTokensOfOwner(token?: ERC721TokenDetailed) { - const { value: tokenIds, ...result } = useERC721TokenIdsOfOwner(token) - const totalTokens = useTokens(token) - - if (!token || !totalTokens.length) - return { - ...result, - value: [], - } - return { - ...result, - value: totalTokens.filter((x) => tokenIds.includes(x.tokenId)), - } -} diff --git a/packages/maskbook/src/plugins/COTM/messages.ts b/packages/maskbook/src/plugins/COTM/messages.ts deleted file mode 100644 index ffc60568d7ad..000000000000 --- a/packages/maskbook/src/plugins/COTM/messages.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { createPluginMessage } from '../utils/createPluginMessage' -import { createPluginRPC } from '../utils/createPluginRPC' -import { COTM_PluginID } from './constants' - -if (module.hot) module.hot.accept() -const COTM_Message = createPluginMessage<{ _: unknown }>(COTM_PluginID) -export const PluginCOTM = createPluginRPC(COTM_PluginID, () => import('./services'), COTM_Message.events._) diff --git a/packages/maskbook/src/plugins/COTM/schema.json b/packages/maskbook/src/plugins/COTM/schema.json deleted file mode 100644 index 729ed8b6df66..000000000000 --- a/packages/maskbook/src/plugins/COTM/schema.json +++ /dev/null @@ -1,59 +0,0 @@ -{ - "type": "object", - "additionalProperties": true, - "properties": { - "sender": { - "type": "object", - "additionalProperties": true, - "properties": { - "address": { - "type": "string", - "title": "address" - }, - "name": { - "type": "string", - "title": "name" - }, - "message": { - "type": "string", - "title": "message" - } - }, - "required": ["address", "message", "name"], - "title": "sender" - }, - "nft_token": { - "type": "object", - "additionalProperties": true, - "properties": { - "address": { - "description": "token address", - "type": "string", - "title": "address" - }, - "name": { - "description": "token name", - "type": "string", - "title": "name" - }, - "symbol": { - "description": "token symbol", - "type": "string", - "title": "symbol" - } - }, - "required": ["address", "name", "symbol"], - "title": "token" - }, - "creation_time": { - "type": "number", - "title": "creation_time" - }, - "network": { - "enum": ["Mainnet", "Rinkeby", "Ropsten"], - "type": "string", - "title": "network" - } - }, - "required": ["sender", "creation_time", "network"] -} diff --git a/packages/maskbook/src/plugins/COTM/services.ts b/packages/maskbook/src/plugins/COTM/services.ts deleted file mode 100644 index 455029944913..000000000000 --- a/packages/maskbook/src/plugins/COTM/services.ts +++ /dev/null @@ -1,32 +0,0 @@ -import * as jwt from 'jsonwebtoken' -import { sha3 } from 'web3-utils' -import Services from '../../extension/service' -import { resolveChainName } from '../../web3/pipes' - -export async function mintCOTM_Token(from: string): Promise<{ mint_transaction_hash: string }> { - const host = 'https://redpacket.gives' - const x = 'a3323cd1-fa42-44cd-b053-e474365ab3da' - - const chainId = await Services.Ethereum.getChainId(from) - const network = resolveChainName(chainId).toLowerCase() - - // skip hi - const auth = await fetch(`${host}/hi?id=${from}&network=${network}`) - if (!auth.ok) throw new Error('Auth failed') - - const verify = await auth.text() - const jwt_encoded: { - recipient: string - validation: string - signature: string - } = { - recipient: from, - validation: sha3(from)!, - signature: await Services.Ethereum.sign(verify, from, chainId), - } - const mintResponse = await fetch( - `${host}/netherlands?payload=${jwt.sign(jwt_encoded, x, { algorithm: 'HS256' })}&network=${network}`, - ) - if (!mintResponse.ok) throw new Error('Claim failed') - return { mint_transaction_hash: await mintResponse.text() } -} diff --git a/packages/maskbook/src/plugins/COTM/types.ts b/packages/maskbook/src/plugins/COTM/types.ts deleted file mode 100644 index 6a2ae316fd34..000000000000 --- a/packages/maskbook/src/plugins/COTM/types.ts +++ /dev/null @@ -1,21 +0,0 @@ -import type { EthereumNetwork } from '../../web3/types' - -export interface COTM_Token { - tokenId: string // sha3(hollandchina + id) - tokenImageURL: string -} - -export interface COTM_JSONPayload { - sender: { - address: string - name: string - message: string - } - ntf_token: { - address: string - name: string - symbol: string - } - creation_time: number - network: EthereumNetwork -} diff --git a/packages/maskbook/src/plugins/Election2020/README.md b/packages/maskbook/src/plugins/Election2020/README.md deleted file mode 100644 index 038f1e87e5d6..000000000000 --- a/packages/maskbook/src/plugins/Election2020/README.md +++ /dev/null @@ -1,9 +0,0 @@ -# Plugin: Election 2020 - -## Feature Set - -- [x] View election packet - -## Prototype - -- diff --git a/packages/maskbook/src/plugins/Election2020/UI/ElectionCard.tsx b/packages/maskbook/src/plugins/Election2020/UI/ElectionCard.tsx deleted file mode 100644 index 9c7a6ef6ccac..000000000000 --- a/packages/maskbook/src/plugins/Election2020/UI/ElectionCard.tsx +++ /dev/null @@ -1,65 +0,0 @@ -import Tilt from 'react-tilt' -import { Card, createStyles, Link, makeStyles } from '@material-ui/core' -import type { ElectionToken } from '../types' -import { Image } from '../../../components/shared/Image' -import { resolveLinkOnEtherscan } from '../../../web3/pipes' -import { useChainId } from '../../../web3/hooks/useChainState' -import { useConstant } from '../../../web3/hooks/useConstant' -import { ELECTION_2020_CONSTANTS } from '../constants' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - borderRadius: 4, - position: 'relative', - backgroundColor: theme.palette.background.paper, - }, - content: {}, - identifier: { - color: theme.palette.common.white, - fontSize: 12, - borderRadius: 4, - display: 'block', - width: '100%', - textAlign: 'center', - backgroundColor: 'rgba(0, 0, 0, 0.3)', - left: 0, - bottom: 0, - position: 'absolute', - }, - }), -) - -export interface ElectionCardProps { - token: ElectionToken - canViewOnEtherscan?: boolean -} - -export function ElectionCard(props: ElectionCardProps) { - const classes = useStyles(props) - const chainId = useChainId() - const ELECTION_TOKEN_ADDRESS = useConstant(ELECTION_2020_CONSTANTS, 'ELECTION_TOKEN_ADDRESS') - - const CardComponent = ( - - - - - - ) - return props.canViewOnEtherscan && props.token.tokenId ? ( - - {CardComponent} - - ) : ( - CardComponent - ) -} diff --git a/packages/maskbook/src/plugins/Election2020/UI/ElectionCompositionDialog.tsx b/packages/maskbook/src/plugins/Election2020/UI/ElectionCompositionDialog.tsx deleted file mode 100644 index c8399cf97e03..000000000000 --- a/packages/maskbook/src/plugins/Election2020/UI/ElectionCompositionDialog.tsx +++ /dev/null @@ -1,153 +0,0 @@ -import { useCallback, useState } from 'react' -import { - Box, - createStyles, - DialogContent, - DialogProps, - FormControl, - InputLabel, - makeStyles, - MenuItem, - Select, - TextField, -} from '@material-ui/core' -import { pick } from 'lodash-es' -import { InjectedDialog } from '../../../components/shared/InjectedDialog' -import { CANDIDATE_TYPE, US_STATE_TYPE, Election2020JSONPayload } from '../types' -import { getEnumAsArray } from '../../../utils/enum' -import { PortalShadowRoot } from '../../../utils/shadow-root/ShadowRootPortal' -import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' -import { useChainId } from '../../../web3/hooks/useChainState' -import type { EthereumNetwork } from '../../../web3/types' -import { useAccount } from '../../../web3/hooks/useAccount' -import { resolveChainName } from '../../../web3/pipes' -import { useConstant } from '../../../web3/hooks/useConstant' -import { Election2020MetaKey, ELECTION_2020_CONSTANTS } from '../constants' -import { getActivatedUI } from '../../../social-network/ui' -import { resolveStateName } from '../pipes' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { useI18N } from '../../../utils/i18n-next-ui' - -const useStyles = makeStyles((theme) => - createStyles({ - control: { - width: '100%', - paddingBottom: theme.spacing(2), - }, - }), -) - -export interface ElectionCompositionDialogProps { - open: boolean - onClose: () => void - DialogProps?: Partial -} - -export function ElectionCompositionDialog(props: ElectionCompositionDialogProps) { - const { t } = useI18N() - const classes = useStyles() - - const account = useAccount() - const chainId = useChainId() - - // fetch the NTF token - const ELECTION_TOKEN_ADDRESS = useConstant(ELECTION_2020_CONSTANTS, 'ELECTION_TOKEN_ADDRESS') - const { value: nftToken } = useERC721TokenDetailed(ELECTION_TOKEN_ADDRESS) - - // payload settings - const [name, setName] = useState('') - const [message, setMessage] = useState('') - const [state, setState] = useState(US_STATE_TYPE.AK) - const [candidate, setCandidate] = useState(CANDIDATE_TYPE.TRUMP) - - const onConfirm = useCallback(() => { - if (!nftToken) return - - // compose payload - const payload: Election2020JSONPayload = { - state, - winner: candidate, - sender: { - address: account, - name, - message, - }, - creation_time: new Date().getTime(), - network: resolveChainName(chainId) as EthereumNetwork, - ntf_token: pick(nftToken, ['address', 'name', 'symbol']), - } - - // update the composition dialog - const ref = getActivatedUI().typedMessageMetadata - const next = new Map(ref.value.entries()) - payload ? next.set(Election2020MetaKey, payload) : next.delete(Election2020MetaKey) - ref.value = next - - // close the dialog - props.onClose() - }, [account, chainId, name, message, state, candidate, nftToken, props.onClose]) - - return ( - - - - Winner - - - - State - - - - setName(e.target.value)} - variant="outlined" - /> - - - setMessage(e.target.value)} - variant="outlined" - /> - - - - - {t('confirm')} - - - - - - ) -} diff --git a/packages/maskbook/src/plugins/Election2020/UI/ElectionPacket.tsx b/packages/maskbook/src/plugins/Election2020/UI/ElectionPacket.tsx deleted file mode 100644 index 94f77620b963..000000000000 --- a/packages/maskbook/src/plugins/Election2020/UI/ElectionPacket.tsx +++ /dev/null @@ -1,310 +0,0 @@ -import { useCallback, useEffect } from 'react' -import { Box, Card, CardContent, CardHeader, createStyles, Link, makeStyles, Typography } from '@material-ui/core' -import OpenInNewIcon from '@material-ui/icons/OpenInNew' -import classNames from 'classnames' -import { ElectionCard } from './ElectionCard' -import type { Election2020JSONPayload } from '../types' -import { useConstant } from '../../../web3/hooks/useConstant' -import { ELECTION_2020_CONSTANTS } from '../constants' -import { resolveCandidateName, resolveCandidateBriefName, resolveStateName } from '../pipes' -import { useAccount } from '../../../web3/hooks/useAccount' -import { resolveChainId, resolveChainName, resolveTokenLinkOnEtherscan } from '../../../web3/pipes' -import { useMintCallback } from '../hooks/useMintCallback' -import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' -import { TransactionStateType } from '../../../web3/hooks/useTransactionState' -import { WalletMessages } from '../../Wallet/messages' -import { useElectionTokens } from '../hooks/useElectionTokens' -import { useElectionTokensOfOwner } from '../hooks/useElectionTokensOfOwner' -import { useShareLink } from '../../../utils/hooks/useShareLink' -import { useAvailability } from '../hooks/useAvailability' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { useI18N } from '../../../utils/i18n-next-ui' -import { usePostLink } from '../../../components/DataSource/usePostInfo' -import { useChainId, useChainIdValid } from '../../../web3/hooks/useChainState' -import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' -import { getAssetAsBlobURL } from '../../../utils/suspends/getAssetAsBlobURL' -import type { CSSProperties } from '@material-ui/core/styles/withStyles' -import { EthereumMessages } from '../../Ethereum/messages' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - userSelect: 'none', - backgroundImage: 'linear-gradient(180deg, #121d76 0%, #2c39b9 100%)', - position: 'relative', - - '&::before, &::after': { - content: '""', - width: '90%', - height: 260, - backgroundImage: 'var(--flag-image)', - backgroundSize: 'contain', - backgroundRepeat: 'no-repeat', - - bottom: '-21%', - position: 'absolute', - zIndex: 0, - }, - '&::before': { - left: '-12%', - }, - '&::after': { - right: '-12%', - transform: 'scaleX(-1)', - }, - }, - header: { - zIndex: 0, - position: 'relative', - padding: 0, - '&::before': { - content: '""', - width: '100%', - height: 260, - backgroundImage: 'var(--fireworks-image)', - backgroundSize: 'contain', - backgroundRepeat: 'no-repeat', - top: 0, - left: 0, - position: 'absolute', - zIndex: 0, - }, - }, - content: { - zIndex: 1, - position: 'relative', - paddingBottom: `${theme.spacing(2)} !important`, - }, - footer: { - paddingTop: theme.spacing(2), - display: 'flex', - justifyContent: 'center', - }, - cards: { - scrollSnapAlign: 'center', - overflow: 'auto', - display: 'flex', - flexDirection: 'row', - marginTop: theme.spacing(3), - scrollbarWidth: 'none', - '&::-webkit-scrollbar': { - display: 'none', - }, - }, - notes: { - fontSize: 12, - fontWeight: 500, - width: '100%', - display: 'flex', - flexDirection: 'row', - justifyContent: 'space-between', - }, - title: { - fontSize: '24px !important', - textAlign: 'center', - }, - card: { - padding: theme.spacing(2, 1), - '&:first-child': { - paddingLeft: 0, - }, - '&:last-child': { - paddingRight: 0, - }, - }, - note: { - color: theme.palette.common.white, - fontSize: 'inherit', - textShadow: [ - '-1px 0 0 #121d76', - '1px 0 0 #121d76', - '0 -1px 0 #121d76', - '0 1px 0 #121d76', - '-1px -1px 0 #121d76', - '1px -1px 0 #121d76', - '-1px 1px 0 #121d76', - '1px 1px 0 #121d76', - ].join(','), - }, - link: { - color: theme.palette.common.white, - display: 'flex', - alignItems: 'center', - }, - }), -) - -export interface ElectionPacketProps { - payload: Election2020JSONPayload -} - -export function ElectionPacket(props: ElectionPacketProps) { - // fetch images - const FireworksImage = getAssetAsBlobURL(new URL('../assets/fireworks.jpg', import.meta.url)) - const FlagImage = getAssetAsBlobURL(new URL('../assets/flag.jpg', import.meta.url)) - - const { payload } = props - - const { t } = useI18N() - const classes = useStyles() - - // fetch the NTF token - const ELECTION_TOKEN_ADDRESS = useConstant(ELECTION_2020_CONSTANTS, 'ELECTION_TOKEN_ADDRESS') - const { value: electionToken } = useERC721TokenDetailed(ELECTION_TOKEN_ADDRESS) - - const { value: remaining, loading: loadingRemaining, retry: revalidateAvailability } = useAvailability( - payload.state, - ) - const tokens = useElectionTokens(payload.state, electionToken) - const { - value: tokensOfOwner, - loading: loadingTokensOfOwner, - retry: revalidateTokensOfOwner, - } = useElectionTokensOfOwner(payload.state, electionToken) - - // context - const account = useAccount() - const chainId = useChainId() - const chainIdValid = useChainIdValid() - - //#region mint - const [mintState, mintCallback, resetMintCallback] = useMintCallback(account, payload.state, payload.winner) - //#endregion - - //#region remote controlled transaction dialog - const postLink = usePostLink() - const shareLink = useShareLink( - [ - `I just received an election special ${resolveCandidateBriefName(payload.winner)} NFT from @${ - payload.sender.name - }. Follow @realmaskbook (mask.io) to get your first NFT on Twitter.`, - '#mask_io #twitternft', - postLink, - ].join('\n'), - ) - - // close the transaction dialog - const [_, setTransactionDialogOpen] = useRemoteControlledDialog( - EthereumMessages.events.transactionDialogUpdated, - (ev) => { - if (ev.open) return - - // reset state - resetMintCallback() - - if (mintState.type !== TransactionStateType.CONFIRMED) return - - // revalidate my tokens - revalidateAvailability() - revalidateTokensOfOwner() - }, - ) - - // open the transaction dialog - useEffect(() => { - if (mintState.type === TransactionStateType.UNKNOWN) return - setTransactionDialogOpen({ - open: true, - shareLink, - state: mintState, - summary: `Getting ${resolveCandidateName(payload.winner)} wins NFT token.`, - }) - }, [mintState /* update tx dialog only if state changed */]) - //#endregion - - //#region remote controlled select provider dialog - const [, setSelectProviderDialogOpen] = useRemoteControlledDialog(WalletMessages.events.selectProviderDialogUpdated) - const onConnect = useCallback(() => { - setSelectProviderDialogOpen({ - open: true, - }) - }, [setSelectProviderDialogOpen]) - //#endregion - - // TODO: - // loading UI - if (!electionToken || !tokens.length) return null - if (loadingRemaining || loadingTokensOfOwner) - return Loading the election result of {resolveStateName(payload.state)}… - if (resolveChainId(payload.network) !== chainId) - return Not available on {resolveChainName(chainId)}. - return ( - <> - - - - - 2020 Presidential Election - -
- {tokensOfOwner.length ? ( - - You've collected {tokensOfOwner.length} {electionToken.symbol}. - - ) : null} - {tokensOfOwner.length === 0 && tokens.length > 0 ? ( - - {remaining > 0 ? `Has been collected ${remaining}/${tokens.length}` : 'None Left'} - - ) : null} - - - - {electionToken.name}({electionToken.symbol}) - - - - -
-
- {(tokensOfOwner.length ? tokensOfOwner : tokens).map((x, i) => ( -
- -
- ))} -
-
- From @{payload.sender.name || 'Unknown'} -
-
-
- - {!account ? ( - - {t('plugin_wallet_connect_a_wallet')} - - ) : !chainIdValid ? ( - - {t('plugin_wallet_invalid_network')} - - ) : remaining > 0 && tokensOfOwner.length === 0 ? ( - - {`Get ${resolveCandidateName(payload.winner)} wins NFT`} - - ) : null} - - - ) -} diff --git a/packages/maskbook/src/plugins/Election2020/UI/ElectionPacketsInspector.tsx b/packages/maskbook/src/plugins/Election2020/UI/ElectionPacketsInspector.tsx deleted file mode 100644 index d433312e7b74..000000000000 --- a/packages/maskbook/src/plugins/Election2020/UI/ElectionPacketsInspector.tsx +++ /dev/null @@ -1,5 +0,0 @@ -export interface ElectionPacketsInspectorProps {} - -export function ElectionPacketsInspector(props: ElectionPacketsInspectorProps) { - return null -} diff --git a/packages/maskbook/src/plugins/Election2020/UI/ElectionTokenAlbum.tsx b/packages/maskbook/src/plugins/Election2020/UI/ElectionTokenAlbum.tsx deleted file mode 100644 index 6d9ab39cb0dd..000000000000 --- a/packages/maskbook/src/plugins/Election2020/UI/ElectionTokenAlbum.tsx +++ /dev/null @@ -1,55 +0,0 @@ -import { createStyles, makeStyles } from '@material-ui/core' -import { useChainIdValid } from '../../../web3/hooks/useChainState' -import { useConstant } from '../../../web3/hooks/useConstant' -import { useERC721TokenDetailed } from '../../../web3/hooks/useERC721TokenDetailed' -import { ELECTION_2020_CONSTANTS } from '../constants' -import { useAllElectionTokensOfOwner } from '../hooks/useAllElectionTokensOfOwner' -import { ElectionCard } from './ElectionCard' - -const useStyles = makeStyles((theme) => - createStyles({ - root: { - padding: theme.spacing(0, 3), - }, - content: { - margin: '0 auto', - display: 'flex', - flexFlow: 'row wrap', - justifyContent: 'flex-start', - scrollSnapAlign: 'center', - '&::after': { - content: '""', - flex: 'auto', - }, - }, - tile: { - padding: theme.spacing(1), - }, - }), -) - -export interface ElectionTokenAlbumProps {} - -export function ElectionTokenAlbum(props: ElectionTokenAlbumProps) { - const classes = useStyles(props) - - // fetch the NFT token - const ELECTION_TOKEN_ADDRESS = useConstant(ELECTION_2020_CONSTANTS, 'ELECTION_TOKEN_ADDRESS') - const { value: electionToken } = useERC721TokenDetailed(ELECTION_TOKEN_ADDRESS) - const tokens = useAllElectionTokensOfOwner(electionToken) - - const chainIdValid = useChainIdValid() - if (!chainIdValid) return null - if (!tokens.value.length) return null - return ( -
-
- {tokens.value.map((token) => ( -
- -
- ))} -
-
- ) -} diff --git a/packages/maskbook/src/plugins/Election2020/assets/fireworks.jpg b/packages/maskbook/src/plugins/Election2020/assets/fireworks.jpg deleted file mode 100644 index 8bbcfb1b012d..000000000000 Binary files a/packages/maskbook/src/plugins/Election2020/assets/fireworks.jpg and /dev/null differ diff --git a/packages/maskbook/src/plugins/Election2020/assets/flag.jpg b/packages/maskbook/src/plugins/Election2020/assets/flag.jpg deleted file mode 100644 index 301a1a18fe09..000000000000 Binary files a/packages/maskbook/src/plugins/Election2020/assets/flag.jpg and /dev/null differ diff --git a/packages/maskbook/src/plugins/Election2020/constants.ts b/packages/maskbook/src/plugins/Election2020/constants.ts deleted file mode 100644 index f6c4528e2193..000000000000 --- a/packages/maskbook/src/plugins/Election2020/constants.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { ChainId } from '../../web3/types' - -export const Election2020MetaKey = 'com.maskbook.election2020:1' -export const Election2020PluginID = 'com.maskbook.election2020' - -export const ELECTION_2020_CONSTANTS = { - ELECTION_TOKEN_ADDRESS: { - [ChainId.Mainnet]: '0x797ce6d5a2e4ba7ed007b01a42f797a050a3cbd8', - [ChainId.Ropsten]: '', - [ChainId.Rinkeby]: '0xdb52de365ad57bf54272e2ea37bb8651b44bb397', - [ChainId.Kovan]: '', - [ChainId.Gorli]: '', - }, -} diff --git a/packages/maskbook/src/plugins/Election2020/contracts/useElectionTokenContract.ts b/packages/maskbook/src/plugins/Election2020/contracts/useElectionTokenContract.ts deleted file mode 100644 index 74f62987a35e..000000000000 --- a/packages/maskbook/src/plugins/Election2020/contracts/useElectionTokenContract.ts +++ /dev/null @@ -1,11 +0,0 @@ -import type { AbiItem } from 'web3-utils' -import ElectionTokenABI from '@dimensiondev/contracts/abis/ElectionToken.json' -import type { ElectionToken } from '@dimensiondev/contracts/types/ElectionToken' -import { useConstant } from '../../../web3/hooks/useConstant' -import { useContract } from '../../../web3/hooks/useContract' -import { ELECTION_2020_CONSTANTS } from '../constants' - -export function useElectionTokenContract() { - const address = useConstant(ELECTION_2020_CONSTANTS, 'ELECTION_TOKEN_ADDRESS') - return useContract(address, ElectionTokenABI as AbiItem[]) -} diff --git a/packages/maskbook/src/plugins/Election2020/define.tsx b/packages/maskbook/src/plugins/Election2020/define.tsx deleted file mode 100644 index 39c21c616fec..000000000000 --- a/packages/maskbook/src/plugins/Election2020/define.tsx +++ /dev/null @@ -1,46 +0,0 @@ -import { SnackbarContent } from '@material-ui/core' -import { Suspense } from 'react' -import MaskbookPluginWrapper from '../MaskbookPluginWrapper' -import { Election2020MetaKey, Election2020PluginID } from './constants' -import { Election2020MetadataReader } from './helpers' -import type { Election2020JSONPayload } from './types' -import { createCompositionDialog } from '../utils/createCompositionDialog' -import { ElectionPacket } from './UI/ElectionPacket' -import { ElectionCompositionDialog } from './UI/ElectionCompositionDialog' -import { resolveStateName, resolveCandidateName } from './pipes' -import { Flags } from '../../utils/flags' -import { PluginConfig, PluginScope, PluginStage } from '../types' - -const [ElectionCompositionEntry, ElectionCompositionUI] = createCompositionDialog('🎫 Election', (props) => ( - -)) - -export const Election2020PluginDefine: PluginConfig = { - pluginName: 'Election 2020', - identifier: Election2020PluginID, - stage: PluginStage.Production, - scope: PluginScope.Public, - successDecryptionInspector: function Comp(props) { - const payload = Election2020MetadataReader(props.message.meta) - if (!payload.ok) return null - return ( - - }> - - - - ) - }, - postDialogMetadataBadge: new Map([ - [ - Election2020MetaKey, - (payload: Election2020JSONPayload) => { - return `A Election Packet of ${resolveStateName(payload.state)} and ${resolveCandidateName( - payload.winner, - )} is the winner.` - }, - ], - ]), - PageComponent: Flags.election2020_composition_dialog_enabled ? ElectionCompositionUI : undefined, - postDialogEntries: Flags.election2020_composition_dialog_enabled ? [ElectionCompositionEntry] : undefined, -} diff --git a/packages/maskbook/src/plugins/Election2020/election.json b/packages/maskbook/src/plugins/Election2020/election.json deleted file mode 100644 index 5ab9af51cffb..000000000000 --- a/packages/maskbook/src/plugins/Election2020/election.json +++ /dev/null @@ -1,54 +0,0 @@ -[ - { "state_type": 0, "state_id": "AL", "state_name": "Alabama", "votes": 9 }, - { "state_type": 1, "state_id": "AK", "state_name": "Alaska", "votes": 3 }, - { "state_type": 2, "state_id": "AZ", "state_name": "Arizona", "votes": 11 }, - { "state_type": 3, "state_id": "AR", "state_name": "Arkansas", "votes": 6 }, - { "state_type": 4, "state_id": "CA", "state_name": "California", "votes": 55 }, - { "state_type": 5, "state_id": "CO", "state_name": "Colorado", "votes": 9 }, - { "state_type": 6, "state_id": "CT", "state_name": "Connecticut", "votes": 7 }, - { "state_type": 7, "state_id": "DC", "state_name": "Washington, D.C.", "votes": 3 }, - { "state_type": 8, "state_id": "DE", "state_name": "Delaware", "votes": 3 }, - { "state_type": 9, "state_id": "FL", "state_name": "Florida", "votes": 29 }, - { "state_type": 10, "state_id": "GA", "state_name": "Georgia", "votes": 16 }, - { "state_type": 11, "state_id": "HI", "state_name": "Hawaii", "votes": 4 }, - { "state_type": 12, "state_id": "ID", "state_name": "Idaho", "votes": 4 }, - { "state_type": 13, "state_id": "IL", "state_name": "Illinois", "votes": 20 }, - { "state_type": 14, "state_id": "IN", "state_name": "Indiana", "votes": 11 }, - { "state_type": 15, "state_id": "IA", "state_name": "Iowa", "votes": 6 }, - { "state_type": 16, "state_id": "KS", "state_name": "Kansas", "votes": 6 }, - { "state_type": 17, "state_id": "KY", "state_name": "Kentucky", "votes": 8 }, - { "state_type": 18, "state_id": "LA", "state_name": "Louisiana", "votes": 8 }, - { "state_type": 19, "state_id": "ME", "state_name": "Maine", "votes": 4 }, - { "state_type": 20, "state_id": "MD", "state_name": "Maryland", "votes": 10 }, - { "state_type": 21, "state_id": "MA", "state_name": "Massachusetts", "votes": 11 }, - { "state_type": 22, "state_id": "MI", "state_name": "Michigan", "votes": 16 }, - { "state_type": 23, "state_id": "MN", "state_name": "Minnesota", "votes": 10 }, - { "state_type": 24, "state_id": "MS", "state_name": "Mississippi", "votes": 6 }, - { "state_type": 25, "state_id": "MO", "state_name": "Missouri", "votes": 10 }, - { "state_type": 26, "state_id": "MT", "state_name": "Montana", "votes": 3 }, - { "state_type": 27, "state_id": "NE", "state_name": "Nebraska", "votes": 5 }, - { "state_type": 28, "state_id": "NV", "state_name": "Nevada", "votes": 6 }, - { "state_type": 29, "state_id": "NH", "state_name": "New Hampshire", "votes": 4 }, - { "state_type": 30, "state_id": "NJ", "state_name": "New Jersey", "votes": 14 }, - { "state_type": 31, "state_id": "NM", "state_name": "New Mexico", "votes": 5 }, - { "state_type": 32, "state_id": "NY", "state_name": "New York", "votes": 29 }, - { "state_type": 33, "state_id": "NC", "state_name": "North Carolina", "votes": 15 }, - { "state_type": 34, "state_id": "ND", "state_name": "North Dakota", "votes": 3 }, - { "state_type": 35, "state_id": "OH", "state_name": "Ohio", "votes": 18 }, - { "state_type": 36, "state_id": "OK", "state_name": "Oklahoma", "votes": 7 }, - { "state_type": 37, "state_id": "OR", "state_name": "Oregon", "votes": 7 }, - { "state_type": 38, "state_id": "PA", "state_name": "Pennsylvania", "votes": 20 }, - { "state_type": 39, "state_id": "RI", "state_name": "Rhode Island", "votes": 4 }, - { "state_type": 40, "state_id": "SC", "state_name": "South Carolina", "votes": 9 }, - { "state_type": 41, "state_id": "SD", "state_name": "South Dakota", "votes": 3 }, - { "state_type": 42, "state_id": "TN", "state_name": "Tennessee", "votes": 11 }, - { "state_type": 43, "state_id": "TX", "state_name": "Texas", "votes": 38 }, - { "state_type": 44, "state_id": "UT", "state_name": "Utah", "votes": 6 }, - { "state_type": 45, "state_id": "VT", "state_name": "Vermont", "votes": 3 }, - { "state_type": 46, "state_id": "VA", "state_name": "Virginia", "votes": 13 }, - { "state_type": 47, "state_id": "WA", "state_name": "Washington", "votes": 12 }, - { "state_type": 48, "state_id": "WV", "state_name": "West Virginia", "votes": 5 }, - { "state_type": 49, "state_id": "WI", "state_name": "Wisconsin", "votes": 10 }, - { "state_type": 50, "state_id": "WY", "state_name": "Wyoming", "votes": 3 }, - { "state_type": 51, "state_id": "ZONE_51", "state_name": "Zone 51", "votes": 51 } -] diff --git a/packages/maskbook/src/plugins/Election2020/helpers.ts b/packages/maskbook/src/plugins/Election2020/helpers.ts deleted file mode 100644 index 9b4e6083673a..000000000000 --- a/packages/maskbook/src/plugins/Election2020/helpers.ts +++ /dev/null @@ -1,10 +0,0 @@ -import { createTypedMessageMetadataReader, createRenderWithMetadata } from '../../protocols/typed-message/metadata' -import { Election2020MetaKey } from './constants' -import type { Election2020JSONPayload } from './types' -import schema from './schema.json' - -export const Election2020MetadataReader = createTypedMessageMetadataReader( - Election2020MetaKey, - schema, -) -export const renderWithElection2020Metadata = createRenderWithMetadata(Election2020MetadataReader) diff --git a/packages/maskbook/src/plugins/Election2020/hooks/useAllElectionTokensOfOwner.ts b/packages/maskbook/src/plugins/Election2020/hooks/useAllElectionTokensOfOwner.ts deleted file mode 100644 index f03d15111e66..000000000000 --- a/packages/maskbook/src/plugins/Election2020/hooks/useAllElectionTokensOfOwner.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { useERC721TokenIdsOfOwner } from '../../../web3/hooks/useERC721TokensOfOwner' -import type { ERC721TokenDetailed } from '../../../web3/types' - -export function useAllElectionTokensOfOwner(token?: ERC721TokenDetailed) { - const { value: tokenIds, ...result } = useERC721TokenIdsOfOwner(token) - return { - ...result, - value: tokenIds.filter(Boolean).map((tokenId) => ({ - tokenId: tokenId ?? '', - tokenImageURL: tokenId ? `${token?.baseURI}${tokenId}.gif` : '', - })), - } -} diff --git a/packages/maskbook/src/plugins/Election2020/hooks/useAvailability.ts b/packages/maskbook/src/plugins/Election2020/hooks/useAvailability.ts deleted file mode 100644 index 50b120334689..000000000000 --- a/packages/maskbook/src/plugins/Election2020/hooks/useAvailability.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { useAsyncRetry } from 'react-use' -import { useElectionTokenContract } from '../contracts/useElectionTokenContract' -import { resolveStateType } from '../pipes' -import type { US_STATE_TYPE } from '../types' - -export function useAvailability(stateType: US_STATE_TYPE) { - const electionTokenContract = useElectionTokenContract() - const { value, ...result } = useAsyncRetry(async () => { - if (!electionTokenContract) return null - return electionTokenContract.methods.check_availability(resolveStateType(stateType)).call() - }, [stateType]) - return { - value: Number.parseInt(value ?? '0', 10), - ...result, - } -} diff --git a/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokens.ts b/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokens.ts deleted file mode 100644 index 0bd43b5c4df1..000000000000 --- a/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokens.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { hexToNumberString, soliditySha3 } from 'web3-utils' -import type { ElectionToken, US_STATE_TYPE } from '../types' -import ELECTION_STATE_VS_VOTES from '../election.json' -import { resolveStateType } from '../pipes' -import type { ERC721TokenDetailed } from '../../../web3/types' - -export function useElectionTokens(stateType: US_STATE_TYPE, token?: ERC721TokenDetailed): ElectionToken[] { - const state = ELECTION_STATE_VS_VOTES.find((x) => x.state_id === stateType) - if (!state) return [] - if (!token) return [] - return new Array(state.votes).fill(0).map((_, idx) => { - const tokenId_ = soliditySha3({ t: 'uint8', v: resolveStateType(stateType) }, { t: 'uint8', v: idx }) - const tokenId = tokenId_ ? hexToNumberString(tokenId_) : '' - return { - tokenId, - tokenImageURL: tokenId ? `${token.baseURI}${tokenId}.gif` : '', - } - }) -} diff --git a/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokensOfOwner.ts b/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokensOfOwner.ts deleted file mode 100644 index f844241db654..000000000000 --- a/packages/maskbook/src/plugins/Election2020/hooks/useElectionTokensOfOwner.ts +++ /dev/null @@ -1,21 +0,0 @@ -import type { US_STATE_TYPE } from '../types' -import ELECTION_STATE_VS_VOTES from '../election.json' -import { useERC721TokenIdsOfOwner } from '../../../web3/hooks/useERC721TokensOfOwner' -import type { ERC721TokenDetailed } from '../../../web3/types' -import { useElectionTokens } from './useElectionTokens' - -export function useElectionTokensOfOwner(stateType: US_STATE_TYPE, token?: ERC721TokenDetailed) { - const state = ELECTION_STATE_VS_VOTES.find((x) => x.state_id === stateType) - const { value: tokenIds, ...result } = useERC721TokenIdsOfOwner(token) - const totalTokens = useElectionTokens(stateType, token) - - if (!state || !token || !totalTokens.length) - return { - ...result, - value: [], - } - return { - ...result, - value: totalTokens.filter((x) => tokenIds.includes(x.tokenId)), - } -} diff --git a/packages/maskbook/src/plugins/Election2020/hooks/useMintCallback.ts b/packages/maskbook/src/plugins/Election2020/hooks/useMintCallback.ts deleted file mode 100644 index 12dfa3ef0622..000000000000 --- a/packages/maskbook/src/plugins/Election2020/hooks/useMintCallback.ts +++ /dev/null @@ -1,79 +0,0 @@ -import { useCallback, useEffect, useState } from 'react' -import { useTransactionReceipt } from '../../../web3/hooks/useTransaction' -import { TransactionStateType, useTransactionState } from '../../../web3/hooks/useTransactionState' -import { useElectionTokenContract } from '../contracts/useElectionTokenContract' -import { PluginElection2020 } from '../messages' -import { resolveStateType } from '../pipes' -import type { CANDIDATE_TYPE, US_STATE_TYPE } from '../types' - -export function useMintCallback(from: string, stateType: US_STATE_TYPE, candidateType: CANDIDATE_TYPE) { - const [txHash, setTxHash] = useState('') - const [mintState, setMintState] = useTransactionState() - const electionTokenContract = useElectionTokenContract() - - const mintCallback = useCallback(async () => { - if (!electionTokenContract) { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - return - } - - // pre-step: start waiting for provider to confirm tx - setMintState({ - type: TransactionStateType.WAIT_FOR_CONFIRMING, - }) - - // step 1: check remaining - const remaining = await electionTokenContract.methods.check_availability(resolveStateType(stateType)).call() - if (Number.parseInt(remaining || '0', 10) <= 0) { - setMintState({ - type: TransactionStateType.FAILED, - error: new Error('There is none NTF token left.'), - }) - return - } - - // step 2: mint by server - try { - const { mint_transaction_hash } = await PluginElection2020.mintElectionPacket( - from, - stateType, - candidateType, - ) - setTxHash(mint_transaction_hash) - } catch (error) { - setMintState({ - type: TransactionStateType.FAILED, - error, - }) - } - }, [from, stateType, candidateType, electionTokenContract]) - - const resetCallback = useCallback(() => { - setMintState({ - type: TransactionStateType.UNKNOWN, - }) - }, []) - - //#region tracking receipt - const receipt = useTransactionReceipt(txHash) - - useEffect(() => { - if (!receipt) return - if (receipt.status) - setMintState({ - type: TransactionStateType.CONFIRMED, - no: 0, - receipt, - }) - else - setMintState({ - type: TransactionStateType.FAILED, - error: new Error('The contract execution was not successful, check your transaction.'), - }) - }, [receipt]) - //#endregion - - return [mintState, mintCallback, resetCallback] as const -} diff --git a/packages/maskbook/src/plugins/Election2020/messages.ts b/packages/maskbook/src/plugins/Election2020/messages.ts deleted file mode 100644 index 2bdcd4d66274..000000000000 --- a/packages/maskbook/src/plugins/Election2020/messages.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { Election2020PluginID } from './constants' -import { createPluginMessage } from '../utils/createPluginMessage' -import { createPluginRPC } from '../utils/createPluginRPC' - -if (module.hot) module.hot.accept() -const ElectionMessage = createPluginMessage<{ _: unknown }>(Election2020PluginID) -export const PluginElection2020 = createPluginRPC( - Election2020PluginID, - () => import('./services'), - ElectionMessage.events._, -) diff --git a/packages/maskbook/src/plugins/Election2020/pipes.ts b/packages/maskbook/src/plugins/Election2020/pipes.ts deleted file mode 100644 index cd7218334241..000000000000 --- a/packages/maskbook/src/plugins/Election2020/pipes.ts +++ /dev/null @@ -1,43 +0,0 @@ -import { CANDIDATE_TYPE, US_PARTY_TYPE, US_STATE_TYPE } from './types' -import ELECTION_STATE_VS_VOTES from './election.json' - -export function resolveCandidateName(candidateType: CANDIDATE_TYPE) { - if (candidateType === CANDIDATE_TYPE.TRUMP) return 'Donald Trump' - return 'Joe Biden' -} - -export function resolveCandidateBriefName(candidateType: CANDIDATE_TYPE) { - if (candidateType === CANDIDATE_TYPE.TRUMP) return 'Trump' - return 'Biden' -} - -export function resolveCandiateType(candidateType: CANDIDATE_TYPE) { - if (candidateType === CANDIDATE_TYPE.TRUMP) return 0 - return 1 -} - -export function resolveCandidateSNSAccount(network: string, candidateType: CANDIDATE_TYPE) { - switch (network) { - case 'twitter.com': - return candidateType === CANDIDATE_TYPE.TRUMP ? 'realDonaldTrump' : 'JoeBiden' - case 'facebook.com': - return '' - default: - return '' - } -} - -export function resolveCandidatePartyType(candidateType: CANDIDATE_TYPE) { - if (candidateType === CANDIDATE_TYPE.TRUMP) return US_PARTY_TYPE.RED - return US_PARTY_TYPE.BLUE -} - -export function resolveStateName(stateType: US_STATE_TYPE) { - const state = ELECTION_STATE_VS_VOTES.find((x) => x.state_id === stateType) - return state?.state_name ?? stateType -} - -export function resolveStateType(stateType: US_STATE_TYPE) { - const state = ELECTION_STATE_VS_VOTES.find((x) => x.state_id === stateType) - return state?.state_type ?? 0 -} diff --git a/packages/maskbook/src/plugins/Election2020/schema.json b/packages/maskbook/src/plugins/Election2020/schema.json deleted file mode 100644 index 6b04a6c371d6..000000000000 --- a/packages/maskbook/src/plugins/Election2020/schema.json +++ /dev/null @@ -1,67 +0,0 @@ -{ - "type": "object", - "additionalProperties": true, - "properties": { - "state": { - "type": "string", - "title": "state" - }, - "winner": { - "type": "string", - "title": "state" - }, - "sender": { - "type": "object", - "additionalProperties": true, - "properties": { - "address": { - "type": "string", - "title": "address" - }, - "name": { - "type": "string", - "title": "name" - }, - "message": { - "type": "string", - "title": "message" - } - }, - "required": ["address", "message", "name"], - "title": "sender" - }, - "nft_token": { - "type": "object", - "additionalProperties": true, - "properties": { - "address": { - "description": "token address", - "type": "string", - "title": "address" - }, - "name": { - "description": "token name", - "type": "string", - "title": "name" - }, - "symbol": { - "description": "token symbol", - "type": "string", - "title": "symbol" - } - }, - "required": ["address", "name", "symbol"], - "title": "token" - }, - "creation_time": { - "type": "number", - "title": "creation_time" - }, - "network": { - "enum": ["Mainnet", "Rinkeby", "Ropsten"], - "type": "string", - "title": "network" - } - }, - "required": ["state", "winner", "sender", "creation_time", "network"] -} diff --git a/packages/maskbook/src/plugins/Election2020/services.ts b/packages/maskbook/src/plugins/Election2020/services.ts deleted file mode 100644 index 87e7abe5a325..000000000000 --- a/packages/maskbook/src/plugins/Election2020/services.ts +++ /dev/null @@ -1,42 +0,0 @@ -import * as jwt from 'jsonwebtoken' -import { sha3 } from 'web3-utils' -import Services from '../../extension/service' -import { resolveChainName } from '../../web3/pipes' -import { resolveCandiateType, resolveStateType } from './pipes' -import type { CANDIDATE_TYPE, US_STATE_TYPE } from './types' - -export async function mintElectionPacket( - from: string, - stateType: US_STATE_TYPE, - candidateType: CANDIDATE_TYPE, -): Promise<{ mint_transaction_hash: string }> { - const host = 'https://redpacket.gives' - const x = 'a3323cd1-fa42-44cd-b053-e474365ab3da' - - const chainId = await Services.Ethereum.getChainId(from) - const network = resolveChainName(chainId).toLowerCase() - - // skip hi - const auth = await fetch(`${host}/hi?id=${from}&network=${network}`) - if (!auth.ok) throw new Error('Auth failed') - - const verify = await auth.text() - const jwt_encoded: { - state: number - winner: number - recipient: string - validation: string - signature: string - } = { - state: resolveStateType(stateType), - winner: resolveCandiateType(candidateType), - recipient: from, - validation: sha3(from)!, - signature: await Services.Ethereum.sign(verify, from, chainId), - } - const mintResponse = await fetch( - `${host}/mrpresident?payload=${jwt.sign(jwt_encoded, x, { algorithm: 'HS256' })}&network=${network}`, - ) - if (!mintResponse.ok) throw new Error('Claim failed') - return { mint_transaction_hash: await mintResponse.text() } -} diff --git a/packages/maskbook/src/plugins/Election2020/types.ts b/packages/maskbook/src/plugins/Election2020/types.ts deleted file mode 100644 index 493dd30e3c63..000000000000 --- a/packages/maskbook/src/plugins/Election2020/types.ts +++ /dev/null @@ -1,88 +0,0 @@ -import type { EthereumNetwork } from '../../web3/types' - -export enum CANDIDATE_TYPE { - TRUMP = 'Trump', - BIDEN = 'Biden', -} - -export enum US_PARTY_TYPE { - RED = 'RED', - BLUE = 'BLUE', -} - -export enum US_STATE_TYPE { - AL = 'AL', - AK = 'AK', - AZ = 'AZ', - AR = 'AR', - CA = 'CA', - CO = 'CO', - CT = 'CT', - DE = 'DE', - FL = 'FL', - GA = 'GA', - HI = 'HI', - ID = 'ID', - IL = 'IL', - IN = 'IN', - IA = 'IA', - KS = 'KS', - KY = 'KY', - LA = 'LA', - ME = 'ME', - MD = 'MD', - MA = 'MA', - MI = 'MI', - MN = 'MN', - MS = 'MS', - MO = 'MO', - MT = 'MT', - NE = 'NE', - NV = 'NV', - NH = 'NH', - NJ = 'NJ', - NM = 'NM', - NY = 'NY', - NC = 'NC', - ND = 'ND', - OH = 'OH', - OK = 'OK', - OR = 'OR', - PA = 'PA', - RI = 'RI', - SC = 'SC', - SD = 'SD', - TN = 'TN', - TX = 'TX', - UT = 'UT', - VT = 'VT', - VA = 'VA', - WA = 'WA', - WV = 'WV', - WI = 'WI', - WY = 'WY', - DC = 'DC', - ZONE_51 = 'ZONE_51', -} - -export interface ElectionToken { - tokenId: string // sha3(state + id) - tokenImageURL: string -} - -export interface Election2020JSONPayload { - state: US_STATE_TYPE - winner: CANDIDATE_TYPE - sender: { - address: string - name: string - message: string - } - ntf_token: { - address: string - name: string - symbol: string - } - creation_time: number - network: EthereumNetwork -} diff --git a/packages/maskbook/src/plugins/Election2020/victory.json b/packages/maskbook/src/plugins/Election2020/victory.json deleted file mode 100644 index 2f2ed9b57d2a..000000000000 --- a/packages/maskbook/src/plugins/Election2020/victory.json +++ /dev/null @@ -1,4 +0,0 @@ -{ - "vote_for_trump_start_at": 20201001, - "vote_for_biden_start_at": 20202001 -} diff --git a/packages/maskbook/src/plugins/Ethereum/README.md b/packages/maskbook/src/plugins/Ethereum/README.md new file mode 100644 index 000000000000..41d6a06bfc79 --- /dev/null +++ b/packages/maskbook/src/plugins/Ethereum/README.md @@ -0,0 +1 @@ +## Plugin: Ethereum Wallet diff --git a/packages/maskbook/src/plugins/Ethereum/UI/SelectTokenDialog.tsx b/packages/maskbook/src/plugins/Ethereum/UI/SelectTokenDialog.tsx index 24b5ebb4f6be..2d845c59db27 100644 --- a/packages/maskbook/src/plugins/Ethereum/UI/SelectTokenDialog.tsx +++ b/packages/maskbook/src/plugins/Ethereum/UI/SelectTokenDialog.tsx @@ -30,16 +30,7 @@ const useStyles = makeStyles((theme: Theme) => }), ) -export interface SelectTokenDialogProps extends withClasses { - open?: boolean - includeTokens?: string[] - excludeTokens?: string[] - selectedTokens?: string[] - tokens?: (ERC20TokenDetailed | EtherTokenDetailed)[] - disableSearchBar?: boolean - onSubmit?: (token: ERC20TokenDetailed) => void - onClose?: () => void -} +export interface SelectTokenDialogProps extends withClasses {} export function SelectTokenDialog(props: SelectTokenDialogProps) { const { t } = useI18N() diff --git a/packages/maskbook/src/plugins/FileService/MainDialog.tsx b/packages/maskbook/src/plugins/FileService/MainDialog.tsx index ea81a69250a2..fea6f782f809 100644 --- a/packages/maskbook/src/plugins/FileService/MainDialog.tsx +++ b/packages/maskbook/src/plugins/FileService/MainDialog.tsx @@ -5,7 +5,7 @@ import { useState } from 'react' import { useBeforeUnload } from 'react-use' import { useStylesExtends } from '../../components/custom-ui-helper' import { InjectedDialog } from '../../components/shared/InjectedDialog' -import { editActivatedPostMetadata } from '../../social-network/ui' +import { editActivatedPostMetadata } from '../../protocols/typed-message/global-state' import { useI18N } from '../../utils/i18n-next-ui' import { Entry } from './components' import { META_KEY_2 } from './constants' diff --git a/packages/maskbook/src/plugins/Gitcoin/UI/DonateDialog.tsx b/packages/maskbook/src/plugins/Gitcoin/UI/DonateDialog.tsx index 40f79df06ff3..230bae4807f2 100644 --- a/packages/maskbook/src/plugins/Gitcoin/UI/DonateDialog.tsx +++ b/packages/maskbook/src/plugins/Gitcoin/UI/DonateDialog.tsx @@ -17,11 +17,10 @@ import { TransactionStateType } from '../../../web3/hooks/useTransactionState' import { InjectedDialog } from '../../../components/shared/InjectedDialog' import { SelectTokenDialogEvent, WalletMessages } from '../../Wallet/messages' import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { usePostLink } from '../../../components/DataSource/usePostInfo' import { Flags } from '../../../utils/flags' import { useEtherTokenDetailed } from '../../../web3/hooks/useEtherTokenDetailed' -import { getActivatedUI } from '../../../social-network/ui' +import { activatedSocialNetworkUI } from '../../../social-network' import { PluginGitcoinMessages } from '../messages' import { EthereumMessages } from '../../Ethereum/messages' import { useTokenBalance } from '../../../web3/hooks/useTokenBalance' @@ -29,6 +28,7 @@ import { EthereumWalletConnectedBoundary } from '../../../web3/UI/EthereumWallet import { EthereumERC20TokenApprovedBoundary } from '../../../web3/UI/EthereumERC20TokenApprovedBoundary' import { useConstant } from '../../../web3/hooks/useConstant' import { GITCOIN_CONSTANT } from '../constants' +import { isTwitter } from '../../../social-network-adaptor/twitter.com/base' const useStyles = makeStyles((theme) => createStyles({ @@ -135,21 +135,23 @@ export function DonateDialog(props: DonateDialogProps) { //#endregion //#region transaction dialog - const cashTag = getActivatedUI()?.networkIdentifier === 'twitter.com' ? '$' : '' + const cashTag = isTwitter(activatedSocialNetworkUI) ? '$' : '' const postLink = usePostLink() - const shareLink = useShareLink( - token - ? [ - `I just donated ${title} with ${formatBalance( - amount, - token.decimals ?? 0, - token.decimals ?? 0, - )} ${cashTag}${token.symbol}. Follow @realMaskbook (mask.io) to donate Gitcoin grants.`, - '#mask_io', - postLink, - ].join('\n') - : '', - ) + const shareLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + token + ? [ + `I just donated ${title} with ${formatBalance( + amount, + token.decimals ?? 0, + token.decimals ?? 0, + )} ${cashTag}${token.symbol}. Follow @realMaskbook (mask.io) to donate Gitcoin grants.`, + '#mask_io', + postLink, + ].join('\n') + : '', + ) + .toString() // close the transaction dialog const [_, setTransactionDialogOpen] = useRemoteControlledDialog( diff --git a/packages/maskbook/src/plugins/Gitcoin/define.tsx b/packages/maskbook/src/plugins/Gitcoin/define.tsx index 3877167a9021..13f389cd4281 100644 --- a/packages/maskbook/src/plugins/Gitcoin/define.tsx +++ b/packages/maskbook/src/plugins/Gitcoin/define.tsx @@ -9,7 +9,7 @@ import { GITCOIN_PLUGIN_ID } from './constants' import { DonateDialog } from './UI/DonateDialog' import { PreviewCard } from './UI/PreviewCard' -const isGitcoin = (x: string): boolean => x.startsWith('https://gitcoin.co/grants') +const isGitcoin = (x: string): boolean => /^https:\/\/gitcoin.co\/grants\/\d+/.test(x) export const GitcoinPluginDefine: PluginConfig = { pluginName: 'Gitcoin', diff --git a/packages/maskbook/src/plugins/ITO/UI/ClaimGuide.tsx b/packages/maskbook/src/plugins/ITO/UI/ClaimGuide.tsx index 78fca697f17b..1a0710d1cc40 100644 --- a/packages/maskbook/src/plugins/ITO/UI/ClaimGuide.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/ClaimGuide.tsx @@ -32,7 +32,7 @@ const useStyles = makeStyles((theme) => interface ClaimGuideProps extends Pick { open: boolean status: ClaimStatus - shareSuccessLink: string + shareSuccessLink: string | undefined isBuyer: boolean retryPayload: () => void onUpdate: (status: ClaimStatus) => void diff --git a/packages/maskbook/src/plugins/ITO/UI/CompositionDialog.tsx b/packages/maskbook/src/plugins/ITO/UI/CompositionDialog.tsx index bb0324220d6f..30777ec83146 100644 --- a/packages/maskbook/src/plugins/ITO/UI/CompositionDialog.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/CompositionDialog.tsx @@ -6,7 +6,7 @@ import { ITO_MetaKey } from '../constants' import { JSON_PayloadInMask, DialogTabs } from '../types' import { useI18N } from '../../../utils/i18n-next-ui' import AbstractTab, { AbstractTabProps } from '../../../extension/options-page/DashboardComponents/AbstractTab' -import { editActivatedPostMetadata } from '../../../social-network/ui' +import { editActivatedPostMetadata } from '../../../protocols/typed-message/global-state' import { CreateGuide } from './CreateGuide' import { payloadOutMask } from '../helpers' import { PoolList } from './PoolList' diff --git a/packages/maskbook/src/plugins/ITO/UI/CreateForm.tsx b/packages/maskbook/src/plugins/ITO/UI/CreateForm.tsx index 60054e4a2d52..765d51be9cb7 100644 --- a/packages/maskbook/src/plugins/ITO/UI/CreateForm.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/CreateForm.tsx @@ -76,7 +76,7 @@ export function CreateForm(props: CreateFormProps) { const currentIdentity = useCurrentIdentity() const senderName = currentIdentity?.identifier.userId ?? currentIdentity?.linkedPersona?.nickname ?? 'Unknown User' - const [isMask, setIsMask] = useState(true) + const [isMask, setIsMask] = useState(Flags.mask_ito_enabled ? true : false) const [message, setMessage] = useState(origin?.title ?? '') const [totalOfPerWallet, setTotalOfPerWallet] = useState( new BigNumber(origin?.limit || '0').isZero() diff --git a/packages/maskbook/src/plugins/ITO/UI/ITO.tsx b/packages/maskbook/src/plugins/ITO/UI/ITO.tsx index 891662a1c09f..5ac35496adbe 100644 --- a/packages/maskbook/src/plugins/ITO/UI/ITO.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/ITO.tsx @@ -20,7 +20,6 @@ import { formatDateTime } from '../../../utils/date' import { getTextUILength } from '../../../utils/getTextUILength' import { ClaimGuide, ClaimStatus } from './ClaimGuide' import { usePostLink } from '../../../components/DataSource/usePostInfo' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { TokenIcon } from '../../../extension/options-page/DashboardComponents/TokenIcon' import { sortTokens } from '../helpers' import { ITO_EXCHANGE_RATION_MAX, TIME_WAIT_BLOCKCHAIN } from '../constants' @@ -30,6 +29,7 @@ import { getAssetAsBlobURL } from '../../../utils/suspends/getAssetAsBlobURL' import { EthereumMessages } from '../../Ethereum/messages' import { usePoolPayload } from '../hooks/usePoolPayload' import Services from '../../../extension/service' +import { activatedSocialNetworkUI } from '../../../social-network' export interface IconProps { size?: number @@ -257,13 +257,15 @@ export function ITO(props: ITO_Props) { const isBuyer = chainId === payload.chain_id && payload.buyers.map((val) => val.address.toLowerCase()).includes(account.toLowerCase()) - const shareSuccessLink = useShareLink( - t('plugin_ito_claim_success_share', { - user: seller.name, - link: postLink, - symbol: token.symbol, - }), - ) + const shareSuccessLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + t('plugin_ito_claim_success_share', { + user: seller.name, + link: postLink, + symbol: token.symbol, + }), + ) + .toString() const canWithdraw = useMemo( () => isAccountSeller && !tradeInfo?.destructInfo && (listOfStatus.includes(ITO_Status.expired) || noRemain), [tradeInfo, listOfStatus, isAccountSeller, noRemain], @@ -284,13 +286,15 @@ export function ITO(props: ITO_Props) { }, [shareSuccessLink]) //#endregion - const shareLink = useShareLink( - t('plugin_ito_claim_foreshow_share', { - link: postLink, - name: token.name, - symbol: token.symbol ?? 'token', - }), - ) + const shareLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + t('plugin_ito_claim_foreshow_share', { + link: postLink, + name: token.name, + symbol: token.symbol ?? 'token', + }), + ) + .toString() const onShare = useCallback(async () => { window.open(shareLink, '_blank', 'noopener noreferrer') }, [shareLink]) @@ -595,15 +599,17 @@ export function ITO(props: ITO_Props) { {t('plugin_ito_unlock_in_advance')} - - - {t('plugin_ito_share')} - - + {shareLink ? ( + + + {t('plugin_ito_share')} + + + ) : undefined} ) : listOfStatus.includes(ITO_Status.started) ? ( diff --git a/packages/maskbook/src/plugins/ITO/UI/ITO_Card.tsx b/packages/maskbook/src/plugins/ITO/UI/ITO_Card.tsx index 03be2ab5c33c..ff55da54d08a 100644 --- a/packages/maskbook/src/plugins/ITO/UI/ITO_Card.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/ITO_Card.tsx @@ -5,9 +5,9 @@ import { useCallback, useEffect } from 'react' import { useStylesExtends } from '../../../components/custom-ui-helper' import { usePostLink } from '../../../components/DataSource/usePostInfo' import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' -import { getActivatedUI } from '../../../social-network/ui' +import { activatedSocialNetworkUI } from '../../../social-network' +import { isTwitter } from '../../../social-network-adaptor/twitter.com/base' import { useRemoteControlledDialog } from '../../../utils/hooks/useRemoteControlledDialog' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { TransactionStateType } from '../../../web3/hooks/useTransactionState' import type { ERC20TokenDetailed } from '../../../web3/types' import { EthereumMessages } from '../../Ethereum/messages' @@ -73,19 +73,21 @@ export function ITO_Card(props: ITO_CardProps) { //#endregion //#region transaction dialog - const cashTag = getActivatedUI()?.networkIdentifier === 'twitter.com' ? '$' : '' + const cashTag = isTwitter(activatedSocialNetworkUI) ? '$' : '' const postLink = usePostLink() - const shareLink = useShareLink( - [ - `I just claimed ${cashTag}${token?.symbol} with ${formatBalance( - new BigNumber(packet?.claimable ?? '0'), - 18, - 6, - )}. Follow @realMaskbook (mask.io) to claim airdrop.`, - '#mask_io', - postLink, - ].join('\n'), - ) + const shareLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + [ + `I just claimed ${cashTag}${token?.symbol} with ${formatBalance( + new BigNumber(packet?.claimable ?? '0'), + 18, + 6, + )}. Follow @realMaskbook (mask.io) to claim airdrop.`, + '#mask_io', + postLink, + ].join('\n'), + ) + .toString() // close the transaction dialog const [_, setTransactionDialogOpen] = useRemoteControlledDialog( diff --git a/packages/maskbook/src/plugins/ITO/UI/ShareDialog.tsx b/packages/maskbook/src/plugins/ITO/UI/ShareDialog.tsx index d2f1a0bdc591..616986734b14 100644 --- a/packages/maskbook/src/plugins/ITO/UI/ShareDialog.tsx +++ b/packages/maskbook/src/plugins/ITO/UI/ShareDialog.tsx @@ -7,7 +7,6 @@ import { useStylesExtends } from '../../../components/custom-ui-helper' import { useI18N } from '../../../utils/i18n-next-ui' import { formatBalance } from '../../../plugins/Wallet/formatter' import { usePostLink } from '../../../components/DataSource/usePostInfo' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { getAssetAsBlobURL } from '../../../utils/suspends/getAssetAsBlobURL' import { useCallback } from 'react' @@ -59,7 +58,7 @@ const useStyles = makeStyles((theme) => ) export interface ShareDialogProps extends withClasses<'root'> { - shareSuccessLink: string + shareSuccessLink: string | undefined token: EtherTokenDetailed | ERC20TokenDetailed actualSwapAmount: BigNumber poolName: string @@ -91,13 +90,15 @@ export function ShareDialog(props: ShareDialogProps) { {actualSwapAmount.isZero() ? t('plugin_ito_out_of_stock_hit') : t('plugin_ito_congratulations')} - - {t('plugin_ito_dialog_claim_share_title')} - + {shareSuccessLink ? ( + + {t('plugin_ito_dialog_claim_share_title')} + + ) : null} diff --git a/packages/maskbook/src/plugins/MaskbookPluginWrapper.tsx b/packages/maskbook/src/plugins/MaskbookPluginWrapper.tsx index 4bef2417c3f7..25ba4164e08d 100644 --- a/packages/maskbook/src/plugins/MaskbookPluginWrapper.tsx +++ b/packages/maskbook/src/plugins/MaskbookPluginWrapper.tsx @@ -1,7 +1,8 @@ import { makeStyles, Typography, ThemeProvider, SnackbarContent } from '@material-ui/core' -import { getActivatedUI } from '../social-network/ui' +import { activatedSocialNetworkUI } from '../social-network' import { MaskbookIcon } from '../resources/MaskbookIcon' -import { Suspense } from 'react' +import { Suspense, useRef } from 'react' +import { isTwitter } from '../social-network-adaptor/twitter.com/base' interface PluginWrapperProps { pluginName: string @@ -10,7 +11,6 @@ interface PluginWrapperProps { } const useStyles = makeStyles((theme) => { - const network = getActivatedUI()?.networkIdentifier return { card: { marginTop: theme.spacing(1), @@ -18,7 +18,7 @@ const useStyles = makeStyles((theme) => { boxSizing: 'border-box', border: `1px solid ${theme.palette.divider}`, cursor: 'default', - ...(network === 'twitter.com' + ...(isTwitter(activatedSocialNetworkUI) ? { borderRadius: 15, overflow: 'hidden', @@ -51,20 +51,25 @@ const useStyles = makeStyles((theme) => { export default function MaskbookPluginWrapper(props: PluginWrapperProps) { const classes = useStyles() const { pluginName, children } = props + const useStableTheme = useRef(activatedSocialNetworkUI.customization.useTheme).current + // eslint-disable-next-line react-hooks/rules-of-hooks + const theme = useStableTheme?.() + + const inner = ( +
ev.stopPropagation()}> +
+ +
+ Mask Plugin + {pluginName} +
+
+
{children}
+
+ ) return ( }> - -
ev.stopPropagation()}> -
- -
- Mask Plugin - {pluginName} -
-
-
{children}
-
-
+ {theme ? {inner} : inner}
) } diff --git a/packages/maskbook/src/plugins/PluginSerivce.ts b/packages/maskbook/src/plugins/PluginSerivce.ts index 04d224b68411..1a794cf4eda7 100644 --- a/packages/maskbook/src/plugins/PluginSerivce.ts +++ b/packages/maskbook/src/plugins/PluginSerivce.ts @@ -16,7 +16,5 @@ import './Polls/utils' import './Transak/messages' import './Trader/messages' import './Polls/utils' -import './Election2020/messages' -import './COTM/messages' import './ITO/messages' import './Airdrop/messages' diff --git a/packages/maskbook/src/plugins/PluginUI.ts b/packages/maskbook/src/plugins/PluginUI.ts index d5e8d15c11d1..b02123fea2db 100644 --- a/packages/maskbook/src/plugins/PluginUI.ts +++ b/packages/maskbook/src/plugins/PluginUI.ts @@ -13,9 +13,7 @@ import { PollsPluginDefine } from './Polls/define' import { StorybookPluginDefine } from './Storybook/define' import { FileServicePluginDefine } from './FileService/UI-define' import { TraderPluginDefine } from './Trader/define' -import { Election2020PluginDefine } from './Election2020/define' import { TransakPluginDefine } from './Transak/define' -import { COTM_PluginDefine } from './COTM/define' import { ITO_PluginDefine } from './ITO/define' import { NFTPluginsDefine } from './NFT/define' import { AirdropPluginDefine } from './Airdrop/define' @@ -32,8 +30,6 @@ sideEffect.then(() => { if (Flags.poll_enabled) plugins.add(PollsPluginDefine) if (Flags.trader_enabled) plugins.add(TraderPluginDefine) if (Flags.transak_enabled) plugins.add(TransakPluginDefine) - if (Flags.election2020_enabled) plugins.add(Election2020PluginDefine) - if (Flags.COTM_enabled) plugins.add(COTM_PluginDefine) if (Flags.airdrop_enabled) plugins.add(AirdropPluginDefine) if (process.env.STORYBOOK) plugins.add(StorybookPluginDefine) }) diff --git a/packages/maskbook/src/plugins/Polls/UI/PollsDialog.tsx b/packages/maskbook/src/plugins/Polls/UI/PollsDialog.tsx index a114ad5f730f..0199252c5c31 100644 --- a/packages/maskbook/src/plugins/Polls/UI/PollsDialog.tsx +++ b/packages/maskbook/src/plugins/Polls/UI/PollsDialog.tsx @@ -20,7 +20,7 @@ import { add as addDate } from 'date-fns' import { PortalShadowRoot } from '../../../utils/shadow-root/ShadowRootPortal' import { useStylesExtends } from '../../../components/custom-ui-helper' import AbstractTab, { AbstractTabProps } from '../../../extension/options-page/DashboardComponents/AbstractTab' -import { editActivatedPostMetadata } from '../../../social-network/ui' +import { editActivatedPostMetadata } from '../../../protocols/typed-message/global-state' import { useCurrentIdentity } from '../../../components/DataSource/useActivatedUI' import type { PollGunDB } from '../Services' import { PollCardUI } from './Polls' diff --git a/packages/maskbook/src/plugins/RedPacket/UI/RedPacket.tsx b/packages/maskbook/src/plugins/RedPacket/UI/RedPacket.tsx index 4ffc3d7f3129..fe76073f3282 100644 --- a/packages/maskbook/src/plugins/RedPacket/UI/RedPacket.tsx +++ b/packages/maskbook/src/plugins/RedPacket/UI/RedPacket.tsx @@ -15,7 +15,6 @@ import { WalletMessages } from '../../Wallet/messages' import { useAvailabilityComputed } from '../hooks/useAvailabilityComputed' import { formatBalance } from '../../Wallet/formatter' import { TransactionStateType } from '../../../web3/hooks/useTransactionState' -import { useShareLink } from '../../../utils/hooks/useShareLink' import { useChainId, useChainIdValid } from '../../../web3/hooks/useChainState' import { useAccount } from '../../../web3/hooks/useAccount' import ActionButton from '../../../extension/options-page/DashboardComponents/ActionButton' @@ -31,6 +30,7 @@ import { MetaMaskIcon } from '../../../resources/MetaMaskIcon' import Services from '../../../extension/service' import { useTokenDetailed } from '../../../web3/hooks/useTokenDetailed' import { EthereumMessages } from '../../Ethereum/messages' +import { activatedSocialNetworkUI } from '../../../social-network' const useStyles = makeStyles((theme) => createStyles({ @@ -173,7 +173,7 @@ export function RedPacket(props: RedPacketProps) { //#region remote controlled transaction dialog const postLink = usePostLink() - const shareLink = useShareLink( + const shareLink = activatedSocialNetworkUI.utils.getShareLinkURL?.( canClaim ? [ `I just claimed a red packet from @${payload.sender.name}. Follow @realMaskbook (mask.io) to claim red packets.`, diff --git a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketDialog.tsx b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketDialog.tsx index f8989986eea3..3a5572e27691 100644 --- a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketDialog.tsx +++ b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketDialog.tsx @@ -2,7 +2,7 @@ import { useState, useCallback } from 'react' import { DialogContent } from '@material-ui/core' import AbstractTab, { AbstractTabProps } from '../../../extension/options-page/DashboardComponents/AbstractTab' import { RedPacketJSONPayload, DialogTabs } from '../types' -import { editActivatedPostMetadata } from '../../../social-network/ui' +import { editActivatedPostMetadata } from '../../../protocols/typed-message/global-state' import { RedPacketMetaKey } from '../constants' import { useI18N } from '../../../utils/i18n-next-ui' import { RedPacketForm } from './RedPacketForm' @@ -59,7 +59,7 @@ export default function RedPacketDialog(props: RedPacketDialogProps) { return ( - + ) diff --git a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketForm.tsx b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketForm.tsx index cf8b977d6c7f..c051b60f3653 100644 --- a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketForm.tsx +++ b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketForm.tsx @@ -62,7 +62,7 @@ const useStyles = makeStyles((theme) => padding: 12, }, selectShrinkLabel: { - transform: 'translate(17px, -12px) scale(0.75) !important', + transform: 'translate(17px, -10px) scale(0.75) !important', }, inputShrinkLabel: { transform: 'translate(17px, -3px) scale(0.75) !important', diff --git a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketInPost.tsx b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketInPost.tsx index 5ad98c9aebeb..bed6150ad4cb 100644 --- a/packages/maskbook/src/plugins/RedPacket/UI/RedPacketInPost.tsx +++ b/packages/maskbook/src/plugins/RedPacket/UI/RedPacketInPost.tsx @@ -1,9 +1,9 @@ import { useEffect } from 'react' import type { RedPacketJSONPayload, RedPacketRecord } from '../types' import { usePostInfoDetails } from '../../../components/DataSource/usePostInfo' -import { getPostUrl } from '../../../social-network/utils/getPostUrl' import { RedPacket } from './RedPacket' import { RedPacketRPC } from '../messages' +import { activatedSocialNetworkUI } from '../../../social-network' export interface RedPacketInPostProps { payload: RedPacketJSONPayload @@ -14,7 +14,10 @@ export function RedPacketInPost(props: RedPacketInPostProps) { //#region discover red packet const postIdentifier = usePostInfoDetails('postIdentifier') - const fromUrl = postIdentifier && !postIdentifier.isUnknown ? getPostUrl(postIdentifier) : undefined + const fromUrl = + postIdentifier && !postIdentifier.isUnknown + ? activatedSocialNetworkUI.utils.getPostURL?.(postIdentifier)?.toString() + : undefined useEffect(() => { if (!fromUrl) return if (!payload.txid && payload.contract_version !== 1) return diff --git a/packages/maskbook/src/plugins/Trader/UI/trader/DataProviderIcon.tsx b/packages/maskbook/src/plugins/Trader/UI/trader/DataProviderIcon.tsx index 6efad20e856d..ee458c42f921 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trader/DataProviderIcon.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trader/DataProviderIcon.tsx @@ -37,7 +37,7 @@ export function DataProviderIcon(props: DataProviderIconProps) { return case DataProvider.COIN_MARKET_CAP: return - case DataProvider.UNISWAP: + case DataProvider.UNISWAP_INFO: return default: unreachable(props.provider) diff --git a/packages/maskbook/src/plugins/Trader/UI/trader/Trader.tsx b/packages/maskbook/src/plugins/Trader/UI/trader/Trader.tsx index e3bd2af659b8..cd936814d863 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trader/Trader.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trader/Trader.tsx @@ -19,7 +19,6 @@ import { TRADE_CONSTANTS } from '../../constants' import { delay } from '../../../../utils/utils' import { TransactionStateType } from '../../../../web3/hooks/useTransactionState' import { useRemoteControlledDialog } from '../../../../utils/hooks/useRemoteControlledDialog' -import { useShareLink } from '../../../../utils/hooks/useShareLink' import { formatBalance } from '../../../Wallet/formatter' import { TradePairViewer } from '../uniswap/TradePairViewer' import { useValueRef } from '../../../../utils/hooks/useValueRef' @@ -27,7 +26,7 @@ import { currentTradeProviderSettings } from '../../settings' import { useTradeCallback } from '../../trader/useTradeCallback' import { useTradeStateComputed } from '../../trader/useTradeStateComputed' import { useTokenBalance } from '../../../../web3/hooks/useTokenBalance' -import { getActivatedUI } from '../../../../social-network/ui' +import { activatedSocialNetworkUI } from '../../../../social-network' import { EthereumMessages } from '../../../Ethereum/messages' import Services from '../../../../extension/service' import { UST } from '../../constants' @@ -35,6 +34,7 @@ import { SelectTokenDialogEvent, WalletMessages } from '../../../Wallet/messages import { useChainId } from '../../../../web3/hooks/useChainState' import { createERC20Token, createEtherToken } from '../../../../web3/helpers' import { PluginTraderRPC } from '../../messages' +import { isTwitter } from '../../../../social-network-adaptor/twitter.com/base' const useStyles = makeStyles((theme) => { return createStyles({ @@ -229,19 +229,21 @@ export function Trader(props: TraderProps) { //#endregion //#region remote controlled transaction dialog - const cashTag = getActivatedUI()?.networkIdentifier === 'twitter.com' ? '$' : '' - const shareLink = useShareLink( - trade && inputToken && outputToken - ? [ - `I just swapped ${formatBalance(trade.inputAmount, inputToken.decimals ?? 0, 6)} ${cashTag}${ - inputToken.symbol - } for ${formatBalance(trade.outputAmount, outputToken.decimals ?? 0, 6)} ${cashTag}${ - outputToken.symbol - }. Follow @realMaskbook (mask.io) to swap cryptocurrencies on Twitter.`, - '#mask_io', - ].join('\n') - : '', - ) + const cashTag = isTwitter(activatedSocialNetworkUI) ? '$' : '' + const shareLink = activatedSocialNetworkUI.utils + .getShareLinkURL?.( + trade && inputToken && outputToken + ? [ + `I just swapped ${formatBalance(trade.inputAmount, inputToken.decimals ?? 0, 6)} ${cashTag}${ + inputToken.symbol + } for ${formatBalance(trade.outputAmount, outputToken.decimals ?? 0, 6)} ${cashTag}${ + outputToken.symbol + }. Follow @realMaskbook (mask.io) to swap cryptocurrencies on Twitter.`, + '#mask_io', + ].join('\n') + : '', + ) + .toString() // close the transaction dialog const [_, setTransactionDialogOpen] = useRemoteControlledDialog( diff --git a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketPanel.tsx b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketPanel.tsx index cef4296eda85..9b30581d17eb 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketPanel.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketPanel.tsx @@ -1,5 +1,5 @@ import { makeStyles, createStyles } from '@material-ui/core' -import { DataProvider, Trending } from '../../types' +import type { DataProvider, Trending } from '../../types' import { CoinMarketTable } from './CoinMarketTable' import { CoinMetadataTable } from './CoinMetadataTable' @@ -24,9 +24,7 @@ export function CoinMarketPanel(props: CoinMarketPanelProps) {

- {dataProvider !== DataProvider.UNISWAP ? ( - - ) : null} +
) } diff --git a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketTable.tsx b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketTable.tsx index a07dfe7bb9bf..c285a13bfb04 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketTable.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMarketTable.tsx @@ -1,16 +1,16 @@ import { - makeStyles, createStyles, - TableContainer, + makeStyles, Paper, Table, - TableRow, + TableBody, TableCell, + TableContainer, TableHead, - TableBody, + TableRow, Typography, } from '@material-ui/core' -import type { DataProvider, Trending } from '../../types' +import { DataProvider, Trending } from '../../types' import { formatCurrency } from '../../../Wallet/formatter' const useStyles = makeStyles((theme) => @@ -43,7 +43,7 @@ export interface CoinMarketTableProps { } export function CoinMarketTable(props: CoinMarketTableProps) { - const { trending } = props + const { trending, dataProvider } = props const classes = useStyles() return ( @@ -51,42 +51,54 @@ export function CoinMarketTable(props: CoinMarketTableProps) { - - - Market Cap - - + {dataProvider !== DataProvider.UNISWAP_INFO ? ( + + + Market Cap + + + ) : null} Volume (24h) - - - Circulating Supply - - - - - Total Supply - - + {dataProvider !== DataProvider.UNISWAP_INFO ? ( + <> + + + Circulating Supply + + + + + Total Supply + + + + ) : null} - - {formatCurrency(trending.market?.market_cap ?? 0, '$')} USD - + {dataProvider !== DataProvider.UNISWAP_INFO ? ( + + {formatCurrency(trending.market?.market_cap ?? 0, '$')} USD + + ) : null} {formatCurrency(trending.market?.total_volume ?? 0, '$')} USD - - {formatCurrency(trending.market?.circulating_supply ?? 0, '$')} USD - - - {formatCurrency(trending.market?.total_supply ?? 0, '$')} USD - + {dataProvider !== DataProvider.UNISWAP_INFO ? ( + <> + + {formatCurrency(trending.market?.circulating_supply ?? 0, '$')} USD + + + {formatCurrency(trending.market?.total_supply ?? 0, '$')} USD + + + ) : null}
diff --git a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMenu.tsx b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMenu.tsx index 70a98d6b376f..03d5b53bf053 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trending/CoinMenu.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trending/CoinMenu.tsx @@ -41,7 +41,11 @@ export function CoinMenu(props: CoinMenuProps) { return ( <>
{children}
- + {options.map((x, i) => ( onSelect(x)}> diff --git a/packages/maskbook/src/plugins/Trader/UI/trending/CoinSaftyAlert.tsx b/packages/maskbook/src/plugins/Trader/UI/trending/CoinSaftyAlert.tsx new file mode 100644 index 000000000000..fa1238297aa0 --- /dev/null +++ b/packages/maskbook/src/plugins/Trader/UI/trending/CoinSaftyAlert.tsx @@ -0,0 +1,66 @@ +import { Alert, AlertTitle, Box, Button, createStyles, Link, makeStyles, Paper } from '@material-ui/core' +import type { Coin } from '../../types' +import { useI18N } from '../../../../utils/i18n-next-ui' +import { useApprovedTokens } from '../../trending/useApprovedTokens' +import { resolveTokenLinkOnEtherscan } from '../../../../web3/pipes' +import { ChainId, EthereumTokenType } from '../../../../web3/types' + +const useStyles = makeStyles((theme) => { + return createStyles({ + root: { + padding: theme.spacing(0, 2, 2, 2), + }, + approve: { + marginLeft: theme.spacing(1), + whiteSpace: 'nowrap', + backgroundColor: theme.palette.error.main, + color: theme.palette.error.contrastText, + '&:hover': { + backgroundColor: theme.palette.error.main, + }, + }, + }) +}) + +export interface CoinSaftyAlertProps { + coin: Coin +} + +export function CoinSaftyAlert(props: CoinSaftyAlertProps) { + const { coin } = props + + const { t } = useI18N() + const classes = useStyles() + const { approvedTokens, onApprove } = useApprovedTokens(coin.eth_address) + + if (!coin.eth_address) return null + if (approvedTokens.some((address) => address === coin.eth_address)) return null + + return ( + + + Token Safety Alert + Anyone can create and name any ERC20 token on Ethereum, including creating fake versions of existing + tokens and tokens that claim to represent projects that do not have a token. Similar to Etherscan, this + site automatically tracks analytics for all ERC20 tokens independent of token integrity. Please do your + own research before interacting with any ERC20 token. + + + View on Etherscan + + + + + + ) +} diff --git a/packages/maskbook/src/plugins/Trader/UI/trending/LBPPanel.tsx b/packages/maskbook/src/plugins/Trader/UI/trending/LBPPanel.tsx index f305cc88acc0..04547820f988 100644 --- a/packages/maskbook/src/plugins/Trader/UI/trending/LBPPanel.tsx +++ b/packages/maskbook/src/plugins/Trader/UI/trending/LBPPanel.tsx @@ -32,6 +32,11 @@ const useStyles = makeStyles((theme: Theme) => margin: theme.spacing(1, 0), padding: theme.spacing(0, 2.5), }, + placeholder: { + color: theme.palette.text.secondary, + padding: theme.spacing(2, 2, 0), + textAlign: 'center', + }, connect: { padding: theme.spacing(0, 2.5), display: 'flex', @@ -73,6 +78,15 @@ export function LBPPanel(props: LBPPanelProps) { 100, ) + if (!pools.length) + return ( +
+
+ No pools found. +
+
+ ) + return (
@@ -99,36 +113,31 @@ export function LBPPanel(props: LBPPanelProps) { } />
- {pools.length ? ( - <> - - Solid blue line illustrates the historical price of {token.symbol ?? 'Token'} on the{' '} - {token.symbol ?? 'Token'}'s LBP. The price could continue to go down if no one buys. Please make - your investment decision wisely. - - - - - What's LBP? - - ,{' '} - - Tutorial - {' '} - and{' '} - - {token.symbol} LBP Pool in Balancer - - . - - - ) : null} + + Solid blue line illustrates the historical price of {token.symbol ?? 'Token'} on the{' '} + {token.symbol ?? 'Token'}'s LBP. The price could continue to go down if no one buys. Please make your + investment decision wisely. + + + + What's LBP? + + ,{' '} + + Tutorial + {' '} + and{' '} + + {token.symbol} LBP Pool in Balancer + + . +