Compare commits
62 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 57dca48f3e | |||
| a768b4ea06 | |||
| 026783924f | |||
| 5917a13317 | |||
| b6f138b423 | |||
| 145ebc1232 | |||
| bfe38e96b4 | |||
| 86978b7e99 | |||
| efd3158b2b | |||
| 07561aaed2 | |||
| 1e2face10e | |||
| b6b9914548 | |||
| 87b0b71cc6 | |||
| 8114fa7d73 | |||
| e454a7eba0 | |||
| 3784b29f14 | |||
| a67be7a31e | |||
| 23dea07f6e | |||
| 906a457ae5 | |||
| 0a69409077 | |||
| 9026e65b1f | |||
| a21e60f2d8 | |||
| 39f59fcc07 | |||
| 75967b2bf5 | |||
| ecaf2054fc | |||
| 9b5b2973cb | |||
| 36da804ca4 | |||
| 8f51aa9e89 | |||
| 0a31a04148 | |||
| eb9e309c8b | |||
| 5d0fbf705d | |||
| cd332c4dfd | |||
| 224fe4ce25 | |||
| 4736e82da5 | |||
| 8baecb783f | |||
| bf24d61584 | |||
| 01351a0380 | |||
| d0b3f1abee | |||
| d9e7560117 | |||
| 3a30ee59d4 | |||
| 7880c802ae | |||
| f4013142af | |||
| 57bff96c99 | |||
| a406e0fc01 | |||
| 1310deb17a | |||
| d8787476ac | |||
| bc82e67561 | |||
| 63e79ae2aa | |||
| 48ce9ba659 | |||
| 9ee1c9107b | |||
| a90b4c439b | |||
| 2187b9001c | |||
| caf5814e96 | |||
| 4f0abefa1d | |||
| 25b65c3fb7 | |||
| d7c59a1495 | |||
| 5ac0f01bf5 | |||
| 362c129abd | |||
| c1e77bfc0d | |||
| e3d03ed4d1 | |||
| 153b007387 | |||
| 2a13da1a6c |
+50
@@ -0,0 +1,50 @@
|
|||||||
|
'use strict'
|
||||||
|
|
||||||
|
module.exports = function (api) {
|
||||||
|
const targets = '>1% and not ie 11 and not dead'
|
||||||
|
api.cache(true)
|
||||||
|
api.cacheDirectory = true
|
||||||
|
|
||||||
|
return {
|
||||||
|
presets: [
|
||||||
|
'@babel/preset-typescript',
|
||||||
|
[
|
||||||
|
'@babel/preset-env',
|
||||||
|
{
|
||||||
|
targets,
|
||||||
|
modules: false,
|
||||||
|
}
|
||||||
|
],
|
||||||
|
['@babel/preset-react', {runtime: 'automatic' }]
|
||||||
|
],
|
||||||
|
plugins: [
|
||||||
|
[
|
||||||
|
"babel-plugin-tsconfig-paths",
|
||||||
|
{
|
||||||
|
"relative": true,
|
||||||
|
"extensions": [
|
||||||
|
".js",
|
||||||
|
".jsx",
|
||||||
|
".ts",
|
||||||
|
".tsx",
|
||||||
|
".es",
|
||||||
|
".es6",
|
||||||
|
".mjs"
|
||||||
|
],
|
||||||
|
"rootDir": ".",
|
||||||
|
"tsconfig": "tsconfig.lib.json",
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"@babel/plugin-proposal-numeric-separator",
|
||||||
|
"syntax-dynamic-import",
|
||||||
|
'@babel/plugin-proposal-class-properties',
|
||||||
|
[
|
||||||
|
'@babel/plugin-transform-runtime',
|
||||||
|
{
|
||||||
|
helpers: false,
|
||||||
|
regenerator: true
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
+17
-1
@@ -1,3 +1,19 @@
|
|||||||
{
|
{
|
||||||
"ignores": ["@types/jest", "@commitlint/config-conventional", "@types/react-router"]
|
"ignores": [
|
||||||
|
"@types/jest",
|
||||||
|
"@commitlint/config-conventional",
|
||||||
|
"@types/react-router",
|
||||||
|
"@babel/core",
|
||||||
|
"@babel/plugin-proposal-class-properties",
|
||||||
|
"@babel/plugin-transform-runtime",
|
||||||
|
"@babel/preset-env",
|
||||||
|
"@babel/preset-react",
|
||||||
|
"@babel/preset-typescript",
|
||||||
|
"babel-loader",
|
||||||
|
"babel-plugin-syntax-dynamic-import",
|
||||||
|
"babel-plugin-tsconfig-paths",
|
||||||
|
"file-loader",
|
||||||
|
"ts-node",
|
||||||
|
"webpack-cli"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
# See config in https://docs.github.com/en/code-security/supply-chain-security/keeping-your-dependencies-updated-automatically/configuration-options-for-dependency-updates
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
# Enable version updates for npm
|
||||||
|
- package-ecosystem: 'npm'
|
||||||
|
# Look for `package.json` and `lock` files in the `root` directory
|
||||||
|
directory: '/'
|
||||||
|
# Check the npm registry for updates every day (weekdays)
|
||||||
|
schedule:
|
||||||
|
interval: 'weekly'
|
||||||
|
# Always increase the version in package.json as well (for patch versions by default only package-lock.json i updated)
|
||||||
|
versioning-strategy: increase
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
# Always validate the PR title, and ignore the commits
|
||||||
|
titleOnly: true
|
||||||
@@ -55,21 +55,37 @@ jobs:
|
|||||||
- name: Dependency check
|
- name: Dependency check
|
||||||
run: npm run depcheck
|
run: npm run depcheck
|
||||||
|
|
||||||
|
- name: Types check
|
||||||
|
run: npm run check:types
|
||||||
|
|
||||||
|
- name: Types build
|
||||||
|
run: npm run compile:types
|
||||||
|
|
||||||
- name: Update supported Bee action
|
- name: Update supported Bee action
|
||||||
uses: ethersphere/update-supported-bee-action@v1
|
uses: ethersphere/update-supported-bee-action@v1
|
||||||
if: github.ref == 'refs/heads/master'
|
if: github.ref == 'refs/heads/master'
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.REPO_GHA_PAT }}
|
token: ${{ secrets.GHA_PAT_BASIC }}
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: npm run build
|
run: npm run build
|
||||||
|
|
||||||
|
- name: Build Component
|
||||||
|
run: npm run build:component
|
||||||
|
|
||||||
- name: Create preview
|
- name: Create preview
|
||||||
uses: ethersphere/beeload-action@v1
|
uses: ethersphere/swarm-actions/pr-preview@v0
|
||||||
with:
|
with:
|
||||||
preview: 'true'
|
bee-url: https://unlimited.gateway.ethswarm.org
|
||||||
|
token: ${{ secrets.GHA_PAT_BASIC }}
|
||||||
|
error-document: index.html
|
||||||
|
headers: "${{ secrets.GATEWAY_AUTHORIZATION_HEADER }}"
|
||||||
|
|
||||||
- name: Upload to testnet
|
- name: Upload to testnet
|
||||||
uses: ethersphere/beeload-action@v1
|
uses: ethersphere/swarm-actions/upload-dir@v0
|
||||||
|
continue-on-error: true
|
||||||
with:
|
with:
|
||||||
|
index-document: index.html
|
||||||
|
error-document: index.html
|
||||||
|
dir: ./build
|
||||||
bee-url: https://api.gateway.testnet.ethswarm.org
|
bee-url: https://api.gateway.testnet.ethswarm.org
|
||||||
|
|||||||
@@ -15,6 +15,8 @@ jobs:
|
|||||||
node-version: 14
|
node-version: 14
|
||||||
registry-url: 'https://registry.npmjs.org'
|
registry-url: 'https://registry.npmjs.org'
|
||||||
- run: npm ci
|
- run: npm ci
|
||||||
|
- run: npm run compile:types
|
||||||
|
- run: npm run build:component
|
||||||
- run: npm publish --access public
|
- run: npm publish --access public
|
||||||
env:
|
env:
|
||||||
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
||||||
|
|||||||
@@ -11,10 +11,10 @@ jobs:
|
|||||||
release-please:
|
release-please:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: GoogleCloudPlatform/release-please-action@v2
|
- uses: GoogleCloudPlatform/release-please-action@v3
|
||||||
id: release
|
id: release
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.REPO_GHA_PAT }}
|
token: ${{ secrets.GHA_PAT_BASIC }}
|
||||||
release-type: node
|
release-type: node
|
||||||
package-name: bee-dashboard
|
package-name: bee-dashboard
|
||||||
bump-minor-pre-major: true
|
bump-minor-pre-major: true
|
||||||
|
|||||||
@@ -10,6 +10,7 @@
|
|||||||
|
|
||||||
# production
|
# production
|
||||||
/build
|
/build
|
||||||
|
/lib
|
||||||
|
|
||||||
# misc
|
# misc
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
|||||||
@@ -1,5 +1,98 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## [0.16.0](https://github.com/ethersphere/bee-dashboard/compare/v0.15.0...v0.16.0) (2022-06-02)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add light node upgrade top up methods ([#372](https://github.com/ethersphere/bee-dashboard/issues/372)) ([a768b4e](https://github.com/ethersphere/bee-dashboard/commit/a768b4ea0675596f6fe49771ef9d0755af00db56))
|
||||||
|
* allow for the port to be configured ([#370](https://github.com/ethersphere/bee-dashboard/issues/370)) ([b6f138b](https://github.com/ethersphere/bee-dashboard/commit/b6f138b423cbe18b078fd38ea64b4c7a839d4e6e))
|
||||||
|
* recognize ens domains ([#351](https://github.com/ethersphere/bee-dashboard/issues/351)) ([5917a13](https://github.com/ethersphere/bee-dashboard/commit/5917a133172c9e2fc0a81fb2fa19ea29ff976d03))
|
||||||
|
|
||||||
|
## [0.15.0](https://github.com/ethersphere/bee-dashboard/compare/v0.14.0...v0.15.0) (2022-05-16)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add aditional information to the stamps overview ([#349](https://github.com/ethersphere/bee-dashboard/issues/349)) ([23dea07](https://github.com/ethersphere/bee-dashboard/commit/23dea07f6e53da91f87078749f07bd95c9e65983))
|
||||||
|
* add bee desktop toolkit ([#311](https://github.com/ethersphere/bee-dashboard/issues/311)) ([ecaf205](https://github.com/ethersphere/bee-dashboard/commit/ecaf2054fc5aaa5fa4f1d0b3fb2753af9d9b233e))
|
||||||
|
* add bee-desktop settings capabilities ([#323](https://github.com/ethersphere/bee-dashboard/issues/323)) ([87b0b71](https://github.com/ethersphere/bee-dashboard/commit/87b0b71cc63098a5d886ff47d52715c250d1b659))
|
||||||
|
* support for bzz.link cids when downloading files ([#350](https://github.com/ethersphere/bee-dashboard/issues/350)) ([3784b29](https://github.com/ethersphere/bee-dashboard/commit/3784b29f148b706d5bc40b69b5ae898efa2c1990))
|
||||||
|
* wait for postage stamp to be usable when bying it ([#352](https://github.com/ethersphere/bee-dashboard/issues/352)) ([1e2face](https://github.com/ethersphere/bee-dashboard/commit/1e2face10e93818f281526d8245f84834e5ecb86))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* app crash caused by inputing non-number characters ([#347](https://github.com/ethersphere/bee-dashboard/issues/347)) ([a67be7a](https://github.com/ethersphere/bee-dashboard/commit/a67be7a31ec88e9ce9c7764ec4523496c157d08a))
|
||||||
|
* connection health indicator values to reflect the current network conditions ([#353](https://github.com/ethersphere/bee-dashboard/issues/353)) ([07561aa](https://github.com/ethersphere/bee-dashboard/commit/07561aaed2ce7f7ffd7ecfd8ae8b5190cc9893bc))
|
||||||
|
* nested directory upload preserves the directory structure ([#365](https://github.com/ethersphere/bee-dashboard/issues/365)) ([86978b7](https://github.com/ethersphere/bee-dashboard/commit/86978b7e999584173b082eef86074af698523752))
|
||||||
|
* remove restrictions on postage stamp label ([#354](https://github.com/ethersphere/bee-dashboard/issues/354)) ([b6b9914](https://github.com/ethersphere/bee-dashboard/commit/b6b9914548a0ac00ed293ea35490ce38e9d6adaa))
|
||||||
|
* show current postage stamp price per block ([#348](https://github.com/ethersphere/bee-dashboard/issues/348)) ([906a457](https://github.com/ethersphere/bee-dashboard/commit/906a457ae5a8683f82d218759fd66dc1b7c9a220))
|
||||||
|
|
||||||
|
## [0.14.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.13.0...v0.14.0) (2022-04-14)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add hook that detects if the bee-dashboard is run within bee-desktop ([#334](https://www.github.com/ethersphere/bee-dashboard/issues/334)) ([eb9e309](https://www.github.com/ethersphere/bee-dashboard/commit/eb9e309c8bc0327d137f190d6873618cb215fece))
|
||||||
|
* detect bee mode and enable/disable status checks accordingly ([#318](https://www.github.com/ethersphere/bee-dashboard/issues/318)) ([8baecb7](https://www.github.com/ethersphere/bee-dashboard/commit/8baecb783f1574af1cd1f17738efae4b0ac9f0c8))
|
||||||
|
* optional status checks (e.g. connected peers > 0 or funded chequebook) ([#331](https://www.github.com/ethersphere/bee-dashboard/issues/331)) ([5d0fbf7](https://www.github.com/ethersphere/bee-dashboard/commit/5d0fbf705dfed6738980c751a9654199d60a3787))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* postage stamp price and TTL calculation ([#305](https://www.github.com/ethersphere/bee-dashboard/issues/305)) ([d0b3f1a](https://www.github.com/ethersphere/bee-dashboard/commit/d0b3f1abee7ea017bdd05954d5fadafb67365efd))
|
||||||
|
|
||||||
|
## [0.13.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.12.0...v0.13.0) (2022-01-28)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add hash based routing ([#287](https://www.github.com/ethersphere/bee-dashboard/issues/287)) ([9ee1c91](https://www.github.com/ethersphere/bee-dashboard/commit/9ee1c9107bb08d1838044f39e4d0dd5817c8f283))
|
||||||
|
* add metadata and preview ([#292](https://www.github.com/ethersphere/bee-dashboard/issues/292)) ([f401314](https://www.github.com/ethersphere/bee-dashboard/commit/f4013142afdb407e699eff9587921e23c971f1db))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* clean up spinner and disabled state on download page ([#294](https://www.github.com/ethersphere/bee-dashboard/issues/294)) ([a406e0f](https://www.github.com/ethersphere/bee-dashboard/commit/a406e0fc014991fcbaca230f27f41cd071d8a863))
|
||||||
|
* correct folder name when uploading multiple files or mix of files & directories ([#291](https://www.github.com/ethersphere/bee-dashboard/issues/291)) ([d878747](https://www.github.com/ethersphere/bee-dashboard/commit/d8787476acf068be6609a77b1fadb2f61d0fd502))
|
||||||
|
* disable feeds page when disconnected ([#293](https://www.github.com/ethersphere/bee-dashboard/issues/293)) ([1310deb](https://www.github.com/ethersphere/bee-dashboard/commit/1310deb17aec91f368f99974aaa245abb0a3e201))
|
||||||
|
* do not print size and name when meta is unknown ([#297](https://www.github.com/ethersphere/bee-dashboard/issues/297)) ([7880c80](https://www.github.com/ethersphere/bee-dashboard/commit/7880c802aea6b0830ca52b47b88540b8df5888cc))
|
||||||
|
* get current price from chain state ([#286](https://www.github.com/ethersphere/bee-dashboard/issues/286)) ([bc82e67](https://www.github.com/ethersphere/bee-dashboard/commit/bc82e6756154b33d01796a6e66e51dcfa1495338))
|
||||||
|
|
||||||
|
## [0.12.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.11.2...v0.12.0) (2021-12-21)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add identity and feed management ([#272](https://www.github.com/ethersphere/bee-dashboard/issues/272)) ([25b65c3](https://www.github.com/ethersphere/bee-dashboard/commit/25b65c3fb770b09c685fe66596e372dfbb616625))
|
||||||
|
|
||||||
|
### [0.11.2](https://www.github.com/ethersphere/bee-dashboard/compare/v0.11.1...v0.11.2) (2021-12-15)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **ci:** add lib folder to the package.json files prop ([#270](https://www.github.com/ethersphere/bee-dashboard/issues/270)) ([5ac0f01](https://www.github.com/ethersphere/bee-dashboard/commit/5ac0f01bf50ee23b474ab9c8d61c6af418544083))
|
||||||
|
|
||||||
|
### [0.11.1](https://www.github.com/ethersphere/bee-dashboard/compare/v0.11.0...v0.11.1) (2021-12-14)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* typo in publish script ([#268](https://www.github.com/ethersphere/bee-dashboard/issues/268)) ([c1e77bf](https://www.github.com/ethersphere/bee-dashboard/commit/c1e77bfc0d3ac442d6bacec7402f576a6422927e))
|
||||||
|
|
||||||
|
## [0.11.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.10.0...v0.11.0) (2021-12-14)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* modularisation ([#244](https://www.github.com/ethersphere/bee-dashboard/issues/244)) ([2a13da1](https://www.github.com/ethersphere/bee-dashboard/commit/2a13da1a6c5925946d22666a84f975cec87df115))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **build:** bee-dashboard component building ([#267](https://www.github.com/ethersphere/bee-dashboard/issues/267)) ([153b007](https://www.github.com/ethersphere/bee-dashboard/commit/153b007387618e34e1d5dc7fd82d49722783e757))
|
||||||
|
|
||||||
## [0.10.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.9.0...v0.10.0) (2021-12-07)
|
## [0.10.0](https://www.github.com/ethersphere/bee-dashboard/compare/v0.9.0...v0.10.0) (2021-12-07)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
**Warning: This project is in alpha state. There might (and most probably will) be changes in the future to its API and
|
**Warning: This project is in alpha state. There might (and most probably will) be changes in the future to its API and
|
||||||
working. Also, no guarantees can be made about its stability, efficiency, and security at this stage.**
|
working. Also, no guarantees can be made about its stability, efficiency, and security at this stage.**
|
||||||
|
|
||||||
This project is intended to be used with **Bee version <!-- SUPPORTED_BEE_START -->1.4.0-8fa696a8<!-- SUPPORTED_BEE_END -->**.
|
This project is intended to be used with **Bee version <!-- SUPPORTED_BEE_START -->1.6.0-6ceadd35<!-- SUPPORTED_BEE_END -->**.
|
||||||
Using it with older or newer Bee versions is not recommended and may not work. Stay up to date by joining the
|
Using it with older or newer Bee versions is not recommended and may not work. Stay up to date by joining the
|
||||||
[official Discord](https://discord.gg/GU22h2utj6) and by keeping an eye on the
|
[official Discord](https://discord.gg/GU22h2utj6) and by keeping an eye on the
|
||||||
[releases tab](https://github.com/ethersphere/bee-dashboard/releases).
|
[releases tab](https://github.com/ethersphere/bee-dashboard/releases).
|
||||||
@@ -60,6 +60,15 @@ bee-dashboard
|
|||||||
|
|
||||||
This should open the webpage on [`http://localhost:8080`](http://localhost:8080)
|
This should open the webpage on [`http://localhost:8080`](http://localhost:8080)
|
||||||
|
|
||||||
|
You can also define your own port with the `PORT` environment variable. E.g.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
export PORT=3005
|
||||||
|
bee-dashboard
|
||||||
|
```
|
||||||
|
|
||||||
|
Will start the bee-dashboard on [`http://localhost:3005`](http://localhost:3005)
|
||||||
|
|
||||||
### Docker
|
### Docker
|
||||||
|
|
||||||
To build Docker image and run it, execute the following from inside project directory:
|
To build Docker image and run it, execute the following from inside project directory:
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
module.exports = {
|
||||||
|
extends: ['@commitlint/config-conventional'],
|
||||||
|
rules: {
|
||||||
|
'body-max-line-length': [0, 'always', Infinity], // disable commit body length restriction
|
||||||
|
},
|
||||||
|
}
|
||||||
Generated
+3340
-797
File diff suppressed because it is too large
Load Diff
+45
-14
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@ethersphere/bee-dashboard",
|
"name": "@ethersphere/bee-dashboard",
|
||||||
"version": "0.10.0",
|
"version": "0.16.0",
|
||||||
"description": "An app which helps users to setup their Bee node and do actions like cash out cheques",
|
"description": "An app which helps users to setup their Bee node and do actions like cash out cheques",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"bee",
|
"bee",
|
||||||
@@ -15,6 +15,8 @@
|
|||||||
"bin": {
|
"bin": {
|
||||||
"bee-dashboard": "./serve.js"
|
"bee-dashboard": "./serve.js"
|
||||||
},
|
},
|
||||||
|
"main": "lib/App.js",
|
||||||
|
"types": "lib/src/App.d.ts",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
"url": "https://github.com/ethersphere/bee-dashboard/issues/"
|
"url": "https://github.com/ethersphere/bee-dashboard/issues/"
|
||||||
},
|
},
|
||||||
@@ -24,13 +26,16 @@
|
|||||||
"url": "https://github.com/ethersphere/bee-dashboard.git"
|
"url": "https://github.com/ethersphere/bee-dashboard.git"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ethersphere/bee-js": "3.0.0",
|
"@ethersphere/bee-js": "^4.1.1",
|
||||||
"@ethersphere/manifest-js": "^1.0.0",
|
"@ethersphere/manifest-js": "1.1.0",
|
||||||
|
"@ethersphere/swarm-cid": "^0.1.0",
|
||||||
"@material-ui/core": "4.12.3",
|
"@material-ui/core": "4.12.3",
|
||||||
"@material-ui/icons": "4.11.2",
|
"@material-ui/icons": "4.11.2",
|
||||||
"@material-ui/lab": "4.0.0-alpha.57",
|
"@material-ui/lab": "4.0.0-alpha.57",
|
||||||
"axios": "0.24.0",
|
"axios": "0.24.0",
|
||||||
"bignumber.js": "9.0.1",
|
"bignumber.js": "9.0.1",
|
||||||
|
"ethereumjs-wallet": "^1.0.2",
|
||||||
|
"ethers": "^5.6.4",
|
||||||
"file-saver": "^2.0.5",
|
"file-saver": "^2.0.5",
|
||||||
"formik": "2.2.9",
|
"formik": "2.2.9",
|
||||||
"formik-material-ui": "3.0.1",
|
"formik-material-ui": "3.0.1",
|
||||||
@@ -39,37 +44,50 @@
|
|||||||
"notistack": "1.0.10",
|
"notistack": "1.0.10",
|
||||||
"opener": "1.5.2",
|
"opener": "1.5.2",
|
||||||
"qrcode.react": "1.0.1",
|
"qrcode.react": "1.0.1",
|
||||||
"react": "17.0.2",
|
"react": ">= 17.0.2",
|
||||||
"react-copy-to-clipboard": "5.0.4",
|
"react-copy-to-clipboard": "5.0.4",
|
||||||
"react-dom": "17.0.2",
|
"react-dom": ">= 17.0.2",
|
||||||
"react-feather": "2.0.9",
|
"react-feather": "2.0.9",
|
||||||
"react-identicons": "1.2.5",
|
"react-identicons": "1.2.5",
|
||||||
"react-router": "5.2.0",
|
"react-router": "6.2.1",
|
||||||
"react-router-dom": "5.2.0",
|
"react-router-dom": "6.2.1",
|
||||||
"react-syntax-highlighter": "15.4.4",
|
"react-syntax-highlighter": "15.4.4",
|
||||||
"semver": "7.3.5",
|
"semver": "7.3.5",
|
||||||
"serve-handler": "6.1.3"
|
"serve-handler": "6.1.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@babel/core": "7.16.0",
|
||||||
|
"@babel/plugin-proposal-class-properties": "7.16.0",
|
||||||
|
"@babel/plugin-transform-runtime": "7.16.4",
|
||||||
|
"@babel/preset-env": "7.16.4",
|
||||||
|
"@babel/preset-react": "7.16.7",
|
||||||
|
"@babel/preset-typescript": "7.16.0",
|
||||||
"@commitlint/config-conventional": "14.1.0",
|
"@commitlint/config-conventional": "14.1.0",
|
||||||
"@testing-library/jest-dom": "5.15.0",
|
"@testing-library/jest-dom": "5.16.4",
|
||||||
"@testing-library/react": "12.1.2",
|
"@testing-library/react": "12.1.2",
|
||||||
"@types/file-saver": "^2.0.4",
|
"@testing-library/react-hooks": "^8.0.0",
|
||||||
|
"@types/cors": "^2.8.12",
|
||||||
|
"@types/express": "^4.17.13",
|
||||||
|
"@types/file-saver": "2.0.4",
|
||||||
"@types/jest": "27.0.2",
|
"@types/jest": "27.0.2",
|
||||||
"@types/qrcode.react": "1.0.2",
|
"@types/qrcode.react": "1.0.2",
|
||||||
"@types/react": "17.0.34",
|
"@types/react": "17.0.34",
|
||||||
"@types/react-copy-to-clipboard": "5.0.2",
|
"@types/react-copy-to-clipboard": "5.0.2",
|
||||||
"@types/react-dom": "17.0.11",
|
"@types/react-dom": "17.0.11",
|
||||||
"@types/react-router": "5.1.17",
|
"@types/react-router": "5.1.18",
|
||||||
"@types/react-router-dom": "5.3.2",
|
"@types/react-router-dom": "5.3.2",
|
||||||
"@types/react-syntax-highlighter": "13.5.2",
|
"@types/react-syntax-highlighter": "13.5.2",
|
||||||
"@types/semver": "7.3.9",
|
"@types/semver": "7.3.9",
|
||||||
"@typescript-eslint/eslint-plugin": "4.33.0",
|
"@typescript-eslint/eslint-plugin": "4.33.0",
|
||||||
"@typescript-eslint/parser": "4.33.0",
|
"@typescript-eslint/parser": "4.33.0",
|
||||||
"babel-eslint": "10.1.0",
|
"babel-eslint": "10.1.0",
|
||||||
"depcheck": "1.4.2",
|
"babel-loader": "8.1.0",
|
||||||
|
"babel-plugin-syntax-dynamic-import": "6.18.0",
|
||||||
|
"babel-plugin-tsconfig-paths": "1.0.2",
|
||||||
|
"cors": "^2.8.5",
|
||||||
|
"depcheck": "^1.4.3",
|
||||||
"eslint": "7.24.0",
|
"eslint": "7.24.0",
|
||||||
"eslint-config-prettier": "8.2.0",
|
"eslint-config-prettier": "8.5.0",
|
||||||
"eslint-config-react-app": "6.0.0",
|
"eslint-config-react-app": "6.0.0",
|
||||||
"eslint-plugin-flowtype": "5.10.0",
|
"eslint-plugin-flowtype": "5.10.0",
|
||||||
"eslint-plugin-import": "2.25.2",
|
"eslint-plugin-import": "2.25.2",
|
||||||
@@ -79,22 +97,35 @@
|
|||||||
"eslint-plugin-react": "7.23.2",
|
"eslint-plugin-react": "7.23.2",
|
||||||
"eslint-plugin-react-hooks": "4.2.0",
|
"eslint-plugin-react-hooks": "4.2.0",
|
||||||
"eslint-plugin-testing-library": "3.10.2",
|
"eslint-plugin-testing-library": "3.10.2",
|
||||||
|
"express": "^4.17.3",
|
||||||
|
"file-loader": "6.2.0",
|
||||||
"prettier": "2.4.1",
|
"prettier": "2.4.1",
|
||||||
"react-scripts": "4.0.3",
|
"react-scripts": "4.0.3",
|
||||||
|
"ts-node": "^10.7.0",
|
||||||
"typescript": "4.4.4",
|
"typescript": "4.4.4",
|
||||||
"web-vitals": "2.1.2"
|
"web-vitals": "2.1.2",
|
||||||
|
"webpack": "4.44.2",
|
||||||
|
"webpack-cli": "^4.9.1"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": ">= 17.0.2",
|
||||||
|
"react-dom": ">= 17.0.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"prepare": "npm run build",
|
"prepare": "npm run build",
|
||||||
"start": "react-scripts start",
|
"start": "react-scripts start",
|
||||||
"build": "react-scripts build",
|
"build": "react-scripts build",
|
||||||
|
"build:component": "TS_NODE_COMPILER_OPTIONS='{\"module\": \"commonjs\"}' webpack --mode=production",
|
||||||
|
"compile:types": "tsc --project tsconfig.lib.json --emitDeclarationOnly --declaration",
|
||||||
"test": "react-scripts test",
|
"test": "react-scripts test",
|
||||||
"serve": "node ./serve.js",
|
"serve": "node ./serve.js",
|
||||||
"depcheck": "depcheck .",
|
"depcheck": "depcheck .",
|
||||||
"lint": "eslint --fix \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --write \"src/**/*.ts\" \"src/**/*.tsx\"",
|
"lint": "eslint --fix \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --write \"src/**/*.ts\" \"src/**/*.tsx\"",
|
||||||
"lint:check": "eslint \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --check \"src/**/*.ts\" \"src/**/*.tsx\""
|
"lint:check": "eslint \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --check \"src/**/*.ts\" \"src/**/*.tsx\"",
|
||||||
|
"check:types": "tsc --project tsconfig.lib.json"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
|
"lib",
|
||||||
"build",
|
"build",
|
||||||
"serve.js"
|
"serve.js"
|
||||||
],
|
],
|
||||||
|
|||||||
+1
-7
@@ -6,13 +6,7 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<meta name="theme-color" content="#000000" />
|
<meta name="theme-color" content="#000000" />
|
||||||
|
|
||||||
<link rel="preconnect" href="https://fonts.gstatic.com">
|
<meta name="description" content="Bee Dashboard" />
|
||||||
<link href="https://fonts.googleapis.com/css2?family=Montserrat:wght@300;400;500;600&display=swap" rel="stylesheet">
|
|
||||||
|
|
||||||
<meta
|
|
||||||
name="description"
|
|
||||||
content="Bee Dashboard"
|
|
||||||
/>
|
|
||||||
<link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
|
<link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
|
||||||
<!--
|
<!--
|
||||||
manifest.json provides metadata used when your web app is installed on a
|
manifest.json provides metadata used when your web app is installed on a
|
||||||
|
|||||||
@@ -1,34 +1,35 @@
|
|||||||
#!/usr/bin/env node
|
#!/usr/bin/env node
|
||||||
|
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const handler = require('serve-handler');
|
const handler = require('serve-handler')
|
||||||
const http = require('http');
|
const http = require('http')
|
||||||
const opener = require('opener')
|
const opener = require('opener')
|
||||||
|
|
||||||
|
const port = process.env.PORT || 8080
|
||||||
|
|
||||||
const serverConfig = {
|
const serverConfig = {
|
||||||
public: path.join(__dirname, 'build'),
|
public: path.join(__dirname, 'build'),
|
||||||
trailingSlash: false,
|
trailingSlash: false,
|
||||||
rewrites: [
|
rewrites: [{ source: '**', destination: '/index.html' }],
|
||||||
{ source: "**", destination: "/index.html" },
|
|
||||||
],
|
|
||||||
headers: [
|
headers: [
|
||||||
{
|
{
|
||||||
source: "*",
|
source: '*',
|
||||||
headers: [{
|
headers: [
|
||||||
key: "Cache-Control",
|
{
|
||||||
value: "max-age=3600"
|
key: 'Cache-Control',
|
||||||
}]
|
value: 'max-age=3600',
|
||||||
}
|
},
|
||||||
]
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
const server = http.createServer((request, response) => {
|
const server = http.createServer((request, response) => {
|
||||||
|
return handler(request, response, serverConfig)
|
||||||
|
})
|
||||||
|
|
||||||
return handler(request, response, serverConfig);
|
server.listen(port, () => {
|
||||||
})
|
console.log(`Starting up Bee Dashboard on address http://localhost:${port}`)
|
||||||
|
|
||||||
server.listen(8080, () => {
|
|
||||||
console.log('Starting up Bee Dashboard on address http://localhost:8080')
|
|
||||||
console.log('Hit CTRL-C to stop the server')
|
console.log('Hit CTRL-C to stop the server')
|
||||||
opener('http://localhost:8080')
|
opener(`http://localhost:${port}`)
|
||||||
})
|
})
|
||||||
|
|||||||
+38
-9
@@ -1,34 +1,63 @@
|
|||||||
@font-face {
|
@font-face {
|
||||||
font-family: "IBMPlexMono500";
|
font-family: 'Work Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 300;
|
||||||
|
font-display: swap;
|
||||||
|
src: url(assets/fonts/WorkSans/WorkSans-Light.ttf) format('truetype');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Work Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
font-display: swap;
|
||||||
|
src: url(assets/fonts/WorkSans/WorkSans-Regular.ttf) format('truetype');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Work Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 500;
|
||||||
|
font-display: swap;
|
||||||
|
src: url(assets/fonts/WorkSans/WorkSans-Medium.ttf) format('truetype');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Work Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 600;
|
||||||
|
font-display: swap;
|
||||||
|
src: url(assets/fonts/WorkSans/WorkSans-SemiBold.ttf) format('truetype');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'IBMPlexMono500';
|
||||||
src: url(assets/fonts/IBMPlexMono500.ttf) format('truetype');
|
src: url(assets/fonts/IBMPlexMono500.ttf) format('truetype');
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "IBMPlexMono600";
|
font-family: 'IBMPlexMono600';
|
||||||
src: url(assets/fonts/IBMPlexMono600.ttf) format('truetype');
|
src: url(assets/fonts/IBMPlexMono600.ttf) format('truetype');
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "IBMPlexMonoregular";
|
font-family: 'IBMPlexMonoregular';
|
||||||
src: url(assets/fonts/IBMPlexMonoregular.ttf) format('truetype');
|
src: url(assets/fonts/IBMPlexMonoregular.ttf) format('truetype');
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "WorkSans-Italic-VariableFont_wght";
|
font-family: 'WorkSans-Italic-VariableFont_wght';
|
||||||
src: url(assets/fonts/WorkSans-Italic-VariableFont_wght.ttf) format('truetype');
|
src: url(assets/fonts/WorkSans-Italic-VariableFont_wght.ttf) format('truetype');
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "WorkSans-VariableFont_wght";
|
font-family: 'WorkSans-VariableFont_wght';
|
||||||
src: url(assets/fonts/WorkSans-VariableFont_wght.ttf) format('truetype');
|
src: url(assets/fonts/WorkSans-VariableFont_wght.ttf) format('truetype');
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
}
|
}
|
||||||
|
|
||||||
.App {
|
.App {
|
||||||
font-family: "Helvetica Neue", HelveticaNeue, Helvetica, Arial, sans-serif;
|
font-family: 'Work Sans', 'Helvetica Neue', HelveticaNeue, Helvetica, Arial, sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
a, button {
|
a,
|
||||||
font-family: "IBMPlexMono500" !important;
|
button {
|
||||||
|
font-family: 'IBMPlexMono500' !important;
|
||||||
color: #dd7700;
|
color: #dd7700;
|
||||||
}
|
}
|
||||||
|
|||||||
+28
-16
@@ -1,37 +1,49 @@
|
|||||||
import CssBaseline from '@material-ui/core/CssBaseline'
|
import CssBaseline from '@material-ui/core/CssBaseline'
|
||||||
import { ThemeProvider } from '@material-ui/core/styles'
|
import { ThemeProvider } from '@material-ui/core/styles'
|
||||||
import { SnackbarProvider } from 'notistack'
|
import { SnackbarProvider } from 'notistack'
|
||||||
import { ReactElement } from 'react'
|
import React, { ReactElement } from 'react'
|
||||||
import { BrowserRouter as Router } from 'react-router-dom'
|
import { HashRouter as Router } from 'react-router-dom'
|
||||||
import './App.css'
|
import './App.css'
|
||||||
import Dashboard from './layout/Dashboard'
|
import Dashboard from './layout/Dashboard'
|
||||||
import { Provider as BeeProvider } from './providers/Bee'
|
import { Provider as BeeProvider } from './providers/Bee'
|
||||||
|
import { Provider as FeedsProvider } from './providers/Feeds'
|
||||||
import { Provider as FileProvider } from './providers/File'
|
import { Provider as FileProvider } from './providers/File'
|
||||||
import { Provider as PlatformProvider } from './providers/Platform'
|
import { Provider as PlatformProvider } from './providers/Platform'
|
||||||
import { Provider as SettingsProvider } from './providers/Settings'
|
import { Provider as SettingsProvider } from './providers/Settings'
|
||||||
import { Provider as StampsProvider } from './providers/Stamps'
|
import { Provider as StampsProvider } from './providers/Stamps'
|
||||||
|
import { Provider as TopUpProvider } from './providers/TopUp'
|
||||||
import BaseRouter from './routes'
|
import BaseRouter from './routes'
|
||||||
import { theme } from './theme'
|
import { theme } from './theme'
|
||||||
|
|
||||||
const App = (): ReactElement => (
|
interface Props {
|
||||||
|
beeApiUrl?: string
|
||||||
|
beeDebugApiUrl?: string
|
||||||
|
lockedApiSettings?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const App = ({ beeApiUrl, beeDebugApiUrl, lockedApiSettings }: Props): ReactElement => (
|
||||||
<div className="App">
|
<div className="App">
|
||||||
<ThemeProvider theme={theme}>
|
<ThemeProvider theme={theme}>
|
||||||
<SettingsProvider>
|
<SettingsProvider beeApiUrl={beeApiUrl} beeDebugApiUrl={beeDebugApiUrl} lockedApiSettings={lockedApiSettings}>
|
||||||
<BeeProvider>
|
<BeeProvider>
|
||||||
<StampsProvider>
|
<StampsProvider>
|
||||||
<FileProvider>
|
<FileProvider>
|
||||||
<PlatformProvider>
|
<FeedsProvider>
|
||||||
<SnackbarProvider>
|
<PlatformProvider>
|
||||||
<Router>
|
<TopUpProvider>
|
||||||
<>
|
<SnackbarProvider>
|
||||||
<CssBaseline />
|
<Router>
|
||||||
<Dashboard>
|
<>
|
||||||
<BaseRouter />
|
<CssBaseline />
|
||||||
</Dashboard>
|
<Dashboard>
|
||||||
</>
|
<BaseRouter />
|
||||||
</Router>
|
</Dashboard>
|
||||||
</SnackbarProvider>
|
</>
|
||||||
</PlatformProvider>
|
</Router>
|
||||||
|
</SnackbarProvider>
|
||||||
|
</TopUpProvider>
|
||||||
|
</PlatformProvider>
|
||||||
|
</FeedsProvider>
|
||||||
</FileProvider>
|
</FileProvider>
|
||||||
</StampsProvider>
|
</StampsProvider>
|
||||||
</BeeProvider>
|
</BeeProvider>
|
||||||
|
|||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -3,7 +3,7 @@ import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
|||||||
import { Alert, AlertTitle } from '@material-ui/lab'
|
import { Alert, AlertTitle } from '@material-ui/lab'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
const LIMIT = 100_000_000 // 100 megabytes
|
const LIMIT = 100000000 // 100 megabytes
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
files: File[]
|
files: File[]
|
||||||
|
|||||||
@@ -1,54 +0,0 @@
|
|||||||
import { ReactElement, useState, useContext } from 'react'
|
|
||||||
import { makeStyles, Theme, createStyles } from '@material-ui/core/styles'
|
|
||||||
import { Alert, AlertTitle } from '@material-ui/lab'
|
|
||||||
import Collapse from '@material-ui/core/Collapse'
|
|
||||||
import IconButton from '@material-ui/core/IconButton'
|
|
||||||
import CloseIcon from '@material-ui/icons/Close'
|
|
||||||
import { Context } from '../providers/Bee'
|
|
||||||
import { SUPPORTED_BEE_VERSION_EXACT } from '@ethersphere/bee-js'
|
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
|
||||||
createStyles({
|
|
||||||
root: {
|
|
||||||
width: '100%',
|
|
||||||
marginBottom: theme.spacing(2),
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
|
|
||||||
export default function VersionAlert(): ReactElement | null {
|
|
||||||
const classes = useStyles()
|
|
||||||
const { isLoading, latestUserVersionExact } = useContext(Context)
|
|
||||||
const [open, setOpen] = useState<boolean>(true)
|
|
||||||
|
|
||||||
const isExactlySupportedBeeVersion = SUPPORTED_BEE_VERSION_EXACT === latestUserVersionExact
|
|
||||||
|
|
||||||
if (isLoading || !latestUserVersionExact) return null
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Collapse in={!isExactlySupportedBeeVersion && open}>
|
|
||||||
<div className={classes.root}>
|
|
||||||
<Alert
|
|
||||||
severity="warning"
|
|
||||||
action={
|
|
||||||
<IconButton
|
|
||||||
aria-label="close"
|
|
||||||
color="inherit"
|
|
||||||
size="small"
|
|
||||||
onClick={() => {
|
|
||||||
setOpen(false)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<CloseIcon fontSize="inherit" />
|
|
||||||
</IconButton>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<AlertTitle>Warning</AlertTitle>
|
|
||||||
Your Bee node version (<code>{latestUserVersionExact}</code>) does not exactly match the Bee version we tested
|
|
||||||
the Bee Dashboard against (<code>{SUPPORTED_BEE_VERSION_EXACT}</code>). Please note that some functionality
|
|
||||||
may not work properly.
|
|
||||||
</Alert>
|
|
||||||
</div>
|
|
||||||
</Collapse>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -6,7 +6,7 @@ import DialogContent from '@material-ui/core/DialogContent'
|
|||||||
import DialogContentText from '@material-ui/core/DialogContentText'
|
import DialogContentText from '@material-ui/core/DialogContentText'
|
||||||
import DialogTitle from '@material-ui/core/DialogTitle'
|
import DialogTitle from '@material-ui/core/DialogTitle'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useState, useContext } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
import { Zap } from 'react-feather'
|
import { Zap } from 'react-feather'
|
||||||
import { Context as SettingsContext } from '../providers/Settings'
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
import EthereumAddress from './EthereumAddress'
|
import EthereumAddress from './EthereumAddress'
|
||||||
@@ -61,7 +61,7 @@ export default function CheckoutModal({ peerId, uncashedAmount }: Props): ReactE
|
|||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<Button variant="contained" onClick={handleClickOpen} startIcon={<Zap size="1rem" />}>
|
<Button variant="contained" onClick={handleClickOpen} startIcon={<Zap size="1rem" />}>
|
||||||
Cash out peer {peerId.substr(0, 8)}[…]
|
Cash out peer {peerId.slice(0, 8)}[…]
|
||||||
</Button>
|
</Button>
|
||||||
<Dialog open={open} onClose={handleClose} aria-labelledby="form-dialog-title">
|
<Dialog open={open} onClose={handleClose} aria-labelledby="form-dialog-title">
|
||||||
<DialogTitle id="form-dialog-title">Cashout Cheque</DialogTitle>
|
<DialogTitle id="form-dialog-title">Cashout Cheque</DialogTitle>
|
||||||
|
|||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import { createStyles, makeStyles, Theme } from '@material-ui/core'
|
||||||
|
import { Close } from '@material-ui/icons'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onClose: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
wrapper: {
|
||||||
|
padding: theme.spacing(1),
|
||||||
|
cursor: 'pointer',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function CloseButton({ onClose }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.wrapper} onClick={onClose}>
|
||||||
|
<Close />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
import { createStyles, makeStyles, Theme } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: string
|
||||||
|
prettify?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
wrapper: {
|
||||||
|
overflow: 'scroll',
|
||||||
|
background: '#ffffff',
|
||||||
|
},
|
||||||
|
pre: {
|
||||||
|
maxHeight: '6em',
|
||||||
|
padding: theme.spacing(2),
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
function prettifyString(string: string): string {
|
||||||
|
try {
|
||||||
|
return JSON.stringify(JSON.parse(string), null, 4)
|
||||||
|
} catch {
|
||||||
|
return string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Code({ children, prettify }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.wrapper}>
|
||||||
|
<pre className={classes.pre}>{prettify ? prettifyString(children) : children}</pre>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
import { createStyles, makeStyles, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: (string | ReactElement)[] | (string | ReactElement)
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
text: {
|
||||||
|
color: '#606060',
|
||||||
|
fontSize: '0.9rem',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function DocumentationText({ children }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
return <Typography className={classes.text}>{children}</Typography>
|
||||||
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
import { Typography } from '@material-ui/core/'
|
import { Typography } from '@material-ui/core/'
|
||||||
import QRCodeModal from './QRCodeModal'
|
|
||||||
import ClipboardCopy from './ClipboardCopy'
|
|
||||||
|
|
||||||
import Identicon from 'react-identicons'
|
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
|
import Identicon from 'react-identicons'
|
||||||
|
import { config } from '../config'
|
||||||
|
import ClipboardCopy from './ClipboardCopy'
|
||||||
|
import QRCodeModal from './QRCodeModal'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
address: string | undefined
|
address: string | undefined
|
||||||
@@ -36,9 +36,7 @@ export default function EthereumAddress(props: Props): ReactElement {
|
|||||||
}
|
}
|
||||||
: { marginRight: '7px' }
|
: { marginRight: '7px' }
|
||||||
}
|
}
|
||||||
href={`${process.env.REACT_APP_BLOCKCHAIN_EXPLORER_URL}/${props.transaction ? 'tx' : 'address'}/${
|
href={`${config.BLOCKCHAIN_EXPLORER_URL}/${props.transaction ? 'tx' : 'address'}/${props.address}`}
|
||||||
props.address
|
|
||||||
}`}
|
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer"
|
rel="noreferrer"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -4,8 +4,12 @@ import { ReactElement, ReactNode } from 'react'
|
|||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
|
wrapper: {
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'row',
|
||||||
|
flexWrap: 'wrap',
|
||||||
|
},
|
||||||
action: {
|
action: {
|
||||||
marginTop: theme.spacing(0.75),
|
|
||||||
marginBottom: theme.spacing(1),
|
marginBottom: theme.spacing(1),
|
||||||
marginRight: theme.spacing(1),
|
marginRight: theme.spacing(1),
|
||||||
},
|
},
|
||||||
@@ -21,16 +25,16 @@ export default function ExpandableListItemActions({ children }: Props): ReactEle
|
|||||||
|
|
||||||
if (Array.isArray(children)) {
|
if (Array.isArray(children)) {
|
||||||
return (
|
return (
|
||||||
<Grid container direction="row">
|
<div className={classes.wrapper}>
|
||||||
{children
|
{children
|
||||||
// Exclude falsy values to allow conditional rendering
|
// Exclude falsy values to allow conditional rendering
|
||||||
.filter(x => x)
|
.filter(x => x)
|
||||||
.map((a, i) => (
|
.map((a, i) => (
|
||||||
<Grid key={i} className={classes.action}>
|
<div key={i} className={classes.action}>
|
||||||
{a}
|
{a}
|
||||||
</Grid>
|
</div>
|
||||||
))}
|
))}
|
||||||
</Grid>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
import { Button, Grid, IconButton, InputBase, ListItem, Typography } from '@material-ui/core'
|
import { Grid, IconButton, InputBase, ListItem, Typography } from '@material-ui/core'
|
||||||
import Collapse from '@material-ui/core/Collapse'
|
import Collapse from '@material-ui/core/Collapse'
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { ChangeEvent, ReactElement, useState } from 'react'
|
import { ChangeEvent, ReactElement, useState } from 'react'
|
||||||
import { Check, Edit, Minus, RotateCcw } from 'react-feather'
|
import { Edit, Minus, Search, X } from 'react-feather'
|
||||||
import ExpandableListItemActions from './ExpandableListItemActions'
|
import ExpandableListItemActions from './ExpandableListItemActions'
|
||||||
import ExpandableListItemNote from './ExpandableListItemNote'
|
import ExpandableListItemNote from './ExpandableListItemNote'
|
||||||
|
import { SwarmButton } from './SwarmButton'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -52,9 +53,11 @@ interface Props {
|
|||||||
expandedOnly?: boolean
|
expandedOnly?: boolean
|
||||||
confirmLabel?: string
|
confirmLabel?: string
|
||||||
confirmLabelDisabled?: boolean
|
confirmLabelDisabled?: boolean
|
||||||
|
loading?: boolean
|
||||||
onChange?: (value: string) => void
|
onChange?: (value: string) => void
|
||||||
onConfirm: (value: string) => void
|
onConfirm?: (value: string) => void
|
||||||
mapperFn?: (value: string) => string
|
mapperFn?: (value: string) => string
|
||||||
|
locked?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function ExpandableListItemKey({
|
export default function ExpandableListItemKey({
|
||||||
@@ -67,7 +70,9 @@ export default function ExpandableListItemKey({
|
|||||||
expandedOnly,
|
expandedOnly,
|
||||||
helperText,
|
helperText,
|
||||||
placeholder,
|
placeholder,
|
||||||
|
loading,
|
||||||
mapperFn,
|
mapperFn,
|
||||||
|
locked,
|
||||||
}: Props): ReactElement | null {
|
}: Props): ReactElement | null {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const [open, setOpen] = useState(Boolean(expandedOnly))
|
const [open, setOpen] = useState(Boolean(expandedOnly))
|
||||||
@@ -96,7 +101,7 @@ export default function ExpandableListItemKey({
|
|||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
<div>
|
<div>
|
||||||
{!open && value}
|
{!open && value}
|
||||||
{!expandedOnly && (
|
{!expandedOnly && !locked && (
|
||||||
<IconButton size="small" className={classes.copyValue}>
|
<IconButton size="small" className={classes.copyValue}>
|
||||||
{open ? (
|
{open ? (
|
||||||
<Minus onClick={toggleOpen} strokeWidth={1} />
|
<Minus onClick={toggleOpen} strokeWidth={1} />
|
||||||
@@ -116,6 +121,7 @@ export default function ExpandableListItemKey({
|
|||||||
fullWidth
|
fullWidth
|
||||||
className={classes.content}
|
className={classes.content}
|
||||||
autoFocus
|
autoFocus
|
||||||
|
hidden={locked}
|
||||||
/>
|
/>
|
||||||
</Collapse>
|
</Collapse>
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -123,26 +129,29 @@ export default function ExpandableListItemKey({
|
|||||||
<Collapse in={open} timeout="auto" unmountOnExit>
|
<Collapse in={open} timeout="auto" unmountOnExit>
|
||||||
{helperText && <ExpandableListItemNote>{helperText}</ExpandableListItemNote>}
|
{helperText && <ExpandableListItemNote>{helperText}</ExpandableListItemNote>}
|
||||||
<ExpandableListItemActions>
|
<ExpandableListItemActions>
|
||||||
<Button
|
<SwarmButton
|
||||||
variant="contained"
|
|
||||||
disabled={
|
disabled={
|
||||||
|
loading ||
|
||||||
inputValue === value ||
|
inputValue === value ||
|
||||||
Boolean(confirmLabelDisabled) || // Disable if external validation is provided
|
Boolean(confirmLabelDisabled) || // Disable if external validation is provided
|
||||||
(inputValue === '' && value === undefined) // Disable if no initial value was not provided and the field is empty. The undefined check is improtant so that it is possible to submit with empty input in other cases
|
(inputValue === '' && value === undefined) // Disable if no initial value was not provided and the field is empty. The undefined check is improtant so that it is possible to submit with empty input in other cases
|
||||||
}
|
}
|
||||||
startIcon={<Check size="1rem" />}
|
loading={loading}
|
||||||
onClick={() => onConfirm(inputValue)}
|
iconType={Search}
|
||||||
|
onClick={() => {
|
||||||
|
if (onConfirm) onConfirm(inputValue)
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{confirmLabel || 'Save'}
|
{confirmLabel || 'Save'}
|
||||||
</Button>
|
</SwarmButton>
|
||||||
<Button
|
<SwarmButton
|
||||||
variant="contained"
|
disabled={loading || inputValue === value || inputValue === ''}
|
||||||
disabled={inputValue === value || inputValue === ''}
|
iconType={X}
|
||||||
startIcon={<RotateCcw size="1rem" />}
|
|
||||||
onClick={() => setInputValue(value || '')}
|
onClick={() => setInputValue(value || '')}
|
||||||
|
cancel
|
||||||
>
|
>
|
||||||
Cancel
|
Cancel
|
||||||
</Button>
|
</SwarmButton>
|
||||||
</ExpandableListItemActions>
|
</ExpandableListItemActions>
|
||||||
</Collapse>
|
</Collapse>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -38,6 +38,7 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
interface Props {
|
interface Props {
|
||||||
label: string
|
label: string
|
||||||
value: string
|
value: string
|
||||||
|
expanded?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const lengthWithoutPrefix = (s: string) => s.replace(/^0x/i, '').length
|
const lengthWithoutPrefix = (s: string) => s.replace(/^0x/i, '').length
|
||||||
@@ -54,9 +55,9 @@ const split = (s: string): string[] => {
|
|||||||
return s.match(/(0x|.{1,8})/gi) || []
|
return s.match(/(0x|.{1,8})/gi) || []
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function ExpandableListItemKey({ label, value }: Props): ReactElement | null {
|
export default function ExpandableListItemKey({ label, value, expanded }: Props): ReactElement | null {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const [open, setOpen] = useState(false)
|
const [open, setOpen] = useState(expanded || false)
|
||||||
const [copied, setCopied] = useState(false)
|
const [copied, setCopied] = useState(false)
|
||||||
const toggleOpen = () => setOpen(!open)
|
const toggleOpen = () => setOpen(!open)
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
|||||||
import { ArrowForward, OpenInNewSharp } from '@material-ui/icons'
|
import { ArrowForward, OpenInNewSharp } from '@material-ui/icons'
|
||||||
import { ReactElement, useState } from 'react'
|
import { ReactElement, useState } from 'react'
|
||||||
import CopyToClipboard from 'react-copy-to-clipboard'
|
import CopyToClipboard from 'react-copy-to-clipboard'
|
||||||
import { useHistory } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -61,7 +61,7 @@ export default function ExpandableListItemLink({
|
|||||||
}: Props): ReactElement | null {
|
}: Props): ReactElement | null {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const [copied, setCopied] = useState(false)
|
const [copied, setCopied] = useState(false)
|
||||||
const history = useHistory()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
const tooltipClickHandler = () => setCopied(true)
|
const tooltipClickHandler = () => setCopied(true)
|
||||||
const tooltipCloseHandler = () => setCopied(false)
|
const tooltipCloseHandler = () => setCopied(false)
|
||||||
@@ -72,7 +72,7 @@ export default function ExpandableListItemLink({
|
|||||||
if (navigationType === 'NEW_WINDOW') {
|
if (navigationType === 'NEW_WINDOW') {
|
||||||
window.open(link || value)
|
window.open(link || value)
|
||||||
} else {
|
} else {
|
||||||
history.push(link || value)
|
navigate(link || value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { Box, createStyles, Grid, makeStyles, Typography } from '@material-ui/core'
|
import { Box, createStyles, Grid, makeStyles, Typography } from '@material-ui/core'
|
||||||
import { ArrowBack } from '@material-ui/icons'
|
import { ArrowBack } from '@material-ui/icons'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { useHistory } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
children: string
|
children: string
|
||||||
@@ -20,10 +20,10 @@ const useStyles = makeStyles(() =>
|
|||||||
|
|
||||||
export function HistoryHeader({ children }: Props): ReactElement {
|
export function HistoryHeader({ children }: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const history = useHistory()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
function goBack() {
|
function goBack() {
|
||||||
history.goBack()
|
navigate(-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -0,0 +1,55 @@
|
|||||||
|
import { createStyles, Grid, makeStyles, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
steps: string[]
|
||||||
|
index: number
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
wrapper: {
|
||||||
|
height: '52px',
|
||||||
|
display: 'flex',
|
||||||
|
flex: 1,
|
||||||
|
justifyContent: 'center',
|
||||||
|
alignItems: 'center',
|
||||||
|
},
|
||||||
|
todo: {
|
||||||
|
background: '#f7f7f7',
|
||||||
|
color: '#c9c9c9',
|
||||||
|
},
|
||||||
|
inProgress: {
|
||||||
|
background: '#ffffff',
|
||||||
|
color: '#242424',
|
||||||
|
height: '52px',
|
||||||
|
},
|
||||||
|
done: {
|
||||||
|
background: '#f7f7f7',
|
||||||
|
color: '#606060',
|
||||||
|
height: '52px',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function ProgressIndicator({ steps, index }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
function pickClass(i: number): string {
|
||||||
|
if (i === index) {
|
||||||
|
return classes.inProgress
|
||||||
|
}
|
||||||
|
|
||||||
|
return i < index ? classes.done : classes.todo
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
{steps.map((x, i) => (
|
||||||
|
<div key={i} className={`${classes.wrapper} ${pickClass(i)}`}>
|
||||||
|
<Typography>{x}</Typography>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</Grid>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -2,9 +2,10 @@ import { Divider, Drawer, Grid, Link as MUILink, List } from '@material-ui/core'
|
|||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { OpenInNewSharp } from '@material-ui/icons'
|
import { OpenInNewSharp } from '@material-ui/icons'
|
||||||
import type { ReactElement } from 'react'
|
import type { ReactElement } from 'react'
|
||||||
import { BookOpen, DollarSign, FileText, Home, Layers, Settings } from 'react-feather'
|
import { Bookmark, BookOpen, Briefcase, DollarSign, FileText, Gift, Home, Layers, Settings } from 'react-feather'
|
||||||
import { Link } from 'react-router-dom'
|
import { Link } from 'react-router-dom'
|
||||||
import Logo from '../assets/logo.svg'
|
import Logo from '../assets/logo.svg'
|
||||||
|
import { config } from '../config'
|
||||||
import { ROUTES } from '../routes'
|
import { ROUTES } from '../routes'
|
||||||
import SideBarItem from './SideBarItem'
|
import SideBarItem from './SideBarItem'
|
||||||
import SideBarStatus from './SideBarStatus'
|
import SideBarStatus from './SideBarStatus'
|
||||||
@@ -20,6 +21,11 @@ const navBarItems = [
|
|||||||
path: ROUTES.UPLOAD,
|
path: ROUTES.UPLOAD,
|
||||||
icon: FileText,
|
icon: FileText,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: 'Feeds',
|
||||||
|
path: ROUTES.FEEDS,
|
||||||
|
icon: Bookmark,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: 'Stamps',
|
label: 'Stamps',
|
||||||
path: ROUTES.STAMPS,
|
path: ROUTES.STAMPS,
|
||||||
@@ -35,6 +41,16 @@ const navBarItems = [
|
|||||||
path: ROUTES.SETTINGS,
|
path: ROUTES.SETTINGS,
|
||||||
icon: Settings,
|
icon: Settings,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: 'Account',
|
||||||
|
path: ROUTES.WALLET,
|
||||||
|
icon: Briefcase,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Gift Wallets',
|
||||||
|
path: ROUTES.GIFT_CODES,
|
||||||
|
icon: Gift,
|
||||||
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
const drawerWidth = 300
|
const drawerWidth = 300
|
||||||
@@ -111,7 +127,7 @@ export default function SideBar(): ReactElement {
|
|||||||
</List>
|
</List>
|
||||||
<Divider className={classes.divider} />
|
<Divider className={classes.divider} />
|
||||||
<List>
|
<List>
|
||||||
<MUILink href={process.env.REACT_APP_BEE_DOCS_HOST} target="_blank" className={classes.link}>
|
<MUILink href={config.BEE_DOCS_HOST} target="_blank" className={classes.link}>
|
||||||
<SideBarItem
|
<SideBarItem
|
||||||
iconStart={<BookOpen className={classes.icon} />}
|
iconStart={<BookOpen className={classes.icon} />}
|
||||||
iconEnd={<OpenInNewSharp className={classes.iconSmall} />}
|
iconEnd={<OpenInNewSharp className={classes.iconSmall} />}
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ interface Props {
|
|||||||
export default function SideBarItem({ iconStart, iconEnd, path, label }: Props): ReactElement {
|
export default function SideBarItem({ iconStart, iconEnd, path, label }: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const location = useLocation()
|
const location = useLocation()
|
||||||
const isSelected = Boolean(matchPath(location.pathname, { path, exact: true }))
|
const isSelected = Boolean(path && matchPath(location.pathname, path))
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<StyledListItem button selected={isSelected} disableRipple>
|
<StyledListItem button selected={isSelected} disableRipple>
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ export default function SideBarItem({ path }: Props): ReactElement {
|
|||||||
const { status, isLoading } = useContext(Context)
|
const { status, isLoading } = useContext(Context)
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const location = useLocation()
|
const location = useLocation()
|
||||||
const isSelected = Boolean(matchPath(location.pathname, { path, exact: true }))
|
const isSelected = Boolean(path && matchPath(location.pathname, path))
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ListItem
|
<ListItem
|
||||||
@@ -66,11 +66,9 @@ export default function SideBarItem({ path }: Props): ReactElement {
|
|||||||
disableRipple
|
disableRipple
|
||||||
>
|
>
|
||||||
<ListItemIcon style={{ marginLeft: '30px' }}>
|
<ListItemIcon style={{ marginLeft: '30px' }}>
|
||||||
<StatusIcon isOk={status.all} isLoading={isLoading} />
|
<StatusIcon checkState={status.all} isLoading={isLoading} />
|
||||||
</ListItemIcon>
|
</ListItemIcon>
|
||||||
<ListItemText
|
<ListItemText primary={<Typography className={classes.smallerText}>{`Node ${status.all}`}</Typography>} />
|
||||||
primary={<Typography className={classes.smallerText}>{`Node ${status.all ? 'OK' : 'Error'}`}</Typography>}
|
|
||||||
/>
|
|
||||||
<ListItemIcon className={classes.icon}>
|
<ListItemIcon className={classes.icon}>
|
||||||
{status.all ? null : <ArrowRight className={classes.iconSmall} />}
|
{status.all ? null : <ArrowRight className={classes.iconSmall} />}
|
||||||
</ListItemIcon>
|
</ListItemIcon>
|
||||||
|
|||||||
@@ -1,23 +1,40 @@
|
|||||||
import type { ReactElement } from 'react'
|
import type { ReactElement } from 'react'
|
||||||
import { CircularProgress } from '@material-ui/core'
|
import { CircularProgress } from '@material-ui/core'
|
||||||
|
import { CheckState } from '../providers/Bee'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
isOk: boolean
|
checkState: CheckState
|
||||||
isLoading?: boolean
|
isLoading?: boolean
|
||||||
size?: number | string
|
size?: number | string
|
||||||
className?: string
|
className?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function StatusIcon({ isOk, size, className, isLoading }: Props): ReactElement {
|
export default function StatusIcon({ checkState, size, className, isLoading }: Props): ReactElement {
|
||||||
const s = size || '1rem'
|
const s = size || '1rem'
|
||||||
|
|
||||||
if (isLoading) return <CircularProgress size={s} className={className} />
|
if (isLoading) return <CircularProgress size={s} className={className} />
|
||||||
|
|
||||||
|
let backgroundColor: string
|
||||||
|
switch (checkState) {
|
||||||
|
case CheckState.OK:
|
||||||
|
backgroundColor = '#1de600'
|
||||||
|
break
|
||||||
|
case CheckState.WARNING:
|
||||||
|
backgroundColor = 'orange'
|
||||||
|
break
|
||||||
|
case CheckState.ERROR:
|
||||||
|
backgroundColor = '#ff3a52'
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
// Default is error
|
||||||
|
backgroundColor = '#ff3a52'
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<span
|
<span
|
||||||
className={className}
|
className={className}
|
||||||
style={{
|
style={{
|
||||||
backgroundColor: isOk ? '#1de600' : '#ff3a52',
|
backgroundColor,
|
||||||
height: s,
|
height: s,
|
||||||
width: s,
|
width: s,
|
||||||
borderRadius: '50%',
|
borderRadius: '50%',
|
||||||
|
|||||||
@@ -9,13 +9,17 @@ interface Props {
|
|||||||
className?: string
|
className?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
loading?: boolean
|
loading?: boolean
|
||||||
|
cancel?: boolean
|
||||||
|
variant?: 'text' | 'contained' | 'outlined'
|
||||||
}
|
}
|
||||||
|
|
||||||
const useStyles = makeStyles(() =>
|
const useStyles = makeStyles(() =>
|
||||||
createStyles({
|
createStyles({
|
||||||
button: {
|
button: {
|
||||||
|
height: '52px',
|
||||||
position: 'relative',
|
position: 'relative',
|
||||||
whiteSpace: 'nowrap',
|
whiteSpace: 'nowrap',
|
||||||
|
color: '#242424',
|
||||||
'&:hover, &:focus': {
|
'&:hover, &:focus': {
|
||||||
'& svg': {
|
'& svg': {
|
||||||
stroke: '#fff',
|
stroke: '#fff',
|
||||||
@@ -23,6 +27,10 @@ const useStyles = makeStyles(() =>
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
cancelButton: {
|
||||||
|
background: '#f7f7f7',
|
||||||
|
color: '#606060',
|
||||||
|
},
|
||||||
spinnerWrapper: {
|
spinnerWrapper: {
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
left: '50%',
|
left: '50%',
|
||||||
@@ -34,24 +42,43 @@ const useStyles = makeStyles(() =>
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
export function SwarmButton({ children, onClick, iconType, className, disabled, loading }: Props): ReactElement {
|
export function SwarmButton({
|
||||||
|
children,
|
||||||
|
onClick,
|
||||||
|
iconType,
|
||||||
|
className,
|
||||||
|
disabled,
|
||||||
|
loading,
|
||||||
|
cancel,
|
||||||
|
variant = 'contained',
|
||||||
|
}: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
|
|
||||||
|
function getIconColor() {
|
||||||
|
if (loading || disabled) {
|
||||||
|
return 'rgba(0, 0, 0, 0.26)'
|
||||||
|
}
|
||||||
|
|
||||||
|
return cancel ? '#606060' : '#dd7700'
|
||||||
|
}
|
||||||
|
|
||||||
|
function getButtonClassName() {
|
||||||
|
return [className, classes.button, cancel && classes.cancelButton].filter(x => x).join(' ')
|
||||||
|
}
|
||||||
|
|
||||||
const icon = React.createElement(iconType, {
|
const icon = React.createElement(iconType, {
|
||||||
size: '1.25rem',
|
size: '1.25rem',
|
||||||
color: disabled ? 'rgba(0, 0, 0, 0.26)' : '#dd7700',
|
color: getIconColor(),
|
||||||
})
|
})
|
||||||
|
|
||||||
const classNames = className ? [className, classes.button].join(' ') : classes.button
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Button
|
<Button
|
||||||
className={classNames}
|
className={getButtonClassName()}
|
||||||
onClick={(event: React.MouseEvent<HTMLButtonElement>) => {
|
onClick={(event: React.MouseEvent<HTMLButtonElement>) => {
|
||||||
onClick()
|
onClick()
|
||||||
event.currentTarget.blur()
|
event.currentTarget.blur()
|
||||||
}}
|
}}
|
||||||
variant="contained"
|
variant={variant}
|
||||||
startIcon={icon}
|
startIcon={icon}
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { Box, Dialog, Grid } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: ReactElement | ReactElement[]
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SwarmDialog({ children }: Props): ReactElement {
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
open={true}
|
||||||
|
PaperProps={{
|
||||||
|
style: { borderRadius: 0, background: '#efefef' },
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Box p={4} sx={{ maxWidth: '100%', width: '650px' }}>
|
||||||
|
<Grid container direction="column">
|
||||||
|
{children}
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import { Box, Divider } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
my?: number
|
||||||
|
mt?: number
|
||||||
|
mb?: number
|
||||||
|
color?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SwarmDivider({ my, mt, mb, color = '#cbcbcb' }: Props): ReactElement {
|
||||||
|
return (
|
||||||
|
<Box my={my} mt={mt} mb={mb}>
|
||||||
|
<Divider style={{ borderColor: color }} />
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,90 @@
|
|||||||
|
import { createStyles, FormHelperText, makeStyles, MenuItem, Select as SimpleSelect, Theme } from '@material-ui/core'
|
||||||
|
import { Field } from 'formik'
|
||||||
|
import { Select } from 'formik-material-ui'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
export type SelectEvent = React.ChangeEvent<{
|
||||||
|
name?: string | undefined
|
||||||
|
value: unknown
|
||||||
|
}>
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
options: { value: string; label: string }[]
|
||||||
|
onChange?: (event: SelectEvent) => void
|
||||||
|
formik?: boolean
|
||||||
|
defaultValue?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
select: {
|
||||||
|
borderRadius: 0,
|
||||||
|
background: theme.palette.background.paper,
|
||||||
|
'& fieldset': {
|
||||||
|
border: 0,
|
||||||
|
},
|
||||||
|
'& .MuiSelect-select': {
|
||||||
|
'&:focus': {
|
||||||
|
background: theme.palette.background.paper,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
option: {
|
||||||
|
height: '52px',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function SwarmSelect({ defaultValue, formik, name, options, onChange, label }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
if (formik) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{label && <FormHelperText>{label}</FormHelperText>}
|
||||||
|
<Field
|
||||||
|
required
|
||||||
|
component={Select}
|
||||||
|
name={name}
|
||||||
|
fullWidth
|
||||||
|
variant="outlined"
|
||||||
|
defaultValue={defaultValue || ''}
|
||||||
|
className={classes.select}
|
||||||
|
placeholder={label}
|
||||||
|
MenuProps={{ MenuListProps: { disablePadding: true }, PaperProps: { square: true } }}
|
||||||
|
>
|
||||||
|
{options.map((x, i) => (
|
||||||
|
<MenuItem key={i} value={x.value} className={classes.option}>
|
||||||
|
{x.label}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</Field>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{label && <FormHelperText>{label}</FormHelperText>}
|
||||||
|
<SimpleSelect
|
||||||
|
required
|
||||||
|
name={name}
|
||||||
|
fullWidth
|
||||||
|
variant="outlined"
|
||||||
|
className={classes.select}
|
||||||
|
defaultValue={defaultValue || ''}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder={label}
|
||||||
|
MenuProps={{ MenuListProps: { disablePadding: true }, PaperProps: { square: true } }}
|
||||||
|
>
|
||||||
|
{options.map((x, i) => (
|
||||||
|
<MenuItem key={i} value={x.value} className={classes.option}>
|
||||||
|
{x.label}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</SimpleSelect>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
import { createStyles, makeStyles, TextField as SimpleTextField, Theme } from '@material-ui/core'
|
||||||
|
import { Field } from 'formik'
|
||||||
|
import { TextField } from 'formik-material-ui'
|
||||||
|
import { ChangeEvent, ReactElement } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
name: string
|
||||||
|
label: string
|
||||||
|
password?: boolean
|
||||||
|
formik?: boolean
|
||||||
|
optional?: boolean
|
||||||
|
defaultValue?: string
|
||||||
|
onChange?: (event: ChangeEvent<HTMLTextAreaElement>) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
field: {
|
||||||
|
background: theme.palette.background.paper,
|
||||||
|
'& fieldset': {
|
||||||
|
border: 0,
|
||||||
|
},
|
||||||
|
'& .Mui-focused': {
|
||||||
|
background: theme.palette.background.paper,
|
||||||
|
},
|
||||||
|
'& .MuiInputBase-root': {
|
||||||
|
background: theme.palette.background.paper,
|
||||||
|
},
|
||||||
|
'& .MuiFilledInput-root': {
|
||||||
|
borderRadius: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function SwarmTextInput({
|
||||||
|
name,
|
||||||
|
label,
|
||||||
|
password,
|
||||||
|
optional,
|
||||||
|
formik,
|
||||||
|
onChange,
|
||||||
|
defaultValue,
|
||||||
|
}: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
if (formik) {
|
||||||
|
return (
|
||||||
|
<Field
|
||||||
|
component={TextField}
|
||||||
|
type={password ? 'password' : undefined}
|
||||||
|
required={!optional}
|
||||||
|
name={name}
|
||||||
|
label={label}
|
||||||
|
fullWidth
|
||||||
|
variant="filled"
|
||||||
|
className={classes.field}
|
||||||
|
defaultValue={defaultValue || ''}
|
||||||
|
InputProps={{ disableUnderline: true }}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SimpleTextField
|
||||||
|
type={password ? 'password' : undefined}
|
||||||
|
required
|
||||||
|
label={label}
|
||||||
|
fullWidth
|
||||||
|
variant="filled"
|
||||||
|
className={classes.field}
|
||||||
|
defaultValue={defaultValue || ''}
|
||||||
|
onChange={onChange}
|
||||||
|
InputProps={{ disableUnderline: true }}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import { createStyles, Grid, makeStyles, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { CloseButton } from './CloseButton'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: string
|
||||||
|
onClose: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
text: {
|
||||||
|
color: '#606060',
|
||||||
|
fontWeight: 'bold',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function TitleWithClose({ children, onClose }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Grid container justifyContent="space-between" alignItems="center">
|
||||||
|
<span> </span>
|
||||||
|
<Typography className={classes.text} align="center">
|
||||||
|
{children}
|
||||||
|
</Typography>
|
||||||
|
<CloseButton onClose={onClose} />
|
||||||
|
</Grid>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
import type { ReactElement } from 'react'
|
import { Button, Grid, Link as MuiLink, Typography } from '@material-ui/core/'
|
||||||
import { Link } from 'react-router-dom'
|
|
||||||
|
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { Button, Grid, Typography, Link as MuiLink } from '@material-ui/core/'
|
import type { ReactElement } from 'react'
|
||||||
import { ROUTES } from '../routes'
|
|
||||||
import { Activity } from 'react-feather'
|
import { Activity } from 'react-feather'
|
||||||
|
import { Link } from 'react-router-dom'
|
||||||
|
import { config } from '../config'
|
||||||
|
import { ROUTES } from '../routes'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -37,11 +37,11 @@ export default function TroubleshootConnectionCard(): ReactElement {
|
|||||||
<Grid item className={classes.content}>
|
<Grid item className={classes.content}>
|
||||||
<Typography align="center">
|
<Typography align="center">
|
||||||
Please check your node status to fix the problem. You can also check out the{' '}
|
Please check your node status to fix the problem. You can also check out the{' '}
|
||||||
<MuiLink href={process.env.REACT_APP_BEE_DOCS_HOST} target="_blank" rel="noreferrer">
|
<MuiLink href={config.BEE_DOCS_HOST} target="_blank" rel="noreferrer">
|
||||||
Swarm Bee Docs
|
Swarm Bee Docs
|
||||||
</MuiLink>{' '}
|
</MuiLink>{' '}
|
||||||
or ask for support on the{' '}
|
or ask for support on the{' '}
|
||||||
<MuiLink href={process.env.REACT_APP_BEE_DISCORD_HOST} target="_blank" rel="noreferrer">
|
<MuiLink href={config.BEE_DISCORD_HOST} target="_blank" rel="noreferrer">
|
||||||
Ethereum Swarm Discord
|
Ethereum Swarm Discord
|
||||||
</MuiLink>
|
</MuiLink>
|
||||||
.
|
.
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
function getProcessEnv(key: string): string | undefined | false {
|
||||||
|
return typeof process === 'object' && process.env[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
class Config {
|
||||||
|
public readonly BEE_API_HOST: string
|
||||||
|
public readonly BEE_DEBUG_API_HOST: string
|
||||||
|
public readonly BLOCKCHAIN_EXPLORER_URL: string
|
||||||
|
public readonly BEE_DOCS_HOST: string
|
||||||
|
public readonly BEE_DISCORD_HOST: string
|
||||||
|
public readonly GITHUB_REPO_URL: string
|
||||||
|
public readonly BEE_DESKTOP_URL: string
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.BEE_API_HOST =
|
||||||
|
sessionStorage.getItem('api_host') || getProcessEnv('REACT_APP_BEE_HOST') || 'http://localhost:1633'
|
||||||
|
this.BEE_DEBUG_API_HOST =
|
||||||
|
sessionStorage.getItem('debug_api_host') || getProcessEnv('REACT_APP_BEE_DEBUG_HOST') || 'http://localhost:1635'
|
||||||
|
this.BLOCKCHAIN_EXPLORER_URL =
|
||||||
|
getProcessEnv('REACT_APP_BLOCKCHAIN_EXPLORER_URL') || 'https://blockscout.com/xdai/mainnet'
|
||||||
|
this.BEE_DOCS_HOST = getProcessEnv('REACT_APP_BEE_DOCS_HOST') || 'https://docs.ethswarm.org/docs/'
|
||||||
|
this.BEE_DISCORD_HOST = getProcessEnv('REACT_APP_BEE_DISCORD_HOST') || 'https://discord.gg/eKr9XPv7'
|
||||||
|
this.GITHUB_REPO_URL =
|
||||||
|
getProcessEnv('REACT_APP_BEE_GITHUB_REPO_URL') || 'https://api.github.com/repos/ethersphere/bee'
|
||||||
|
this.BEE_DESKTOP_URL = getProcessEnv('REACT_APP_BEE_DESKTOP_URL') || window.location.origin
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const config = new Config()
|
||||||
|
|
||||||
|
export default config
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
export const META_FILE_NAME = '.swarmgatewaymeta.json'
|
||||||
|
export const PREVIEW_FILE_NAME = '.swarmgatewaypreview.jpeg'
|
||||||
|
export const PREVIEW_DIMENSIONS = { maxWidth: 250, maxHeight: 175 }
|
||||||
|
export const BZZ_LINK_DOMAIN = process.env.REACT_APP_BZZ_LINK_DOMAIN || 'bzz.link'
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
import { renderHook } from '@testing-library/react-hooks'
|
||||||
|
import express from 'express'
|
||||||
|
import cors from 'cors'
|
||||||
|
import type { Server } from 'http'
|
||||||
|
import { useIsBeeDesktop } from './apiHooks'
|
||||||
|
|
||||||
|
interface AddressInfo {
|
||||||
|
address: string
|
||||||
|
family: string
|
||||||
|
port: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mockServer(data: Record<string | number | symbol, string>): Promise<Server> {
|
||||||
|
const app = express()
|
||||||
|
app.use(cors())
|
||||||
|
|
||||||
|
app.get('/info', (req, res) => {
|
||||||
|
res.send(data)
|
||||||
|
})
|
||||||
|
|
||||||
|
return new Promise(resolve => {
|
||||||
|
const server = app.listen(() => {
|
||||||
|
resolve(server)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
let serverCorrect: Server
|
||||||
|
let serverWrong: Server
|
||||||
|
|
||||||
|
let serverCorrectURL: string
|
||||||
|
let serverWrongURL: string
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
serverCorrect = await mockServer({ name: 'bee-desktop' })
|
||||||
|
const portServerCorrect = (serverCorrect.address() as AddressInfo).port
|
||||||
|
serverCorrectURL = `http://localhost:${portServerCorrect}`
|
||||||
|
|
||||||
|
serverWrong = await mockServer({ foo: 'bar' })
|
||||||
|
const portServerWrong = (serverWrong.address() as AddressInfo).port
|
||||||
|
serverWrongURL = `http://localhost:${portServerWrong}`
|
||||||
|
})
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
await new Promise(resolve => serverCorrect.close(resolve))
|
||||||
|
await new Promise(resolve => serverWrong.close(resolve))
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('useIsBeeDesktop', () => {
|
||||||
|
it('should fail when connected to wrong server', async () => {
|
||||||
|
const { result, waitFor } = renderHook(() => useIsBeeDesktop({ BEE_DESKTOP_URL: serverWrongURL }))
|
||||||
|
|
||||||
|
expect(result.current.isLoading).toBe(true)
|
||||||
|
expect(result.current.isBeeDesktop).toBe(false)
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(result.current.isLoading).toBe(false)
|
||||||
|
})
|
||||||
|
expect(result.current.isBeeDesktop).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return isBeeDesktop true when connected to bee-desktop', async () => {
|
||||||
|
const { result, waitFor } = renderHook(() => useIsBeeDesktop({ BEE_DESKTOP_URL: serverCorrectURL }))
|
||||||
|
|
||||||
|
expect(result.current.isLoading).toBe(true)
|
||||||
|
expect(result.current.isBeeDesktop).toBe(false)
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(result.current.isLoading).toBe(false)
|
||||||
|
})
|
||||||
|
expect(result.current.isBeeDesktop).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
||||||
+88
-2
@@ -1,5 +1,7 @@
|
|||||||
import { useState, useEffect } from 'react'
|
|
||||||
import axios from 'axios'
|
import axios from 'axios'
|
||||||
|
import { useEffect, useState } from 'react'
|
||||||
|
import { config } from '../config'
|
||||||
|
import { getJson } from '../utils/net'
|
||||||
|
|
||||||
export interface LatestBeeReleaseHook {
|
export interface LatestBeeReleaseHook {
|
||||||
latestBeeRelease: LatestBeeRelease | null
|
latestBeeRelease: LatestBeeRelease | null
|
||||||
@@ -7,6 +9,90 @@ export interface LatestBeeReleaseHook {
|
|||||||
error: Error | null
|
error: Error | null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface IsBeeDesktopHook {
|
||||||
|
isBeeDesktop: boolean
|
||||||
|
isLoading: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Config {
|
||||||
|
BEE_DESKTOP_URL: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detect if the dashboard is run within bee-desktop
|
||||||
|
*
|
||||||
|
* @returns isBeeDesktop true if this is run within bee-desktop
|
||||||
|
*/
|
||||||
|
export const useIsBeeDesktop = (conf: Config = config): IsBeeDesktopHook => {
|
||||||
|
const [isBeeDesktop, setIsBeeDesktop] = useState<boolean>(false)
|
||||||
|
const [isLoading, setLoading] = useState<boolean>(true)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
axios
|
||||||
|
.get(`${conf.BEE_DESKTOP_URL}/info`)
|
||||||
|
.then(res => {
|
||||||
|
if (res.data?.name === 'bee-desktop') setIsBeeDesktop(true)
|
||||||
|
else setIsBeeDesktop(false)
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
setIsBeeDesktop(false)
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
setLoading(false)
|
||||||
|
})
|
||||||
|
}, [conf])
|
||||||
|
|
||||||
|
return { isBeeDesktop, isLoading }
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface BeeConfig {
|
||||||
|
'api-addr': string
|
||||||
|
'debug-api-addr': string
|
||||||
|
'debug-api-enable': boolean
|
||||||
|
password: string
|
||||||
|
'swap-enable': boolean
|
||||||
|
'swap-initial-deposit': bigint
|
||||||
|
mainnet: boolean
|
||||||
|
'full-node': boolean
|
||||||
|
'chain-enable': boolean
|
||||||
|
'cors-allowed-origins': string
|
||||||
|
'resolver-options': string
|
||||||
|
'use-postage-snapshot': boolean
|
||||||
|
'data-dir': string
|
||||||
|
transaction: string
|
||||||
|
'block-hash': string
|
||||||
|
'swap-endpoint'?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface GetBeeConfig {
|
||||||
|
config: BeeConfig | null
|
||||||
|
isLoading: boolean
|
||||||
|
error: Error | null
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useGetBeeConfig = (conf: Config = config): GetBeeConfig => {
|
||||||
|
const [beeConfig, setBeeConfig] = useState<BeeConfig | null>(null)
|
||||||
|
const [isLoading, setLoading] = useState<boolean>(true)
|
||||||
|
const [error, setError] = useState<Error | null>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
getJson<BeeConfig>(`${conf.BEE_DESKTOP_URL}/config`)
|
||||||
|
.then(beeConf => {
|
||||||
|
setBeeConfig(beeConf)
|
||||||
|
setError(null)
|
||||||
|
})
|
||||||
|
.catch((err: Error) => {
|
||||||
|
setError(err)
|
||||||
|
setBeeConfig(null)
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
setLoading(false)
|
||||||
|
})
|
||||||
|
}, [conf])
|
||||||
|
|
||||||
|
return { config: beeConfig, isLoading, error }
|
||||||
|
}
|
||||||
|
|
||||||
export const useLatestBeeRelease = (): LatestBeeReleaseHook => {
|
export const useLatestBeeRelease = (): LatestBeeReleaseHook => {
|
||||||
const [latestBeeRelease, setLatestBeeRelease] = useState<LatestBeeRelease | null>(null)
|
const [latestBeeRelease, setLatestBeeRelease] = useState<LatestBeeRelease | null>(null)
|
||||||
const [isLoadingLatestBeeRelease, setLoading] = useState<boolean>(false)
|
const [isLoadingLatestBeeRelease, setLoading] = useState<boolean>(false)
|
||||||
@@ -14,7 +100,7 @@ export const useLatestBeeRelease = (): LatestBeeReleaseHook => {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
axios
|
axios
|
||||||
.get(`${process.env.REACT_APP_BEE_GITHUB_REPO_URL}/releases/latest`)
|
.get(`${config.GITHUB_REPO_URL}/releases/latest`)
|
||||||
.then(res => {
|
.then(res => {
|
||||||
setLatestBeeRelease(res.data)
|
setLatestBeeRelease(res.data)
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import { useContext, ReactElement } from 'react'
|
import { CircularProgress, Container } from '@material-ui/core'
|
||||||
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
|
import { ReactElement, useContext } from 'react'
|
||||||
import ErrorBoundary from '../components/ErrorBoundary'
|
import ErrorBoundary from '../components/ErrorBoundary'
|
||||||
import AlertVersion from '../components/AlertVersion'
|
|
||||||
import { Container, CircularProgress } from '@material-ui/core'
|
|
||||||
|
|
||||||
import { createStyles, Theme, makeStyles } from '@material-ui/core/styles'
|
|
||||||
|
|
||||||
import SideBar from '../components/SideBar'
|
import SideBar from '../components/SideBar'
|
||||||
|
|
||||||
import { Context } from '../providers/Bee'
|
import { Context } from '../providers/Bee'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
@@ -33,7 +29,6 @@ const Dashboard = (props: Props): ReactElement => {
|
|||||||
<Container className={classes.content}>
|
<Container className={classes.content}>
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<>
|
<>
|
||||||
<AlertVersion />
|
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<div style={{ textAlign: 'center', width: '100%' }}>
|
<div style={{ textAlign: 'center', width: '100%' }}>
|
||||||
<CircularProgress />
|
<CircularProgress />
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
import { BigNumber } from 'bignumber.js'
|
||||||
|
import { Token } from './Token'
|
||||||
|
|
||||||
|
export class BzzToken extends Token {
|
||||||
|
constructor(amount: BigNumber | string | BigInt) {
|
||||||
|
super(amount, 16)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
import { BigNumber } from 'bignumber.js'
|
||||||
|
import { Token } from './Token'
|
||||||
|
|
||||||
|
export class DaiToken extends Token {
|
||||||
|
constructor(amount: BigNumber | string | BigInt) {
|
||||||
|
super(amount, 18)
|
||||||
|
}
|
||||||
|
}
|
||||||
+31
-1
@@ -13,7 +13,9 @@ export class Token {
|
|||||||
constructor(amount: BigNumber | string | BigInt, decimals: digits = BZZ_DECIMALS) {
|
constructor(amount: BigNumber | string | BigInt, decimals: digits = BZZ_DECIMALS) {
|
||||||
const a = makeBigNumber(amount)
|
const a = makeBigNumber(amount)
|
||||||
|
|
||||||
if (!isInteger(a) || !POSSIBLE_DECIMALS.includes(decimals)) throw new TypeError('Not a valid token values')
|
if (!isInteger(a) || !POSSIBLE_DECIMALS.includes(decimals)) {
|
||||||
|
throw new TypeError(`Not a valid token values: ${amount} ${decimals}`)
|
||||||
|
}
|
||||||
|
|
||||||
this.amount = a
|
this.amount = a
|
||||||
this.decimals = decimals
|
this.decimals = decimals
|
||||||
@@ -57,4 +59,32 @@ export class Token {
|
|||||||
toFixedDecimal(digits = 7): string {
|
toFixedDecimal(digits = 7): string {
|
||||||
return this.toDecimal.toFixed(digits)
|
return this.toDecimal.toFixed(digits)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
toSignificantDigits(digits = 4): string {
|
||||||
|
const asString = this.toDecimal.toFixed(this.decimals)
|
||||||
|
|
||||||
|
let indexOfSignificantDigit = -1
|
||||||
|
let reachedDecimalPoint = false
|
||||||
|
|
||||||
|
for (let i = 0; i < asString.length; i++) {
|
||||||
|
const char = asString[i]
|
||||||
|
|
||||||
|
if (char === '.') {
|
||||||
|
reachedDecimalPoint = true
|
||||||
|
indexOfSignificantDigit = i + 1
|
||||||
|
} else if (reachedDecimalPoint && char !== '0') {
|
||||||
|
indexOfSignificantDigit = i
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return asString.slice(0, indexOfSignificantDigit + digits)
|
||||||
|
}
|
||||||
|
|
||||||
|
minusBaseUnits(amount: string): Token {
|
||||||
|
return new Token(
|
||||||
|
this.toBigNumber.minus(new BigNumber(amount).multipliedBy(new BigNumber(10).pow(this.decimals))),
|
||||||
|
this.decimals,
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,9 @@
|
|||||||
import type { ReactElement } from 'react'
|
import type { ReactElement } from 'react'
|
||||||
|
import CashoutModal from '../../components/CashoutModal'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
import ExpandableListItem from '../../components/ExpandableListItem'
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
|
||||||
import CashoutModal from '../../components/CashoutModal'
|
|
||||||
import { Accounting } from '../../hooks/accounting'
|
import { Accounting } from '../../hooks/accounting'
|
||||||
import type { Token } from '../../models/Token'
|
import type { Token } from '../../models/Token'
|
||||||
|
|
||||||
@@ -25,7 +23,7 @@ export default function PeerBalances({ accounting, isLoadingUncashed, totalUncas
|
|||||||
{accounting?.map(({ peer, balance, received, sent, uncashedAmount, total }) => (
|
{accounting?.map(({ peer, balance, received, sent, uncashedAmount, total }) => (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
key={peer}
|
key={peer}
|
||||||
label={`Peer ${peer.substr(0, 8)}[…]`}
|
label={`Peer ${peer.slice(0, 8)}[…]`}
|
||||||
level={1}
|
level={1}
|
||||||
info={`${uncashedAmount.toFixedDecimal()} BZZ (uncashed)`}
|
info={`${uncashedAmount.toFixedDecimal()} BZZ (uncashed)`}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { ReactElement, useContext } from 'react'
|
|||||||
|
|
||||||
import PeerBalances from './PeerBalances'
|
import PeerBalances from './PeerBalances'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { useAccounting } from '../../hooks/accounting'
|
import { useAccounting } from '../../hooks/accounting'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
@@ -19,7 +19,7 @@ export default function Accounting(): ReactElement {
|
|||||||
|
|
||||||
const { accounting, totalUncashed, isLoadingUncashed } = useAccounting(beeDebugApi, settlements, peerBalances)
|
const { accounting, totalUncashed, isLoadingUncashed } = useAccounting(beeDebugApi, settlements, peerBalances)
|
||||||
|
|
||||||
if (!status.all) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
@@ -0,0 +1,133 @@
|
|||||||
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
|
import { Form, Formik } from 'formik'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { Check, X } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmSelect } from '../../components/SwarmSelect'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { Context as FeedsContext, IdentityType } from '../../providers/Feeds'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { convertWalletToIdentity, generateWallet, persistIdentity } from '../../utils/identity'
|
||||||
|
|
||||||
|
interface FormValues {
|
||||||
|
identityName?: string
|
||||||
|
type?: IdentityType
|
||||||
|
password?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const initialValues: FormValues = {
|
||||||
|
identityName: '',
|
||||||
|
type: 'PRIVATE_KEY',
|
||||||
|
password: '',
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function CreateNewFeed(): ReactElement {
|
||||||
|
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
||||||
|
const { identities, setIdentities } = useContext(FeedsContext)
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
async function onSubmit(values: FormValues) {
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
|
if (!beeApi) {
|
||||||
|
enqueueSnackbar(<span>Bee API unavailabe</span>, { variant: 'error' })
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const wallet = generateWallet()
|
||||||
|
const stamps = await beeDebugApi?.getAllPostageBatch()
|
||||||
|
|
||||||
|
if (!stamps || !stamps.length) {
|
||||||
|
enqueueSnackbar(<span>No stamp available</span>, { variant: 'error' })
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!values.identityName || !values.type) {
|
||||||
|
enqueueSnackbar(<span>Form is unfinished</span>, { variant: 'error' })
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const identity = await convertWalletToIdentity(wallet, values.type, values.identityName, values.password)
|
||||||
|
persistIdentity(identities, identity)
|
||||||
|
setIdentities(identities)
|
||||||
|
navigate(ROUTES.FEEDS)
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
function cancel() {
|
||||||
|
navigate(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<HistoryHeader>Create new feed</HistoryHeader>
|
||||||
|
<Box mb={4}>
|
||||||
|
<DocumentationText>
|
||||||
|
To create a feed you will need to create an identity. Please refer to the{' '}
|
||||||
|
<a
|
||||||
|
href="https://docs.ethswarm.org/api/#tag/Feed/paths/~1feeds~1{owner}~1{topic}/post"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
official Bee documentation
|
||||||
|
</a>{' '}
|
||||||
|
to understand how feeds work.
|
||||||
|
</DocumentationText>
|
||||||
|
</Box>
|
||||||
|
<Formik initialValues={initialValues} onSubmit={onSubmit}>
|
||||||
|
{({ submitForm, values }) => (
|
||||||
|
<Form>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<SwarmTextInput name="identityName" label="Identity name" formik />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<SwarmSelect
|
||||||
|
formik
|
||||||
|
name="type"
|
||||||
|
options={[
|
||||||
|
{ label: 'Keypair Only', value: 'PRIVATE_KEY' },
|
||||||
|
{ label: 'Password Protected', value: 'V3' },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
{values.type === 'V3' && <SwarmTextInput name="password" label="Password" password formik />}
|
||||||
|
<Box mt={2}>
|
||||||
|
<ExpandableListItemKey label="Topic" value={'00'.repeat(32)} />
|
||||||
|
</Box>
|
||||||
|
<Box mt={2} sx={{ bgcolor: '#fcf2e8' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Feeds name</Typography>
|
||||||
|
<Typography>{values.identityName} Website</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<Box mt={1.25}>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton onClick={submitForm} iconType={Check} disabled={loading} loading={loading}>
|
||||||
|
Create Feed
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={cancel} iconType={X} disabled={loading} cancel>
|
||||||
|
Cancel
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Box>
|
||||||
|
</Form>
|
||||||
|
)}
|
||||||
|
</Formik>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { Trash, X } from 'react-feather'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDialog } from '../../components/SwarmDialog'
|
||||||
|
import { TitleWithClose } from '../../components/TitleWithClose'
|
||||||
|
import { Identity } from '../../providers/Feeds'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
identity: Identity
|
||||||
|
onConfirm: (identity: Identity) => void
|
||||||
|
onClose: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export function DeleteFeedDialog({ identity, onConfirm, onClose }: Props): ReactElement {
|
||||||
|
return (
|
||||||
|
<SwarmDialog>
|
||||||
|
<Box mb={4}>
|
||||||
|
<TitleWithClose onClose={onClose}>Delete</TitleWithClose>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography align="center">{`You are about to delete feed ${identity.name} Website. It is strongly advised to export this feed first.`}</Typography>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={Trash} onClick={() => onConfirm(identity)}>
|
||||||
|
Delete
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={X} onClick={onClose} cancel>
|
||||||
|
Cancel
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</SwarmDialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,71 @@
|
|||||||
|
import { Box, createStyles, makeStyles, Typography } from '@material-ui/core'
|
||||||
|
import { saveAs } from 'file-saver'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { Clipboard, Download } from 'react-feather'
|
||||||
|
import { Code } from '../../components/Code'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDialog } from '../../components/SwarmDialog'
|
||||||
|
import { TitleWithClose } from '../../components/TitleWithClose'
|
||||||
|
import { Identity } from '../../providers/Feeds'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
identity: Identity
|
||||||
|
onClose: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
wrapper: {
|
||||||
|
maxWidth: '100%',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function ExportFeedDialog({ identity, onClose }: Props): ReactElement {
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
function onDownload() {
|
||||||
|
saveAs(
|
||||||
|
new Blob([identity.identity], {
|
||||||
|
type: 'application/json',
|
||||||
|
}),
|
||||||
|
identity.name + '.json',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getExportText() {
|
||||||
|
return identity.type === 'V3' ? 'JSON file' : 'the private key string'
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCopy() {
|
||||||
|
navigator.clipboard
|
||||||
|
.writeText(identity.identity)
|
||||||
|
.then(() => enqueueSnackbar('Copied to Clipboard', { variant: 'success' }))
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SwarmDialog>
|
||||||
|
<Box mb={4}>
|
||||||
|
<TitleWithClose onClose={onClose}>Export</TitleWithClose>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography align="center">{`We exported the identity associated with this feed as ${getExportText()}.`}</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4} className={classes.wrapper}>
|
||||||
|
<Code prettify>{identity.identity}</Code>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={Download} onClick={onDownload}>
|
||||||
|
Download JSON File
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={Clipboard} onClick={onCopy}>
|
||||||
|
Copy To Clipboard
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</SwarmDialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement, useState } from 'react'
|
||||||
|
import { Check, X } from 'react-feather'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDialog } from '../../components/SwarmDialog'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { TitleWithClose } from '../../components/TitleWithClose'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
feedName: string
|
||||||
|
onProceed: (password: string) => void
|
||||||
|
onCancel: () => void
|
||||||
|
loading: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FeedPasswordDialog({ feedName, onProceed, onCancel, loading }: Props): ReactElement {
|
||||||
|
const [password, setPassword] = useState('')
|
||||||
|
|
||||||
|
function onProceedClick() {
|
||||||
|
return onProceed(password)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SwarmDialog>
|
||||||
|
<Box mb={4}>
|
||||||
|
<TitleWithClose onClose={onCancel}>Update Feed</TitleWithClose>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography>Please enter the password for “{feedName}”:</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<SwarmTextInput
|
||||||
|
label="Password"
|
||||||
|
name="password"
|
||||||
|
onChange={event => {
|
||||||
|
setPassword(event.target.value)
|
||||||
|
}}
|
||||||
|
password
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={Check} onClick={onProceedClick} disabled={loading} loading={loading}>
|
||||||
|
Proceed
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={X} onClick={onCancel} cancel disabled={loading}>
|
||||||
|
Cancel
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</SwarmDialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,90 @@
|
|||||||
|
import * as swarmCid from '@ethersphere/swarm-cid'
|
||||||
|
import { Box } from '@material-ui/core'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { X } from 'react-feather'
|
||||||
|
import { useParams, useNavigate } from 'react-router-dom'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as IdentityContext } from '../../providers/Feeds'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { UploadArea } from '../files/UploadArea'
|
||||||
|
|
||||||
|
export function FeedSubpage(): ReactElement {
|
||||||
|
const { identities } = useContext(IdentityContext)
|
||||||
|
const { uuid } = useParams()
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
const { status } = useContext(BeeContext)
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
const [available, setAvailable] = useState(false)
|
||||||
|
|
||||||
|
const identity = identities.find(x => x.uuid === uuid)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!identity || !identity.feedHash) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
beeApi?.downloadData(identity.feedHash).then(() => setAvailable(true))
|
||||||
|
} catch {
|
||||||
|
setAvailable(false)
|
||||||
|
}
|
||||||
|
}, [beeApi, uuid, identity])
|
||||||
|
|
||||||
|
if (!identity || !status.all) {
|
||||||
|
navigate(ROUTES.FEEDS, { replace: true })
|
||||||
|
|
||||||
|
return <></>
|
||||||
|
}
|
||||||
|
|
||||||
|
function onClose() {
|
||||||
|
navigate(ROUTES.FEEDS)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<HistoryHeader>{`${identity.name} Website`}</HistoryHeader>
|
||||||
|
<UploadArea showHelp={false} uploadOrigin={{ origin: 'FEED', uuid }} />
|
||||||
|
{available && identity.feedHash ? (
|
||||||
|
<>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItemKey label="Feed hash" value={identity.feedHash} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ExpandableListItemLink
|
||||||
|
label="BZZ Link"
|
||||||
|
value={`https://${swarmCid.encodeFeedReference(identity.feedHash)}.bzz.link`}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<Box mb={4}>
|
||||||
|
<DocumentationText>
|
||||||
|
This feed is curently not pointing anywhere, you can update the feed to fix this. Please refer to the{' '}
|
||||||
|
<a
|
||||||
|
href="https://docs.ethswarm.org/api/#tag/Feed/paths/~1feeds~1{owner}~1{topic}/post"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
official Bee documentation
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</DocumentationText>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={X} onClick={onClose} cancel>
|
||||||
|
Close
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,113 @@
|
|||||||
|
import { Box, createStyles, makeStyles, TextareaAutosize, Theme } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import React, { ReactElement, useContext, useRef, useState } from 'react'
|
||||||
|
import { Check, Upload } from 'react-feather'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDialog } from '../../components/SwarmDialog'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { TitleWithClose } from '../../components/TitleWithClose'
|
||||||
|
import { Context, Identity } from '../../providers/Feeds'
|
||||||
|
import { importIdentity, persistIdentity } from '../../utils/identity'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onClose: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
textarea: {
|
||||||
|
width: '100%',
|
||||||
|
border: 0,
|
||||||
|
padding: theme.spacing(1),
|
||||||
|
},
|
||||||
|
displayNone: {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function ImportFeedDialog({ onClose }: Props): ReactElement {
|
||||||
|
const [textareaValue, setTextareaValue] = useState('')
|
||||||
|
const [name, setName] = useState('')
|
||||||
|
const fileInputRef = useRef(null)
|
||||||
|
|
||||||
|
const { identities, setIdentities } = useContext(Context)
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
async function onImport() {
|
||||||
|
const feed = await importIdentity(name, textareaValue)
|
||||||
|
|
||||||
|
if (feed) {
|
||||||
|
onFeedReady(feed)
|
||||||
|
} else {
|
||||||
|
enqueueSnackbar('Feed is not valid', { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onUploadIdentityFile() {
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
const input = fileInputRef.current as HTMLInputElement
|
||||||
|
input.click()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onIdentityFileSelected(event: React.ChangeEvent<HTMLInputElement>) {
|
||||||
|
const fileReader = new FileReader()
|
||||||
|
const file = event.target?.files?.[0]
|
||||||
|
fileReader.onload = async event => {
|
||||||
|
const string = event.target?.result
|
||||||
|
|
||||||
|
if (string) {
|
||||||
|
const feed = await importIdentity(name, string as string)
|
||||||
|
|
||||||
|
if (feed) {
|
||||||
|
onFeedReady(feed)
|
||||||
|
} else {
|
||||||
|
enqueueSnackbar('Feed is not valid', { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file) {
|
||||||
|
fileReader.readAsText(file)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onFeedReady(identity: Identity) {
|
||||||
|
persistIdentity(identities, identity)
|
||||||
|
setIdentities(identities)
|
||||||
|
enqueueSnackbar('Feed imported successfully', { variant: 'success' })
|
||||||
|
onClose()
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SwarmDialog>
|
||||||
|
<input onChange={onIdentityFileSelected} ref={fileInputRef} className={classes.displayNone} type="file" />
|
||||||
|
<Box mb={4}>
|
||||||
|
<TitleWithClose onClose={onClose}>Import</TitleWithClose>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput label="Identity Name" name="name" onChange={event => setName(event.target.value)} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<TextareaAutosize
|
||||||
|
className={classes.textarea}
|
||||||
|
minRows={5}
|
||||||
|
onChange={event => setTextareaValue(event.target.value)}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={Upload} onClick={onUploadIdentityFile}>
|
||||||
|
Upload Json File
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={Check} onClick={onImport}>
|
||||||
|
Use Pasted Text
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</SwarmDialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,148 @@
|
|||||||
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { Bookmark, X } from 'react-feather'
|
||||||
|
import { useParams, useNavigate } from 'react-router'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SelectEvent, SwarmSelect } from '../../components/SwarmSelect'
|
||||||
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as IdentityContext, Identity } from '../../providers/Feeds'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Context as StampContext } from '../../providers/Stamps'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { persistIdentity, updateFeed } from '../../utils/identity'
|
||||||
|
import { FeedPasswordDialog } from './FeedPasswordDialog'
|
||||||
|
|
||||||
|
export default function UpdateFeed(): ReactElement {
|
||||||
|
const { identities, setIdentities } = useContext(IdentityContext)
|
||||||
|
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
||||||
|
const { stamps, refresh } = useContext(StampContext)
|
||||||
|
const { status } = useContext(BeeContext)
|
||||||
|
const { hash } = useParams()
|
||||||
|
|
||||||
|
const [selectedStamp, setSelectedStamp] = useState<string | null>(null)
|
||||||
|
const [selectedIdentity, setSelectedIdentity] = useState<Identity | null>(null)
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const [showPasswordPrompt, setShowPasswordPrompt] = useState(false)
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
refresh()
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
function onFeedChange(event: SelectEvent) {
|
||||||
|
const uuid = event.target.value
|
||||||
|
setSelectedIdentity(identities.find(x => x.uuid === uuid) || null)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onStampChange(event: SelectEvent) {
|
||||||
|
const batchId = event.target.value as string
|
||||||
|
setSelectedStamp(batchId)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCancel() {
|
||||||
|
navigate(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onBeginUpdatingFeed() {
|
||||||
|
if (!selectedIdentity) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedIdentity.type === 'V3') {
|
||||||
|
setShowPasswordPrompt(true)
|
||||||
|
} else {
|
||||||
|
onFeedUpdate(selectedIdentity)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onFeedUpdate(identity: Identity, password?: string) {
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
|
if (!beeApi || !beeDebugApi || !selectedStamp) {
|
||||||
|
enqueueSnackbar(<span>Bee API unavailabe</span>, { variant: 'error' })
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await updateFeed(beeApi, identity, hash!, selectedStamp, password as string) // eslint-disable-line
|
||||||
|
persistIdentity(identities, identity)
|
||||||
|
setIdentities([...identities])
|
||||||
|
navigate(ROUTES.FEEDS_PAGE.replace(':uuid', identity.uuid))
|
||||||
|
} catch (error: unknown) {
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
|
const message = (typeof error === 'object' && error !== null && Reflect.get(error, 'message')) || ''
|
||||||
|
|
||||||
|
if (message.includes('possibly wrong passphrase')) {
|
||||||
|
enqueueSnackbar('Wrong password, please try again', { variant: 'error' })
|
||||||
|
} else {
|
||||||
|
enqueueSnackbar('Could not update feed at this time, please try again later', { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!status.all) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
{showPasswordPrompt && selectedIdentity && (
|
||||||
|
<FeedPasswordDialog
|
||||||
|
feedName={selectedIdentity.name + ' Website'}
|
||||||
|
onCancel={() => {
|
||||||
|
setShowPasswordPrompt(false)
|
||||||
|
}}
|
||||||
|
onProceed={(password: string) => {
|
||||||
|
onFeedUpdate(selectedIdentity, password)
|
||||||
|
}}
|
||||||
|
loading={loading}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<HistoryHeader>Update feed</HistoryHeader>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Grid container>
|
||||||
|
<SwarmSelect
|
||||||
|
options={identities.map(x => ({ value: x.uuid, label: `${x.name} Website` }))}
|
||||||
|
onChange={onFeedChange}
|
||||||
|
label="Feed"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
<Box mb={4}>
|
||||||
|
<Grid container>
|
||||||
|
{stamps ? (
|
||||||
|
<SwarmSelect
|
||||||
|
options={stamps.map(x => ({ value: x.batchID, label: x.batchID.slice(0, 8) }))}
|
||||||
|
onChange={onStampChange}
|
||||||
|
label="Stamp"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Typography>You need to buy a stamp first to be able to update a feed.</Typography>
|
||||||
|
)}
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton
|
||||||
|
onClick={onBeginUpdatingFeed}
|
||||||
|
iconType={Bookmark}
|
||||||
|
loading={!showPasswordPrompt && loading}
|
||||||
|
disabled={loading || !selectedStamp || !selectedIdentity}
|
||||||
|
>
|
||||||
|
Update Selected Feed
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={onCancel} iconType={X} disabled={loading} cancel>
|
||||||
|
Close
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,116 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { Download, Info, PlusSquare, Trash } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
|
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as IdentityContext, Identity } from '../../providers/Feeds'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { formatEnum } from '../../utils'
|
||||||
|
import { persistIdentitiesWithoutUpdate } from '../../utils/identity'
|
||||||
|
import { DeleteFeedDialog } from './DeleteFeedDialog'
|
||||||
|
import { ExportFeedDialog } from './ExportFeedDialog'
|
||||||
|
import { ImportFeedDialog } from './ImportFeedDialog'
|
||||||
|
|
||||||
|
export default function Feeds(): ReactElement {
|
||||||
|
const { identities, setIdentities } = useContext(IdentityContext)
|
||||||
|
const { status } = useContext(BeeContext)
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
const [selectedIdentity, setSelectedIdentity] = useState<Identity | null>(null)
|
||||||
|
const [showImport, setShowImport] = useState(false)
|
||||||
|
const [showExport, setShowExport] = useState(false)
|
||||||
|
const [showDelete, setShowDelete] = useState(false)
|
||||||
|
|
||||||
|
function createNewFeed() {
|
||||||
|
return navigate(ROUTES.FEEDS_NEW)
|
||||||
|
}
|
||||||
|
|
||||||
|
function viewFeed(uuid: string) {
|
||||||
|
navigate(ROUTES.FEEDS_PAGE.replace(':uuid', uuid))
|
||||||
|
}
|
||||||
|
|
||||||
|
function onDialogClose() {
|
||||||
|
setShowDelete(false)
|
||||||
|
setShowExport(false)
|
||||||
|
setShowImport(false)
|
||||||
|
setSelectedIdentity(null)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onDelete(identity: Identity) {
|
||||||
|
onDialogClose()
|
||||||
|
const updatedFeeds = identities.filter(x => x.uuid !== identity.uuid)
|
||||||
|
setIdentities(updatedFeeds)
|
||||||
|
persistIdentitiesWithoutUpdate(updatedFeeds)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onShowExport(identity: Identity) {
|
||||||
|
setSelectedIdentity(identity)
|
||||||
|
setShowExport(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onShowDelete(identity: Identity) {
|
||||||
|
setSelectedIdentity(identity)
|
||||||
|
setShowDelete(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
{showImport && <ImportFeedDialog onClose={() => setShowImport(false)} />}
|
||||||
|
{showExport && selectedIdentity && <ExportFeedDialog identity={selectedIdentity} onClose={onDialogClose} />}
|
||||||
|
{showDelete && selectedIdentity && (
|
||||||
|
<DeleteFeedDialog
|
||||||
|
identity={selectedIdentity}
|
||||||
|
onClose={onDialogClose}
|
||||||
|
onConfirm={(identity: Identity) => onDelete(identity)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography variant="h1">Feeds</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={PlusSquare} onClick={createNewFeed}>
|
||||||
|
Create New Feed
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={PlusSquare} onClick={() => setShowImport(true)}>
|
||||||
|
Import Feed
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Box>
|
||||||
|
{identities.map((x, i) => (
|
||||||
|
<ExpandableList key={i} label={`${x.name} Website`} defaultOpen>
|
||||||
|
<Box mb={0.5}>
|
||||||
|
<ExpandableList label={x.name} level={1}>
|
||||||
|
<ExpandableListItemKey label="Identity address" value={x.address} />
|
||||||
|
<ExpandableListItem label="Identity type" value={formatEnum(x.type)} />
|
||||||
|
</ExpandableList>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemKey label="Topic" value={'00'.repeat(32)} />
|
||||||
|
{x.feedHash && <ExpandableListItemKey label="Feed hash" value={x.feedHash} />}
|
||||||
|
<Box mt={0.75}>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton onClick={() => viewFeed(x.uuid)} iconType={Info}>
|
||||||
|
View Feed Page
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={() => onShowExport(x)} iconType={Download}>
|
||||||
|
Export...
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={() => onShowDelete(x)} iconType={Trash}>
|
||||||
|
Delete...
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Box>
|
||||||
|
</ExpandableList>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,99 +1,58 @@
|
|||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
import { Web } from '@material-ui/icons'
|
import { Web } from '@material-ui/icons'
|
||||||
import { ReactElement, useEffect, useState } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { File, Folder } from 'react-feather'
|
import { File, Folder } from 'react-feather'
|
||||||
import { FitImage } from '../../components/FitImage'
|
import { FitImage } from '../../components/FitImage'
|
||||||
import { detectIndexHtml, getAssetNameFromFiles, getHumanReadableFileSize } from '../../utils/file'
|
import { shortenText } from '../../utils'
|
||||||
import { SwarmFile } from '../../utils/SwarmFile'
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
|
import { shortenHash } from '../../utils/hash'
|
||||||
import { AssetIcon } from './AssetIcon'
|
import { AssetIcon } from './AssetIcon'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
assetName?: string
|
previewUri?: string
|
||||||
files: SwarmFile[]
|
metadata?: Metadata
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: add optional prop for indexDocument when it is already known (e.g. downloading a manifest)
|
// TODO: add optional prop for indexDocument when it is already known (e.g. downloading a manifest)
|
||||||
|
|
||||||
export function AssetPreview({ assetName, files }: Props): ReactElement {
|
export function AssetPreview({ metadata, previewUri }: Props): ReactElement | null {
|
||||||
const [previewComponent, setPreviewComponent] = useState<ReactElement | undefined>(undefined)
|
let previewComponent = <File />
|
||||||
const [previewUri, setPreviewUri] = useState<string | undefined>(undefined)
|
let type = metadata?.type
|
||||||
|
|
||||||
useEffect(() => {
|
if (metadata?.isWebsite) {
|
||||||
if (files.length === 1) {
|
previewComponent = <Web />
|
||||||
// single image
|
type = 'Website'
|
||||||
if (files[0].type.startsWith('image/')) {
|
} else if (metadata?.type === 'folder') {
|
||||||
files[0].arrayBuffer().then(value => {
|
previewComponent = <Folder />
|
||||||
const blob = new Blob([value])
|
type = 'Folder'
|
||||||
setPreviewUri(URL.createObjectURL(blob))
|
|
||||||
})
|
|
||||||
// single non-image
|
|
||||||
} else {
|
|
||||||
setPreviewUri(undefined)
|
|
||||||
setPreviewComponent(<AssetIcon icon={<File />} />)
|
|
||||||
}
|
|
||||||
// collection
|
|
||||||
} else if (detectIndexHtml(files)) {
|
|
||||||
setPreviewUri(undefined)
|
|
||||||
setPreviewComponent(<AssetIcon icon={<Web />} />)
|
|
||||||
} else {
|
|
||||||
setPreviewUri(undefined)
|
|
||||||
setPreviewComponent(<AssetIcon icon={<Folder />} />)
|
|
||||||
}
|
|
||||||
}, [files])
|
|
||||||
|
|
||||||
const getPrimaryText = () => {
|
|
||||||
const name = getAssetNameFromFiles(files)
|
|
||||||
|
|
||||||
if (files.length === 1) {
|
|
||||||
return 'Filename: ' + (assetName || name)
|
|
||||||
}
|
|
||||||
|
|
||||||
return 'Folder name: ' + (assetName || name)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const getKind = () => {
|
|
||||||
if (files.length === 1) {
|
|
||||||
return files[0].type
|
|
||||||
}
|
|
||||||
|
|
||||||
if (detectIndexHtml(files)) {
|
|
||||||
return 'Website'
|
|
||||||
}
|
|
||||||
|
|
||||||
return 'Folder'
|
|
||||||
}
|
|
||||||
|
|
||||||
const isFolder = () => ['Folder', 'Website'].includes(getKind())
|
|
||||||
|
|
||||||
const getSize = () => {
|
|
||||||
const bytes = files.reduce((total, item) => total + item.size, 0)
|
|
||||||
|
|
||||||
return getHumanReadableFileSize(bytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
const size = getSize()
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<Box bgcolor="background.paper">
|
<Box bgcolor="background.paper">
|
||||||
<Grid container direction="row">
|
<Grid container direction="row">
|
||||||
{previewComponent ? (
|
{previewUri ? (
|
||||||
previewComponent
|
|
||||||
) : (
|
|
||||||
<FitImage maxWidth="250px" maxHeight="175px" alt="Upload Preview" src={previewUri} />
|
<FitImage maxWidth="250px" maxHeight="175px" alt="Upload Preview" src={previewUri} />
|
||||||
|
) : (
|
||||||
|
<AssetIcon icon={previewComponent} />
|
||||||
)}
|
)}
|
||||||
<Box p={2}>
|
<Box p={2}>
|
||||||
<Typography>{getPrimaryText()}</Typography>
|
{metadata?.hash && <Typography>Swarm Hash: {shortenHash(metadata.hash)}</Typography>}
|
||||||
<Typography>Kind: {getKind()}</Typography>
|
{metadata?.name && metadata?.name !== metadata?.hash && (
|
||||||
{size !== '0 bytes' && <Typography>Size: {size}</Typography>}
|
<Typography>
|
||||||
|
{metadata?.type === 'folder' ? 'Folder Name' : 'Filename'}: {shortenText(metadata?.name)}
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
<Typography>Kind: {type}</Typography>
|
||||||
|
{metadata?.size ? <Typography>Size: {getHumanReadableFileSize(metadata.size)}</Typography> : null}
|
||||||
</Box>
|
</Box>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Box>
|
</Box>
|
||||||
{isFolder() && (
|
{metadata?.type === 'folder' && metadata.count && (
|
||||||
<Box mt={0.25} p={2} bgcolor="background.paper">
|
<Box mt={0.25} p={2} bgcolor="background.paper">
|
||||||
<Grid container justifyContent="space-between" alignItems="center" direction="row">
|
<Grid container justifyContent="space-between" alignItems="center" direction="row">
|
||||||
<Typography variant="subtitle2">Folder content</Typography>
|
<Typography variant="subtitle2">Folder content</Typography>
|
||||||
<Typography variant="subtitle2">{files.length} items</Typography>
|
<Typography variant="subtitle2">{metadata.count} items</Typography>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,24 +1,40 @@
|
|||||||
import { Box, Typography } from '@material-ui/core'
|
import * as swarmCid from '@ethersphere/swarm-cid'
|
||||||
|
import { Box } from '@material-ui/core'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
|
import { Utils } from '@ethersphere/bee-js'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
hash: string
|
isWebsite?: boolean
|
||||||
|
reference: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export function AssetSummary({ hash }: Props): ReactElement {
|
export function AssetSummary({ isWebsite, reference }: Props): ReactElement {
|
||||||
|
const isHash = Utils.isHexString(reference) && reference.length === 64
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<ExpandableListItemKey label="Swarm hash" value={hash} />
|
{isHash && <ExpandableListItemKey label="Swarm hash" value={reference} />}
|
||||||
<ExpandableListItemLink label="Share on Swarm Gateway" value={`https://gateway.ethswarm.org/access/${hash}`} />
|
{!isHash && <ExpandableListItemLink label="ENS" value={reference} />}
|
||||||
|
<ExpandableListItemLink
|
||||||
|
label="Share on Swarm Gateway"
|
||||||
|
value={`https://gateway.ethswarm.org/access/${reference}`}
|
||||||
|
/>
|
||||||
|
{isWebsite && isHash && (
|
||||||
|
<ExpandableListItemLink
|
||||||
|
label="BZZ Link"
|
||||||
|
value={`https://${swarmCid.encodeManifestReference(reference).toString()}.bzz.link`}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
<Typography>
|
<DocumentationText>
|
||||||
The Swarm Gateway is graciously provided by the Swarm Foundation. This service is under development and provided
|
The Swarm Gateway is graciously provided by the Swarm Foundation. This service is under development and provided
|
||||||
for testing purposes only. Learn more at{' '}
|
for testing purposes only. Learn more at{' '}
|
||||||
<a href="https://gateway.ethswarm.org/">https://gateway.ethswarm.org/</a>.
|
<a href="https://gateway.ethswarm.org/">https://gateway.ethswarm.org/</a>.
|
||||||
</Typography>
|
</DocumentationText>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,12 +2,13 @@ import { Utils } from '@ethersphere/bee-js'
|
|||||||
import { ManifestJs } from '@ethersphere/manifest-js'
|
import { ManifestJs } from '@ethersphere/manifest-js'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useState } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
import { useHistory } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
||||||
import { History } from '../../components/History'
|
import { History } from '../../components/History'
|
||||||
|
import { Context, defaultUploadOrigin } from '../../providers/File'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { extractSwarmHash } from '../../utils'
|
import { recognizeEnsOrSwarmHash, regexpEns } from '../../utils'
|
||||||
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
||||||
import { FileNavigation } from './FileNavigation'
|
import { FileNavigation } from './FileNavigation'
|
||||||
|
|
||||||
@@ -16,24 +17,42 @@ export function Download(): ReactElement {
|
|||||||
const { beeApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const [referenceError, setReferenceError] = useState<string | undefined>(undefined)
|
const [referenceError, setReferenceError] = useState<string | undefined>(undefined)
|
||||||
|
|
||||||
|
const { setUploadOrigin } = useContext(Context)
|
||||||
|
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
const history = useHistory()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
const validateChange = (value: string) => {
|
const validateChange = (value: string) => {
|
||||||
if (Utils.isHexString(value, 64) || Utils.isHexString(value, 128) || !value.trim().length) {
|
if (
|
||||||
|
Utils.isHexString(value, 64) ||
|
||||||
|
Utils.isHexString(value, 128) ||
|
||||||
|
!value.trim().length ||
|
||||||
|
regexpEns.test(value)
|
||||||
|
) {
|
||||||
setReferenceError(undefined)
|
setReferenceError(undefined)
|
||||||
} else {
|
} else {
|
||||||
setReferenceError('Incorrect format of swarm hash. Expected 64 or 128 hexstring characters.')
|
setReferenceError(
|
||||||
|
'Incorrect format of swarm hash. Expected 64 or 128 hexstring characters, bzz.link url or ENS domain.',
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function onSwarmIdentifier(identifier: string) {
|
async function onSwarmIdentifier(identifier: string) {
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
if (!beeApi) {
|
if (!beeApi) {
|
||||||
|
setLoading(false)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const manifestJs = new ManifestJs(beeApi)
|
const manifestJs = new ManifestJs(beeApi)
|
||||||
|
const feedIdentifier = await manifestJs.resolveFeedManifest(identifier)
|
||||||
|
|
||||||
|
if (feedIdentifier) {
|
||||||
|
identifier = feedIdentifier
|
||||||
|
}
|
||||||
const isManifest = await manifestJs.isManifest(identifier)
|
const isManifest = await manifestJs.isManifest(identifier)
|
||||||
|
|
||||||
if (!isManifest) {
|
if (!isManifest) {
|
||||||
@@ -41,7 +60,8 @@ export function Download(): ReactElement {
|
|||||||
}
|
}
|
||||||
const indexDocument = await manifestJs.getIndexDocumentPath(identifier)
|
const indexDocument = await manifestJs.getIndexDocumentPath(identifier)
|
||||||
putHistory(HISTORY_KEYS.DOWNLOAD_HISTORY, identifier, determineHistoryName(identifier, indexDocument))
|
putHistory(HISTORY_KEYS.DOWNLOAD_HISTORY, identifier, determineHistoryName(identifier, indexDocument))
|
||||||
history.push(ROUTES.HASH.replace(':hash', identifier))
|
setUploadOrigin(defaultUploadOrigin)
|
||||||
|
navigate(ROUTES.HASH.replace(':hash', identifier))
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
let message = typeof error === 'object' && error !== null && Reflect.get(error, 'message')
|
let message = typeof error === 'object' && error !== null && Reflect.get(error, 'message')
|
||||||
|
|
||||||
@@ -58,20 +78,6 @@ export function Download(): ReactElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function recognizeSwarmHash(value: string) {
|
|
||||||
if (value.length < 64) {
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
|
|
||||||
const hash = extractSwarmHash(value)
|
|
||||||
|
|
||||||
if (hash) {
|
|
||||||
return hash
|
|
||||||
}
|
|
||||||
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<FileNavigation active="DOWNLOAD" />
|
<FileNavigation active="DOWNLOAD" />
|
||||||
@@ -80,11 +86,12 @@ export function Download(): ReactElement {
|
|||||||
onConfirm={value => onSwarmIdentifier(value)}
|
onConfirm={value => onSwarmIdentifier(value)}
|
||||||
onChange={validateChange}
|
onChange={validateChange}
|
||||||
helperText={referenceError}
|
helperText={referenceError}
|
||||||
confirmLabel={'Search'}
|
confirmLabel={'Find'}
|
||||||
confirmLabelDisabled={Boolean(referenceError) || loading}
|
confirmLabelDisabled={Boolean(referenceError) || loading}
|
||||||
placeholder="e.g. 31fb0362b1a42536134c86bc58b97ac0244e5c6630beec3e27c2d1cecb38c605"
|
placeholder="e.g. 31fb0362b1a42536134c86bc58b97ac0244e5c6630beec3e27c2d1cecb38c605"
|
||||||
expandedOnly
|
expandedOnly
|
||||||
mapperFn={value => recognizeSwarmHash(value)}
|
mapperFn={value => recognizeEnsOrSwarmHash(value)}
|
||||||
|
loading={loading}
|
||||||
/>
|
/>
|
||||||
<History title="Download History" localStorageKey={HISTORY_KEYS.DOWNLOAD_HISTORY} />
|
<History title="Download History" localStorageKey={HISTORY_KEYS.DOWNLOAD_HISTORY} />
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -1,32 +1,46 @@
|
|||||||
import { Button } from '@material-ui/core'
|
import { Box, Grid } from '@material-ui/core'
|
||||||
import { Clear } from '@material-ui/icons'
|
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { Download, Link } from 'react-feather'
|
import { Bookmark, Download, Link, X } from 'react-feather'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
onOpen: () => void
|
onOpen: () => void
|
||||||
onDownload: () => void
|
|
||||||
onCancel: () => void
|
onCancel: () => void
|
||||||
|
onDownload: () => void
|
||||||
|
onUpdateFeed: () => void
|
||||||
hasIndexDocument: boolean
|
hasIndexDocument: boolean
|
||||||
loading: boolean
|
loading: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export function DownloadActionBar({ onOpen, onDownload, onCancel, hasIndexDocument, loading }: Props): ReactElement {
|
export function DownloadActionBar({
|
||||||
|
onOpen,
|
||||||
|
onCancel,
|
||||||
|
onDownload,
|
||||||
|
onUpdateFeed,
|
||||||
|
hasIndexDocument,
|
||||||
|
loading,
|
||||||
|
}: Props): ReactElement {
|
||||||
return (
|
return (
|
||||||
<ExpandableListItemActions>
|
<Grid container justifyContent="space-between">
|
||||||
{hasIndexDocument && (
|
<ExpandableListItemActions>
|
||||||
<SwarmButton onClick={onOpen} iconType={Link} disabled={loading}>
|
{hasIndexDocument && (
|
||||||
View Website
|
<SwarmButton onClick={onOpen} iconType={Link} disabled={loading}>
|
||||||
|
View Website
|
||||||
|
</SwarmButton>
|
||||||
|
)}
|
||||||
|
<SwarmButton onClick={onDownload} iconType={Download} disabled={loading} loading={loading}>
|
||||||
|
Download
|
||||||
</SwarmButton>
|
</SwarmButton>
|
||||||
)}
|
<SwarmButton onClick={onCancel} iconType={X} disabled={loading} cancel>
|
||||||
<SwarmButton onClick={onDownload} iconType={Download} disabled={loading} loading={loading}>
|
Close
|
||||||
Download
|
</SwarmButton>
|
||||||
</SwarmButton>
|
</ExpandableListItemActions>
|
||||||
<Button onClick={onCancel} variant="contained" startIcon={<Clear />} disabled={loading}>
|
<Box mb={1} mr={1}>
|
||||||
Close
|
<SwarmButton onClick={onUpdateFeed} iconType={Bookmark} disabled={loading}>
|
||||||
</Button>
|
Update Feed
|
||||||
</ExpandableListItemActions>
|
</SwarmButton>
|
||||||
|
</Box>
|
||||||
|
</Grid>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { createStyles, makeStyles, Tab, Tabs, Theme } from '@material-ui/core'
|
import { createStyles, makeStyles, Tab, Tabs, Theme } from '@material-ui/core'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { useHistory } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -24,10 +24,10 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
|
|
||||||
export function FileNavigation({ active }: Props): ReactElement {
|
export function FileNavigation({ active }: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const history = useHistory()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
function onChange(event: React.ChangeEvent<Record<string, never>>, newValue: number) {
|
function onChange(event: React.ChangeEvent<Record<string, never>>, newValue: number) {
|
||||||
history.push(newValue === 1 ? ROUTES.DOWNLOAD : ROUTES.UPLOAD)
|
navigate(newValue === 1 ? ROUTES.DOWNLOAD : ROUTES.UPLOAD)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
|
import { Button, ListItemIcon, Menu, MenuItem, Typography } from '@material-ui/core'
|
||||||
import React, { ReactElement } from 'react'
|
import React, { ReactElement } from 'react'
|
||||||
import { Button, ListItemIcon, Typography, Menu, MenuItem } from '@material-ui/core'
|
|
||||||
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -35,7 +35,7 @@ export default function SimpleMenu({ stamps, selectedStamp, setSelected }: Props
|
|||||||
selected={stamp.batchID === selectedStamp?.batchID}
|
selected={stamp.batchID === selectedStamp?.batchID}
|
||||||
>
|
>
|
||||||
<ListItemIcon>{stamp.usageText}</ListItemIcon>
|
<ListItemIcon>{stamp.usageText}</ListItemIcon>
|
||||||
<Typography variant="body2">{stamp.batchID.substr(0, 8)}[…]</Typography>
|
<Typography variant="body2">{stamp.batchID.slice(0, 8)}[…]</Typography>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
))}
|
))}
|
||||||
</Menu>
|
</Menu>
|
||||||
|
|||||||
+75
-30
@@ -1,37 +1,43 @@
|
|||||||
import { ManifestJs } from '@ethersphere/manifest-js'
|
import { ManifestJs } from '@ethersphere/manifest-js'
|
||||||
import { Box } from '@material-ui/core'
|
import { Box, Typography } from '@material-ui/core'
|
||||||
import { saveAs } from 'file-saver'
|
import { saveAs } from 'file-saver'
|
||||||
import JSZip from 'jszip'
|
import JSZip from 'jszip'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
import { RouteComponentProps, useHistory } from 'react-router-dom'
|
import { useNavigate, useParams } from 'react-router-dom'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { Loading } from '../../components/Loading'
|
import { Loading } from '../../components/Loading'
|
||||||
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
|
import config from '../../config'
|
||||||
|
import { META_FILE_NAME, PREVIEW_FILE_NAME } from '../../constants'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { convertBeeFileToBrowserFile, convertManifestToFiles } from '../../utils/file'
|
|
||||||
import { shortenHash } from '../../utils/hash'
|
|
||||||
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
||||||
import { SwarmFile } from '../../utils/SwarmFile'
|
|
||||||
import { AssetPreview } from './AssetPreview'
|
import { AssetPreview } from './AssetPreview'
|
||||||
import { AssetSummary } from './AssetSummary'
|
import { AssetSummary } from './AssetSummary'
|
||||||
import { DownloadActionBar } from './DownloadActionBar'
|
import { DownloadActionBar } from './DownloadActionBar'
|
||||||
|
|
||||||
interface MatchParams {
|
export function Share(): ReactElement {
|
||||||
hash: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export function Share(props: RouteComponentProps<MatchParams>): ReactElement {
|
|
||||||
const { apiUrl, beeApi } = useContext(SettingsContext)
|
const { apiUrl, beeApi } = useContext(SettingsContext)
|
||||||
const reference = props.match.params.hash
|
const { status } = useContext(BeeContext)
|
||||||
const history = useHistory()
|
|
||||||
|
const { hash } = useParams()
|
||||||
|
const reference = hash! // eslint-disable-line
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const [downloading, setDownloading] = useState(false)
|
const [downloading, setDownloading] = useState(false)
|
||||||
const [files, setFiles] = useState<SwarmFile[]>([])
|
|
||||||
const [swarmEntries, setSwarmEntries] = useState<Record<string, string>>({})
|
const [swarmEntries, setSwarmEntries] = useState<Record<string, string>>({})
|
||||||
const [indexDocument, setIndexDocument] = useState<string | null>(null)
|
const [indexDocument, setIndexDocument] = useState<string | null>(null)
|
||||||
|
const [notFound, setNotFound] = useState(false)
|
||||||
|
const [preview, setPreview] = useState<string | undefined>(undefined)
|
||||||
|
const [metadata, setMetadata] = useState<Metadata | undefined>()
|
||||||
|
|
||||||
async function prepare() {
|
async function prepare() {
|
||||||
if (!beeApi) {
|
if (!beeApi || !status.all) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -39,19 +45,43 @@ export function Share(props: RouteComponentProps<MatchParams>): ReactElement {
|
|||||||
const isManifest = await manifestJs.isManifest(reference)
|
const isManifest = await manifestJs.isManifest(reference)
|
||||||
|
|
||||||
if (!isManifest) {
|
if (!isManifest) {
|
||||||
throw Error('The specified hash does not contain valid content.')
|
setNotFound(true)
|
||||||
|
enqueueSnackbar('The specified hash does not contain valid content.', { variant: 'error' })
|
||||||
|
|
||||||
|
return
|
||||||
}
|
}
|
||||||
const entries = await manifestJs.getHashes(reference)
|
const entries = await manifestJs.getHashes(reference)
|
||||||
setSwarmEntries(entries)
|
|
||||||
const indexDocument = await manifestJs.getIndexDocumentPath(reference)
|
const indexDocument = await manifestJs.getIndexDocumentPath(reference)
|
||||||
setIndexDocument(indexDocument)
|
setIndexDocument(indexDocument)
|
||||||
|
|
||||||
if (Object.keys(entries).length === 1) {
|
const previewFile = entries[PREVIEW_FILE_NAME]
|
||||||
const response = await beeApi.downloadFile(reference)
|
|
||||||
setFiles([new SwarmFile(convertBeeFileToBrowserFile(response) as File)])
|
delete entries[META_FILE_NAME]
|
||||||
} else {
|
delete entries[PREVIEW_FILE_NAME]
|
||||||
setFiles(convertManifestToFiles(entries))
|
setSwarmEntries(entries)
|
||||||
|
|
||||||
|
const count = Object.keys(entries).length
|
||||||
|
|
||||||
|
let metadata: Metadata | undefined = {
|
||||||
|
hash,
|
||||||
|
size: 0,
|
||||||
|
type: count > 1 ? 'folder' : 'unknown',
|
||||||
|
name: reference,
|
||||||
|
isWebsite: Boolean(indexDocument) && count > 1,
|
||||||
|
count,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const mtdt = await beeApi.downloadFile(reference, META_FILE_NAME)
|
||||||
|
const remoteMetadata = mtdt.data.text()
|
||||||
|
metadata = { ...metadata, ...(JSON.parse(remoteMetadata) as Metadata) }
|
||||||
|
} catch (e) {} // eslint-disable-line no-empty
|
||||||
|
|
||||||
|
if (previewFile) {
|
||||||
|
setPreview(`${config.BEE_API_HOST}/bzz/${reference}/${PREVIEW_FILE_NAME}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
setMetadata(metadata)
|
||||||
}
|
}
|
||||||
|
|
||||||
function onOpen() {
|
function onOpen() {
|
||||||
@@ -59,17 +89,22 @@ export function Share(props: RouteComponentProps<MatchParams>): ReactElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function onClose() {
|
function onClose() {
|
||||||
// POP means there is no history - nowhere to go back yet
|
if (navigate.length > 0) {
|
||||||
if (history.action === 'POP') {
|
// There is at least one different route in browser history that we can return to
|
||||||
history.push(ROUTES.UPLOAD)
|
navigate(-1)
|
||||||
} else {
|
} else {
|
||||||
history.goBack()
|
// This is the first page user opened, navigate to upload page instead of going back
|
||||||
|
navigate(ROUTES.UPLOAD)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onUpdateFeed() {
|
||||||
|
navigate(ROUTES.FEEDS_UPDATE.replace(':hash', reference))
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setLoading(true)
|
setLoading(true)
|
||||||
prepare().then(() => {
|
prepare().finally(() => {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
})
|
})
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
@@ -95,25 +130,35 @@ export function Share(props: RouteComponentProps<MatchParams>): ReactElement {
|
|||||||
setDownloading(false)
|
setDownloading(false)
|
||||||
}
|
}
|
||||||
|
|
||||||
const assetName = shortenHash(reference)
|
if (!status.all) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
if (loading) {
|
if (loading) {
|
||||||
return <Loading />
|
return <Loading />
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (notFound) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Not Found</HistoryHeader>
|
||||||
|
<Typography>The specified hash is not found.</Typography>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<AssetPreview files={files} assetName={assetName} />
|
<AssetPreview metadata={metadata} previewUri={preview} />
|
||||||
</Box>
|
</Box>
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<AssetSummary hash={reference} />
|
<AssetSummary isWebsite={metadata?.isWebsite} reference={reference} />
|
||||||
</Box>
|
</Box>
|
||||||
<DownloadActionBar
|
<DownloadActionBar
|
||||||
onOpen={onOpen}
|
onOpen={onOpen}
|
||||||
onCancel={onClose}
|
onCancel={onClose}
|
||||||
onDownload={onDownload}
|
onDownload={onDownload}
|
||||||
hasIndexDocument={Boolean(indexDocument && files.length > 1)}
|
onUpdateFeed={onUpdateFeed}
|
||||||
|
hasIndexDocument={Boolean(metadata?.isWebsite)}
|
||||||
loading={downloading}
|
loading={downloading}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
|
|||||||
+161
-41
@@ -1,54 +1,141 @@
|
|||||||
|
import { Box } from '@material-ui/core'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
import { useHistory } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
||||||
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
|
import { META_FILE_NAME, PREVIEW_FILE_NAME } from '../../constants'
|
||||||
|
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as IdentityContext, Identity } from '../../providers/Feeds'
|
||||||
import { Context as FileContext } from '../../providers/File'
|
import { Context as FileContext } from '../../providers/File'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context, EnrichedPostageBatch } from '../../providers/Stamps'
|
import { Context as StampsContext, EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { detectIndexHtml, getAssetNameFromFiles } from '../../utils/file'
|
import { detectIndexHtml, getAssetNameFromFiles, packageFile } from '../../utils/file'
|
||||||
|
import { persistIdentity, updateFeed } from '../../utils/identity'
|
||||||
import { HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
||||||
import { CreatePostageStampModal } from '../stamps/CreatePostageStampModal'
|
import { FeedPasswordDialog } from '../feeds/FeedPasswordDialog'
|
||||||
import { SelectPostageStampModal } from '../stamps/SelectPostageStampModal'
|
import { PostageStampCreation } from '../stamps/PostageStampCreation'
|
||||||
|
import { PostageStampSelector } from '../stamps/PostageStampSelector'
|
||||||
import { AssetPreview } from './AssetPreview'
|
import { AssetPreview } from './AssetPreview'
|
||||||
import { StampPreview } from './StampPreview'
|
import { StampPreview } from './StampPreview'
|
||||||
import { UploadActionBar } from './UploadActionBar'
|
import { UploadActionBar } from './UploadActionBar'
|
||||||
|
|
||||||
export function Upload(): ReactElement {
|
export function Upload(): ReactElement {
|
||||||
const [isBuyingStamp, setBuyingStamp] = useState(false)
|
const [step, setStep] = useState(0)
|
||||||
const [isSelectingStamp, setSelectingStamp] = useState(false)
|
const [stampMode, setStampMode] = useState<'SELECT' | 'BUY'>('SELECT')
|
||||||
const [stamp, setStamp] = useState<EnrichedPostageBatch | null>(null)
|
const [stamp, setStamp] = useState<EnrichedPostageBatch | null>(null)
|
||||||
const [isUploading, setUploading] = useState(false)
|
const [isUploading, setUploading] = useState(false)
|
||||||
|
const [showPasswordPrompt, setShowPasswordPrompt] = useState(false)
|
||||||
|
|
||||||
const { stamps, refresh } = useContext(Context)
|
const { refresh } = useContext(StampsContext)
|
||||||
const { beeApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { files, setFiles } = useContext(FileContext)
|
const { files, setFiles, uploadOrigin, metadata, previewUri, previewBlob } = useContext(FileContext)
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { identities, setIdentities } = useContext(IdentityContext)
|
||||||
const history = useHistory()
|
const { status } = useContext(BeeContext)
|
||||||
|
|
||||||
if (!files.length) {
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
setFiles([])
|
const navigate = useNavigate()
|
||||||
history.replace(ROUTES.UPLOAD)
|
|
||||||
}
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
refresh()
|
refresh()
|
||||||
}, []) // eslint-disable-line react-hooks/exhaustive-deps
|
}, []) // eslint-disable-line react-hooks/exhaustive-deps
|
||||||
|
|
||||||
const uploadFiles = () => {
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
if (!beeApi || !files.length || !stamp) {
|
|
||||||
|
if (!files.length) {
|
||||||
|
setFiles([])
|
||||||
|
navigate(ROUTES.UPLOAD, { replace: true })
|
||||||
|
|
||||||
|
return <></>
|
||||||
|
}
|
||||||
|
|
||||||
|
const identity = uploadOrigin.uuid ? identities.find(x => x.uuid === uploadOrigin.uuid) : null
|
||||||
|
|
||||||
|
const onUpload = () => {
|
||||||
|
if (uploadOrigin.origin === 'UPLOAD') {
|
||||||
|
uploadFiles()
|
||||||
|
} else {
|
||||||
|
if ((identity as Identity).type === 'PRIVATE_KEY') {
|
||||||
|
uploadFiles()
|
||||||
|
} else {
|
||||||
|
setShowPasswordPrompt(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const uploadFiles = (password?: string) => {
|
||||||
|
if (!beeApi || !files.length || !stamp || !metadata) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
const indexDocument = files.length === 1 ? files[0].name : detectIndexHtml(files) || undefined
|
let fls: FilePath[] = files.map(f => packageFile(f)) // Apart from packaging, this is needed to not modify the original files array as it can trigger effects
|
||||||
|
let indexDocument: string | undefined = undefined // This means we assume it's folder
|
||||||
|
|
||||||
|
if (files.length === 1) indexDocument = files[0].name
|
||||||
|
else if (files.length > 1) {
|
||||||
|
const idx = detectIndexHtml(files)
|
||||||
|
|
||||||
|
// This is a website
|
||||||
|
if (idx) {
|
||||||
|
// The website is in some directory, remove it
|
||||||
|
if (idx.commonPrefix) {
|
||||||
|
const substrStart = idx.commonPrefix.length
|
||||||
|
indexDocument = idx.indexPath.slice(substrStart)
|
||||||
|
fls = files.map(f => {
|
||||||
|
const path = (f.path as string).slice(substrStart)
|
||||||
|
|
||||||
|
return packageFile(f, path)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// The website is not packed in a directory
|
||||||
|
indexDocument = idx.indexPath
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const lastModified = files[0].lastModified
|
||||||
|
|
||||||
|
// We want to store only some metadata
|
||||||
|
const mtd: SwarmMetadata = {
|
||||||
|
name: metadata.name,
|
||||||
|
size: metadata.size,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type of the file only makes sense for a single file
|
||||||
|
if (files.length === 1) mtd.type = metadata.type
|
||||||
|
|
||||||
|
const metafile = new File([JSON.stringify(mtd)], META_FILE_NAME, {
|
||||||
|
type: 'application/json',
|
||||||
|
lastModified,
|
||||||
|
})
|
||||||
|
fls.push(packageFile(metafile))
|
||||||
|
|
||||||
|
if (previewBlob) {
|
||||||
|
const previewFile = new File([previewBlob], PREVIEW_FILE_NAME, {
|
||||||
|
type: 'image/jpeg',
|
||||||
|
lastModified,
|
||||||
|
})
|
||||||
|
fls.push(packageFile(previewFile))
|
||||||
|
}
|
||||||
|
|
||||||
setUploading(true)
|
setUploading(true)
|
||||||
|
|
||||||
beeApi
|
beeApi
|
||||||
.uploadFiles(stamp.batchID, files as unknown as File[], { indexDocument })
|
.uploadFiles(stamp.batchID, fls, { indexDocument })
|
||||||
.then(hash => {
|
.then(hash => {
|
||||||
putHistory(HISTORY_KEYS.UPLOAD_HISTORY, hash.reference, getAssetNameFromFiles(files))
|
putHistory(HISTORY_KEYS.UPLOAD_HISTORY, hash.reference, getAssetNameFromFiles(files))
|
||||||
history.replace(ROUTES.HASH.replace(':hash', hash.reference))
|
|
||||||
|
if (uploadOrigin.origin === 'UPLOAD') {
|
||||||
|
navigate(ROUTES.HASH.replace(':hash', hash.reference), { replace: true })
|
||||||
|
} else {
|
||||||
|
updateFeed(beeApi, identity as Identity, hash.reference, stamp.batchID, password as string).then(() => {
|
||||||
|
persistIdentity(identities, identity as Identity)
|
||||||
|
setIdentities([...identities])
|
||||||
|
navigate(ROUTES.FEEDS_PAGE.replace(':uuid', uploadOrigin.uuid as string), { replace: true })
|
||||||
|
})
|
||||||
|
}
|
||||||
})
|
})
|
||||||
.catch(e => {
|
.catch(e => {
|
||||||
enqueueSnackbar(`Error uploading: ${e.message}`, { variant: 'error' })
|
enqueueSnackbar(`Error uploading: ${e.message}`, { variant: 'error' })
|
||||||
@@ -57,36 +144,69 @@ export function Upload(): ReactElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const reset = () => {
|
const reset = () => {
|
||||||
|
setStep(0)
|
||||||
setFiles([])
|
setFiles([])
|
||||||
setStamp(null)
|
setStamp(null)
|
||||||
setUploading(false)
|
setUploading(false)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const onFeedPasswordGiven = (password: string) => {
|
||||||
|
uploadFiles(password)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<HistoryHeader>Upload</HistoryHeader>
|
{showPasswordPrompt && (
|
||||||
{files.length && <AssetPreview files={files} />}
|
<FeedPasswordDialog
|
||||||
{stamp !== null ? <StampPreview stamp={stamp} /> : null}
|
loading={isUploading}
|
||||||
{files.length && (
|
feedName={(identity as Identity).name}
|
||||||
<UploadActionBar
|
onCancel={() => setShowPasswordPrompt(false)}
|
||||||
canSelectStamp={stamps !== null && stamps.length > 0}
|
onProceed={onFeedPasswordGiven}
|
||||||
hasSelectedStamp={stamp !== null}
|
|
||||||
onCancel={reset}
|
|
||||||
onBuy={() => setBuyingStamp(true)}
|
|
||||||
onSelect={() => setSelectingStamp(true)}
|
|
||||||
onUpload={uploadFiles}
|
|
||||||
onClearStamp={() => setStamp(null)}
|
|
||||||
isUploading={isUploading}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{isBuyingStamp ? <CreatePostageStampModal onClose={() => setBuyingStamp(false)} /> : null}
|
{identity && <HistoryHeader>{`Update "${identity.name}"`}</HistoryHeader>}
|
||||||
{stamps && isSelectingStamp ? (
|
{!identity && <HistoryHeader>Upload</HistoryHeader>}
|
||||||
<SelectPostageStampModal
|
<Box mb={4}>
|
||||||
stamps={stamps}
|
<ProgressIndicator steps={['Preview', 'Add postage stamp', 'Upload to node']} index={step} />
|
||||||
onClose={() => setSelectingStamp(false)}
|
</Box>
|
||||||
onSelect={stamp => setStamp(stamp)}
|
{(step === 0 || step === 2) && <AssetPreview metadata={metadata} previewUri={previewUri} />}
|
||||||
/>
|
{step === 1 && (
|
||||||
) : null}
|
<>
|
||||||
|
<Box mb={2}>
|
||||||
|
{stampMode === 'SELECT' ? (
|
||||||
|
<PostageStampSelector onSelect={stamp => setStamp(stamp)} defaultValue={stamp?.batchID} />
|
||||||
|
) : (
|
||||||
|
<PostageStampCreation onFinished={() => setStampMode('SELECT')} />
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<DocumentationText>
|
||||||
|
Please refer to the{' '}
|
||||||
|
<a
|
||||||
|
href="https://docs.ethswarm.org/debug-api/#tag/Postage-Stamps/paths/~1stamps~1{amount}~1{depth}/post"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
official Bee documentation
|
||||||
|
</a>{' '}
|
||||||
|
to understand these values.
|
||||||
|
</DocumentationText>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{step === 2 && stamp && <StampPreview stamp={stamp} />}
|
||||||
|
<UploadActionBar
|
||||||
|
step={step}
|
||||||
|
onCancel={reset}
|
||||||
|
onGoBack={() => setStep(step => step - 1)}
|
||||||
|
onProceed={() => setStep(step => step + 1)}
|
||||||
|
onUpload={onUpload}
|
||||||
|
isUploading={isUploading}
|
||||||
|
hasStamp={Boolean(stamp)}
|
||||||
|
uploadLabel={identity ? 'Update Feed' : 'Upload To Your Node'}
|
||||||
|
stampMode={stampMode}
|
||||||
|
setStampMode={setStampMode}
|
||||||
|
/>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,69 +1,88 @@
|
|||||||
import { Button, Typography } from '@material-ui/core'
|
import { Box, Grid } from '@material-ui/core'
|
||||||
import { Clear } from '@material-ui/icons'
|
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { Check, Layers, PlusSquare, RefreshCcw } from 'react-feather'
|
import { ArrowLeft, Check, Layers, PlusSquare, X } from 'react-feather'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
canSelectStamp: boolean
|
step: number
|
||||||
hasSelectedStamp: boolean
|
|
||||||
onUpload: () => void
|
onUpload: () => void
|
||||||
onBuy: () => void
|
|
||||||
onSelect: () => void
|
|
||||||
onCancel: () => void
|
onCancel: () => void
|
||||||
onClearStamp: () => void
|
onGoBack: () => void
|
||||||
|
onProceed: () => void
|
||||||
isUploading: boolean
|
isUploading: boolean
|
||||||
|
hasStamp: boolean
|
||||||
|
uploadLabel: string
|
||||||
|
stampMode: 'BUY' | 'SELECT'
|
||||||
|
setStampMode: (mode: 'BUY' | 'SELECT') => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export function UploadActionBar({
|
export function UploadActionBar({
|
||||||
canSelectStamp,
|
step,
|
||||||
hasSelectedStamp,
|
|
||||||
onUpload,
|
onUpload,
|
||||||
onBuy,
|
|
||||||
onSelect,
|
|
||||||
onCancel,
|
onCancel,
|
||||||
onClearStamp,
|
onGoBack,
|
||||||
|
onProceed,
|
||||||
isUploading,
|
isUploading,
|
||||||
|
hasStamp,
|
||||||
|
uploadLabel,
|
||||||
|
stampMode,
|
||||||
|
setStampMode,
|
||||||
}: Props): ReactElement {
|
}: Props): ReactElement {
|
||||||
const showBuy = !hasSelectedStamp
|
if (step === 0) {
|
||||||
const showSelect = canSelectStamp && !hasSelectedStamp
|
return (
|
||||||
const showUpload = hasSelectedStamp
|
<>
|
||||||
const showChange = canSelectStamp && hasSelectedStamp
|
<Box mb={1}>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton onClick={onProceed} iconType={Layers}>
|
||||||
|
Add Postage Stamp
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={onCancel} iconType={X} cancel>
|
||||||
|
Cancel
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Box>
|
||||||
|
<DocumentationText>You need a postage stamp to upload.</DocumentationText>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
if (step === 1) {
|
||||||
<>
|
return (
|
||||||
|
<Grid container direction="row" justifyContent="space-between">
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
{stampMode === 'SELECT' && (
|
||||||
|
<SwarmButton onClick={onProceed} iconType={Check} disabled={!hasStamp}>
|
||||||
|
Proceed With Selected Stamp
|
||||||
|
</SwarmButton>
|
||||||
|
)}
|
||||||
|
<SwarmButton onClick={onGoBack} iconType={ArrowLeft} cancel>
|
||||||
|
Back To Preview
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
<SwarmButton
|
||||||
|
onClick={() => setStampMode(stampMode === 'BUY' ? 'SELECT' : 'BUY')}
|
||||||
|
iconType={stampMode === 'BUY' ? Layers : PlusSquare}
|
||||||
|
>
|
||||||
|
{stampMode === 'BUY' ? 'Use Existing Stamp' : 'Buy New Stamp'}
|
||||||
|
</SwarmButton>
|
||||||
|
</Grid>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (step === 2) {
|
||||||
|
return (
|
||||||
<ExpandableListItemActions>
|
<ExpandableListItemActions>
|
||||||
{showBuy ? (
|
<SwarmButton onClick={onUpload} iconType={Check} disabled={isUploading} loading={isUploading}>
|
||||||
<SwarmButton onClick={onBuy} iconType={PlusSquare}>
|
{uploadLabel}
|
||||||
Buy New Postage Stamp
|
</SwarmButton>
|
||||||
</SwarmButton>
|
<SwarmButton onClick={onGoBack} iconType={ArrowLeft} disabled={isUploading} cancel>
|
||||||
) : null}
|
Change Postage Stamp
|
||||||
{showSelect ? (
|
</SwarmButton>
|
||||||
<SwarmButton onClick={onSelect} iconType={Layers}>
|
|
||||||
Use Existing Postage Stamp
|
|
||||||
</SwarmButton>
|
|
||||||
) : null}
|
|
||||||
{showUpload ? (
|
|
||||||
<SwarmButton onClick={onUpload} iconType={Check} disabled={isUploading} loading={isUploading}>
|
|
||||||
Upload To Your Node
|
|
||||||
</SwarmButton>
|
|
||||||
) : null}
|
|
||||||
{showChange ? (
|
|
||||||
<SwarmButton onClick={onClearStamp} iconType={RefreshCcw} disabled={isUploading}>
|
|
||||||
Change Postage Stamp
|
|
||||||
</SwarmButton>
|
|
||||||
) : null}
|
|
||||||
<Button onClick={onCancel} variant="contained" startIcon={<Clear />}>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
</ExpandableListItemActions>
|
</ExpandableListItemActions>
|
||||||
{showSelect ? (
|
)
|
||||||
<Typography>
|
}
|
||||||
You need a postage stamp to upload. Please refer to the official Bee documentation to understand how postage
|
|
||||||
stamps work.
|
return <></>
|
||||||
</Typography>
|
|
||||||
) : null}
|
|
||||||
</>
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,19 +1,22 @@
|
|||||||
import { createStyles, makeStyles, Theme, Typography } from '@material-ui/core'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core'
|
||||||
import { DropzoneArea } from 'material-ui-dropzone'
|
import { DropzoneArea } from 'material-ui-dropzone'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useState } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
import { FilePlus, FolderPlus, PlusCircle } from 'react-feather'
|
import { FilePlus, FolderPlus, PlusCircle } from 'react-feather'
|
||||||
import { useHistory } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { Context } from '../../providers/File'
|
import { Context, UploadOrigin } from '../../providers/File'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { detectIndexHtml } from '../../utils/file'
|
import { detectIndexHtml } from '../../utils/file'
|
||||||
import { SwarmFile } from '../../utils/SwarmFile'
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
maximumSizeInBytes: number
|
uploadOrigin: UploadOrigin
|
||||||
|
showHelp: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const MAX_FILE_SIZE = 1_000_000_000 // 1 gigabyte
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
areaWrapper: { position: 'relative', marginBottom: theme.spacing(2) },
|
areaWrapper: { position: 'relative', marginBottom: theme.spacing(2) },
|
||||||
@@ -44,10 +47,10 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
export function UploadArea({ maximumSizeInBytes }: Props): ReactElement {
|
export function UploadArea({ uploadOrigin, showHelp }: Props): ReactElement {
|
||||||
const { setFiles } = useContext(Context)
|
const { setFiles, setUploadOrigin } = useContext(Context)
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const history = useHistory()
|
const navigate = useNavigate()
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
const [strictWebsiteMode, setStrictWebsiteMode] = useState(false)
|
const [strictWebsiteMode, setStrictWebsiteMode] = useState(false)
|
||||||
const [version, setVersion] = useState(0)
|
const [version, setVersion] = useState(0)
|
||||||
@@ -95,8 +98,8 @@ export function UploadArea({ maximumSizeInBytes }: Props): ReactElement {
|
|||||||
|
|
||||||
const handleChange = (files?: File[]) => {
|
const handleChange = (files?: File[]) => {
|
||||||
if (files) {
|
if (files) {
|
||||||
const swarmFiles = files.map(x => new SwarmFile(x))
|
const FilePaths = files as FilePath[]
|
||||||
const indexDocument = files.length === 1 ? files[0].name : detectIndexHtml(swarmFiles) || undefined
|
const indexDocument = files.length === 1 ? files[0].name : detectIndexHtml(FilePaths) || undefined
|
||||||
|
|
||||||
if (files.length && strictWebsiteMode && !indexDocument) {
|
if (files.length && strictWebsiteMode && !indexDocument) {
|
||||||
enqueueSnackbar('To upload a website, there must be an index.html or index.htm in the root of the folder.', {
|
enqueueSnackbar('To upload a website, there must be an index.html or index.htm in the root of the folder.', {
|
||||||
@@ -107,10 +110,11 @@ export function UploadArea({ maximumSizeInBytes }: Props): ReactElement {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
setFiles(swarmFiles)
|
setFiles(FilePaths)
|
||||||
|
|
||||||
if (files.length) {
|
if (files.length) {
|
||||||
history.push(ROUTES.UPLOAD_IN_PROGRESS)
|
setUploadOrigin(uploadOrigin)
|
||||||
|
navigate(ROUTES.UPLOAD_IN_PROGRESS)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -123,7 +127,7 @@ export function UploadArea({ maximumSizeInBytes }: Props): ReactElement {
|
|||||||
dropzoneClass={classes.dropzone}
|
dropzoneClass={classes.dropzone}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
filesLimit={1e9}
|
filesLimit={1e9}
|
||||||
maxFileSize={maximumSizeInBytes}
|
maxFileSize={MAX_FILE_SIZE}
|
||||||
showPreviews={false}
|
showPreviews={false}
|
||||||
/>
|
/>
|
||||||
<div className={classes.buttonWrapper}>
|
<div className={classes.buttonWrapper}>
|
||||||
@@ -138,10 +142,12 @@ export function UploadArea({ maximumSizeInBytes }: Props): ReactElement {
|
|||||||
</SwarmButton>
|
</SwarmButton>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<Typography>
|
{showHelp && (
|
||||||
You can click the buttons above or simply drag and drop to add a file or folder. To upload a website to Swarm,
|
<DocumentationText>
|
||||||
make sure that your folder contains an “index.html” file.
|
You can click the buttons above or simply drag and drop to add a file or folder. To upload a website to Swarm,
|
||||||
</Typography>
|
make sure that your folder contains an “index.html” file.
|
||||||
|
</DocumentationText>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,21 @@
|
|||||||
import { ReactElement } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { History } from '../../components/History'
|
import { History } from '../../components/History'
|
||||||
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { defaultUploadOrigin } from '../../providers/File'
|
||||||
import { HISTORY_KEYS } from '../../utils/local-storage'
|
import { HISTORY_KEYS } from '../../utils/local-storage'
|
||||||
import { FileNavigation } from './FileNavigation'
|
import { FileNavigation } from './FileNavigation'
|
||||||
import { UploadArea } from './UploadArea'
|
import { UploadArea } from './UploadArea'
|
||||||
|
|
||||||
const MAX_FILE_SIZE = 1_000_000_000 // 1 gigabyte
|
|
||||||
|
|
||||||
export function UploadLander(): ReactElement {
|
export function UploadLander(): ReactElement {
|
||||||
|
const { status } = useContext(BeeContext)
|
||||||
|
|
||||||
|
if (!status.all) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<FileNavigation active="UPLOAD" />
|
<FileNavigation active="UPLOAD" />
|
||||||
<UploadArea maximumSizeInBytes={MAX_FILE_SIZE} />
|
<UploadArea showHelp={true} uploadOrigin={defaultUploadOrigin} />
|
||||||
<History title="Upload History" localStorageKey={HISTORY_KEYS.UPLOAD_HISTORY} />
|
<History title="Upload History" localStorageKey={HISTORY_KEYS.UPLOAD_HISTORY} />
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -0,0 +1,98 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { Check, X } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { Loading } from '../../components/Loading'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as TopUpContext } from '../../providers/TopUp'
|
||||||
|
import { createGiftWallet } from '../../utils/desktop'
|
||||||
|
import { generateWallet } from '../../utils/identity'
|
||||||
|
import { ResolvedWallet } from '../../utils/wallet'
|
||||||
|
|
||||||
|
export default function Index(): ReactElement {
|
||||||
|
const { giftWallets, addGiftWallet } = useContext(TopUpContext)
|
||||||
|
const { balance } = useContext(BeeContext)
|
||||||
|
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [balances, setBalances] = useState<ResolvedWallet[]>([])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
async function mapGiftWallets() {
|
||||||
|
const results = []
|
||||||
|
for (const giftWallet of giftWallets) {
|
||||||
|
results.push(await ResolvedWallet.make(giftWallet))
|
||||||
|
}
|
||||||
|
setBalances(results)
|
||||||
|
}
|
||||||
|
|
||||||
|
mapGiftWallets()
|
||||||
|
}, [giftWallets])
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
async function onCreate() {
|
||||||
|
enqueueSnackbar('Sending funds to gift wallet...')
|
||||||
|
setLoading(true)
|
||||||
|
try {
|
||||||
|
const wallet = generateWallet()
|
||||||
|
addGiftWallet(wallet)
|
||||||
|
await createGiftWallet(wallet.getAddressString())
|
||||||
|
enqueueSnackbar('Succesfully funded gift wallet', { variant: 'success' })
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Failed to fund gift wallet: ${error}`, { variant: 'error' })
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCancel() {
|
||||||
|
navigate(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!balance) {
|
||||||
|
return <Loading />
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Invite to Swarm...</HistoryHeader>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
Generate and share a gift wallet that anyone can use to set-up their light node with Swarm Desktop. This will
|
||||||
|
use 1 XDAI and 5 BZZ from your node wallet.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItem label="XDAI balance" value={`${balance.dai.toSignificantDigits(4)} XDAI`} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<ExpandableListItem label="BZZ balance" value={`${balance.bzz.toSignificantDigits(4)} BZZ`} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
{balances.map((x, i) => (
|
||||||
|
<Box mb={2} key={i}>
|
||||||
|
<ExpandableListItemKey label={`swarm${String(i).padStart(3, '0')}`} value={x.privateKey} />
|
||||||
|
<ExpandableListItemKey label="Address" value={x.address} />
|
||||||
|
<ExpandableListItem label="XDAI balance" value={`${x.dai.toSignificantDigits(4)} XDAI`} />
|
||||||
|
<ExpandableListItem label="BZZ balance" value={`${x.bzz.toSignificantDigits(4)} BZZ`} />
|
||||||
|
</Box>
|
||||||
|
))}
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton onClick={onCreate} iconType={Check} loading={loading} disabled={loading}>
|
||||||
|
Generate gift wallet
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={onCancel} cancel iconType={X} disabled={loading}>
|
||||||
|
Cancel
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -2,7 +2,7 @@ import { ReactElement, useContext } from 'react'
|
|||||||
import { Button } from '@material-ui/core'
|
import { Button } from '@material-ui/core'
|
||||||
|
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
import ExpandableListItem from '../../components/ExpandableListItem'
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
@@ -17,13 +17,15 @@ export default function Status(): ReactElement {
|
|||||||
topology,
|
topology,
|
||||||
nodeAddresses,
|
nodeAddresses,
|
||||||
chequebookAddress,
|
chequebookAddress,
|
||||||
|
nodeInfo,
|
||||||
} = useContext(BeeContext)
|
} = useContext(BeeContext)
|
||||||
|
|
||||||
if (!status.all) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<ExpandableList label="Bee Node" defaultOpen>
|
<ExpandableList label="Bee Node" defaultOpen>
|
||||||
|
<ExpandableListItem label="Mode" value={nodeInfo?.beeMode} />
|
||||||
<ExpandableListItem
|
<ExpandableListItem
|
||||||
label="Agent"
|
label="Agent"
|
||||||
value={
|
value={
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
import { BeeModes } from '@ethersphere/bee-js'
|
||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { Loading } from '../../components/Loading'
|
||||||
|
import { Context } from '../../providers/Bee'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
|
export default function Settings(): ReactElement {
|
||||||
|
const [startedAt] = useState(Date.now())
|
||||||
|
const { apiHealth, nodeInfo } = useContext(Context)
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (Date.now() - startedAt < 45_000) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (apiHealth && nodeInfo?.beeMode === BeeModes.LIGHT) {
|
||||||
|
navigate(ROUTES.INFO)
|
||||||
|
}
|
||||||
|
}, [startedAt, navigate, nodeInfo, apiHealth])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Loading />
|
||||||
|
</Box>
|
||||||
|
<Typography>Your node is being upgraded to light mode... postage syncing may take up to 10 minutes.</Typography>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { Loading } from '../../components/Loading'
|
||||||
|
import { Context } from '../../providers/Bee'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
|
export default function Settings(): ReactElement {
|
||||||
|
const [waited, setWaited] = useState(false)
|
||||||
|
const { apiHealth } = useContext(Context)
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (waited) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const timeout = setTimeout(() => setWaited(true), 5_000)
|
||||||
|
|
||||||
|
return () => clearTimeout(timeout)
|
||||||
|
}, [waited])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!waited) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (apiHealth) {
|
||||||
|
navigate(ROUTES.INFO)
|
||||||
|
}
|
||||||
|
}, [navigate, waited, apiHealth])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Loading />
|
||||||
|
</Box>
|
||||||
|
<Typography>You will be redirected automatically once your node is up and running.</Typography>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { Box, createStyles, Grid, makeStyles, Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { Battery, BatteryCharging, Check, Gift } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
checkWrapper: {
|
||||||
|
background: 'rgba(0, 230, 118, 0.25)',
|
||||||
|
borderRadius: 99999,
|
||||||
|
width: '180px',
|
||||||
|
height: '180px',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export default function Confirmation(): ReactElement {
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
const styles = useStyles()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Connect to the blockchain</HistoryHeader>
|
||||||
|
<Grid container direction="column" alignItems="center">
|
||||||
|
<Box mb={6}>
|
||||||
|
<div className={styles.checkWrapper}>
|
||||||
|
<Check size={100} color="#ededed" />
|
||||||
|
</div>
|
||||||
|
</Box>
|
||||||
|
<Box mb={1}>
|
||||||
|
<Typography style={{ fontWeight: 'bold' }}>Your node's RPC endpoint is set up correctly!</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography align="center">Lastly, you will need to top-up your node wallet.</Typography>
|
||||||
|
<Typography align="center">
|
||||||
|
If you're not familiar with cryptocurrencies, you can start with a bank card.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton iconType={Battery} onClick={() => navigate(ROUTES.TOP_UP_BANK_CARD)}>
|
||||||
|
Get started with bank card
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={BatteryCharging} onClick={() => navigate(ROUTES.TOP_UP_CRYPTO)}>
|
||||||
|
Use DAI
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={Gift} onClick={() => navigate(ROUTES.TOP_UP_GIFT_CODE)}>
|
||||||
|
Use a gift code
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Grid>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { Check } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { Context } from '../../providers/TopUp'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { Rpc } from '../../utils/rpc'
|
||||||
|
|
||||||
|
export default function Index(): ReactElement {
|
||||||
|
const { jsonRpcProvider, setJsonRpcProvider } = useContext(Context)
|
||||||
|
|
||||||
|
const [provider, setProvider] = useState(jsonRpcProvider)
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
async function onSubmit() {
|
||||||
|
try {
|
||||||
|
await Rpc.eth_getBlockByNumber(provider)
|
||||||
|
enqueueSnackbar('Connected to RPC provider successfully.', { variant: 'success' })
|
||||||
|
setJsonRpcProvider(provider)
|
||||||
|
navigate(ROUTES.CONFIRMATION)
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar('Could not connect to RPC provider.', { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Connect to the blockchain</HistoryHeader>
|
||||||
|
<Box mb={1}>
|
||||||
|
<Typography style={{ fontWeight: 'bold' }}>Set up RPC endpoint</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
To connect to and retrieve data from the blockchain, you'll need to connect to a publicly-provided node
|
||||||
|
via the node's RPC endpoint. If you're not familiar with this, you may use{' '}
|
||||||
|
<a href="https://getblock.io/" target="_blank" rel="noreferrer">
|
||||||
|
https://getblock.io/
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput
|
||||||
|
name="rpc-endpoint"
|
||||||
|
label="RPC Endpoint"
|
||||||
|
onChange={event => setProvider(event.target.value)}
|
||||||
|
defaultValue={jsonRpcProvider}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<SwarmButton iconType={Check} onClick={onSubmit}>
|
||||||
|
Connect
|
||||||
|
</SwarmButton>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,15 +1,46 @@
|
|||||||
|
import CircularProgress from '@material-ui/core/CircularProgress'
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
|
||||||
export default function Settings(): ReactElement {
|
export default function SettingsPage(): ReactElement {
|
||||||
const { apiUrl, apiDebugUrl, setApiUrl, setDebugApiUrl } = useContext(SettingsContext)
|
const { apiUrl, apiDebugUrl, setApiUrl, setDebugApiUrl, lockedApiSettings, config, isLoading } =
|
||||||
|
useContext(SettingsContext)
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return (
|
||||||
|
<div style={{ textAlign: 'center', width: '100%' }}>
|
||||||
|
<CircularProgress />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run within Bee Desktop, display read only config
|
||||||
|
if (config) {
|
||||||
|
return (
|
||||||
|
<ExpandableList label="Bee Desktop Settings" defaultOpen>
|
||||||
|
<ExpandableListItemInput label="Bee API" value={config['api-addr']} locked />
|
||||||
|
<ExpandableListItemInput label="Bee Debug API" value={config['debug-api-addr']} locked />
|
||||||
|
<ExpandableListItemInput label="CORS" value={config['cors-allowed-origins']} locked />
|
||||||
|
<ExpandableListItemInput label="Data DIR" value={config['data-dir']} locked />
|
||||||
|
<ExpandableListItemInput label="ENS resolver URL" value={config['resolver-options']} locked />
|
||||||
|
{config['swap-endpoint'] && (
|
||||||
|
<ExpandableListItemInput label="SWAP endpoint" value={config['swap-endpoint']} locked />
|
||||||
|
)}
|
||||||
|
</ExpandableList>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList label="API Settings" defaultOpen>
|
<ExpandableList label="API Settings" defaultOpen>
|
||||||
<ExpandableListItemInput label="Bee API" value={apiUrl} onConfirm={setApiUrl} />
|
<ExpandableListItemInput label="Bee API" value={apiUrl} onConfirm={setApiUrl} locked={lockedApiSettings} />
|
||||||
<ExpandableListItemInput label="Bee Debug API" value={apiDebugUrl} onConfirm={setDebugApiUrl} />
|
<ExpandableListItemInput
|
||||||
|
label="Bee Debug API"
|
||||||
|
value={apiDebugUrl}
|
||||||
|
onConfirm={setDebugApiUrl}
|
||||||
|
locked={lockedApiSettings}
|
||||||
|
/>
|
||||||
</ExpandableList>
|
</ExpandableList>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,152 +0,0 @@
|
|||||||
import Button from '@material-ui/core/Button'
|
|
||||||
import CircularProgress from '@material-ui/core/CircularProgress'
|
|
||||||
import Dialog from '@material-ui/core/Dialog'
|
|
||||||
import DialogActions from '@material-ui/core/DialogActions'
|
|
||||||
import DialogContent from '@material-ui/core/DialogContent'
|
|
||||||
import DialogContentText from '@material-ui/core/DialogContentText'
|
|
||||||
import DialogTitle from '@material-ui/core/DialogTitle'
|
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
|
||||||
import BigNumber from 'bignumber.js'
|
|
||||||
import { Field, Form, Formik, FormikHelpers } from 'formik'
|
|
||||||
import { TextField } from 'formik-material-ui'
|
|
||||||
import { useSnackbar } from 'notistack'
|
|
||||||
import React, { ReactElement, useContext } from 'react'
|
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
|
||||||
import { Context } from '../../providers/Stamps'
|
|
||||||
|
|
||||||
interface FormValues {
|
|
||||||
depth?: string
|
|
||||||
amount?: string
|
|
||||||
label?: string
|
|
||||||
}
|
|
||||||
type FormErrors = Partial<FormValues>
|
|
||||||
const initialFormValues: FormValues = {
|
|
||||||
depth: '',
|
|
||||||
amount: '',
|
|
||||||
label: '',
|
|
||||||
}
|
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
|
||||||
createStyles({
|
|
||||||
wrapper: {
|
|
||||||
margin: theme.spacing(1),
|
|
||||||
position: 'relative',
|
|
||||||
},
|
|
||||||
field: {
|
|
||||||
marginTop: theme.spacing(1),
|
|
||||||
marginBottom: theme.spacing(1),
|
|
||||||
},
|
|
||||||
buttonProgress: {
|
|
||||||
position: 'absolute',
|
|
||||||
top: '50%',
|
|
||||||
left: '50%',
|
|
||||||
marginTop: -12,
|
|
||||||
marginBottom: -12,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
onClose: () => void
|
|
||||||
}
|
|
||||||
|
|
||||||
export function CreatePostageStampModal({ onClose }: Props): ReactElement {
|
|
||||||
const classes = useStyles()
|
|
||||||
const { refresh } = useContext(Context)
|
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Formik
|
|
||||||
initialValues={initialFormValues}
|
|
||||||
onSubmit={async (values: FormValues, actions: FormikHelpers<FormValues>) => {
|
|
||||||
try {
|
|
||||||
// This is really just a typeguard, the validation pretty much guarantees these will have the right values
|
|
||||||
if (!values.depth || !values.amount) return
|
|
||||||
|
|
||||||
if (!beeDebugApi) return
|
|
||||||
|
|
||||||
const amount = BigInt(values.amount)
|
|
||||||
const depth = Number.parseInt(values.depth)
|
|
||||||
const options = values.label ? { label: values.label } : undefined
|
|
||||||
await beeDebugApi.createPostageBatch(amount.toString(), depth, options)
|
|
||||||
actions.resetForm()
|
|
||||||
await refresh()
|
|
||||||
onClose()
|
|
||||||
} catch (e) {
|
|
||||||
enqueueSnackbar(`Error: ${(e as Error).message}`, { variant: 'error' })
|
|
||||||
actions.setSubmitting(false)
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
validate={(values: FormValues) => {
|
|
||||||
const errors: FormErrors = {}
|
|
||||||
|
|
||||||
// Depth
|
|
||||||
if (!values.depth) errors.depth = 'Required field'
|
|
||||||
else {
|
|
||||||
const depth = new BigNumber(values.depth)
|
|
||||||
|
|
||||||
if (!depth.isInteger()) errors.depth = 'Depth must be an integer'
|
|
||||||
else if (depth.isLessThan(16)) errors.depth = 'Minimal depth is 16'
|
|
||||||
else if (depth.isGreaterThan(255)) errors.depth = 'Depth has to be at most 255'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Amount
|
|
||||||
if (!values.amount) errors.amount = 'Required field'
|
|
||||||
else {
|
|
||||||
const amount = new BigNumber(values.amount)
|
|
||||||
|
|
||||||
if (!amount.isInteger()) errors.amount = 'Amount must be an integer'
|
|
||||||
else if (amount.isLessThanOrEqualTo(0)) errors.amount = 'Amount must be greater than 0'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Label
|
|
||||||
if (values.label && !/^[0-9a-z]*$/i.test(values.label)) errors.label = 'Label must be an alphanumeric string'
|
|
||||||
|
|
||||||
return errors
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{({ submitForm, isValid, isSubmitting, values }) => (
|
|
||||||
<Form>
|
|
||||||
<Dialog open={true} onClose={onClose} aria-labelledby="form-dialog-title">
|
|
||||||
<DialogTitle id="form-dialog-title">Buy new postage stamp</DialogTitle>
|
|
||||||
<DialogContent>
|
|
||||||
<Field
|
|
||||||
component={TextField}
|
|
||||||
required
|
|
||||||
name="depth"
|
|
||||||
autoFocus
|
|
||||||
label="Depth"
|
|
||||||
fullWidth
|
|
||||||
className={classes.field}
|
|
||||||
/>
|
|
||||||
<Field component={TextField} required name="amount" label="Amount" fullWidth className={classes.field} />
|
|
||||||
<Field component={TextField} name="label" label="Label" fullWidth className={classes.field} />
|
|
||||||
</DialogContent>
|
|
||||||
<DialogActions>
|
|
||||||
<Button onClick={onClose} variant="contained">
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<div className={classes.wrapper}>
|
|
||||||
<Button
|
|
||||||
disabled={isSubmitting || !isValid || !values.amount || !values.depth}
|
|
||||||
type="submit"
|
|
||||||
variant="contained"
|
|
||||||
onClick={submitForm}
|
|
||||||
>
|
|
||||||
Create
|
|
||||||
{isSubmitting && <CircularProgress size={24} className={classes.buttonProgress} />}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</DialogActions>
|
|
||||||
<DialogContent>
|
|
||||||
<DialogContentText>
|
|
||||||
Please refer to the official Bee documentation to understand these values.
|
|
||||||
</DialogContentText>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
</Form>
|
|
||||||
)}
|
|
||||||
</Formik>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { PostageStampCreation } from './PostageStampCreation'
|
||||||
|
|
||||||
|
export function CreatePostageStampPage(): ReactElement {
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
function onFinished() {
|
||||||
|
navigate(ROUTES.STAMPS)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<HistoryHeader>Buy new postage stamp</HistoryHeader>
|
||||||
|
<PostageStampCreation onFinished={onFinished} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -9,10 +9,13 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function PostageStamp({ stamp, shorten }: Props): ReactElement {
|
export function PostageStamp({ stamp, shorten }: Props): ReactElement {
|
||||||
|
const batchId = shorten ? stamp.batchID.slice(0, 8) : stamp.batchID
|
||||||
|
const label = `${batchId}${stamp.label ? ` - ${stamp.label}` : ''}`
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box p={2} width="100%">
|
<Box p={2} width="100%">
|
||||||
<Grid container justifyContent="space-between" alignItems="center" direction="row">
|
<Grid container justifyContent="space-between" alignItems="center" direction="row">
|
||||||
<Typography variant="subtitle2">{shorten ? stamp.batchID.slice(0, 8) : stamp.batchID}</Typography>
|
<Typography variant="subtitle2">{label}</Typography>
|
||||||
<Capacity width="100px" usage={stamp.usage} />
|
<Capacity width="100px" usage={stamp.usage} />
|
||||||
</Grid>
|
</Grid>
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -0,0 +1,173 @@
|
|||||||
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
|
import BigNumber from 'bignumber.js'
|
||||||
|
import { Form, Formik, FormikHelpers } from 'formik'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext } from 'react'
|
||||||
|
import { Check } from 'react-feather'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Context as StampsContext } from '../../providers/Stamps'
|
||||||
|
import {
|
||||||
|
calculateStampPrice,
|
||||||
|
convertAmountToSeconds,
|
||||||
|
convertDepthToBytes,
|
||||||
|
secondsToTimeString,
|
||||||
|
waitUntilStampUsable,
|
||||||
|
} from '../../utils'
|
||||||
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
|
|
||||||
|
interface FormValues {
|
||||||
|
depth?: string
|
||||||
|
amount?: string
|
||||||
|
label?: string
|
||||||
|
}
|
||||||
|
type FormErrors = Partial<FormValues>
|
||||||
|
const initialFormValues: FormValues = {
|
||||||
|
depth: '',
|
||||||
|
amount: '',
|
||||||
|
label: '',
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onFinished: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export function PostageStampCreation({ onFinished }: Props): ReactElement {
|
||||||
|
const { chainState } = useContext(BeeContext)
|
||||||
|
const { refresh } = useContext(StampsContext)
|
||||||
|
const { beeDebugApi } = useContext(SettingsContext)
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
function getFileSize(depth: number): string {
|
||||||
|
if (isNaN(depth) || depth < 17 || depth > 255) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
return `~${getHumanReadableFileSize(convertDepthToBytes(depth))}`
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTtl(amount: number): string {
|
||||||
|
const isCurrentPriceAvailable = chainState && chainState.currentPrice
|
||||||
|
|
||||||
|
if (amount <= 0 || !isCurrentPriceAvailable) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
const pricePerBlock = Number.parseInt(chainState.currentPrice, 10)
|
||||||
|
|
||||||
|
return `${secondsToTimeString(
|
||||||
|
convertAmountToSeconds(amount, pricePerBlock),
|
||||||
|
)} (with price of ${pricePerBlock.toFixed(0)} per block)`
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPrice(depth: number, amount: bigint): string {
|
||||||
|
const hasInvalidInput = amount <= 0 || isNaN(depth) || depth < 17 || depth > 255
|
||||||
|
|
||||||
|
if (hasInvalidInput) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
const price = calculateStampPrice(depth, amount)
|
||||||
|
|
||||||
|
return `${price.toSignificantDigits()} BZZ`
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Formik
|
||||||
|
initialValues={initialFormValues}
|
||||||
|
onSubmit={async (values: FormValues, actions: FormikHelpers<FormValues>) => {
|
||||||
|
try {
|
||||||
|
// This is really just a typeguard, the validation pretty much guarantees these will have the right values
|
||||||
|
if (!values.depth || !values.amount) return
|
||||||
|
|
||||||
|
if (!beeDebugApi) return
|
||||||
|
|
||||||
|
const amount = BigInt(values.amount)
|
||||||
|
const depth = Number.parseInt(values.depth)
|
||||||
|
const options = values.label ? { label: values.label } : undefined
|
||||||
|
const batch = await beeDebugApi.createPostageBatch(amount.toString(), depth, options)
|
||||||
|
await waitUntilStampUsable(batch, beeDebugApi)
|
||||||
|
actions.resetForm()
|
||||||
|
await refresh()
|
||||||
|
onFinished()
|
||||||
|
} catch (e) {
|
||||||
|
enqueueSnackbar(`Error: ${(e as Error).message}`, { variant: 'error' })
|
||||||
|
actions.setSubmitting(false)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
validate={(values: FormValues) => {
|
||||||
|
const errors: FormErrors = {}
|
||||||
|
|
||||||
|
// Depth
|
||||||
|
if (!values.depth) errors.depth = 'Required field'
|
||||||
|
else {
|
||||||
|
const depth = new BigNumber(values.depth)
|
||||||
|
|
||||||
|
if (!depth.isInteger()) errors.depth = 'Depth must be an integer'
|
||||||
|
else if (depth.isLessThan(16)) errors.depth = 'Minimal depth is 16'
|
||||||
|
else if (depth.isGreaterThan(255)) errors.depth = 'Depth has to be at most 255'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Amount
|
||||||
|
if (!values.amount) errors.amount = 'Required field'
|
||||||
|
else {
|
||||||
|
const amount = new BigNumber(values.amount)
|
||||||
|
|
||||||
|
if (!amount.isInteger()) errors.amount = 'Amount must be an integer'
|
||||||
|
else if (amount.isLessThanOrEqualTo(0)) errors.amount = 'Amount must be greater than 0'
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{({ submitForm, isValid, isSubmitting, values, errors }) => (
|
||||||
|
<Form>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="depth" label="Depth" formik />
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Corresponding file size</Typography>
|
||||||
|
<Typography>{!errors.depth && values.depth ? getFileSize(parseInt(values.depth, 10)) : '-'}</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="amount" label="Amount" formik />
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Corresponding TTL (Time to live)</Typography>
|
||||||
|
<Typography>
|
||||||
|
{!errors.amount && values.amount ? getTtl(Number.parseInt(values.amount, 10)) : '-'}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="label" label="Label" optional formik />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4} sx={{ bgcolor: '#fcf2e8' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Indicative Price</Typography>
|
||||||
|
<Typography>
|
||||||
|
{!errors.amount && !errors.depth && values.amount && values.depth
|
||||||
|
? getPrice(parseInt(values.depth, 10), BigInt(values.amount))
|
||||||
|
: '-'}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<SwarmButton
|
||||||
|
disabled={isSubmitting || !isValid || !values.amount || !values.depth}
|
||||||
|
onClick={submitForm}
|
||||||
|
iconType={Check}
|
||||||
|
loading={isSubmitting}
|
||||||
|
>
|
||||||
|
Buy New Stamp
|
||||||
|
</SwarmButton>
|
||||||
|
</Form>
|
||||||
|
)}
|
||||||
|
</Formik>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import React, { ReactElement, useContext } from 'react'
|
||||||
|
import { SwarmSelect } from '../../components/SwarmSelect'
|
||||||
|
import { Context, EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onSelect: (stamp: EnrichedPostageBatch) => void
|
||||||
|
defaultValue?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function PostageStampSelector({ onSelect, defaultValue }: Props): ReactElement {
|
||||||
|
const { stamps } = useContext(Context)
|
||||||
|
|
||||||
|
function onChange(stampId: string) {
|
||||||
|
if (!stamps) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const stamp = stamps.find(x => x.batchID === stampId)
|
||||||
|
|
||||||
|
if (stamp) {
|
||||||
|
onSelect(stamp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SwarmSelect
|
||||||
|
options={(stamps || []).map(x => ({ label: x.batchID.slice(0, 8), value: x.batchID }))}
|
||||||
|
onChange={event => onChange(event.target.value as string)}
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { createStyles, FormControl, makeStyles, MenuItem, Select, Theme } from '@material-ui/core'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core'
|
||||||
import Button from '@material-ui/core/Button'
|
import Button from '@material-ui/core/Button'
|
||||||
import Dialog from '@material-ui/core/Dialog'
|
import Dialog from '@material-ui/core/Dialog'
|
||||||
import DialogContent from '@material-ui/core/DialogContent'
|
import DialogContent from '@material-ui/core/DialogContent'
|
||||||
@@ -6,6 +6,7 @@ import DialogTitle from '@material-ui/core/DialogTitle'
|
|||||||
import { Check, Clear } from '@material-ui/icons'
|
import { Check, Clear } from '@material-ui/icons'
|
||||||
import React, { ReactElement, useState } from 'react'
|
import React, { ReactElement, useState } from 'react'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import { SwarmSelect } from '../../components/SwarmSelect'
|
||||||
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -26,14 +27,6 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
color: '#606060',
|
color: '#606060',
|
||||||
textAlign: 'center',
|
textAlign: 'center',
|
||||||
},
|
},
|
||||||
select: {
|
|
||||||
background: theme.palette.background.paper,
|
|
||||||
borderRadius: 0,
|
|
||||||
border: 0,
|
|
||||||
},
|
|
||||||
option: {
|
|
||||||
height: '52px',
|
|
||||||
},
|
|
||||||
hint: {
|
hint: {
|
||||||
marginBottom: '16px',
|
marginBottom: '16px',
|
||||||
},
|
},
|
||||||
@@ -72,21 +65,10 @@ export function SelectPostageStampModal({ stamps, onSelect, onClose }: Props): R
|
|||||||
Select postage stamp
|
Select postage stamp
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent>
|
<DialogContent>
|
||||||
<FormControl fullWidth>
|
<SwarmSelect
|
||||||
<Select
|
options={stamps.map(x => ({ label: x.batchID, value: x.batchID }))}
|
||||||
onChange={event => onChange(event.target.value as string)}
|
onChange={event => onChange(event.target.value as string)}
|
||||||
fullWidth
|
/>
|
||||||
variant="outlined"
|
|
||||||
className={classes.select}
|
|
||||||
defaultValue=""
|
|
||||||
>
|
|
||||||
{stamps.map(x => (
|
|
||||||
<MenuItem key={x.batchID} value={x.batchID} className={classes.option}>
|
|
||||||
{x.batchID.slice(0, 8)}
|
|
||||||
</MenuItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogContent>
|
<DialogContent>
|
||||||
<ExpandableListItemActions>
|
<ExpandableListItemActions>
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
import type { ReactElement } from 'react'
|
import type { ReactElement } from 'react'
|
||||||
import ExpandableElement from '../../components/ExpandableElement'
|
import ExpandableElement from '../../components/ExpandableElement'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
|
import { secondsToTimeString } from '../../utils'
|
||||||
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
import { PostageStamp } from './PostageStamp'
|
import { PostageStamp } from './PostageStamp'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -17,7 +20,27 @@ function StampsTable({ postageStamps }: Props): ReactElement | null {
|
|||||||
{postageStamps.map(stamp => (
|
{postageStamps.map(stamp => (
|
||||||
<ExpandableElement
|
<ExpandableElement
|
||||||
key={stamp.batchID}
|
key={stamp.batchID}
|
||||||
expandable={<ExpandableListItemKey label="Batch ID" value={stamp.batchID} />}
|
expandable={
|
||||||
|
<>
|
||||||
|
<ExpandableListItemKey label="Batch ID" value={stamp.batchID} />
|
||||||
|
<ExpandableListItem label="Depth" value={String(stamp.depth)} />
|
||||||
|
<ExpandableListItem
|
||||||
|
label="Capacity"
|
||||||
|
value={`${getHumanReadableFileSize(2 ** stamp.depth * 4096 * stamp.usage)} / ${getHumanReadableFileSize(
|
||||||
|
2 ** stamp.depth * 4096,
|
||||||
|
)}`}
|
||||||
|
/>
|
||||||
|
<ExpandableListItem label="Amount" value={parseInt(stamp.amount, 10).toLocaleString()} />
|
||||||
|
<ExpandableListItem
|
||||||
|
label="Expires in"
|
||||||
|
value={stamp.batchTTL === -1 ? 'does not expire' : `${secondsToTimeString(stamp.batchTTL)}`}
|
||||||
|
/>
|
||||||
|
<ExpandableListItem label="Label" value={stamp.label} />
|
||||||
|
<ExpandableListItem label="Usable" value={stamp.usable ? 'yes' : 'no'} />
|
||||||
|
<ExpandableListItem label="Exists" value={stamp.exists ? 'yes' : 'no'} />
|
||||||
|
<ExpandableListItem label="Purchase Block Number" value={stamp.blockNumber} />
|
||||||
|
</>
|
||||||
|
}
|
||||||
>
|
>
|
||||||
<PostageStamp stamp={stamp} shorten={true} />
|
<PostageStamp stamp={stamp} shorten={true} />
|
||||||
</ExpandableElement>
|
</ExpandableElement>
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
import { CircularProgress, Container } from '@material-ui/core'
|
import { CircularProgress, Container } from '@material-ui/core'
|
||||||
import { createStyles, makeStyles } from '@material-ui/core/styles'
|
import { createStyles, makeStyles } from '@material-ui/core/styles'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect } from 'react'
|
||||||
import { PlusSquare } from 'react-feather'
|
import { PlusSquare } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as StampsContext } from '../../providers/Stamps'
|
import { Context as StampsContext } from '../../providers/Stamps'
|
||||||
import { CreatePostageStampModal } from './CreatePostageStampModal'
|
import { ROUTES } from '../../routes'
|
||||||
import StampsTable from './StampsTable'
|
import StampsTable from './StampsTable'
|
||||||
|
|
||||||
const useStyles = makeStyles(() =>
|
const useStyles = makeStyles(() =>
|
||||||
@@ -28,7 +29,7 @@ const useStyles = makeStyles(() =>
|
|||||||
export default function Stamp(): ReactElement {
|
export default function Stamp(): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
|
|
||||||
const [isBuyingStamp, setBuyingStamp] = useState(false)
|
const navigate = useNavigate()
|
||||||
|
|
||||||
const { stamps, isLoading, error, start, stop } = useContext(StampsContext)
|
const { stamps, isLoading, error, start, stop } = useContext(StampsContext)
|
||||||
const { status } = useContext(BeeContext)
|
const { status } = useContext(BeeContext)
|
||||||
@@ -40,7 +41,11 @@ export default function Stamp(): ReactElement {
|
|||||||
return () => stop()
|
return () => stop()
|
||||||
}, [status]) // eslint-disable-line react-hooks/exhaustive-deps
|
}, [status]) // eslint-disable-line react-hooks/exhaustive-deps
|
||||||
|
|
||||||
if (!status.all) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
|
function navigateToNewStamp() {
|
||||||
|
navigate(ROUTES.STAMPS_NEW)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={classes.root}>
|
<div className={classes.root}>
|
||||||
@@ -52,9 +57,7 @@ export default function Stamp(): ReactElement {
|
|||||||
{!error && (
|
{!error && (
|
||||||
<>
|
<>
|
||||||
<div className={classes.actions}>
|
<div className={classes.actions}>
|
||||||
{isBuyingStamp ? <CreatePostageStampModal onClose={() => setBuyingStamp(false)} /> : null}
|
<SwarmButton onClick={navigateToNewStamp} iconType={PlusSquare}>
|
||||||
|
|
||||||
<SwarmButton onClick={() => setBuyingStamp(true)} iconType={PlusSquare}>
|
|
||||||
Buy New Postage Stamp
|
Buy New Postage Stamp
|
||||||
</SwarmButton>
|
</SwarmButton>
|
||||||
<div style={{ height: '5px' }}>{isLoading && <CircularProgress />}</div>
|
<div style={{ height: '5px' }}>{isLoading && <CircularProgress />}</div>
|
||||||
|
|||||||
@@ -1,39 +1,59 @@
|
|||||||
import { useContext } from 'react'
|
import { useContext } from 'react'
|
||||||
import DepositModal from '../../../containers/DepositModal'
|
import DepositModal from '../../../containers/DepositModal'
|
||||||
import type { ReactElement } from 'react'
|
import type { ReactElement, ReactNode } from 'react'
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemActions from '../../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
const ChequebookDeployFund = (): ReactElement | null => {
|
const ChequebookDeployFund = (): ReactElement | null => {
|
||||||
const { status, isLoading, chequebookAddress } = useContext(Context)
|
const { status, isLoading, chequebookAddress } = useContext(Context)
|
||||||
const isOk = status.chequebook
|
const { checkState, isEnabled } = status.chequebook
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
|
let text: ReactNode
|
||||||
|
|
||||||
|
switch (checkState) {
|
||||||
|
case CheckState.OK:
|
||||||
|
text = 'Your chequebook is deployed and funded'
|
||||||
|
break
|
||||||
|
case CheckState.WARNING:
|
||||||
|
text = (
|
||||||
|
<>
|
||||||
|
Your chequebook is not funded. Please deposit some xBZZ to your chequebook address. You may need to aquire BZZ
|
||||||
|
(e.g. <a href="https://bzz.exchange/">bzz.exchange</a>) and bridge it to the xDai network through the{' '}
|
||||||
|
<a href="https://omni.xdaichain.com/bridge">omni bridge</a>. To pay the transaction fees, you will also need
|
||||||
|
xDAI token. You can purchase DAI on the network and bridge it to xDai network through the{' '}
|
||||||
|
<a href="https://bridge.xdaichain.com/">xDai Bridge</a>. See the{' '}
|
||||||
|
<a href="https://www.xdaichain.com/#xdai-stable-chain">official xDai website</a> for more information.
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
text = (
|
||||||
|
<>
|
||||||
|
Your chequebook is either not deployed nor funded. To run the node you will need xDAI and xBZZ on the xDai
|
||||||
|
network. You may need to aquire BZZ (e.g. <a href="https://bzz.exchange/">bzz.exchange</a>) and bridge it to
|
||||||
|
the xDai network through the <a href="https://omni.xdaichain.com/bridge">omni bridge</a>. To pay the
|
||||||
|
transaction fees, you will also need xDAI token. You can purchase DAI on the network and bridge it to xDai
|
||||||
|
network through the <a href="https://bridge.xdaichain.com/">xDai Bridge</a>. See the{' '}
|
||||||
|
<a href="https://www.xdaichain.com/#xdai-stable-chain">official xDai website</a> for more information.
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Chequebook Deployment & Funding
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Chequebook Deployment & Funding
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>{text}</ExpandableListItemNote>
|
||||||
{isOk ? (
|
|
||||||
'Your chequebook is deployed and funded'
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
Your chequebook is either not deployed or funded. To run the node you will need xDAI and xBZZ on the xDai
|
|
||||||
network. You may need to aquire BZZ (e.g. <a href="https://bzz.exchange/">bzz.exchange</a>) and bridge it to
|
|
||||||
the xDai network through the <a href="https://omni.xdaichain.com/bridge">omni bridge</a>. To pay the
|
|
||||||
transaction fees, you will also need xDAI token. You can purchase DAI on the network and bridge it to xDai
|
|
||||||
network through the <a href="https://bridge.xdaichain.com/">xDai Bridge</a>. See the{' '}
|
|
||||||
<a href="https://www.xdaichain.com/#xdai-stable-chain">official xDai website</a> for more information.
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</ExpandableListItemNote>
|
|
||||||
{chequebookAddress && (
|
{chequebookAddress && (
|
||||||
<>
|
<>
|
||||||
<ExpandableListItemKey label="Chequebook Address" value={chequebookAddress.chequebookAddress} />
|
<ExpandableListItemKey label="Chequebook Address" value={chequebookAddress.chequebookAddress} />
|
||||||
|
|||||||
@@ -6,30 +6,32 @@ import ExpandableListItem from '../../../components/ExpandableListItem'
|
|||||||
import ExpandableListItemInput from '../../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../../components/ExpandableListItemInput'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../../providers/Settings'
|
import { Context as SettingsContext } from '../../../providers/Settings'
|
||||||
|
|
||||||
export default function NodeConnectionCheck(): ReactElement | null {
|
export default function NodeConnectionCheck(): ReactElement | null {
|
||||||
const { status, isLoading } = useContext(Context)
|
const { status, isLoading } = useContext(Context)
|
||||||
const { setDebugApiUrl, apiDebugUrl } = useContext(SettingsContext)
|
const { setDebugApiUrl, apiDebugUrl } = useContext(SettingsContext)
|
||||||
const isOk = status.debugApiConnection
|
const { checkState, isEnabled } = status.debugApiConnection
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Connection to Bee Debug API
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Bee Debug API
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>
|
||||||
{isOk
|
{checkState === CheckState.OK
|
||||||
? 'The connection to the Bee nodes debug API has been successful'
|
? 'The connection to the Bee nodes debug API has been successful'
|
||||||
: 'We cannot connect to your nodes debug API. Please check the following to troubleshoot your issue.'}
|
: 'We cannot connect to your nodes debug API. Please check the following to troubleshoot your issue.'}
|
||||||
</ExpandableListItemNote>
|
</ExpandableListItemNote>
|
||||||
<ExpandableListItemInput label="Bee Debug API" value={apiDebugUrl} onConfirm={setDebugApiUrl} />
|
<ExpandableListItemInput label="Bee Debug API" value={apiDebugUrl} onConfirm={setDebugApiUrl} />
|
||||||
|
|
||||||
{!isOk && (
|
{checkState === CheckState.ERROR && (
|
||||||
<ExpandableList level={1} label="Troubleshoot">
|
<ExpandableList level={1} label="Troubleshoot">
|
||||||
<ExpandableListItem
|
<ExpandableListItem
|
||||||
label={
|
label={
|
||||||
|
|||||||
@@ -3,22 +3,24 @@ import ExpandableList from '../../../components/ExpandableList'
|
|||||||
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
export default function EthereumConnectionCheck(): ReactElement | null {
|
export default function EthereumConnectionCheck(): ReactElement | null {
|
||||||
const { status, isLoading, nodeAddresses } = useContext(Context)
|
const { status, isLoading, nodeAddresses } = useContext(Context)
|
||||||
const isOk = status.blockchainConnection
|
const { checkState, isEnabled } = status.blockchainConnection
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Connection to Blockchain
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Blockchain
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>
|
||||||
{isOk ? (
|
{checkState === CheckState.OK ? (
|
||||||
'Your node is connected to the xDai blockchain'
|
'Your node is connected to the xDai blockchain'
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
|
|||||||
@@ -7,28 +7,30 @@ import ExpandableListItem from '../../../components/ExpandableListItem'
|
|||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import ExpandableListItemInput from '../../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../../components/ExpandableListItemInput'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
export default function NodeConnectionCheck(): ReactElement | null {
|
export default function NodeConnectionCheck(): ReactElement | null {
|
||||||
const { setApiUrl, apiUrl } = useContext(SettingsContext)
|
const { setApiUrl, apiUrl } = useContext(SettingsContext)
|
||||||
const { status, isLoading } = useContext(Context)
|
const { status, isLoading } = useContext(Context)
|
||||||
const isOk = status.apiConnection
|
const { isEnabled, checkState } = status.apiConnection
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Connection to Bee API
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Bee API
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>
|
||||||
{isOk
|
{checkState === CheckState.OK
|
||||||
? 'The connection to the Bee nodes API has been successful'
|
? 'The connection to the Bee nodes API has been successful'
|
||||||
: 'Could not connect to your Bee nodes API. Please check the troubleshoot below on how you may resolve it.'}
|
: 'Could not connect to your Bee nodes API. Please check the troubleshoot below on how you may resolve it.'}
|
||||||
</ExpandableListItemNote>
|
</ExpandableListItemNote>
|
||||||
<ExpandableListItemInput label="Bee API" value={apiUrl} onConfirm={setApiUrl} />
|
<ExpandableListItemInput label="Bee API" value={apiUrl} onConfirm={setApiUrl} />
|
||||||
{!isOk && (
|
{checkState === CheckState.ERROR && (
|
||||||
<ExpandableList level={1} label="Troubleshoot">
|
<ExpandableList level={1} label="Troubleshoot">
|
||||||
<ExpandableListItem
|
<ExpandableListItem
|
||||||
label={
|
label={
|
||||||
|
|||||||
@@ -1,27 +1,37 @@
|
|||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, ReactNode, useContext } from 'react'
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import TopologyStats from '../../../components/TopologyStats'
|
import TopologyStats from '../../../components/TopologyStats'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
export default function PeerConnection(): ReactElement | null {
|
export default function PeerConnection(): ReactElement | null {
|
||||||
const { status, isLoading, topology } = useContext(Context)
|
const { status, isLoading, topology } = useContext(Context)
|
||||||
const isOk = status.topology
|
const { isEnabled, checkState } = status.topology
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
|
let text: ReactNode
|
||||||
|
switch (checkState) {
|
||||||
|
case CheckState.OK:
|
||||||
|
text = 'You are connected to other Bee nodes'
|
||||||
|
break
|
||||||
|
|
||||||
|
// Both error state and warning state
|
||||||
|
default:
|
||||||
|
text =
|
||||||
|
'Your node is not connected to any peers. Please wait a bit if you just started the node, otherwise review your configuration file.'
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Connection to Peers
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Peers
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>{text}</ExpandableListItemNote>
|
||||||
{isOk
|
|
||||||
? 'You are connected to other Bee nodes'
|
|
||||||
: 'Your node is not connected to any peers. Please wait a bit if you just started the node, otherwise review your configuration file.'}
|
|
||||||
</ExpandableListItemNote>
|
|
||||||
|
|
||||||
<TopologyStats topology={topology} />
|
<TopologyStats topology={topology} />
|
||||||
</ExpandableList>
|
</ExpandableList>
|
||||||
|
|||||||
@@ -4,22 +4,24 @@ import ExpandableList from '../../../components/ExpandableList'
|
|||||||
import ExpandableListItem from '../../../components/ExpandableListItem'
|
import ExpandableListItem from '../../../components/ExpandableListItem'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
import { Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
export default function VersionCheck(): ReactElement | null {
|
export default function VersionCheck(): ReactElement | null {
|
||||||
const { status, isLoading, latestUserVersion, latestPublishedVersion, latestBeeVersionUrl } = useContext(Context)
|
const { status, isLoading, latestUserVersion, latestPublishedVersion, latestBeeVersionUrl } = useContext(Context)
|
||||||
const isOk = status.version
|
const { isEnabled, checkState } = status.version
|
||||||
|
|
||||||
|
if (!isEnabled) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={
|
label={
|
||||||
<>
|
<>
|
||||||
<StatusIcon isOk={isOk} isLoading={isLoading} /> Bee Version
|
<StatusIcon checkState={checkState} isLoading={isLoading} /> Bee Version
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ExpandableListItemNote>
|
<ExpandableListItemNote>
|
||||||
{isOk ? (
|
{checkState === CheckState.OK ? (
|
||||||
'You are running the latest version of Bee.'
|
'You are running the latest version of Bee.'
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
import { Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import Index from '.'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
|
export function BankCardTopUpIndex(): ReactElement {
|
||||||
|
return (
|
||||||
|
<Index
|
||||||
|
header={'Top-up with bank card'}
|
||||||
|
title={'Use a bank card to buy xDAI to the funding wallet address below'}
|
||||||
|
p={
|
||||||
|
<Typography>
|
||||||
|
It's recommended to buy an amount equivalent to 5 to 10 EUR maximum. If you're not familiar with
|
||||||
|
cryptocurrencies, you may use{' '}
|
||||||
|
<a href="https://ramp.network/buy/" rel="noreferrer" target="_blank">
|
||||||
|
https://ramp.network/buy/
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</Typography>
|
||||||
|
}
|
||||||
|
next={ROUTES.TOP_UP_BANK_CARD_SWAP}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { Typography } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
import Index from '.'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
|
export function CryptoTopUpIndex(): ReactElement {
|
||||||
|
return (
|
||||||
|
<Index
|
||||||
|
header={'Top-up with cryptocurrencies'}
|
||||||
|
title={'Send xDAI to the funding wallet below'}
|
||||||
|
p={
|
||||||
|
<Typography>
|
||||||
|
For security reasons it is recommended to send maximum 5 to 10 xDAI. To get xDAI from DAI you may use{' '}
|
||||||
|
<a href="https://bridge.xdaichain.com/" rel="noreferrer" target="_blank">
|
||||||
|
https://bridge.xdaichain.com/
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</Typography>
|
||||||
|
}
|
||||||
|
next={ROUTES.TOP_UP_CRYPTO_SWAP}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,107 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { ArrowDown, Check } from 'react-feather'
|
||||||
|
import { useNavigate, useParams } from 'react-router'
|
||||||
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { Loading } from '../../components/Loading'
|
||||||
|
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDivider } from '../../components/SwarmDivider'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as TopUpContext } from '../../providers/TopUp'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { sleepMs } from '../../utils'
|
||||||
|
import { restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
||||||
|
import { ResolvedWallet } from '../../utils/wallet'
|
||||||
|
|
||||||
|
export function GiftCardFund(): ReactElement {
|
||||||
|
const { nodeAddresses, balance } = useContext(BeeContext)
|
||||||
|
const { jsonRpcProvider } = useContext(TopUpContext)
|
||||||
|
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [wallet, setWallet] = useState<ResolvedWallet | null>(null)
|
||||||
|
|
||||||
|
const { privateKeyString } = useParams()
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!privateKeyString) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ResolvedWallet.make(privateKeyString).then(setWallet)
|
||||||
|
}, [privateKeyString])
|
||||||
|
|
||||||
|
if (!wallet || !balance) {
|
||||||
|
return <Loading />
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onFund() {
|
||||||
|
if (!wallet || !nodeAddresses) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
|
try {
|
||||||
|
await wallet.transfer(nodeAddresses.ethereum)
|
||||||
|
enqueueSnackbar('Successfully funded node, restarting...', { variant: 'success' })
|
||||||
|
await sleepMs(5_000)
|
||||||
|
await upgradeToLightNode(jsonRpcProvider)
|
||||||
|
await restartBeeNode()
|
||||||
|
navigate(ROUTES.RESTART_LIGHT)
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Failed to fund/restart node: ${error}`, { variant: 'error' })
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Top-up with gift code</HistoryHeader>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ProgressIndicator index={1} steps={['Paste gift code', 'Fund your node']} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography style={{ fontWeight: 'bold' }}>Send funds to your Bee node</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
Deposit all the funds from the gift wallet to your node wallet address. You can use the button below to
|
||||||
|
transfer all funds to your node.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<SwarmDivider mb={4} />
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItemKey label="Gift wallet address" value={wallet.address || 'N/A'} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItem label="XDAI balance" value={`${wallet.dai.toSignificantDigits(4)} XDAI`} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ExpandableListItem label="BZZ balance" value={`${wallet.bzz.toSignificantDigits(4)} BZZ`} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ArrowDown size={24} color="#aaaaaa" />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItemKey label="Node wallet address" value={nodeAddresses?.ethereum || 'N/A'} expanded />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItem label="XDAI balance" value={`${balance.dai.toSignificantDigits(4)} XDAI`} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<ExpandableListItem label="BZZ balance" value={`${balance.bzz.toSignificantDigits(4)} BZZ`} />
|
||||||
|
</Box>
|
||||||
|
<SwarmButton iconType={Check} onClick={onFund} disabled={loading} loading={loading}>
|
||||||
|
Send all funds to your node
|
||||||
|
</SwarmButton>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,71 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useState } from 'react'
|
||||||
|
import { ArrowRight } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDivider } from '../../components/SwarmDivider'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { BzzToken } from '../../models/BzzToken'
|
||||||
|
import { DaiToken } from '../../models/DaiToken'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { getWalletFromPrivateKeyString } from '../../utils/identity'
|
||||||
|
import { Rpc } from '../../utils/rpc'
|
||||||
|
|
||||||
|
export function GiftCardTopUpIndex(): ReactElement {
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [giftCode, setGiftCode] = useState('')
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
async function onProceed() {
|
||||||
|
setLoading(true)
|
||||||
|
try {
|
||||||
|
const wallet = getWalletFromPrivateKeyString(giftCode)
|
||||||
|
const dai = new DaiToken(await Rpc._eth_getBalance(wallet.getAddressString()))
|
||||||
|
const bzz = new BzzToken(await Rpc._eth_getBalanceERC20(wallet.getAddressString()))
|
||||||
|
|
||||||
|
if (dai.toDecimal.lt(0.001) || bzz.toDecimal.lt(0.001)) {
|
||||||
|
throw Error('Gift wallet does not have enough funds')
|
||||||
|
}
|
||||||
|
enqueueSnackbar('Successfully verified gift wallet', { variant: 'success' })
|
||||||
|
navigate(ROUTES.TOP_UP_GIFT_CODE_FUND.replace(':privateKeyString', giftCode))
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Gift wallet could not be verified: ${error}`, { variant: 'error' })
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>Top-up with gift code</HistoryHeader>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ProgressIndicator index={0} steps={['Paste gift code', 'Fund your node']} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography style={{ fontWeight: 'bold' }}>Please paste your gift code below</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
A gift code is a unique key to a gift wallet that you can use to fund your node. Please don't share your
|
||||||
|
gift code as it can only be used once.
|
||||||
|
</Box>
|
||||||
|
<SwarmDivider mb={4} />
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput
|
||||||
|
label="Gift code"
|
||||||
|
name="gift-code"
|
||||||
|
onChange={event => {
|
||||||
|
setGiftCode(event.target.value)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<SwarmButton iconType={ArrowRight} loading={loading} disabled={loading} onClick={onProceed}>
|
||||||
|
Proceed
|
||||||
|
</SwarmButton>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,127 @@
|
|||||||
|
import { Box, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { ArrowDown, Check } from 'react-feather'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { Loading } from '../../components/Loading'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmDivider } from '../../components/SwarmDivider'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { BzzToken } from '../../models/BzzToken'
|
||||||
|
import { DaiToken } from '../../models/DaiToken'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as TopUpContext } from '../../providers/TopUp'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { sleepMs } from '../../utils'
|
||||||
|
import { performSwap, restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
||||||
|
import { TopUpProgressIndicator } from './TopUpProgressIndicator'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
header: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Swap({ header }: Props): ReactElement {
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [hasSwapped, setSwapped] = useState(false)
|
||||||
|
|
||||||
|
const { jsonRpcProvider } = useContext(TopUpContext)
|
||||||
|
const { balance } = useContext(BeeContext)
|
||||||
|
|
||||||
|
const navigate = useNavigate()
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
if (!balance) {
|
||||||
|
return <Loading />
|
||||||
|
}
|
||||||
|
|
||||||
|
const daiToSwap = balance.dai.minusBaseUnits('1')
|
||||||
|
|
||||||
|
const daiAfterSwap = new DaiToken(balance.dai.toBigNumber.minus(daiToSwap.toBigNumber))
|
||||||
|
const bzzAfterSwap = new BzzToken(daiToSwap.toBigNumber.dividedToIntegerBy(200))
|
||||||
|
|
||||||
|
async function onSwap() {
|
||||||
|
if (hasSwapped) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setLoading(true)
|
||||||
|
setSwapped(true)
|
||||||
|
try {
|
||||||
|
await performSwap(daiToSwap.toString)
|
||||||
|
enqueueSnackbar('Successfully swapped, restarting...', { variant: 'success' })
|
||||||
|
await sleepMs(5_000)
|
||||||
|
await upgradeToLightNode(jsonRpcProvider)
|
||||||
|
await restartBeeNode()
|
||||||
|
navigate(ROUTES.RESTART_LIGHT)
|
||||||
|
enqueueSnackbar('Upgraded to light node', { variant: 'success' })
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Failed to swap: ${error}`, { variant: 'error' })
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HistoryHeader>{header}</HistoryHeader>
|
||||||
|
<Box mb={4}>
|
||||||
|
<TopUpProgressIndicator index={1} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Typography style={{ fontWeight: 'bold' }}>Swap some xDAI to BZZ</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
You need to swap xDAI to BZZ in order to use Swarm. Make sure to keep at least 1 xDAI in order to pay for
|
||||||
|
transaction costs on the network.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<SwarmDivider mb={4} />
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
Your current balance is {balance.dai.toSignificantDigits(4)} xDAI and {balance.bzz.toSignificantDigits(4)}{' '}
|
||||||
|
BZZ.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<SwarmTextInput
|
||||||
|
label="Amount to swap"
|
||||||
|
defaultValue={`${daiToSwap.toSignificantDigits(4)} XDAI`}
|
||||||
|
name="x"
|
||||||
|
onChange={() => false}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<ArrowDown size={24} color="#aaaaaa" />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItemKey label="Funding wallet address" value={balance.address} expanded />
|
||||||
|
</Box>
|
||||||
|
<Box mb={0.25}>
|
||||||
|
<ExpandableListItem
|
||||||
|
label="Resulting XDAI balance after swap"
|
||||||
|
value={`${daiAfterSwap.toSignificantDigits(4)} XDAI`}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<ExpandableListItem
|
||||||
|
label="Resulting BZZ balance after swap"
|
||||||
|
value={`${bzzAfterSwap.toSignificantDigits(4)} BZZ`}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<SwarmButton
|
||||||
|
iconType={Check}
|
||||||
|
onClick={onSwap}
|
||||||
|
disabled={hasSwapped || loading || balance.dai.toDecimal.lte(1)}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
Swap Now
|
||||||
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user