diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 0000000000..674766d7ca --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,5 @@ +@raviteja83 +@KaustubhKumar05 +@amar-1995 +@hdz-666 +@ygit \ No newline at end of file diff --git a/.github/CODE_OF_CONDUCT.md b/.github/CODE_OF_CONDUCT.md new file mode 100644 index 0000000000..6ba377a75a --- /dev/null +++ b/.github/CODE_OF_CONDUCT.md @@ -0,0 +1,76 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +In the interest of fostering an open and welcoming environment, we as +contributors and maintainers pledge to make participation in our project and +our community a harassment-free experience for everyone, regardless of age, body +size, disability, ethnicity, sex characteristics, gender identity and expression, +level of experience, education, socio-economic status, nationality, personal +appearance, race, religion, or sexual identity and orientation. + +## Our Standards + +Examples of behaviour that contributes to creating a positive environment +include: + +* Using welcoming and inclusive language +* Being respectful of differing viewpoints and experiences +* Gracefully accepting constructive criticism +* Focusing on what is best for the community +* Showing empathy towards other community members + +Examples of unacceptable behaviour by participants include: + +* The use of sexualized language or imagery and unwelcome sexual attention or + advances +* Trolling, insulting/derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or electronic + address, without explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Our Responsibilities + +Project maintainers are responsible for clarifying the standards of acceptable +behaviour and are expected to take appropriate and fair corrective action in +response to any instances of unacceptable behaviour. + +Project maintainers have the right and responsibility to remove, edit, or +reject comments, commits, code, wiki edits, issues, and other contributions +that are not aligned to this Code of Conduct, or to ban temporarily or +permanently any contributor for other behaviours that they deem inappropriate, +threatening, offensive, or harmful. + +## Scope + +This Code of Conduct applies both within project spaces and in public spaces +when an individual is representing the project or its community. Examples of +representing a project or community include using an official project e-mail +address, posting via an official social media account or acting as an appointed +representative at an online or offline event. Representation of a project may be +further defined and clarified by project maintainers. + +## Enforcement + +Instances of abusive, harassing or otherwise unacceptable behaviour may be +reported by contacting the 100ms team. All +complaints will be reviewed and investigated and will result in a response that +is deemed necessary and appropriate to the circumstances. The project team is +obligated to maintain confidentiality concerning the reporter of an incident. +Further details of specific enforcement policies may be posted separately. + +Project maintainers who do not follow or enforce the Code of Conduct in good +faith may face temporary or permanent repercussions as determined by other +members of the project's leadership. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, +available at + +[homepage]: https://www.contributor-covenant.org + +For answers to common questions about this code of conduct, see + diff --git a/.github/ISSUE_TEMPLATE/bug_report.yml b/.github/ISSUE_TEMPLATE/bug_report.yml index 049d4b5ba6..a58e397fe4 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.yml +++ b/.github/ISSUE_TEMPLATE/bug_report.yml @@ -30,5 +30,4 @@ body: - Firefox - Chrome - Safari - - Microsoft Edge - \ No newline at end of file + - Microsoft Edge \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml new file mode 100644 index 0000000000..852ba8201e --- /dev/null +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -0,0 +1,11 @@ +blank_issues_enabled: false +contact_links: + - name: 📃 100ms Documentation + url: https://www.100ms.live/docs + about: Explore the 100ms capabilities with our Popular Guides, Demos & Blogs on 100ms Documentation page + - name: 💫 Register on 100ms Dashboard + url: https://dashboard.100ms.live/register + about: Try the gold-standard for adding live audio-video to your apps for free. + - name: 🗣️ Talk to us + url: https://www.100ms.live/contact + about: Get in touch with the 100ms team today! We are committed to helping our customers maximize the potential of our cutting-edge live video platform. Whether you need advice on how to leverage live streaming for your business, pricing information, or want to learn more about the 100ms platform, our team is here to answer all your questions and provide tailored solutions to meet your unique needs. diff --git a/.github/ISSUE_TEMPLATE/feature_request.yml b/.github/ISSUE_TEMPLATE/feature_request.yml index f8e83701f3..33f595c361 100644 --- a/.github/ISSUE_TEMPLATE/feature_request.yml +++ b/.github/ISSUE_TEMPLATE/feature_request.yml @@ -13,5 +13,4 @@ body: label: What's the feature? description: Describe the feature, who it would help, and link to any examples from other apps. validations: - required: true - \ No newline at end of file + required: true \ No newline at end of file diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md index 737e45d20a..684a291339 100644 --- a/.github/pull_request_template.md +++ b/.github/pull_request_template.md @@ -1,6 +1,29 @@ -### Details(context, link the issue, how was the bug fixed, what does the new feature do) +# Description + +_Replace this paragraph with a description of what this PR is changing or adding, and why. Consider including before/after screenshots._ + +_List which issues are fixed by this PR. You must list at least one issue._ + +--- + +## Implementation note, gotchas, related work and Future TODOs (optional) + + - -- -### Implementation note, gotchas, related work and Future TODOs (optional) +### Pre-launch Checklist + +- [ ] The [Documentation] is updated accordingly, or this PR doesn't require it. +- [ ] I updated/added relevant documentation. +- [ ] I listed at least one issue that this PR fixes in the description above. +- [ ] I added new tests to check the change I am making, or this PR is test-exempt. +- [ ] All existing and new tests are passing. + +### Merging: +- Squash merge to dev +- Merge commit to publish-alpha and main + + + +[Documentation]: https://www.100ms.live/docs \ No newline at end of file diff --git a/.github/workflows/alpha-release.yml b/.github/workflows/alpha-release.yml index 200a536c34..a1388d3b95 100644 --- a/.github/workflows/alpha-release.yml +++ b/.github/workflows/alpha-release.yml @@ -9,7 +9,7 @@ jobs: bump_versions: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: fetch-depth: 0 @@ -41,10 +41,8 @@ jobs: needs: bump_versions steps: - name: Trigger Publish Packages workflow - uses: aurelien-baudet/workflow-dispatch@v2.1.1 + uses: aurelien-baudet/workflow-dispatch@v4.0.0 with: workflow: publish.yml token: ${{ secrets.GITHUB_TOKEN }} - inputs: '{ "publishFlag": "alpha" }' - - \ No newline at end of file + inputs: '{ "publishFlag": "alpha" }' \ No newline at end of file diff --git a/.github/workflows/create-release-pr.yml b/.github/workflows/create-release-pr.yml index 1aac560706..25b42a8a6f 100644 --- a/.github/workflows/create-release-pr.yml +++ b/.github/workflows/create-release-pr.yml @@ -1,4 +1,4 @@ -name: Create release PR +name: Create Release PR on: workflow_dispatch: inputs: @@ -19,7 +19,7 @@ jobs: if: github.event.inputs.versionBump != 'prerelease' && github.ref != 'refs/heads/dev' run: exit 1 - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: fetch-depth: 0 @@ -48,7 +48,7 @@ jobs: echo $STORE_VERSION echo "::set-output name=store_version::$(echo $STORE_VERSION)" - - uses: peter-evans/create-pull-request@v4 + - uses: peter-evans/create-pull-request@v7 with: commit-message: 'ci: update versions for release' title: 'ci: update versions for release' diff --git a/.github/workflows/cypress.yml b/.github/workflows/cypress.yml index 34ed21abac..0ace350abd 100644 --- a/.github/workflows/cypress.yml +++ b/.github/workflows/cypress.yml @@ -8,8 +8,8 @@ jobs: runs-on: ubuntu-latest if: github.event.pull_request.draft != true steps: - - uses: actions/checkout@v2 - - uses: dorny/paths-filter@v2 + - uses: actions/checkout@v4 + - uses: dorny/paths-filter@v3 id: filter with: filters: | diff --git a/.github/workflows/firstinteraction.yml b/.github/workflows/firstinteraction.yml new file mode 100644 index 0000000000..345e65ee85 --- /dev/null +++ b/.github/workflows/firstinteraction.yml @@ -0,0 +1,34 @@ +name: first-interaction + +on: + workflow_dispatch: {} + issues: + types: [opened] + pull_request: + branches: + - main + - develop + types: [opened] + +jobs: + check_for_first_interaction: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/first-interaction@main + with: + repo-token: ${{ secrets.GITHUB_TOKEN }} + issue-message: | + Hello! Thank you for filing an issue. + + Please include relevant logs or detailed description for faster resolutions. + + We really appreciate your contribution! + pr-message: | + Hello! Thank you for your contribution. + + If you are fixing a bug, please reference the issue number in the description. + + If you are implementing a feature request, please check with the maintainers that the feature will be accepted first. + + We really appreciate your contribution! diff --git a/.github/workflows/generate-docs.yml b/.github/workflows/generate-docs.yml index cf434339c9..113382818a 100644 --- a/.github/workflows/generate-docs.yml +++ b/.github/workflows/generate-docs.yml @@ -3,7 +3,7 @@ on: workflow_dispatch: workflow_call: secrets: - DOCKER_GIT_TOKEN: + DOCKER_GIT_TOKEN: required: true jobs: @@ -11,10 +11,10 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout repo - uses: actions/checkout@v2 + uses: actions/checkout@v4 - name: Use Node ${{ matrix.node }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: ${{ matrix.node }} cache: 'yarn' @@ -34,7 +34,7 @@ jobs: run: yarn docs - name: checkout 100ms-docs - uses: actions/checkout@v2 + uses: actions/checkout@v4 with: token: ${{ secrets.DOCKER_GIT_TOKEN }} repository: 100mslive/100ms-docs @@ -52,9 +52,9 @@ jobs: rm -r v2 mv docs v2 mv react/docs v2/react-hooks - + - name: Create PR - uses: peter-evans/create-pull-request@v3 + uses: peter-evans/create-pull-request@v7 with: path: 100ms-docs token: ${{ secrets.DOCKER_GIT_TOKEN }} diff --git a/.github/workflows/lint-test-build.yml b/.github/workflows/lint-test-build.yml index c81b8cbf5b..92c54ad683 100644 --- a/.github/workflows/lint-test-build.yml +++ b/.github/workflows/lint-test-build.yml @@ -1,18 +1,18 @@ name: Lint, Test and Build -on: +on: push: merge_group: types: [checks_requested] - + jobs: build: runs-on: ubuntu-latest steps: - name: Checkout repo - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Setup Node - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: 18 cache: 'yarn' diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 64ad12c0dd..3935706aeb 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -17,8 +17,8 @@ jobs: publish_packages: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v2 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: 18 registry-url: https://registry.npmjs.org/ @@ -30,7 +30,7 @@ jobs: - name: Notify slack starting if: github.event.inputs.publishFlag == 'latest' && success() id: slack # IMPORTANT: reference this step ID value in future Slack steps - env: + env: SLACK_BOT_TOKEN: ${{ secrets.SLACK_DEPLOY_BOT_TOKEN }} uses: voxmedia/github-action-slack-notify-build@v1 with: diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml index 246d2fb95b..8e9e78c8c8 100644 --- a/.github/workflows/stale.yml +++ b/.github/workflows/stale.yml @@ -8,7 +8,7 @@ jobs: stale: runs-on: ubuntu-latest steps: - - uses: actions/stale@v8 + - uses: actions/stale@v9 with: stale-issue-message: "This issue is stale because it has been open 30 days with no activity. Remove stale label or comment or this will be closed in 5 days." stale-pr-message: "This PR is stale because it has been open 45 days with no activity. Remove stale label or comment or this will be closed in 10 days." diff --git a/.github/workflows/sync-alpha-to-main.yml b/.github/workflows/sync-alpha-to-main.yml index 34dee4d4ab..0557263d8f 100644 --- a/.github/workflows/sync-alpha-to-main.yml +++ b/.github/workflows/sync-alpha-to-main.yml @@ -9,7 +9,7 @@ jobs: pull-request: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v1 + - uses: actions/checkout@v4 - name: pull-request uses: repo-sync/pull-request@v2 if: github.event.pull_request.merged == true diff --git a/README.md b/README.md index c7c247dcd9..07fd77dc2a 100644 --- a/README.md +++ b/README.md @@ -1,16 +1,16 @@ - # Web SDKs -This monorepo contains all the packages required to integrate 100ms on the web. +This monorepo contains all the packages required to integrate 100ms on the web. ## What is included? + The packages folder contains all the SDK's of 100ms. Here is a brief overview of them: | Directory | Package | Description | Link | |--|--|--|--| -| `hms-video-store` | `@100mslive/hms-video-store` | This package contains the core SDK and the reactive store parts. | [README](./packages/hms-video-store) | -| `react-icons` | `@100mslive/react-icons` | This contains all the icons used in the 100ms prebuilt. | [README](./packages/react-icons) | +| `hms-video-store` | `@100mslive/hms-video-store` | This package contains the core SDK and the reactive store parts. | [README](./packages/hms-video-store) | +| `react-icons` | `@100mslive/react-icons` | This contains all the icons used in the 100ms prebuilt. | [README](./packages/react-icons) | | `react-sdk` | `@100mslive/react-sdk` | This contains the base React Hooks and some commonly used functionalities as React Hooks. | [README](./packages/react-sdk) | -| `roomkit-react` | `@100mslive/roomkit-react`| This contains the React components used in the Prebuilt and the Prebuilt component itself. | [README](./packages/roomkit-react) | +| `roomkit-react` | `@100mslive/roomkit-react`| This contains the React components used in the Prebuilt and the Prebuilt component itself. | [README](./packages/roomkit-react) | | `roomkit-web` | `@100mslive/roomkit-web` | This is a web component port of the `HMSPrebuilt` component from the `roomkit-react`. If you are not using React,this can be used as a web component. | [README](./packages/roomkit-web)| For full documentation, visit [100ms.live/docs](https://www.100ms.live/docs) @@ -18,21 +18,22 @@ For full documentation, visit [100ms.live/docs](https://www.100ms.live/docs)
## How to integrate? -The 100ms SDK gives you everything you need to build scalable, high-quality live video and audio experiences. + +The 100ms SDK gives you everything you need to build scalable, high-quality live video and audio experiences. **There are two ways you can add 100ms to your apps:** 1. ## Custom UI - - 100ms SDKs are powerful and highly extensible to build and support all custom experiences and UI. - - **Related packages include:** `@100mslive/react-sdk`, `@100mslive/hms-video-store` and `@100mslive/react-icons`. - - Get started with integrating the SDK using the [How to Guide](https://www.100ms.live/docs/javascript/v2/how-to-guides/install-the-sdk/integration).
+ - 100ms SDKs are powerful and highly extensible to build and support all custom experiences and UI. + - **Related packages include:** `@100mslive/react-sdk`, `@100mslive/hms-video-store` and `@100mslive/react-icons`. + - Get started with integrating the SDK using the [How to Guide](https://www.100ms.live/docs/javascript/v2/how-to-guides/install-the-sdk/integration).
> Navigate to `react-sdk` for the base React Hooks and some commonly used functionalities by clicking [here](./packages/react-sdk). -2. ## 100ms Prebuilt - - 100ms Prebuilt is a high-level abstraction with no-code customization that enables you to embed video conferencing and/or live streaming UI—with a few lines of code. - - **Related packages include:** `roomkit-react` and `roomkit-web`. - - Get started with 100ms Prebuilt using the [Prebuilt Quickstart for Web](https://www.100ms.live/docs/javascript/v2/quickstart/prebuilt-quickstart).
+2. ## 100ms Prebuilt + - 100ms Prebuilt is a high-level abstraction with no-code customization that enables you to embed video conferencing and/or live streaming UI—with a few lines of code. + - **Related packages include:** `roomkit-react` and `roomkit-web`. + - Get started with 100ms Prebuilt using the [Prebuilt Quickstart for Web](https://www.100ms.live/docs/javascript/v2/quickstart/prebuilt-quickstart).
> Navigate to `roomkit-react` for the React components used in Prebuilt and the Prebuilt component itself by clicking [here](./packages/roomkit-react). @@ -40,16 +41,15 @@ The 100ms SDK gives you everything you need to build scalable, high-quality live ![Banner](prebuilt-banner.png) +### 100ms Prebuilt Cross Platform Support -### 100ms Prebuilt Cross Platform Support -| Client | Repository | Docs | Example | -|--|--|--|--| -| Web | [web-sdks](https://github.com/100mslive/web-sdks/tree/main/packages/roomkit-react) | [Link](https://www.100ms.live/docs/javascript/v2/quickstart/prebuilt-quickstart) | [prebuilt-react-integration](https://github.com/100mslive/web-sdks/tree/main/examples/prebuilt-react-integration) -| Android | [100ms-android](https://github.com/100mslive/100ms-android/tree/release-v2/room-kit) | [Link](https://www.100ms.live/docs/android/v2/quickstart/prebuilt-android) | [AndroidPrebuiltDemo](https://github.com/100mslive/AndroidPrebuiltDemo) -| iOS | [100ms-roomkit-ios](https://github.com/100mslive/100ms-roomkit-ios) | [Link](https://www.100ms.live/docs/ios/v2/quickstart/prebuilt) | [100ms-roomkit-example](https://github.com/100mslive/100ms-roomkit-example) -| Flutter | [100ms-flutter](https://github.com/100mslive/100ms-flutter/tree/main/packages/hms_room_kit)| [Link](https://www.100ms.live/docs/flutter/v2/quickstart/prebuilt) | [hms_room_kit/example](https://github.com/100mslive/100ms-flutter/tree/main/packages/hms_room_kit/example) -| React Native | [100ms-react-native](https://github.com/100mslive/100ms-react-native/tree/main/packages/react-native-room-kit)| [Link](https://www.100ms.live/docs/react-native/v2/quickstart/prebuilt) | [react-native-room-kit/example](https://github.com/100mslive/100ms-react-native/tree/main/packages/react-native-room-kit/example) - +| Client | Repository | Docs | Example | +| ------------ | -------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------- | +| Web | [web-sdks](https://github.com/100mslive/web-sdks/tree/main/packages/roomkit-react) | [Link](https://www.100ms.live/docs/javascript/v2/quickstart/prebuilt-quickstart) | [prebuilt-react-integration](https://github.com/100mslive/web-sdks/tree/main/examples/prebuilt-react-integration) | +| Android | [100ms-android](https://github.com/100mslive/100ms-android/tree/release-v2/room-kit) | [Link](https://www.100ms.live/docs/android/v2/quickstart/prebuilt-android) | [AndroidPrebuiltDemo](https://github.com/100mslive/AndroidPrebuiltDemo) | +| iOS | [100ms-roomkit-ios](https://github.com/100mslive/100ms-roomkit-ios) | [Link](https://www.100ms.live/docs/ios/v2/quickstart/prebuilt) | [100ms-roomkit-example](https://github.com/100mslive/100ms-roomkit-example) | +| Flutter | [100ms-flutter](https://github.com/100mslive/100ms-flutter/tree/main/packages/hms_room_kit) | [Link](https://www.100ms.live/docs/flutter/v2/quickstart/prebuilt) | [hms_room_kit/example](https://github.com/100mslive/100ms-flutter/tree/main/packages/hms_room_kit/example) | +| React Native | [100ms-react-native](https://github.com/100mslive/100ms-react-native/tree/main/packages/react-native-room-kit) | [Link](https://www.100ms.live/docs/react-native/v2/quickstart/prebuilt) | [react-native-room-kit/example](https://github.com/100mslive/100ms-react-native/tree/main/packages/react-native-room-kit/example) |

@@ -62,7 +62,6 @@ The 100ms SDK gives you everything you need to build scalable, high-quality live if you are using a different version in other projects, use [nvm](https://github.com/nvm-sh/nvm?tab=readme-ov-file#installing-and-updating) to manage node versions. - ``` git clone https://github.com/100mslive/web-sdks.git cd web-sdks @@ -81,16 +80,14 @@ yarn dev Once you run `yarn dev`, the localhost link with the port will be generated automatically. Just get the roomCode from [100ms dashboard](https://dashboard.100ms.live) and append at the end - ### Testing Changes Locally + Run `yarn start` by navigating to the package you are making changes to, the changes should reflect in the above sample app. For example, if you are making changes in roomkit-react(prebuilt), run `yarn start` in that package. The sample app should auto reload. > Note: Make sure `yarn build` is run atleast once before using `yarn start` - -
### Deploying Your Changes @@ -100,8 +97,8 @@ Once you have forked the repository and tested your changes on the local build, - Import the fork repository - Set `examples/prebuilt-react-integration` as the root directory - Use the Create React App preset and update the build and install commands `Project Settings` to use the root level scripts - - install: `cd ../../ && yarn install` - - build: `cd ../../ && yarn build` + - install: `cd ../../ && yarn install` + - build: `cd ../../ && yarn build` For reference: @@ -109,7 +106,7 @@ For reference:
-Once the app has been deployed, you can append the room code at the end of the deployment URL to preview your changes +Once the app has been deployed, you can append the room code at the end of the deployment URL to preview your changes
@@ -142,14 +139,12 @@ import { HMSPrebuilt } from '@100mslive/roomkit-react'; ``` ## Contributing -We welcome external contributors or anyone excited to help improve 100ms SDKs. If you'd like to get involved, check out our [contribution guide](./DEVELOPER.MD), and get started exploring the codebase. -Please join us [on Discord](https://discord.com/invite/kGdmszyzq2) to discuss any new ideas and/or PRs. +We welcome external contributors or anyone excited to help improve 100ms SDKs. If you'd like to get involved, check out our [contribution guide](./DEVELOPER.MD), and get started exploring the codebase.
## Community & Support -- [GitHub Issues](https://github.com/100mslive/web-sdks/issues) - Submit any bugs or errors you encounter using the Web SDKs. -- [Discord](https://discord.com/invite/kGdmszyzq2) - Hang out with the community members, share your projects or ask questions to get help from the 100ms team. -- [Contact](https://www.100ms.live/contact) - Reach out to 100ms team to get pricing information, understand how we can help you go live, or to learn more about the platform. +- [GitHub Issues](https://github.com/100mslive/web-sdks/issues) - Submit any bugs or errors you encounter using the Web SDKs. +- [Contact](https://www.100ms.live/contact) - Reach out to 100ms team to get pricing information, understand how we can help you go live, or to learn more about the platform. diff --git a/examples/prebuilt-react-integration/README.md b/examples/prebuilt-react-integration/README.md index c0ba926c6d..244c6693f5 100644 --- a/examples/prebuilt-react-integration/README.md +++ b/examples/prebuilt-react-integration/README.md @@ -6,6 +6,3 @@ - Run `yarn && yarn build` at the root level of the repo (./web-sdks) - Then, navigate to this folder and run `yarn dev` - -### Facing a problem? -Join us on the [100ms Discord](https://discord.com/invite/kGdmszyzq2) to ask questions and get help from the 100ms team. diff --git a/examples/prebuilt-react-integration/package.json b/examples/prebuilt-react-integration/package.json index afe09a3856..8c838ca36b 100644 --- a/examples/prebuilt-react-integration/package.json +++ b/examples/prebuilt-react-integration/package.json @@ -10,7 +10,7 @@ "preview": "vite preview" }, "dependencies": { - "@100mslive/roomkit-react": "0.3.23", + "@100mslive/roomkit-react": "0.3.24", "react": "^18.2.0", "react-dom": "^18.2.0" }, diff --git a/packages/hls-player/package.json b/packages/hls-player/package.json index 13751620ea..af5c74f56e 100644 --- a/packages/hls-player/package.json +++ b/packages/hls-player/package.json @@ -1,6 +1,6 @@ { "name": "@100mslive/hls-player", - "version": "0.3.23", + "version": "0.3.24", "description": "HLS client library which uses HTML5 Video element and Media Source Extension for playback", "main": "dist/index.cjs.js", "module": "dist/index.js", @@ -36,7 +36,7 @@ "author": "100ms", "license": "MIT", "dependencies": { - "@100mslive/hls-stats": "0.4.23", + "@100mslive/hls-stats": "0.4.24", "eventemitter2": "^6.4.9", "hls.js": "1.4.12" } diff --git a/packages/hls-stats/package.json b/packages/hls-stats/package.json index 467ff2f149..fc3522ae93 100644 --- a/packages/hls-stats/package.json +++ b/packages/hls-stats/package.json @@ -1,6 +1,6 @@ { "name": "@100mslive/hls-stats", - "version": "0.4.23", + "version": "0.4.24", "description": "A simple library that provides stats for your hls stream", "main": "dist/index.cjs.js", "module": "dist/index.js", diff --git a/packages/hms-video-store/package.json b/packages/hms-video-store/package.json index dc3f4b7925..dcfed593f6 100644 --- a/packages/hms-video-store/package.json +++ b/packages/hms-video-store/package.json @@ -1,5 +1,5 @@ { - "version": "0.12.23", + "version": "0.12.24", "license": "MIT", "repository": { "type": "git", diff --git a/packages/hms-video-store/src/IHMSActions.ts b/packages/hms-video-store/src/IHMSActions.ts index d1ab46e0ff..01cc905201 100644 --- a/packages/hms-video-store/src/IHMSActions.ts +++ b/packages/hms-video-store/src/IHMSActions.ts @@ -25,6 +25,7 @@ import { TokenRequestOptions, } from './internal'; import { + DebugInfo, HMSChangeMultiTrackStateParams, HMSGenericTypes, HMSMessageID, @@ -581,4 +582,9 @@ export interface IHMSActions { - HMSLogger.d(this.TAG, 'muted natively'); + HMSLogger.d(this.TAG, 'muted natively', document.visibilityState); + const reason = document.visibilityState === 'hidden' ? 'visibility-change' : 'incoming-call'; this.eventBus.analytics.publish( this.sendInterruptionEvent({ started: true, - reason: 'incoming-call', + reason: reason, }), ); this.eventBus.localVideoEnabled.publish({ enabled: false, track: this }); @@ -538,10 +541,12 @@ export class HMSLocalVideoTrack extends HMSVideoTrack { /** @internal */ handleTrackUnmuteNatively = async () => { HMSLogger.d(this.TAG, 'unmuted natively'); + const reason = document.visibilityState === 'hidden' ? 'visibility-change' : 'incoming-call'; + this.eventBus.analytics.publish( this.sendInterruptionEvent({ started: false, - reason: 'incoming-call', + reason: reason, }), ); this.handleTrackUnmute(); diff --git a/packages/hms-video-store/src/media/tracks/HMSTrackExceptionTrackType.ts b/packages/hms-video-store/src/media/tracks/HMSTrackExceptionTrackType.ts new file mode 100644 index 0000000000..5bca9518b2 --- /dev/null +++ b/packages/hms-video-store/src/media/tracks/HMSTrackExceptionTrackType.ts @@ -0,0 +1,6 @@ +export enum HMSTrackExceptionTrackType { + AUDIO = 'audio', + VIDEO = 'video', + AUDIO_VIDEO = 'audio, video', + SCREEN = 'screen', +} diff --git a/packages/hms-video-store/src/media/tracks/HMSVideoTrack.ts b/packages/hms-video-store/src/media/tracks/HMSVideoTrack.ts index 03e6d7e0b7..8e577d4579 100644 --- a/packages/hms-video-store/src/media/tracks/HMSVideoTrack.ts +++ b/packages/hms-video-store/src/media/tracks/HMSVideoTrack.ts @@ -87,9 +87,9 @@ export class HMSVideoTrack extends HMSTrack { this.sinkCount++; } - handleTrackUnmute() { + handleTrackUnmute = () => { this.getSinks().forEach(videoElement => this.reTriggerPlay({ videoElement })); - } + }; private reTriggerPlay = ({ videoElement }: { videoElement: HTMLVideoElement }) => { setTimeout(() => { diff --git a/packages/hms-video-store/src/plugins/video/HMSMediaStreamPluginsManager.ts b/packages/hms-video-store/src/plugins/video/HMSMediaStreamPluginsManager.ts index 1991355757..5bc32bcfcf 100644 --- a/packages/hms-video-store/src/plugins/video/HMSMediaStreamPluginsManager.ts +++ b/packages/hms-video-store/src/plugins/video/HMSMediaStreamPluginsManager.ts @@ -63,4 +63,8 @@ export class HMSMediaStreamPluginsManager { getPlugins(): string[] { return Array.from(this.plugins).map(plugin => plugin.getName()); } + + async cleanup() { + this.removePlugins(Array.from(this.plugins)); + } } diff --git a/packages/hms-video-store/src/reactive-store/HMSSDKActions.ts b/packages/hms-video-store/src/reactive-store/HMSSDKActions.ts index 70d21c9fa4..a6f77818a4 100644 --- a/packages/hms-video-store/src/reactive-store/HMSSDKActions.ts +++ b/packages/hms-video-store/src/reactive-store/HMSSDKActions.ts @@ -845,6 +845,10 @@ export class HMSSDKActions this.transport.isFlagEnabled(flag)); + const initEndpoint = this.store.getConfig()?.initEndpoint; + return { + websocketURL, + enabledFlags, + initEndpoint, + }; + } + getSessionStore() { return this.sessionStore; } diff --git a/packages/hms-video-store/src/transport/RetryScheduler.ts b/packages/hms-video-store/src/transport/RetryScheduler.ts index 0feaf8c006..7230612a99 100644 --- a/packages/hms-video-store/src/transport/RetryScheduler.ts +++ b/packages/hms-video-store/src/transport/RetryScheduler.ts @@ -114,17 +114,7 @@ export class RetryScheduler { } } - const timeElapsedSinceError = Date.now() - failedAt; - if (timeElapsedSinceError >= maxRetryTime || hasFailedDependency) { - error.description += `. [${TFC[category]}] Could not recover after ${timeElapsedSinceError} milliseconds`; - - if (hasFailedDependency) { - error.description += ` Could not recover all of it's required dependencies - [${(dependencies as Array) - .map(dep => TFC[dep]) - .toString()}]`; - } - error.isTerminal = true; - + const handleTerminalError = (error: HMSException) => { // @NOTE: Don't reject to throw error for dependencies, use onStateChange // const taskPromise = this.inProgress.get(category); this.inProgress.delete(category); @@ -140,6 +130,19 @@ export class RetryScheduler { } return; + }; + + const timeElapsedSinceError = Date.now() - failedAt; + if (timeElapsedSinceError >= maxRetryTime || hasFailedDependency) { + error.description += `. [${TFC[category]}] Could not recover after ${timeElapsedSinceError} milliseconds`; + + if (hasFailedDependency) { + error.description += ` Could not recover all of it's required dependencies - [${(dependencies as Array) + .map(dep => TFC[dep]) + .toString()}]`; + } + error.isTerminal = true; + return handleTerminalError(error); } if (changeState) { @@ -158,11 +161,18 @@ export class RetryScheduler { taskSucceeded = await this.setTimeoutPromise(task, delay); } catch (ex) { taskSucceeded = false; - HMSLogger.w( - this.TAG, - `[${TFC[category]}] Un-caught exception ${(ex as HMSException).name} in retry-task, initiating retry`, - ex, - ); + const error = ex as HMSException; + + if (error.isTerminal) { + HMSLogger.e(this.TAG, `[${TFC[category]}] Un-caught terminal exception ${error.name} in retry-task`, ex); + return handleTerminalError(error); + } else { + HMSLogger.w( + this.TAG, + `[${TFC[category]}] Un-caught exception ${error.name} in retry-task, initiating retry`, + ex, + ); + } } if (taskSucceeded) { diff --git a/packages/hms-video-store/src/utils/media.ts b/packages/hms-video-store/src/utils/media.ts index 330b544b14..05d60e443f 100644 --- a/packages/hms-video-store/src/utils/media.ts +++ b/packages/hms-video-store/src/utils/media.ts @@ -1,12 +1,13 @@ import HMSLogger from './logger'; -import { BuildGetMediaError, HMSGetMediaActions } from '../error/utils'; +import { BuildGetMediaError } from '../error/utils'; +import { HMSTrackExceptionTrackType } from '../media/tracks/HMSTrackExceptionTrackType'; export async function getLocalStream(constraints: MediaStreamConstraints): Promise { try { const stream = await navigator.mediaDevices.getUserMedia(constraints); return stream; } catch (err) { - throw BuildGetMediaError(err as Error, HMSGetMediaActions.AV); + throw BuildGetMediaError(err as Error, HMSTrackExceptionTrackType.AUDIO_VIDEO); } } @@ -16,7 +17,7 @@ export async function getLocalScreen(constraints: MediaStreamConstraints['video' const stream = await navigator.mediaDevices.getDisplayMedia({ video: constraints, audio: false }); return stream; } catch (err) { - throw BuildGetMediaError(err as Error, HMSGetMediaActions.SCREEN); + throw BuildGetMediaError(err as Error, HMSTrackExceptionTrackType.SCREEN); } } @@ -37,7 +38,7 @@ export async function getLocalDevices(): Promise { devices.forEach(device => deviceGroups[device.kind].push(device)); return deviceGroups; } catch (err) { - throw BuildGetMediaError(err as Error, HMSGetMediaActions.AV); + throw BuildGetMediaError(err as Error, HMSTrackExceptionTrackType.AUDIO_VIDEO); } } @@ -73,7 +74,7 @@ export enum HMSAudioDeviceCategory { export const getAudioDeviceCategory = (deviceLabel?: string) => { if (!deviceLabel) { - HMSLogger.e('[DeviceManager]:', 'No device label provided'); + HMSLogger.w('[DeviceManager]:', 'No device label provided'); return HMSAudioDeviceCategory.SPEAKERPHONE; } const label = deviceLabel.toLowerCase(); diff --git a/packages/hms-video-store/src/utils/track.ts b/packages/hms-video-store/src/utils/track.ts index fea545dd08..6af65ce30f 100644 --- a/packages/hms-video-store/src/utils/track.ts +++ b/packages/hms-video-store/src/utils/track.ts @@ -1,5 +1,6 @@ -import { BuildGetMediaError, HMSGetMediaActions } from '../error/utils'; +import { BuildGetMediaError } from '../error/utils'; import { HMSAudioTrackSettings, HMSVideoTrackSettings } from '../media/settings'; +import { HMSTrackExceptionTrackType } from '../media/tracks/HMSTrackExceptionTrackType'; export async function getAudioTrack(settings: HMSAudioTrackSettings): Promise { try { @@ -8,7 +9,7 @@ export async function getAudioTrack(settings: HMSAudioTrackSettings): PromiseSomething's gone wrong.

Sorry, we encountered an error. Please refresh the page to continue. If you keep seeing this error, you - can ask for help on Discord. + can ask for help on Dashboard.

{shouldShowError && (
diff --git a/packages/react-icons/package.json b/packages/react-icons/package.json index c9868e9ff4..757f6c0f1a 100644 --- a/packages/react-icons/package.json +++ b/packages/react-icons/package.json @@ -4,7 +4,7 @@ "main": "dist/index.cjs.js", "module": "dist/index.js", "typings": "dist/index.d.ts", - "version": "0.10.23", + "version": "0.10.24", "author": "100ms", "license": "MIT", "repository": { diff --git a/packages/react-sdk/package.json b/packages/react-sdk/package.json index 996614b58e..eaa962bbec 100644 --- a/packages/react-sdk/package.json +++ b/packages/react-sdk/package.json @@ -4,7 +4,7 @@ "main": "dist/index.cjs.js", "module": "dist/index.js", "typings": "dist/index.d.ts", - "version": "0.10.23", + "version": "0.10.24", "author": "100ms", "license": "MIT", "repository": { @@ -48,7 +48,7 @@ "react": ">=16.8 <19.0.0" }, "dependencies": { - "@100mslive/hms-video-store": "0.12.23", + "@100mslive/hms-video-store": "0.12.24", "react-resize-detector": "^7.0.0", "zustand": "^3.6.2" } diff --git a/packages/react-sdk/src/hooks/useAwayNotifications.ts b/packages/react-sdk/src/hooks/useAwayNotifications.ts index cd72f9c3c5..2cd5365727 100644 --- a/packages/react-sdk/src/hooks/useAwayNotifications.ts +++ b/packages/react-sdk/src/hooks/useAwayNotifications.ts @@ -8,10 +8,12 @@ export const useAwayNotifications = () => { const vanillaStore = useHMSVanillaStore(); const requestPermission = useCallback(async () => { // Headless check for beam - if (navigator.webdriver) { + if (!('Notification' in window) || navigator.webdriver) { + console.debug('Request Permsissions : Notifications not supported or headless browser'); + // Notifications not supported return; } - if (!Notification || Notification?.permission === 'granted' || Notification?.permission === 'denied') { + if (Notification?.permission === 'granted' || Notification?.permission === 'denied') { return; } const unsubscribe = vanillaStore.subscribe(async role => { @@ -23,8 +25,12 @@ export const useAwayNotifications = () => { }, [vanillaStore]); const showNotification = useCallback((title: string, options?: NotificationOptions) => { + // Notifications not supported + if (!('Notification' in window)) { + console.debug('Show Notifications: Notifications not supported or headless browser'); + return; + } if ( - !Notification || Notification?.permission === 'denied' || /** * document.visibilityState is still 'visible' when the tab is active but window is not open diff --git a/packages/roomkit-react/README.md b/packages/roomkit-react/README.md index 92bff98b9c..b4a8f1f1ae 100644 --- a/packages/roomkit-react/README.md +++ b/packages/roomkit-react/README.md @@ -1,7 +1,6 @@ - ![Banner](https://github.com/100mslive/web-sdks/blob/06c65259912db6ccd8617f2ecb6fef51429251ec/prebuilt-banner.png) -# Room Kit React Library +# Room Kit React Library 100ms Room Kit provides simple & easy to use UI components to build Live Streaming & Video Conferencing experiences in your apps. @@ -44,6 +43,7 @@ export default App() { For additional props, refer the [docs](https://www.100ms.live/docs/javascript/v2/quickstart/prebuilt-quickstart#props-for-hmsprebuilt) ## Customisation + While we offer [a no-code way to customize Prebuilt](https://www.100ms.live/docs/get-started/v2/get-started/prebuilt/overview#customize-prebuilt), you can fork your copy of the Prebuilt component and make changes to the code to allow for more fine-tuning. Prebuilt customisations are available on [100ms Dashboard](https://dashboard.100ms.live). @@ -56,15 +56,14 @@ The `Prebuilt` folder contains the full Prebuilt implementation. ### Major Components in Prebuilt -| Component | Description | -|--|--| -| [RoomLayoutProvider](src/Prebuilt/provider/roomLayoutProvider/index.tsx) | This is a context that contains the configuration from the dashboard [customiser](https://dashboard.100ms.live/). Whatever changes are made in the dashboard customiser are available the next time you join.| -|[AppStateContext](src/Prebuilt/AppStateContext.tsx) | Contains the logic to switch between different screens, for example, Preview to Meeting, Meeting to Leave. These transitions are based on the roomState that is available from the reactive store (`useHMSStore(selectHMSRoomState)`). | -| [PreviewScreen](src/Prebuilt/components/Preview/PreviewScreen.tsx) | Contains the Preview implementation. Contains the Video tile, video, audio toggles and Virtual background and settings along with the name input.| -| [ConferenceScreen](src/Prebuilt/components/ConferenceScreen.tsx) | This contains the screen once you finish Preview and enter the meeting. This contains the header and footer and the main content.| -| [VideoStreamingSection](src/Prebuilt/layouts/VideoStreamingSection.tsx) |This is the component that contains the main video rendering components and a sidebar (Interactive features like Chat and Polls are displayed here) | -| [LeaveScreen](src/Prebuilt/components/LeaveScreen.tsx) |This is the screen that is shown when you leave the meeting | - +| Component | Description | +| ------------------------------------------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| [RoomLayoutProvider](src/Prebuilt/provider/roomLayoutProvider/index.tsx) | This is a context that contains the configuration from the dashboard [customiser](https://dashboard.100ms.live/). Whatever changes are made in the dashboard customiser are available the next time you join. | +| [AppStateContext](src/Prebuilt/AppStateContext.tsx) | Contains the logic to switch between different screens, for example, Preview to Meeting, Meeting to Leave. These transitions are based on the roomState that is available from the reactive store (`useHMSStore(selectHMSRoomState)`). | +| [PreviewScreen](src/Prebuilt/components/Preview/PreviewScreen.tsx) | Contains the Preview implementation. Contains the Video tile, video, audio toggles and Virtual background and settings along with the name input. | +| [ConferenceScreen](src/Prebuilt/components/ConferenceScreen.tsx) | This contains the screen once you finish Preview and enter the meeting. This contains the header and footer and the main content. | +| [VideoStreamingSection](src/Prebuilt/layouts/VideoStreamingSection.tsx) | This is the component that contains the main video rendering components and a sidebar (Interactive features like Chat and Polls are displayed here) | +| [LeaveScreen](src/Prebuilt/components/LeaveScreen.tsx) | This is the screen that is shown when you leave the meeting | ### Customising the Styles @@ -74,14 +73,13 @@ When [`HMSThemeProvider`](./src/Theme/ThemeProvider.tsx) is used at the top leve For components created using the base components like `Box`, `Flex`, `Button` etc, css Prop is available to modify the styles. Within the css prop, you can access the variables from the [base config](./src/Theme/base.config.ts). - ## Contributing - Make sure whatever new Component/file you create is in `Typescript`. - Don't forget to add data-testid for actionables like buttons, menus etc. -- Setup proper tooling ([ESLint](https://eslint.org/) and [Prettier](https://prettier.io/)) in your editor. +- Setup proper tooling ([ESLint](https://eslint.org/) and [Prettier](https://prettier.io/)) in your editor. - `yarn lint` will be run before you push the changes, so whenever a push fails, check if there are any lint errors. @@ -89,6 +87,6 @@ Read this [doc](../../DEVELOPER.MD) for the coding guidelines. ## Community & Support -- [GitHub Issues](https://github.com/100mslive/web-sdks/issues) - Submit any bugs or errors you encounter using the Web SDKs. -- [Discord](https://discord.com/invite/kGdmszyzq2) - Hang out with the community members, share your projects or ask questions to get help from the 100ms team. -- [Contact](https://www.100ms.live/contact) - Reach out to 100ms team to get pricing information, understand how we can help you go live, or to learn more about the platform. +- [GitHub Issues](https://github.com/100mslive/web-sdks/issues) - Submit any bugs or errors you encounter using the Web SDKs. +- [Dashboard](https://dashboard.100ms.live/dashboard) - ask questions to get help from the 100ms team. +- [Contact](https://www.100ms.live/contact) - Reach out to 100ms team to get pricing information, understand how we can help you go live, or to learn more about the platform. diff --git a/packages/roomkit-react/package.json b/packages/roomkit-react/package.json index 6224388f97..a816c1bea7 100644 --- a/packages/roomkit-react/package.json +++ b/packages/roomkit-react/package.json @@ -10,7 +10,7 @@ "prebuilt", "roomkit" ], - "version": "0.3.23", + "version": "0.3.24", "author": "100ms", "license": "MIT", "repository": { @@ -75,12 +75,12 @@ "react": ">=17.0.2 <19.0.0" }, "dependencies": { - "@100mslive/hls-player": "0.3.23", + "@100mslive/hls-player": "0.3.24", "@100mslive/hms-noise-cancellation": "0.0.1", - "@100mslive/hms-virtual-background": "1.13.23", - "@100mslive/hms-whiteboard": "0.0.13", - "@100mslive/react-icons": "0.10.23", - "@100mslive/react-sdk": "0.10.23", + "@100mslive/hms-virtual-background": "1.13.24", + "@100mslive/hms-whiteboard": "0.0.14", + "@100mslive/react-icons": "0.10.24", + "@100mslive/react-sdk": "0.10.24", "@100mslive/types-prebuilt": "0.12.12", "@emoji-mart/data": "^1.0.6", "@emoji-mart/react": "^1.0.1", @@ -99,7 +99,7 @@ "@radix-ui/react-tabs": "1.0.0", "@radix-ui/react-toast": "1.0.0", "@radix-ui/react-tooltip": "1.0.6", - "@stitches/react": "^1.2.8", + "@stitches/react": "1.3.1-1", "emoji-mart": "^5.2.2", "eventemitter2": "^6.4.9", "lodash.merge": "^4.6.2", diff --git a/packages/roomkit-react/src/Avatar/Avatar.tsx b/packages/roomkit-react/src/Avatar/Avatar.tsx index c9dd366d9e..470e0a63a8 100644 --- a/packages/roomkit-react/src/Avatar/Avatar.tsx +++ b/packages/roomkit-react/src/Avatar/Avatar.tsx @@ -13,7 +13,7 @@ export const StyledAvatar = styled('div', { ...flexCenter, color: '$colors$on_primary_high', fontFamily: '$sans', - aspectRatio: 1, + aspectRatio: '1', fontWeight: 600, fontSize: '$space$9', variants: { diff --git a/packages/roomkit-react/src/Layout/Flex.tsx b/packages/roomkit-react/src/Layout/Flex.tsx index 2a83ed6603..a5ea8cf1d3 100644 --- a/packages/roomkit-react/src/Layout/Flex.tsx +++ b/packages/roomkit-react/src/Layout/Flex.tsx @@ -48,7 +48,7 @@ export const Flex = styled('div', { baseline: { alignItems: 'baseline', }, - strech: { + stretch: { alignItems: 'stretch', }, }, diff --git a/packages/roomkit-react/src/Popover/Popover.stories.tsx b/packages/roomkit-react/src/Popover/Popover.stories.tsx index 24a8b51b81..b955386e62 100644 --- a/packages/roomkit-react/src/Popover/Popover.stories.tsx +++ b/packages/roomkit-react/src/Popover/Popover.stories.tsx @@ -32,7 +32,7 @@ const Template: ComponentStory = () => (
diff --git a/packages/roomkit-react/src/Prebuilt/components/Footer/WhiteboardToggle.tsx b/packages/roomkit-react/src/Prebuilt/components/Footer/WhiteboardToggle.tsx index 3db4f2b32b..68505503fe 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Footer/WhiteboardToggle.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Footer/WhiteboardToggle.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { useState } from 'react'; import { selectPeerScreenSharing, useHMSStore, useWhiteboard } from '@100mslive/react-sdk'; import { PencilDrawIcon } from '@100mslive/react-icons'; import { Tooltip } from '../../..'; @@ -11,6 +11,7 @@ export const WhiteboardToggle = () => { const { toggle, open, isOwner } = useWhiteboard(); const peerSharing = useHMSStore(selectPeerScreenSharing); const disabled = !!peerSharing || (open && !isOwner); + const [isLoading, setLoading] = useState(false); if (!toggle) { return null; @@ -25,17 +26,23 @@ export const WhiteboardToggle = () => { > { - if (disabled) { + if (disabled || isLoading) { return; } try { - await toggle(); + if (!open) { + setLoading(true); + await toggle(); + setTimeout(() => setLoading(false), 500); + } else { + await toggle(); + } } catch (error) { ToastManager.addToast({ title: (error as Error).message, variant: 'error' }); } }} active={!open} - disabled={disabled} + disabled={disabled || isLoading} data-testid="whiteboard_btn" > diff --git a/packages/roomkit-react/src/Prebuilt/components/HMSVideo/HMSVideo.jsx b/packages/roomkit-react/src/Prebuilt/components/HMSVideo/HMSVideo.jsx index 5ba399c1d4..d48854ab21 100644 --- a/packages/roomkit-react/src/Prebuilt/components/HMSVideo/HMSVideo.jsx +++ b/packages/roomkit-react/src/Prebuilt/components/HMSVideo/HMSVideo.jsx @@ -1,7 +1,7 @@ import React, { forwardRef, useEffect, useState } from 'react'; import { Flex } from '../../../Layout'; -export const HMSVideo = forwardRef(({ children, ...props }, videoRef) => { +export const HMSVideo = forwardRef(({ children, isFullScreen, ...props }, videoRef) => { const [width, setWidth] = useState('auto'); useEffect(() => { @@ -22,7 +22,7 @@ export const HMSVideo = forwardRef(({ children, ...props }, videoRef) => { return () => { videoEl.removeEventListener('loadedmetadata', updatingVideoWidth); }; - }, []); + }, [videoRef, width]); return ( { transition: 'all 0.3s ease-in-out', '@md': { '& video': { - height: props.isFullScreen ? '' : '$60 !important', + height: isFullScreen ? '' : '$60 !important', }, }, '& video::cue': { diff --git a/packages/roomkit-react/src/Prebuilt/components/InsetTile.tsx b/packages/roomkit-react/src/Prebuilt/components/InsetTile.tsx index e20fc32dc0..252d1d41df 100644 --- a/packages/roomkit-react/src/Prebuilt/components/InsetTile.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/InsetTile.tsx @@ -105,7 +105,7 @@ export const InsetTile = ({ peerId }: { peerId?: string }) => { r: '$2', ...(!minimised ? { - aspectRatio: aspectRatio, + aspectRatio: `${aspectRatio}`, h: height, } : {}), diff --git a/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/DesktopOptions.tsx b/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/DesktopOptions.tsx index a642567526..410b6d699e 100644 --- a/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/DesktopOptions.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/DesktopOptions.tsx @@ -130,6 +130,9 @@ export const DesktopOptions = ({ { + e.preventDefault(); + }} align="end" css={{ py: '$0', diff --git a/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/MwebOptions.tsx b/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/MwebOptions.tsx index c5e4257ff6..ebc5f4559a 100644 --- a/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/MwebOptions.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/MoreSettings/SplitComponents/MwebOptions.tsx @@ -43,6 +43,8 @@ import { StopRecordingInSheet } from '../../Header/StreamActions'; // @ts-ignore: No implicit any import SettingsModal from '../../Settings/SettingsModal'; // @ts-ignore: No implicit any +import { StatsForNerds } from '../../StatsForNerds'; +// @ts-ignore: No implicit any import { ToastManager } from '../../Toast/ToastManager'; // @ts-ignore: No implicit any import { ActionTile } from '../ActionTile'; @@ -94,6 +96,7 @@ export const MwebOptions = ({ const [openModals, setOpenModals] = useState(new Set()); const [openOptionsSheet, setOpenOptionsSheet] = useState(false); const [openSettingsSheet, setOpenSettingsSheet] = useState(false); + const [openStatsForNerdsSheet, setOpenStatsForNerdsSheet] = useState(false); const [showEmojiCard, setShowEmojiCard] = useState(false); const [showRecordingOn, setShowRecordingOn] = useState(false); const toggleParticipants = useSidepaneToggle(SIDE_PANE_OPTIONS.PARTICIPANTS); @@ -269,7 +272,15 @@ export const MwebOptions = ({ Settings - + { + setOpenStatsForNerdsSheet(true); + setOpenOptionsSheet(false); + }} + > + + Stats For Nerds + {isConnected && permissions?.browserRecording ? ( + + {openStatsForNerdsSheet && ( + + )} + {openModals.has(MODALS.MUTE_ALL) && ( updateState(MODALS.MUTE_ALL, value)} isMobile /> )} diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceChangeNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceChangeNotifications.tsx new file mode 100644 index 0000000000..2ec629c2ce --- /dev/null +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceChangeNotifications.tsx @@ -0,0 +1,18 @@ +import { useEffect } from 'react'; +import { HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; + +export const DeviceChangeNotifications = () => { + const notification = useHMSNotifications(HMSNotificationTypes.DEVICE_CHANGE_UPDATE); + + useEffect(() => { + if (notification) { + ToastManager.addToast({ + title: notification.message, + }); + } + }, [notification]); + + return null; +}; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceInUseError.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceInUseError.tsx index 3249d845ca..4dcceb3821 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceInUseError.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/DeviceInUseError.tsx @@ -1,5 +1,10 @@ import React, { useEffect, useState } from 'react'; -import { HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +import { + HMSNotificationTypes, + HMSTrackException, + HMSTrackExceptionTrackType, + useHMSNotifications, +} from '@100mslive/react-sdk'; import { Button, Dialog, Text } from '../../..'; // @ts-ignore: No implicit Any import { DialogContent, DialogRow } from '../../primitives/DialogContent'; @@ -24,6 +29,11 @@ export function DeviceInUseError() { if (!error || error.code !== 3003) { return; } + const errorTrackExceptionType = (error as HMSTrackException)?.trackType; + const hasAudio = errorTrackExceptionType === HMSTrackExceptionTrackType.AUDIO; + const hasVideo = errorTrackExceptionType === HMSTrackExceptionTrackType.VIDEO; + const hasAudioVideo = errorTrackExceptionType === HMSTrackExceptionTrackType.AUDIO_VIDEO; + const hasScreen = errorTrackExceptionType === HMSTrackExceptionTrackType.SCREEN; const errorMessage = error?.message; ToastManager.addToast({ @@ -35,10 +45,7 @@ export function DeviceInUseError() { ), }); - const hasAudio = errorMessage.includes('audio'); - const hasVideo = errorMessage.includes('video'); - const hasScreen = errorMessage.includes('screen'); - if (hasAudio && hasVideo) { + if (hasAudioVideo) { setDeviceType('camera and microphone'); } else if (hasAudio) { setDeviceType('microphone'); diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/ErrorNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/ErrorNotifications.tsx new file mode 100644 index 0000000000..e9dbf17fbd --- /dev/null +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/ErrorNotifications.tsx @@ -0,0 +1,56 @@ +import React, { useEffect } from 'react'; +import { HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +import { GroupIcon } from '@100mslive/react-icons'; +import { Box } from '../../../Layout'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; +// @ts-ignore: No implicit Any +import { useSubscribedNotifications } from '../AppData/useUISettings'; + +export const ErrorNotifications = () => { + const notification = useHMSNotifications(HMSNotificationTypes.ERROR); + const subscribedNotifications = useSubscribedNotifications() || {}; + + useEffect(() => { + if (!notification || !notification.data) return; + + const { isTerminal, action, code, message, description } = notification.data; + + if (isTerminal && action !== 'INIT') { + if ([500, 6008].includes(code)) { + ToastManager.addToast({ + title: `Error: ${message}`, + }); + } else if (message === 'role limit reached') { + ToastManager.addToast({ + title: 'The room is currently full, try joining later', + close: true, + icon: ( + + + + ), + }); + } else { + ToastManager.addToast({ + title: message || 'We couldn’t reconnect you. When you’re back online, try joining the room.', + close: false, + }); + } + return; + } + // Autoplay error or user denied screen share (cancelled browser pop-up) + if ([3008, 3001, 3011].includes(code)) { + return; + } + if (action === 'INIT') { + return; + } + if (!subscribedNotifications.ERROR) return; + ToastManager.addToast({ + title: `Error: ${message} - ${description}`, + }); + }, [notification, subscribedNotifications.ERROR]); + + return null; +}; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/MessageNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/MessageNotifications.tsx new file mode 100644 index 0000000000..faea7f5cde --- /dev/null +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/MessageNotifications.tsx @@ -0,0 +1,24 @@ +import { useEffect } from 'react'; +import { HMSNotificationTypes, selectIsLocalScreenShared } from '@100mslive/hms-video-store'; +import { useAwayNotifications, useHMSNotifications, useHMSStore } from '@100mslive/react-sdk'; +import { useRoomLayout } from '../../provider/roomLayoutProvider'; +import { usePIPWindow } from '../PIP/usePIPWindow'; + +export const MessageNotifications = () => { + const notification = useHMSNotifications(HMSNotificationTypes.NEW_MESSAGE); + const amIScreenSharing = useHMSStore(selectIsLocalScreenShared); + const logoURL = useRoomLayout()?.logo?.url; + const { pipWindow } = usePIPWindow(); + const { showNotification } = useAwayNotifications(); + + useEffect(() => { + if (notification && amIScreenSharing && !notification.data?.ignored && !pipWindow) { + showNotification(`New message from ${notification.data.senderName}`, { + body: notification.data.message, + icon: logoURL, + }); + } + }, [notification]); + + return null; +}; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/Notifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/Notifications.tsx index c9e9c529fc..c14ff9c091 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/Notifications.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/Notifications.tsx @@ -1,71 +1,32 @@ /* eslint-disable no-case-declarations */ -import React, { useCallback, useEffect } from 'react'; -import { - HMSNotificationTypes, - HMSRoleChangeRequest, - HMSRoomState, - selectIsLocalScreenShared, - selectLocalPeerID, - selectPeerNameByID, - selectRoomState, - useAwayNotifications, - useCustomEvent, - useHMSNotifications, - useHMSStore, - useHMSVanillaStore, -} from '@100mslive/react-sdk'; -import { GroupIcon } from '@100mslive/react-icons'; -import { Box, Button } from '../../..'; -import { useRoomLayout, useUpdateRoomLayout } from '../../provider/roomLayoutProvider'; +import React, { useCallback } from 'react'; +import { HMSRoleChangeRequest, HMSRoomState, selectRoomState, useCustomEvent, useHMSStore } from '@100mslive/react-sdk'; // @ts-ignore: No implicit Any import { ToastManager } from '../Toast/ToastManager'; import { AutoplayBlockedModal } from './AutoplayBlockedModal'; import { ChatNotifications } from './ChatNotifications'; +import { DeviceChangeNotifications } from './DeviceChangeNotifications'; import { DeviceInUseError } from './DeviceInUseError'; +import { ErrorNotifications } from './ErrorNotifications'; import { HandRaisedNotifications } from './HandRaisedNotifications'; import { InitErrorModal } from './InitErrorModal'; +import { MessageNotifications } from './MessageNotifications'; import { PeerNotifications } from './PeerNotifications'; import { PermissionErrorNotificationModal } from './PermissionErrorModal'; +import { PollNotificationModal } from './PollNotificationModal'; import { ReconnectNotifications } from './ReconnectNotifications'; +import { RoleChangeNotification } from './RoleChangeNotification'; import { TrackBulkUnmuteModal } from './TrackBulkUnmuteModal'; import { TrackNotifications } from './TrackNotifications'; import { TrackUnmuteModal } from './TrackUnmuteModal'; import { TranscriptionNotifications } from './TranscriptionNotifications'; -import { useRoomLayoutConferencingScreen } from '../../provider/roomLayoutProvider/hooks/useRoomLayoutScreen'; // @ts-ignore: No implicit Any -import { usePollViewToggle } from '../AppData/useSidepane'; -// @ts-ignore: No implicit Any -import { useIsNotificationDisabled, useSubscribedNotifications } from '../AppData/useUISettings'; -import { usePIPWindow } from '../PIP/usePIPWindow'; +import { useIsNotificationDisabled } from '../AppData/useUISettings'; import { ROLE_CHANGE_DECLINED } from '../../common/constants'; -const pollToastKey: Record = {}; - export function Notifications() { - const localPeerID = useHMSStore(selectLocalPeerID); - const notification = useHMSNotifications([ - HMSNotificationTypes.NAME_UPDATED, - HMSNotificationTypes.ERROR, - HMSNotificationTypes.ROLE_UPDATED, - HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST, - HMSNotificationTypes.REMOVED_FROM_ROOM, - HMSNotificationTypes.ROOM_ENDED, - HMSNotificationTypes.DEVICE_CHANGE_UPDATE, - HMSNotificationTypes.POLL_STARTED, - HMSNotificationTypes.POLL_STOPPED, - HMSNotificationTypes.NEW_MESSAGE, - ]); - const subscribedNotifications = useSubscribedNotifications() || {}; const roomState = useHMSStore(selectRoomState); - const updateRoomLayoutForRole = useUpdateRoomLayout(); const isNotificationDisabled = useIsNotificationDisabled(); - const screenProps = useRoomLayoutConferencingScreen(); - const vanillaStore = useHMSVanillaStore(); - const togglePollView = usePollViewToggle(); - const { showNotification } = useAwayNotifications(); - const amIScreenSharing = useHMSStore(selectIsLocalScreenShared); - const logoURL = useRoomLayout()?.logo?.url; - const { pipWindow } = usePIPWindow(); const handleRoleChangeDenied = useCallback((request: HMSRoleChangeRequest & { peerName: string }) => { ToastManager.addToast({ @@ -76,130 +37,6 @@ export function Notifications() { useCustomEvent({ type: ROLE_CHANGE_DECLINED, onEvent: handleRoleChangeDenied }); - useEffect(() => { - if (!notification || isNotificationDisabled) { - return; - } - switch (notification.type) { - case HMSNotificationTypes.NAME_UPDATED: - console.log(notification.data.id + ' changed their name to ' + notification.data.name); - break; - case HMSNotificationTypes.ERROR: - if (notification.data?.isTerminal && notification.data?.action !== 'INIT') { - if ([500, 6008].includes(notification.data?.code)) { - ToastManager.addToast({ - title: `Error: ${notification.data?.message}`, - }); - } else if (notification.data?.message === 'role limit reached') { - ToastManager.addToast({ - title: 'The room is currently full, try joining later', - close: true, - icon: ( - - - - ), - }); - } else { - ToastManager.addToast({ - title: - notification.data?.message || - 'We couldn’t reconnect you. When you’re back online, try joining the room.', - close: false, - }); - } - return; - } - // Autoplay error or user denied screen share (cancelled browser pop-up) - if (notification.data?.code === 3008 || notification.data?.code === 3001 || notification.data?.code === 3011) { - return; - } - if (notification.data?.action === 'INIT') { - return; - } - if (!subscribedNotifications.ERROR) return; - ToastManager.addToast({ - title: `Error: ${notification.data?.message} - ${notification.data?.description}`, - duration: 8000, - }); - break; - case HMSNotificationTypes.ROLE_UPDATED: { - if (notification.data?.isLocal && notification.data?.roleName) { - ToastManager.addToast({ - title: `You are now a ${notification.data.roleName}`, - }); - updateRoomLayoutForRole?.(notification.data.roleName); - } - break; - } - case HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST: - const track = notification.data?.track; - if (!notification.data.enabled) { - ToastManager.addToast({ - title: `Your ${track.source} ${track.type} was muted by - ${notification.data.requestedBy?.name}.`, - }); - } - break; - case HMSNotificationTypes.REMOVED_FROM_ROOM: - case HMSNotificationTypes.ROOM_ENDED: - ToastManager.addToast({ - title: `${notification.message}. - ${notification.data.reason && `Reason: ${notification.data.reason}`}`, - }); - break; - case HMSNotificationTypes.DEVICE_CHANGE_UPDATE: - ToastManager.addToast({ - title: notification.message, - }); - break; - - case HMSNotificationTypes.POLL_STARTED: - if (notification.data.startedBy !== localPeerID && screenProps.screenType !== 'hls_live_streaming') { - const pollStartedBy = vanillaStore.getState(selectPeerNameByID(notification.data.startedBy)) || 'Participant'; - - const pollToastID = ToastManager.addToast({ - title: `${pollStartedBy} started a ${notification.data.type}: ${notification.data.title}`, - action: ( - - ), - duration: Infinity, - }); - pollToastKey[notification.data.id] = pollToastID; - } - break; - case HMSNotificationTypes.POLL_STOPPED: - const pollID = notification?.data.id; - if (pollID && pollToastKey?.[pollID]) { - ToastManager.removeToast(pollToastKey?.[notification.data.id]); - delete pollToastKey[notification?.data.id]; - } - break; - case HMSNotificationTypes.NEW_MESSAGE: - if (amIScreenSharing && !notification.data?.ignored && !pipWindow) { - showNotification(`New message from ${notification.data.senderName}`, { - body: notification.data.message, - icon: logoURL, - }); - } - break; - default: - break; - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [notification, subscribedNotifications.ERROR, subscribedNotifications.METADATA_UPDATED]); - if (isNotificationDisabled) { return null; } @@ -210,7 +47,12 @@ export function Notifications() { {roomState === HMSRoomState.Connected ? : null} + + + + + diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/PeerNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/PeerNotifications.tsx index 4487b13836..fe3ee15adb 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/PeerNotifications.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/PeerNotifications.tsx @@ -7,7 +7,11 @@ import { useSetSubscribedChatSelector, useSubscribedNotifications } from '../App // @ts-ignore: No implicit Any import { CHAT_SELECTOR, SUBSCRIBED_NOTIFICATIONS } from '../../common/constants'; -const notificationTypes = [HMSNotificationTypes.PEER_JOINED, HMSNotificationTypes.PEER_LEFT]; +const notificationTypes = [ + HMSNotificationTypes.PEER_JOINED, + HMSNotificationTypes.PEER_LEFT, + HMSNotificationTypes.NAME_UPDATED, +]; export const PeerNotifications = () => { const notification = useHMSNotifications(notificationTypes); @@ -35,11 +39,14 @@ export const PeerNotifications = () => { return; } break; + case HMSNotificationTypes.NAME_UPDATED: + console.log(notification.data.id + ' changed their name to ' + notification.data.name); + return; default: return; } ToastBatcher.showToast({ notification }); - }, [notification, isPeerJoinSubscribed, isPeerLeftSubscribed, selectedPeer.id, setPeerSelector]); + }, [notification]); return null; }; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/PermissionErrorModal.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/PermissionErrorModal.tsx index 982fa900a5..c605dd5b3e 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/PermissionErrorModal.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/PermissionErrorModal.tsx @@ -1,6 +1,12 @@ import React, { useEffect, useState } from 'react'; import { useMedia } from 'react-use'; -import { HMSException, HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +import { + HMSException, + HMSNotificationTypes, + HMSTrackException, + HMSTrackExceptionTrackType, + useHMSNotifications, +} from '@100mslive/react-sdk'; import { Button, config as cssConfig, Dialog, Flex, Text } from '../../..'; // @ts-ignore: No implicit Any import androidPermissionAlert from '../../images/android-perm-1.png'; @@ -26,11 +32,13 @@ export const PermissionErrorModal = ({ error }: { error?: HMSException }) => { ) { return; } - const errorMessage = error?.message; - const hasAudio = errorMessage.includes('audio'); - const hasVideo = errorMessage.includes('video'); - const hasScreen = errorMessage.includes('screen'); - if (hasAudio && hasVideo) { + + const errorTrackExceptionType = (error as HMSTrackException)?.trackType; + const hasAudio = errorTrackExceptionType === HMSTrackExceptionTrackType.AUDIO; + const hasVideo = errorTrackExceptionType === HMSTrackExceptionTrackType.VIDEO; + const hasAudioVideo = errorTrackExceptionType === HMSTrackExceptionTrackType.AUDIO_VIDEO; + const hasScreen = errorTrackExceptionType === HMSTrackExceptionTrackType.SCREEN; + if (hasAudioVideo) { setDeviceType('camera and microphone'); } else if (hasAudio) { setDeviceType('microphone'); diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/PollNotificationModal.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/PollNotificationModal.tsx new file mode 100644 index 0000000000..4b960842e5 --- /dev/null +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/PollNotificationModal.tsx @@ -0,0 +1,71 @@ +import React, { useEffect } from 'react'; +import { + HMSNotificationTypes, + selectLocalPeerID, + selectPeerNameByID, + useHMSNotifications, + useHMSStore, + useHMSVanillaStore, +} from '@100mslive/react-sdk'; +import { Button } from '../../../Button'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; +import { useRoomLayoutConferencingScreen } from '../../provider/roomLayoutProvider/hooks/useRoomLayoutScreen'; +// @ts-ignore: No implicit Any +import { usePollViewToggle } from '../AppData/useSidepane'; + +const notificationTypes = [HMSNotificationTypes.POLL_STARTED, HMSNotificationTypes.POLL_STOPPED]; + +const pollToastKey: Record = {}; + +export const PollNotificationModal = () => { + const togglePollView = usePollViewToggle(); + const localPeerID = useHMSStore(selectLocalPeerID); + const vanillaStore = useHMSVanillaStore(); + const screenProps = useRoomLayoutConferencingScreen(); + + const notification = useHMSNotifications(notificationTypes); + + useEffect(() => { + switch (notification?.type) { + case HMSNotificationTypes.POLL_STARTED: + if (notification.data.startedBy !== localPeerID && screenProps.screenType !== 'hls_live_streaming') { + const pollStartedBy = vanillaStore.getState(selectPeerNameByID(notification.data.startedBy)) || 'Participant'; + + const pollToastID = ToastManager.addToast({ + title: `${pollStartedBy} started a ${notification.data.type}: ${notification.data.title}`, + action: ( + + ), + duration: Infinity, + }); + pollToastKey[notification.data.id] = pollToastID; + } + break; + case HMSNotificationTypes.POLL_STOPPED: + { + const pollID = notification?.data.id; + if (pollID && pollToastKey?.[pollID]) { + ToastManager.removeToast(pollToastKey?.[notification.data.id]); + delete pollToastKey[notification?.data.id]; + } + } + break; + default: + break; + } + }, [notification]); + + return null; +}; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/ReconnectNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/ReconnectNotifications.tsx index 71511f55c9..406fbfd65d 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/ReconnectNotifications.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/ReconnectNotifications.tsx @@ -5,11 +5,7 @@ import { ToastConfig } from '../Toast/ToastConfig'; // @ts-ignore: No implicit Any import { ToastManager } from '../Toast/ToastManager'; -const notificationTypes = [ - HMSNotificationTypes.RECONNECTED, - HMSNotificationTypes.RECONNECTING, - HMSNotificationTypes.ERROR, -]; +const notificationTypes = [HMSNotificationTypes.RECONNECTED, HMSNotificationTypes.RECONNECTING]; let notificationId: string | null = null; export const ReconnectNotifications = () => { diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/RoleChangeNotification.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/RoleChangeNotification.tsx new file mode 100644 index 0000000000..00dfab79db --- /dev/null +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/RoleChangeNotification.tsx @@ -0,0 +1,24 @@ +import { useEffect } from 'react'; +import { HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +import { useUpdateRoomLayout } from '../../provider/roomLayoutProvider'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; + +export const RoleChangeNotification = () => { + const notification = useHMSNotifications(HMSNotificationTypes.ROLE_UPDATED); + const updateRoomLayoutForRole = useUpdateRoomLayout(); + + useEffect(() => { + if (!notification?.data) { + return; + } + if (notification.data?.isLocal && notification.data?.roleName) { + ToastManager.addToast({ + title: `You are now a ${notification.data.roleName}`, + }); + updateRoomLayoutForRole?.(notification.data.roleName); + } + }, [notification]); + + return null; +}; diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.tsx index 63bc1f8ee6..89dabac4a8 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.tsx @@ -9,14 +9,16 @@ import { MicOnIcon } from '@100mslive/react-icons'; // @ts-ignore: No implicit Any import { RequestDialog } from '../../primitives/DialogContent'; +const notificationTypes = [ + HMSNotificationTypes.CHANGE_MULTI_TRACK_STATE_REQUEST, + HMSNotificationTypes.ROOM_ENDED, + HMSNotificationTypes.REMOVED_FROM_ROOM, +]; + export const TrackBulkUnmuteModal = () => { const hmsActions = useHMSActions(); const [muteNotification, setMuteNotification] = useState(null); - const notification = useHMSNotifications([ - HMSNotificationTypes.CHANGE_MULTI_TRACK_STATE_REQUEST, - HMSNotificationTypes.ROOM_ENDED, - HMSNotificationTypes.REMOVED_FROM_ROOM, - ]); + const notification = useHMSNotifications(notificationTypes); useEffect(() => { switch (notification?.type) { diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackNotifications.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackNotifications.tsx index 01beae1716..bbd234643f 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackNotifications.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackNotifications.tsx @@ -1,17 +1,39 @@ import { useEffect } from 'react'; import { HMSNotificationTypes, useHMSNotifications } from '@100mslive/react-sdk'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; const notificationTypes = [ HMSNotificationTypes.TRACK_ADDED, HMSNotificationTypes.TRACK_REMOVED, HMSNotificationTypes.TRACK_MUTED, HMSNotificationTypes.TRACK_UNMUTED, + HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST, ]; + export const TrackNotifications = () => { const notification = useHMSNotifications(notificationTypes); useEffect(() => { if (notification) { - console.debug(`[${notification.type}]`, notification); + switch (notification.type) { + case HMSNotificationTypes.TRACK_ADDED: + case HMSNotificationTypes.TRACK_REMOVED: + case HMSNotificationTypes.TRACK_MUTED: + case HMSNotificationTypes.TRACK_UNMUTED: + console.debug(`[${notification.type}]`, notification); + break; + case HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST: + { + const track = notification.data?.track; + if (!notification.data.enabled) { + ToastManager.addToast({ + title: `Your ${track.source} ${track.type} was muted by + ${notification.data.requestedBy?.name}.`, + }); + } + } + break; + } } }, [notification]); diff --git a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackUnmuteModal.tsx b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackUnmuteModal.tsx index 3b0cce23bd..7d8142701c 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackUnmuteModal.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Notifications/TrackUnmuteModal.tsx @@ -8,20 +8,28 @@ import { import { MicOnIcon } from '@100mslive/react-icons'; // @ts-ignore: No implicit Any import { RequestDialog } from '../../primitives/DialogContent'; +// @ts-ignore: No implicit Any +import { ToastManager } from '../Toast/ToastManager'; + +const notificationTypes = [ + HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST, + HMSNotificationTypes.ROOM_ENDED, + HMSNotificationTypes.REMOVED_FROM_ROOM, +]; export const TrackUnmuteModal = () => { const hmsActions = useHMSActions(); - const notification = useHMSNotifications([ - HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST, - HMSNotificationTypes.ROOM_ENDED, - HMSNotificationTypes.REMOVED_FROM_ROOM, - ]); + const notification = useHMSNotifications(notificationTypes); const [muteNotification, setMuteNotification] = useState(null); useEffect(() => { switch (notification?.type) { case HMSNotificationTypes.REMOVED_FROM_ROOM: case HMSNotificationTypes.ROOM_ENDED: + ToastManager.addToast({ + title: `${notification.message}. + ${notification.data.reason && `Reason: ${notification.data.reason}`}`, + }); setMuteNotification(null); break; case HMSNotificationTypes.CHANGE_TRACK_STATE_REQUEST: diff --git a/packages/roomkit-react/src/Prebuilt/components/Preview/PreviewJoin.tsx b/packages/roomkit-react/src/Prebuilt/components/Preview/PreviewJoin.tsx index 7dc9797ceb..4b3a7618c2 100644 --- a/packages/roomkit-react/src/Prebuilt/components/Preview/PreviewJoin.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/Preview/PreviewJoin.tsx @@ -61,7 +61,7 @@ const useLocalTileAspectRatio = () => { } else { aspectRatio = isMobile ? 9 / 16 : 16 / 9; } - return aspectRatio; + return aspectRatio.toString(); }; const PreviewJoin = ({ @@ -113,6 +113,7 @@ const PreviewJoin = ({ const { elements = {} } = useRoomLayoutPreviewScreen(); const { preview_header: previewHeader = {}, virtual_background } = elements || {}; const aspectRatio = useLocalTileAspectRatio(); + useEffect(() => { if (authToken) { if (skipPreview) { @@ -158,7 +159,7 @@ const PreviewJoin = ({ {toggleVideo ? : null} - + { +export const StatsForNerds = ({ open, onOpenChange }) => { + const mediaQueryLg = cssConfig.media.md; + const isMobile = useMedia(mediaQueryLg); + const tracksWithLabels = useTracksWithLabel(); const statsOptions = useMemo( () => [{ id: 'local-peer', label: 'Local Peer Stats' }, ...tracksWithLabels], [tracksWithLabels], ); + const hmsActions = useHMSActions(); + const details = hmsActions.getDebugInfo(); const [selectedStat, setSelectedStat] = useState(statsOptions[0]); const [showStatsOnTiles, setShowStatsOnTiles] = useSetUiSettings(UI_SETTINGS.showStatsOnTiles); - const [open, setOpen] = useState(false); + const [openDropdown, setOpenDropdown] = useState(false); const ref = useRef(); const selectionBg = useDropdownSelection(); @@ -39,14 +49,106 @@ export const StatsForNerds = ({ onOpenChange }) => { } }, [tracksWithLabels, selectedStat]); - return ( - + return isMobile ? ( + + + + + + + Stats For Nerds + + + + + + + + + + + Show Stats on Tiles + + + {/* Select */} + + + + + + + {statsOptions.map(option => { + const isSelected = option.id === selectedStat.id && option.layer === selectedStat.layer; + return ( + { + setSelectedStat(option); + }} + css={{ + px: '$9', + bg: isSelected ? selectionBg : undefined, + c: isSelected ? '$on_primary_high' : '$on_primary_high', + }} + > + {option.label} + + ); + })} + + + + + {/* Stats */} + {selectedStat.id === 'local-peer' ? ( + + ) : ( + + )} + + + + + + + ) : ( + @@ -70,6 +172,7 @@ export const StatsForNerds = ({ onOpenChange }) => { {/* Select */} + { - + @@ -119,6 +222,9 @@ export const StatsForNerds = ({ onOpenChange }) => { ) : ( )} + + + @@ -229,8 +335,29 @@ const TrackStats = ({ trackID, layer, local }) => { ); }; -const StatsRow = React.memo(({ label, value }) => ( - +const DebugInfo = ({ details }) => { + return ( + + + + + + + + + + + + + + + ); +}; + +const StatsRow = React.memo(({ label, value, css }) => ( + {tiles?.map(tile => { diff --git a/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/GridLayout.tsx b/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/GridLayout.tsx index b7097f49ca..82b1d47186 100644 --- a/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/GridLayout.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/GridLayout.tsx @@ -9,7 +9,6 @@ import { useHMSStore, useHMSVanillaStore, } from '@100mslive/react-sdk'; -// @ts-ignore: No implicit Any import { EqualProminence } from './EqualProminence'; import { RoleProminence } from './RoleProminence'; import { ScreenshareLayout } from './ScreenshareLayout'; diff --git a/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/ProminenceLayout.tsx b/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/ProminenceLayout.tsx index 428ea1b49e..5b996cc336 100644 --- a/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/ProminenceLayout.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/VideoLayouts/ProminenceLayout.tsx @@ -70,9 +70,9 @@ const SecondarySection = ({ rootCSS={{ padding: 0, maxWidth: 240, - aspectRatio: 16 / 9, + aspectRatio: '16 / 9', ...(hasSidebar ? { w: '100%' } : { h: '100%' }), - '@md': { aspectRatio: 1 }, + '@md': { aspectRatio: '1' }, }} objectFit="contain" {...tileLayoutProps} diff --git a/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBOption.tsx b/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBOption.tsx index 58b4294170..17340ee391 100644 --- a/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBOption.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBOption.tsx @@ -25,7 +25,9 @@ const Root = ({ bg: '$surface_bright', border: `4px solid ${isActive ? '$primary_default' : '$surface_dim'}`, cursor: 'pointer', - '&:hover': { border: '4px solid $primary_dim' }, + '@media (hover:hover)': { + '&:hover': { border: '4px solid $primary_dim' }, + }, ...(mediaURL ? { height: '$20', backgroundImage: `url("${mediaURL}")`, backgroundSize: 'cover' } : {}), }} onClick={async () => { diff --git a/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBPicker.tsx b/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBPicker.tsx index 30a1dc70ad..2ad687cf2f 100644 --- a/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBPicker.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/VirtualBackground/VBPicker.tsx @@ -180,6 +180,9 @@ export const VBPicker = ({ backgroundMedia = [] }: { backgroundMedia: VirtualBac onClick: async () => { await VBHandler.removeEffects(); hmsActions.setAppData(APP_DATA.background, HMSVirtualBackgroundTypes.NONE); + if (isMobile) { + toggleVB(); + } }, supported: true, }, @@ -235,6 +238,9 @@ export const VBPicker = ({ backgroundMedia = [] }: { backgroundMedia: VirtualBac onClick: async () => { await VBHandler?.setBackground(mediaURL); hmsActions.setAppData(APP_DATA.background, mediaURL); + if (isMobile) { + toggleVB(); + } }, supported: true, }))} diff --git a/packages/roomkit-react/src/Prebuilt/components/hooks/useTileLayout.tsx b/packages/roomkit-react/src/Prebuilt/components/hooks/useTileLayout.tsx index 17b63f888d..cf38dd7af1 100644 --- a/packages/roomkit-react/src/Prebuilt/components/hooks/useTileLayout.tsx +++ b/packages/roomkit-react/src/Prebuilt/components/hooks/useTileLayout.tsx @@ -115,6 +115,7 @@ export const useTileLayout = ({ } } } + for (let i = 0; i < row.length; i++) { row[i].width = tileWidth; row[i].height = tileHeight; diff --git a/packages/roomkit-react/src/Prebuilt/layouts/HLSView.jsx b/packages/roomkit-react/src/Prebuilt/layouts/HLSView.jsx index a90071082d..405f16fd3c 100644 --- a/packages/roomkit-react/src/Prebuilt/layouts/HLSView.jsx +++ b/packages/roomkit-react/src/Prebuilt/layouts/HLSView.jsx @@ -46,7 +46,6 @@ const toastMap = {}; const ToggleChat = ({ isFullScreen = false }) => { const { elements } = useRoomLayoutConferencingScreen(); const sidepane = useHMSStore(selectAppData(APP_DATA.sidePane)); - const toggleChat = useSidepaneToggle(SIDE_PANE_OPTIONS.CHAT); const showChat = !!elements?.chat; const isMobile = useMedia(config.media.md); const hmsActions = useHMSActions(); @@ -57,7 +56,7 @@ const ToggleChat = ({ isFullScreen = false }) => { hmsActions.setAppData(APP_DATA.sidePane, ''); }) .with({ isMobile: true, showChat: true, sidepane: P.when(value => !value) }, () => { - toggleChat(); + hmsActions.setAppData(APP_DATA.sidePane, SIDE_PANE_OPTIONS.CHAT); }) .with({ showChat: false, isMobile: true, sidepane: SIDE_PANE_OPTIONS.CHAT }, () => { hmsActions.setAppData(APP_DATA.sidePane, ''); @@ -65,7 +64,7 @@ const ToggleChat = ({ isFullScreen = false }) => { .otherwise(() => { //do nothing }); - }, [sidepane, isMobile, toggleChat, showChat, hmsActions, isFullScreen]); + }, [sidepane, isMobile, showChat, hmsActions, isFullScreen]); return null; }; const HLSView = () => { @@ -286,6 +285,7 @@ const HLSView = () => { hlsPlayer.reset(); }; } + // eslint-disable-next-line react-hooks/exhaustive-deps }, [hlsUrl, vanillaStore, hmsActions]); /** diff --git a/packages/roomkit-react/src/Prebuilt/layouts/PDFView.jsx b/packages/roomkit-react/src/Prebuilt/layouts/PDFView.jsx index 851738a2be..881454471f 100644 --- a/packages/roomkit-react/src/Prebuilt/layouts/PDFView.jsx +++ b/packages/roomkit-react/src/Prebuilt/layouts/PDFView.jsx @@ -19,6 +19,7 @@ export const PDFView = () => { return () => { resetConfig(); }; + // eslint-disable-next-line react-hooks/exhaustive-deps }, []); useEffect(() => { (async () => { diff --git a/packages/roomkit-react/src/Prebuilt/layouts/VideoStreamingSection.tsx b/packages/roomkit-react/src/Prebuilt/layouts/VideoStreamingSection.tsx index 8acd07d015..d2be65f114 100644 --- a/packages/roomkit-react/src/Prebuilt/layouts/VideoStreamingSection.tsx +++ b/packages/roomkit-react/src/Prebuilt/layouts/VideoStreamingSection.tsx @@ -1,4 +1,4 @@ -import React, { Suspense, useEffect, useState } from 'react'; +import React, { useEffect, useState } from 'react'; import { ControlPosition } from 'react-draggable'; import { useMedia } from 'react-use'; import { @@ -15,16 +15,16 @@ import { useHMSStore, } from '@100mslive/react-sdk'; import { PeopleAddIcon, ShareScreenIcon } from '@100mslive/react-icons'; -import FullPageProgress from '../components/FullPageProgress'; import { GridLayout } from '../components/VideoLayouts/GridLayout'; import { Box, Flex } from '../../Layout'; import { config } from '../../Theme'; // @ts-ignore: No implicit Any import { EmbedView } from './EmbedView'; // @ts-ignore: No implicit Any +import HLSView from './HLSView'; +// @ts-ignore: No implicit Any import { PDFView } from './PDFView'; import SidePane from './SidePane'; -// @ts-ignore: No implicit Any import { WaitingView } from './WaitingView'; import { CaptionsViewer } from '../plugins/CaptionsViewer'; // @ts-ignore: No implicit Any @@ -32,8 +32,6 @@ import { usePDFConfig, useUrlToEmbed } from '../components/AppData/useUISettings import { useCloseScreenshareWhiteboard } from '../components/hooks/useCloseScreenshareWhiteboard'; import { useLandscapeHLSStream, useMobileHLSStream, useWaitingRoomInfo } from '../common/hooks'; import { SESSION_STORE_KEY } from '../common/constants'; -// @ts-ignore: No implicit Any -const HLSView = React.lazy(() => import('./HLSView')); export const VideoStreamingSection = ({ screenType, @@ -79,90 +77,88 @@ export const VideoStreamingSection = ({ } return ( - }> - , 'row' | 'column'>({ isLandscapeHLSStream, isMobileHLSStream }) + .with({ isLandscapeHLSStream: true }, () => 'row') + .with({ isMobileHLSStream: true }, () => 'column') + .otherwise(() => 'row')} + > + {match({ + screenType, + isNotAllowedToPublish, + isScreenOnlyPublishParams, + hasSubscribedRolePublishing, + isSharingScreen, + pdfAnnotatorActive, + urlToIframe, + }) + .with( + { + screenType: 'hls_live_streaming', + }, + () => , + ) + .when( + ({ isNotAllowedToPublish, hasSubscribedRolePublishing }) => + isNotAllowedToPublish && !hasSubscribedRolePublishing, + () => ( + } + /> + ), + ) + .when( + ({ isScreenOnlyPublishParams, isSharingScreen, hasSubscribedRolePublishing }) => + isScreenOnlyPublishParams && !isSharingScreen && !hasSubscribedRolePublishing, + () => ( + } + /> + ), + ) + .when( + ({ pdfAnnotatorActive }) => !!pdfAnnotatorActive, + () => , + ) + .when( + ({ urlToIframe }) => !!urlToIframe, + () => , + ) + + .otherwise(() => { + // @ts-ignore + return ; + })} + + '1 1 0') + .with({ isMobileHLSStream: true }, () => '2 1 0') + .otherwise(() => undefined), position: 'relative', - gap: isMobileHLSStream || isLandscapeHLSStream ? '0' : '$4', + height: !isMobileHLSStream ? '100%' : undefined, + maxHeight: '100%', + '&:empty': { display: 'none' }, + overflowY: 'clip', }} - direction={match, 'row' | 'column'>({ isLandscapeHLSStream, isMobileHLSStream }) - .with({ isLandscapeHLSStream: true }, () => 'row') - .with({ isMobileHLSStream: true }, () => 'column') - .otherwise(() => 'row')} > - {match({ - screenType, - isNotAllowedToPublish, - isScreenOnlyPublishParams, - hasSubscribedRolePublishing, - isSharingScreen, - pdfAnnotatorActive, - urlToIframe, - }) - .with( - { - screenType: 'hls_live_streaming', - }, - () => , - ) - .when( - ({ isNotAllowedToPublish, hasSubscribedRolePublishing }) => - isNotAllowedToPublish && !hasSubscribedRolePublishing, - () => ( - } - /> - ), - ) - .when( - ({ isScreenOnlyPublishParams, isSharingScreen, hasSubscribedRolePublishing }) => - isScreenOnlyPublishParams && !isSharingScreen && !hasSubscribedRolePublishing, - () => ( - } - /> - ), - ) - .when( - ({ pdfAnnotatorActive }) => !!pdfAnnotatorActive, - () => , - ) - .when( - ({ urlToIframe }) => !!urlToIframe, - () => , - ) - - .otherwise(() => { - // @ts-ignore - return ; - })} - - '1 1 0') - .with({ isMobileHLSStream: true }, () => '2 1 0') - .otherwise(() => undefined), - position: 'relative', - height: !isMobileHLSStream ? '100%' : undefined, - maxHeight: '100%', - '&:empty': { display: 'none' }, - overflowY: 'clip', - }} - > - - - - + + + ); }; diff --git a/packages/roomkit-react/src/Prebuilt/layouts/WaitingView.tsx b/packages/roomkit-react/src/Prebuilt/layouts/WaitingView.tsx index aac75c4895..378617e972 100644 --- a/packages/roomkit-react/src/Prebuilt/layouts/WaitingView.tsx +++ b/packages/roomkit-react/src/Prebuilt/layouts/WaitingView.tsx @@ -36,12 +36,12 @@ export const WaitingView = React.memo( gap: '$4', }} > - + {title} {subtitle} diff --git a/packages/roomkit-web/package.json b/packages/roomkit-web/package.json index b160993a49..b8ed70a12e 100644 --- a/packages/roomkit-web/package.json +++ b/packages/roomkit-web/package.json @@ -1,6 +1,6 @@ { "name": "@100mslive/roomkit-web", - "version": "0.2.23", + "version": "0.2.24", "description": "A web component implementation of 100ms Prebuilt component", "keywords": [ "web-components", @@ -33,7 +33,7 @@ "build": "rm -rf dist && node ../../scripts/build-webapp" }, "dependencies": { - "@100mslive/roomkit-react": "0.3.23", + "@100mslive/roomkit-react": "0.3.24", "@r2wc/react-to-web-component": "2.0.2" } } diff --git a/scripts/build-webapp.js b/scripts/build-webapp.js index 3c8a0919b3..9eb775acc8 100644 --- a/scripts/build-webapp.js +++ b/scripts/build-webapp.js @@ -16,7 +16,7 @@ async function main() { const pkg = JSON.parse(fs.readFileSync('package.json', 'utf8')); const source = pkg.name === '@100mslive/roomkit-web' ? './src/index.js' : './src/index.ts'; const external = [...Object.keys(pkg.dependencies || {}), ...Object.keys(pkg.peerDependencies || {})]; - const loader = { '.js': 'jsx', '.svg': 'dataurl', '.png': 'dataurl' }; + const loader = { '.js': 'jsx', '.svg': 'copy', '.png': 'copy' }; const define = { 'process.env': JSON.stringify(process.env) }; const target = 'es6'; const plugins = [ diff --git a/scripts/dev-webapp.js b/scripts/dev-webapp.js index 98bea395b1..b30d4a2df2 100644 --- a/scripts/dev-webapp.js +++ b/scripts/dev-webapp.js @@ -5,7 +5,7 @@ async function main() { const pkg = JSON.parse(fs.readFileSync('package.json', 'utf8')); const source = 'src/App.js'; const external = Object.keys(pkg.dependencies || {}); - const loader = { '.js': 'jsx', '.svg': 'dataurl', '.png': 'dataurl' }; + const loader = { '.js': 'jsx', '.svg': 'copy', '.png': 'copy' }; require('dotenv').config(); const define = { 'process.env': JSON.stringify(process.env) }; const commonOptions = { diff --git a/yarn.lock b/yarn.lock index 091e434c67..a4ab58ed6b 100644 --- a/yarn.lock +++ b/yarn.lock @@ -4451,10 +4451,10 @@ ignore "^5.1.8" p-map "^4.0.0" -"@stitches/react@^1.2.8": - version "1.2.8" - resolved "https://registry.yarnpkg.com/@stitches/react/-/react-1.2.8.tgz#954f8008be8d9c65c4e58efa0937f32388ce3a38" - integrity sha512-9g9dWI4gsSVe8bNLlb+lMkBYsnIKCZTmvqvDG+Avnn69XfmHZKiaMrx7cgTaddq7aTPPmXiTsbFcUy0xgI4+wA== +"@stitches/react@1.3.1-1": + version "1.3.1-1" + resolved "https://registry.yarnpkg.com/@stitches/react/-/react-1.3.1-1.tgz#d8fcbadd4ea06506117ddfcd5a583a598acc18d8" + integrity sha512-ErptbQehV25Da6LtZuM/51kGNK/UvlRY2da9IhhfXQ9h4bmf2f+lYFiJQ9j43O1kwYr6iYJIBRM47FEbsUWffw== "@storybook/addon-a11y@^7.0.27": version "7.2.3"