mirror of
				https://github.com/cinnyapp/cinny.git
				synced 2025-11-04 14:30:29 +03:00 
			
		
		
		
	Merge branch 'dev' into dev
This commit is contained in:
		
						commit
						fef259c5ec
					
				
					 197 changed files with 9996 additions and 1922 deletions
				
			
		
							
								
								
									
										2
									
								
								.github/workflows/build-pull-request.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build-pull-request.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -14,7 +14,7 @@ jobs:
 | 
			
		|||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4.2.0
 | 
			
		||||
      - name: Setup node
 | 
			
		||||
        uses: actions/setup-node@v4.3.0
 | 
			
		||||
        uses: actions/setup-node@v4.4.0
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 20.12.2
 | 
			
		||||
          cache: 'npm'
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										4
									
								
								.github/workflows/deploy-pull-request.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/deploy-pull-request.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -15,7 +15,7 @@ jobs:
 | 
			
		|||
    if: ${{ github.event.workflow_run.conclusion == 'success' }}
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Download pr number
 | 
			
		||||
        uses: dawidd6/action-download-artifact@07ab29fd4a977ae4d2b275087cf67563dfdf0295
 | 
			
		||||
        uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5
 | 
			
		||||
        with:
 | 
			
		||||
          workflow: ${{ github.event.workflow.id }}
 | 
			
		||||
          run_id: ${{ github.event.workflow_run.id }}
 | 
			
		||||
| 
						 | 
				
			
			@ -24,7 +24,7 @@ jobs:
 | 
			
		|||
        id: pr
 | 
			
		||||
        run: echo "id=$(<pr.txt)" >> $GITHUB_OUTPUT
 | 
			
		||||
      - name: Download artifact
 | 
			
		||||
        uses: dawidd6/action-download-artifact@07ab29fd4a977ae4d2b275087cf67563dfdf0295
 | 
			
		||||
        uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5
 | 
			
		||||
        with:
 | 
			
		||||
          workflow: ${{ github.event.workflow.id }}
 | 
			
		||||
          run_id: ${{ github.event.workflow_run.id }}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										2
									
								
								.github/workflows/docker-pr.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/docker-pr.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -13,7 +13,7 @@ jobs:
 | 
			
		|||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4.2.0
 | 
			
		||||
      - name: Build Docker image
 | 
			
		||||
        uses: docker/build-push-action@v6.15.0
 | 
			
		||||
        uses: docker/build-push-action@v6.18.0
 | 
			
		||||
        with:
 | 
			
		||||
          context: .
 | 
			
		||||
          push: false
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										2
									
								
								.github/workflows/netlify-dev.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/netlify-dev.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -13,7 +13,7 @@ jobs:
 | 
			
		|||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4.2.0
 | 
			
		||||
      - name: Setup node
 | 
			
		||||
        uses: actions/setup-node@v4.3.0
 | 
			
		||||
        uses: actions/setup-node@v4.4.0
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 20.12.2
 | 
			
		||||
          cache: 'npm'
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										12
									
								
								.github/workflows/prod-deploy.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/prod-deploy.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -12,7 +12,7 @@ jobs:
 | 
			
		|||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4.2.0
 | 
			
		||||
      - name: Setup node
 | 
			
		||||
        uses: actions/setup-node@v4.3.0
 | 
			
		||||
        uses: actions/setup-node@v4.4.0
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 20.12.2
 | 
			
		||||
          cache: 'npm'
 | 
			
		||||
| 
						 | 
				
			
			@ -52,7 +52,7 @@ jobs:
 | 
			
		|||
          gpg --export | xxd -p
 | 
			
		||||
          echo '${{ secrets.GNUPG_PASSPHRASE }}' | gpg --batch --yes --pinentry-mode loopback --passphrase-fd 0 --armor --detach-sign cinny-${{ steps.vars.outputs.tag }}.tar.gz
 | 
			
		||||
      - name: Upload tagged release
 | 
			
		||||
        uses: softprops/action-gh-release@c95fe1489396fe8a9eb87c0abf8aa5b2ef267fda
 | 
			
		||||
        uses: softprops/action-gh-release@72f2c25fcb47643c292f7107632f7a47c1df5cd8
 | 
			
		||||
        with:
 | 
			
		||||
          files: |
 | 
			
		||||
            cinny-${{ steps.vars.outputs.tag }}.tar.gz
 | 
			
		||||
| 
						 | 
				
			
			@ -72,25 +72,25 @@ jobs:
 | 
			
		|||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3.10.0
 | 
			
		||||
      - name: Login to Docker Hub
 | 
			
		||||
        uses: docker/login-action@v3.4.0
 | 
			
		||||
        uses: docker/login-action@v3.5.0
 | 
			
		||||
        with:
 | 
			
		||||
          username: ${{ secrets.DOCKER_USERNAME }}
 | 
			
		||||
          password: ${{ secrets.DOCKER_PASSWORD }}
 | 
			
		||||
      - name: Login to the Container registry
 | 
			
		||||
        uses: docker/login-action@v3.4.0
 | 
			
		||||
        uses: docker/login-action@v3.5.0
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.actor }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
      - name: Extract metadata (tags, labels) for Docker
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5.7.0
 | 
			
		||||
        uses: docker/metadata-action@v5.8.0
 | 
			
		||||
        with:
 | 
			
		||||
          images: |
 | 
			
		||||
            ${{ secrets.DOCKER_USERNAME }}/cinny
 | 
			
		||||
            ghcr.io/${{ github.repository }}
 | 
			
		||||
      - name: Build and push Docker image
 | 
			
		||||
        uses: docker/build-push-action@v6.15.0
 | 
			
		||||
        uses: docker/build-push-action@v6.18.0
 | 
			
		||||
        with:
 | 
			
		||||
          context: .
 | 
			
		||||
          platforms: linux/amd64,linux/arm64
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -11,7 +11,7 @@ RUN npm run build
 | 
			
		|||
 | 
			
		||||
 | 
			
		||||
## App
 | 
			
		||||
FROM nginx:1.27.4-alpine
 | 
			
		||||
FROM nginx:1.29.0-alpine
 | 
			
		||||
 | 
			
		||||
COPY --from=builder /src/dist /app
 | 
			
		||||
COPY --from=builder /src/docker-nginx.conf /etc/nginx/conf.d/default.conf
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										114
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										114
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							| 
						 | 
				
			
			@ -1,12 +1,12 @@
 | 
			
		|||
{
 | 
			
		||||
  "name": "cinny",
 | 
			
		||||
  "version": "4.6.0",
 | 
			
		||||
  "version": "4.9.1",
 | 
			
		||||
  "lockfileVersion": 3,
 | 
			
		||||
  "requires": true,
 | 
			
		||||
  "packages": {
 | 
			
		||||
    "": {
 | 
			
		||||
      "name": "cinny",
 | 
			
		||||
      "version": "4.6.0",
 | 
			
		||||
      "version": "4.9.1",
 | 
			
		||||
      "license": "AGPL-3.0-only",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "@atlaskit/pragmatic-drag-and-drop": "1.1.6",
 | 
			
		||||
| 
						 | 
				
			
			@ -21,6 +21,7 @@
 | 
			
		|||
        "@vanilla-extract/recipes": "0.3.0",
 | 
			
		||||
        "@vanilla-extract/vite-plugin": "3.7.1",
 | 
			
		||||
        "await-to-js": "3.0.0",
 | 
			
		||||
        "badwords-list": "2.0.1-4",
 | 
			
		||||
        "blurhash": "2.0.4",
 | 
			
		||||
        "browser-encrypt-attachment": "0.3.0",
 | 
			
		||||
        "chroma-js": "3.1.2",
 | 
			
		||||
| 
						 | 
				
			
			@ -33,7 +34,7 @@
 | 
			
		|||
        "file-saver": "2.0.5",
 | 
			
		||||
        "flux": "4.0.3",
 | 
			
		||||
        "focus-trap-react": "10.0.2",
 | 
			
		||||
        "folds": "2.1.0",
 | 
			
		||||
        "folds": "2.2.0",
 | 
			
		||||
        "formik": "2.4.6",
 | 
			
		||||
        "html-dom-parser": "4.0.0",
 | 
			
		||||
        "html-react-parser": "4.2.0",
 | 
			
		||||
| 
						 | 
				
			
			@ -45,7 +46,7 @@
 | 
			
		|||
        "jotai": "2.6.0",
 | 
			
		||||
        "linkify-react": "4.1.3",
 | 
			
		||||
        "linkifyjs": "4.1.3",
 | 
			
		||||
        "matrix-js-sdk": "35.0.0",
 | 
			
		||||
        "matrix-js-sdk": "37.5.0",
 | 
			
		||||
        "millify": "6.1.0",
 | 
			
		||||
        "pdfjs-dist": "4.2.67",
 | 
			
		||||
        "prismjs": "1.30.0",
 | 
			
		||||
| 
						 | 
				
			
			@ -98,7 +99,7 @@
 | 
			
		|||
        "prettier": "2.8.1",
 | 
			
		||||
        "sass": "1.56.2",
 | 
			
		||||
        "typescript": "4.9.4",
 | 
			
		||||
        "vite": "5.4.15",
 | 
			
		||||
        "vite": "5.4.19",
 | 
			
		||||
        "vite-plugin-pwa": "0.20.5",
 | 
			
		||||
        "vite-plugin-static-copy": "1.0.4",
 | 
			
		||||
        "vite-plugin-top-level-await": "1.4.4"
 | 
			
		||||
| 
						 | 
				
			
			@ -2263,17 +2264,19 @@
 | 
			
		|||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/@matrix-org/matrix-sdk-crypto-wasm": {
 | 
			
		||||
      "version": "11.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@matrix-org/matrix-sdk-crypto-wasm/-/matrix-sdk-crypto-wasm-11.1.0.tgz",
 | 
			
		||||
      "integrity": "sha512-JPuO9RCVDklDjbFzMvZfQb7PuiFkLY72bniRSu81lRzkkrcbZtmKqBFMm9H4f2FSz+tHVkDnmsvn12I4sdJJ5A==",
 | 
			
		||||
      "version": "14.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@matrix-org/matrix-sdk-crypto-wasm/-/matrix-sdk-crypto-wasm-14.1.0.tgz",
 | 
			
		||||
      "integrity": "sha512-vcSxHJIr6lP0Fgo8jl0sTHg+OZxZn+skGjiyB62erfgw/R2QqJl0ZVSY8SRcbk9LtHo/ZGld1tnaOyjL2e3cLQ==",
 | 
			
		||||
      "license": "Apache-2.0",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">= 10"
 | 
			
		||||
        "node": ">= 18"
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/@matrix-org/olm": {
 | 
			
		||||
      "version": "3.2.15",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@matrix-org/olm/-/olm-3.2.15.tgz",
 | 
			
		||||
      "integrity": "sha512-S7lOrndAK9/8qOtaTq/WhttJC/o4GAzdfK0MUPpo8ApzsJEC0QjtwrkC3KBXdFP1cD1MXi/mlKR7aaoVMKgs6Q=="
 | 
			
		||||
      "integrity": "sha512-S7lOrndAK9/8qOtaTq/WhttJC/o4GAzdfK0MUPpo8ApzsJEC0QjtwrkC3KBXdFP1cD1MXi/mlKR7aaoVMKgs6Q==",
 | 
			
		||||
      "license": "Apache-2.0"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/@nodelib/fs.scandir": {
 | 
			
		||||
      "version": "2.1.5",
 | 
			
		||||
| 
						 | 
				
			
			@ -4589,7 +4592,8 @@
 | 
			
		|||
    "node_modules/@types/events": {
 | 
			
		||||
      "version": "3.0.3",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.3.tgz",
 | 
			
		||||
      "integrity": "sha512-trOc4AAUThEz9hapPtSd7wf5tiQKvTtu5b371UxXdTuqzIh0ArcRspRP0i0Viu+LXstIQ1z96t1nsPxT9ol01g=="
 | 
			
		||||
      "integrity": "sha512-trOc4AAUThEz9hapPtSd7wf5tiQKvTtu5b371UxXdTuqzIh0ArcRspRP0i0Viu+LXstIQ1z96t1nsPxT9ol01g==",
 | 
			
		||||
      "license": "MIT"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/@types/file-saver": {
 | 
			
		||||
      "version": "2.0.5",
 | 
			
		||||
| 
						 | 
				
			
			@ -4678,7 +4682,8 @@
 | 
			
		|||
    "node_modules/@types/retry": {
 | 
			
		||||
      "version": "0.12.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@types/retry/-/retry-0.12.0.tgz",
 | 
			
		||||
      "integrity": "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA=="
 | 
			
		||||
      "integrity": "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA==",
 | 
			
		||||
      "license": "MIT"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/@types/sanitize-html": {
 | 
			
		||||
      "version": "2.9.0",
 | 
			
		||||
| 
						 | 
				
			
			@ -5088,7 +5093,8 @@
 | 
			
		|||
    "node_modules/another-json": {
 | 
			
		||||
      "version": "0.2.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/another-json/-/another-json-0.2.0.tgz",
 | 
			
		||||
      "integrity": "sha512-/Ndrl68UQLhnCdsAzEXLMFuOR546o2qbYRqCglaNHbjXrwG1ayTcdwr3zkSGOGtGXDyR5X9nCFfnyG2AFJIsqg=="
 | 
			
		||||
      "integrity": "sha512-/Ndrl68UQLhnCdsAzEXLMFuOR546o2qbYRqCglaNHbjXrwG1ayTcdwr3zkSGOGtGXDyR5X9nCFfnyG2AFJIsqg==",
 | 
			
		||||
      "license": "Apache-2.0"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/ansi-regex": {
 | 
			
		||||
      "version": "5.0.1",
 | 
			
		||||
| 
						 | 
				
			
			@ -5431,6 +5437,12 @@
 | 
			
		|||
        "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0"
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/badwords-list": {
 | 
			
		||||
      "version": "2.0.1-4",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/badwords-list/-/badwords-list-2.0.1-4.tgz",
 | 
			
		||||
      "integrity": "sha512-FxfZUp7B9yCnesNtFQS9v6PvZdxTYa14Q60JR6vhjdQdWI4naTjJIyx22JzoER8ooeT8SAAKoHLjKfCV7XgYUQ==",
 | 
			
		||||
      "license": "MIT"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/balanced-match": {
 | 
			
		||||
      "version": "1.0.2",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
 | 
			
		||||
| 
						 | 
				
			
			@ -5438,9 +5450,10 @@
 | 
			
		|||
      "devOptional": true
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/base-x": {
 | 
			
		||||
      "version": "5.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/base-x/-/base-x-5.0.0.tgz",
 | 
			
		||||
      "integrity": "sha512-sMW3VGSX1QWVFA6l8U62MLKz29rRfpTlYdCqLdpLo1/Yd4zZwSbnUaDfciIAowAqvq7YFnWq9hrhdg1KYgc1lQ=="
 | 
			
		||||
      "version": "5.0.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/base-x/-/base-x-5.0.1.tgz",
 | 
			
		||||
      "integrity": "sha512-M7uio8Zt++eg3jPj+rHMfCC+IuygQHHCOU+IYsVtik6FWjuYpVt/+MRKcgsAMHh8mMFAwnB+Bs+mTrFiXjMzKg==",
 | 
			
		||||
      "license": "MIT"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/base64-js": {
 | 
			
		||||
      "version": "1.5.1",
 | 
			
		||||
| 
						 | 
				
			
			@ -5546,6 +5559,7 @@
 | 
			
		|||
      "version": "6.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/bs58/-/bs58-6.0.0.tgz",
 | 
			
		||||
      "integrity": "sha512-PD0wEnEYg6ijszw/u8s+iI3H17cTymlrwkKhDhPZq+Sokl3AU4htyBFTjAeNAlCCmg0f53g6ih3jATyCKftTfw==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "base-x": "^5.0.0"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -5848,6 +5862,7 @@
 | 
			
		|||
      "version": "1.0.5",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
 | 
			
		||||
      "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">= 0.6"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -6999,6 +7014,7 @@
 | 
			
		|||
      "version": "3.3.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz",
 | 
			
		||||
      "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">=0.8.x"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -7249,15 +7265,16 @@
 | 
			
		|||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/folds": {
 | 
			
		||||
      "version": "2.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/folds/-/folds-2.1.0.tgz",
 | 
			
		||||
      "integrity": "sha512-KwAG8bH3jsyZ9FKPMg+6ABV2YOcpp4nL0cCelsalnaPeRThkc5fgG1Xj5mhmdffYKjEXpEbERi5qmGbepgJryg==",
 | 
			
		||||
      "version": "2.2.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/folds/-/folds-2.2.0.tgz",
 | 
			
		||||
      "integrity": "sha512-uOfck5eWEIK11rhOAEdSoPIvMXwv+D1Go03pxSAKezWVb+uRoBdmE6LEqiLOF+ac4DGmZRMPvpdDsXCg7EVNIg==",
 | 
			
		||||
      "license": "Apache-2.0",
 | 
			
		||||
      "peerDependencies": {
 | 
			
		||||
        "@vanilla-extract/css": "^1.9.2",
 | 
			
		||||
        "@vanilla-extract/recipes": "^0.3.0",
 | 
			
		||||
        "classnames": "^2.3.2",
 | 
			
		||||
        "react": "^17.0.0",
 | 
			
		||||
        "react-dom": "^17.0.0"
 | 
			
		||||
        "@vanilla-extract/css": "1.9.2",
 | 
			
		||||
        "@vanilla-extract/recipes": "0.3.0",
 | 
			
		||||
        "classnames": "2.3.2",
 | 
			
		||||
        "react": "17.0.0",
 | 
			
		||||
        "react-dom": "17.0.0"
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/for-each": {
 | 
			
		||||
| 
						 | 
				
			
			@ -8557,6 +8574,7 @@
 | 
			
		|||
      "version": "4.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-4.0.0.tgz",
 | 
			
		||||
      "integrity": "sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">=18"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -8689,6 +8707,7 @@
 | 
			
		|||
      "version": "1.9.2",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.9.2.tgz",
 | 
			
		||||
      "integrity": "sha512-HgMmCqIJSAKqo68l0rS2AanEWfkxaZ5wNiEFb5ggm08lDs9Xl2KxBlX3PTcaD2chBM1gXAYf491/M2Rv8Jwayg==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">= 0.6.0"
 | 
			
		||||
      },
 | 
			
		||||
| 
						 | 
				
			
			@ -8764,21 +8783,23 @@
 | 
			
		|||
    "node_modules/matrix-events-sdk": {
 | 
			
		||||
      "version": "0.0.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/matrix-events-sdk/-/matrix-events-sdk-0.0.1.tgz",
 | 
			
		||||
      "integrity": "sha512-1QEOsXO+bhyCroIe2/A5OwaxHvBm7EsSQ46DEDn8RBIfQwN5HWBpFvyWWR4QY0KHPPnnJdI99wgRiAl7Ad5qaA=="
 | 
			
		||||
      "integrity": "sha512-1QEOsXO+bhyCroIe2/A5OwaxHvBm7EsSQ46DEDn8RBIfQwN5HWBpFvyWWR4QY0KHPPnnJdI99wgRiAl7Ad5qaA==",
 | 
			
		||||
      "license": "Apache-2.0"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/matrix-js-sdk": {
 | 
			
		||||
      "version": "35.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/matrix-js-sdk/-/matrix-js-sdk-35.0.0.tgz",
 | 
			
		||||
      "integrity": "sha512-X8hIsd/8x1SC9vRr8DiNKQxmdrfRujtvEWPz8mY4FxVDJG8HEGDHvqUmaSy2jrtnOUn4oHzGQVLFO3DnhsSf8w==",
 | 
			
		||||
      "version": "37.5.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/matrix-js-sdk/-/matrix-js-sdk-37.5.0.tgz",
 | 
			
		||||
      "integrity": "sha512-5tyuAi5hnKud1UkVq8Z2/3c22hWGELBZzErJPZkE6Hju2uGUfGtrIx6uj6puv0ZjvsUU3X6Qgm8vdReKO1PGig==",
 | 
			
		||||
      "license": "Apache-2.0",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "@babel/runtime": "^7.12.5",
 | 
			
		||||
        "@matrix-org/matrix-sdk-crypto-wasm": "^11.0.0",
 | 
			
		||||
        "@matrix-org/matrix-sdk-crypto-wasm": "^14.0.1",
 | 
			
		||||
        "@matrix-org/olm": "3.2.15",
 | 
			
		||||
        "another-json": "^0.2.0",
 | 
			
		||||
        "bs58": "^6.0.0",
 | 
			
		||||
        "content-type": "^1.0.4",
 | 
			
		||||
        "jwt-decode": "^4.0.0",
 | 
			
		||||
        "loglevel": "^1.7.1",
 | 
			
		||||
        "loglevel": "^1.9.2",
 | 
			
		||||
        "matrix-events-sdk": "0.0.1",
 | 
			
		||||
        "matrix-widget-api": "^1.10.0",
 | 
			
		||||
        "oidc-client-ts": "^3.0.1",
 | 
			
		||||
| 
						 | 
				
			
			@ -8792,21 +8813,23 @@
 | 
			
		|||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/matrix-js-sdk/node_modules/uuid": {
 | 
			
		||||
      "version": "11.0.5",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.5.tgz",
 | 
			
		||||
      "integrity": "sha512-508e6IcKLrhxKdBbcA2b4KQZlLVp2+J5UwQ6F7Drckkc5N9ZJwFa4TgWtsww9UG8fGHbm6gbV19TdM5pQ4GaIA==",
 | 
			
		||||
      "version": "11.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.1.0.tgz",
 | 
			
		||||
      "integrity": "sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A==",
 | 
			
		||||
      "funding": [
 | 
			
		||||
        "https://github.com/sponsors/broofa",
 | 
			
		||||
        "https://github.com/sponsors/ctavan"
 | 
			
		||||
      ],
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "bin": {
 | 
			
		||||
        "uuid": "dist/esm/bin/uuid"
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/matrix-widget-api": {
 | 
			
		||||
      "version": "1.12.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/matrix-widget-api/-/matrix-widget-api-1.12.0.tgz",
 | 
			
		||||
      "integrity": "sha512-6JRd9fJGGvuBRhcTg9wX+Skn/Q1wox3jdp5yYQKJ6pPw4urW9bkTR90APBKVDB1vorJKT44jml+lCzkDMRBjww==",
 | 
			
		||||
      "version": "1.13.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/matrix-widget-api/-/matrix-widget-api-1.13.1.tgz",
 | 
			
		||||
      "integrity": "sha512-mkOHUVzaN018TCbObfGOSaMW2GoUxOfcxNNlTVx5/HeMk3OSQPQM0C9oEME5Liiv/dBUoSrEB64V8wF7e/gb1w==",
 | 
			
		||||
      "license": "Apache-2.0",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "@types/events": "^3.0.0",
 | 
			
		||||
        "events": "^3.2.0"
 | 
			
		||||
| 
						 | 
				
			
			@ -9198,9 +9221,10 @@
 | 
			
		|||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/oidc-client-ts": {
 | 
			
		||||
      "version": "3.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/oidc-client-ts/-/oidc-client-ts-3.1.0.tgz",
 | 
			
		||||
      "integrity": "sha512-IDopEXjiwjkmJLYZo6BTlvwOtnlSniWZkKZoXforC/oLZHC9wkIxd25Kwtmo5yKFMMVcsp3JY6bhcNJqdYk8+g==",
 | 
			
		||||
      "version": "3.2.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/oidc-client-ts/-/oidc-client-ts-3.2.1.tgz",
 | 
			
		||||
      "integrity": "sha512-hS5AJ5s/x4bXhHvNJT1v+GGvzHUwdRWqNQQbSrp10L1IRmzfRGKQ3VWN3dstJb+oF3WtAyKezwD2+dTEIyBiAA==",
 | 
			
		||||
      "license": "Apache-2.0",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "jwt-decode": "^4.0.0"
 | 
			
		||||
      },
 | 
			
		||||
| 
						 | 
				
			
			@ -9288,6 +9312,7 @@
 | 
			
		|||
      "version": "4.6.2",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-4.6.2.tgz",
 | 
			
		||||
      "integrity": "sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "@types/retry": "0.12.0",
 | 
			
		||||
        "retry": "^0.13.1"
 | 
			
		||||
| 
						 | 
				
			
			@ -10051,6 +10076,7 @@
 | 
			
		|||
      "version": "0.13.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/retry/-/retry-0.13.1.tgz",
 | 
			
		||||
      "integrity": "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "engines": {
 | 
			
		||||
        "node": ">= 4"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -10264,6 +10290,7 @@
 | 
			
		|||
      "version": "2.15.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/sdp-transform/-/sdp-transform-2.15.0.tgz",
 | 
			
		||||
      "integrity": "sha512-KrOH82c/W+GYQ0LHqtr3caRpM3ITglq3ljGUIb8LTki7ByacJZ9z+piSGiwZDsRyhQbYBOBJgr2k6X4BZXi3Kw==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "bin": {
 | 
			
		||||
        "sdp-verify": "checker.js"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			@ -11172,7 +11199,8 @@
 | 
			
		|||
    "node_modules/unhomoglyph": {
 | 
			
		||||
      "version": "1.0.6",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/unhomoglyph/-/unhomoglyph-1.0.6.tgz",
 | 
			
		||||
      "integrity": "sha512-7uvcWI3hWshSADBu4JpnyYbTVc7YlhF5GDW/oPD5AxIxl34k4wXR3WDkPnzLxkN32LiTCTKMQLtKVZiwki3zGg=="
 | 
			
		||||
      "integrity": "sha512-7uvcWI3hWshSADBu4JpnyYbTVc7YlhF5GDW/oPD5AxIxl34k4wXR3WDkPnzLxkN32LiTCTKMQLtKVZiwki3zGg==",
 | 
			
		||||
      "license": "MIT"
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/unicode-canonical-property-names-ecmascript": {
 | 
			
		||||
      "version": "2.0.1",
 | 
			
		||||
| 
						 | 
				
			
			@ -11303,9 +11331,9 @@
 | 
			
		|||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/vite": {
 | 
			
		||||
      "version": "5.4.15",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.15.tgz",
 | 
			
		||||
      "integrity": "sha512-6ANcZRivqL/4WtwPGTKNaosuNJr5tWiftOC7liM7G9+rMb8+oeJeyzymDu4rTN93seySBmbjSfsS3Vzr19KNtA==",
 | 
			
		||||
      "version": "5.4.19",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.19.tgz",
 | 
			
		||||
      "integrity": "sha512-qO3aKv3HoQC8QKiNSTuUM1l9o/XX3+c+VTgLHbJWHZGeTPVAg2XwazI9UWzoxjIJCGCV2zU60uqMzjeLZuULqA==",
 | 
			
		||||
      "license": "MIT",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "esbuild": "^0.21.3",
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,6 +1,6 @@
 | 
			
		|||
{
 | 
			
		||||
  "name": "cinny",
 | 
			
		||||
  "version": "4.6.0",
 | 
			
		||||
  "version": "4.9.1",
 | 
			
		||||
  "description": "Yet another matrix client",
 | 
			
		||||
  "main": "index.js",
 | 
			
		||||
  "type": "module",
 | 
			
		||||
| 
						 | 
				
			
			@ -32,6 +32,7 @@
 | 
			
		|||
    "@vanilla-extract/recipes": "0.3.0",
 | 
			
		||||
    "@vanilla-extract/vite-plugin": "3.7.1",
 | 
			
		||||
    "await-to-js": "3.0.0",
 | 
			
		||||
    "badwords-list": "2.0.1-4",
 | 
			
		||||
    "blurhash": "2.0.4",
 | 
			
		||||
    "browser-encrypt-attachment": "0.3.0",
 | 
			
		||||
    "chroma-js": "3.1.2",
 | 
			
		||||
| 
						 | 
				
			
			@ -44,7 +45,7 @@
 | 
			
		|||
    "file-saver": "2.0.5",
 | 
			
		||||
    "flux": "4.0.3",
 | 
			
		||||
    "focus-trap-react": "10.0.2",
 | 
			
		||||
    "folds": "2.1.0",
 | 
			
		||||
    "folds": "2.2.0",
 | 
			
		||||
    "formik": "2.4.6",
 | 
			
		||||
    "html-dom-parser": "4.0.0",
 | 
			
		||||
    "html-react-parser": "4.2.0",
 | 
			
		||||
| 
						 | 
				
			
			@ -56,7 +57,7 @@
 | 
			
		|||
    "jotai": "2.6.0",
 | 
			
		||||
    "linkify-react": "4.1.3",
 | 
			
		||||
    "linkifyjs": "4.1.3",
 | 
			
		||||
    "matrix-js-sdk": "35.0.0",
 | 
			
		||||
    "matrix-js-sdk": "37.5.0",
 | 
			
		||||
    "millify": "6.1.0",
 | 
			
		||||
    "pdfjs-dist": "4.2.67",
 | 
			
		||||
    "prismjs": "1.30.0",
 | 
			
		||||
| 
						 | 
				
			
			@ -109,7 +110,7 @@
 | 
			
		|||
    "prettier": "2.8.1",
 | 
			
		||||
    "sass": "1.56.2",
 | 
			
		||||
    "typescript": "4.9.4",
 | 
			
		||||
    "vite": "5.4.15",
 | 
			
		||||
    "vite": "5.4.19",
 | 
			
		||||
    "vite-plugin-pwa": "0.20.5",
 | 
			
		||||
    "vite-plugin-static-copy": "1.0.4",
 | 
			
		||||
    "vite-plugin-top-level-await": "1.4.4"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,10 +4,25 @@ import PropTypes from 'prop-types';
 | 
			
		|||
import dateFormat from 'dateformat';
 | 
			
		||||
import { isInSameDay } from '../../../util/common';
 | 
			
		||||
 | 
			
		||||
function Time({ timestamp, fullTime }) {
 | 
			
		||||
/**
 | 
			
		||||
 * Renders a formatted timestamp.
 | 
			
		||||
 *
 | 
			
		||||
 * Displays the time in hour:minute format if the message is from today or yesterday, unless `fullTime` is true.
 | 
			
		||||
 * For older messages, it shows the date and time.
 | 
			
		||||
 *
 | 
			
		||||
 * @param {number} timestamp - The timestamp to display.
 | 
			
		||||
 * @param {boolean} [fullTime=false] - If true, always show the full date and time.
 | 
			
		||||
 * @param {boolean} hour24Clock - Whether to use 24-hour time format.
 | 
			
		||||
 * @param {string} dateFormatString - Format string for the date part.
 | 
			
		||||
 * @returns {JSX.Element} A <time> element with the formatted date/time.
 | 
			
		||||
 */
 | 
			
		||||
function Time({ timestamp, fullTime, hour24Clock, dateFormatString }) {
 | 
			
		||||
  const date = new Date(timestamp);
 | 
			
		||||
 | 
			
		||||
  const formattedFullTime = dateFormat(date, 'dd mmmm yyyy, hh:MM TT');
 | 
			
		||||
  const formattedFullTime = dateFormat(
 | 
			
		||||
    date,
 | 
			
		||||
    hour24Clock ? 'dd mmmm yyyy, HH:MM' : 'dd mmmm yyyy, hh:MM TT'
 | 
			
		||||
  );
 | 
			
		||||
  let formattedDate = formattedFullTime;
 | 
			
		||||
 | 
			
		||||
  if (!fullTime) {
 | 
			
		||||
| 
						 | 
				
			
			@ -16,17 +31,19 @@ function Time({ timestamp, fullTime }) {
 | 
			
		|||
    compareDate.setDate(compareDate.getDate() - 1);
 | 
			
		||||
    const isYesterday = isInSameDay(date, compareDate);
 | 
			
		||||
 | 
			
		||||
    formattedDate = dateFormat(date, isToday || isYesterday ? 'hh:MM TT' : 'dd/mm/yyyy');
 | 
			
		||||
    const timeFormat = hour24Clock ? 'HH:MM' : 'hh:MM TT';
 | 
			
		||||
 | 
			
		||||
    formattedDate = dateFormat(
 | 
			
		||||
      date,
 | 
			
		||||
      isToday || isYesterday ? timeFormat : dateFormatString.toLowerCase()
 | 
			
		||||
    );
 | 
			
		||||
    if (isYesterday) {
 | 
			
		||||
      formattedDate = `Yesterday, ${formattedDate}`;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <time
 | 
			
		||||
      dateTime={date.toISOString()}
 | 
			
		||||
      title={formattedFullTime}
 | 
			
		||||
    >
 | 
			
		||||
    <time dateTime={date.toISOString()} title={formattedFullTime}>
 | 
			
		||||
      {formattedDate}
 | 
			
		||||
    </time>
 | 
			
		||||
  );
 | 
			
		||||
| 
						 | 
				
			
			@ -39,6 +56,8 @@ Time.defaultProps = {
 | 
			
		|||
Time.propTypes = {
 | 
			
		||||
  timestamp: PropTypes.number.isRequired,
 | 
			
		||||
  fullTime: PropTypes.bool,
 | 
			
		||||
  hour24Clock: PropTypes.bool.isRequired,
 | 
			
		||||
  dateFormatString: PropTypes.string.isRequired,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export default Time;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,36 +0,0 @@
 | 
			
		|||
import { ReactNode, useCallback, useEffect } from 'react';
 | 
			
		||||
import { Capabilities } from 'matrix-js-sdk';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../hooks/useAsyncCallback';
 | 
			
		||||
import { useMatrixClient } from '../hooks/useMatrixClient';
 | 
			
		||||
import { MediaConfig } from '../hooks/useMediaConfig';
 | 
			
		||||
import { promiseFulfilledResult } from '../utils/common';
 | 
			
		||||
 | 
			
		||||
type CapabilitiesAndMediaConfigLoaderProps = {
 | 
			
		||||
  children: (capabilities?: Capabilities, mediaConfig?: MediaConfig) => ReactNode;
 | 
			
		||||
};
 | 
			
		||||
export function CapabilitiesAndMediaConfigLoader({
 | 
			
		||||
  children,
 | 
			
		||||
}: CapabilitiesAndMediaConfigLoaderProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
 | 
			
		||||
  const [state, load] = useAsyncCallback<
 | 
			
		||||
    [Capabilities | undefined, MediaConfig | undefined],
 | 
			
		||||
    unknown,
 | 
			
		||||
    []
 | 
			
		||||
  >(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      const result = await Promise.allSettled([mx.getCapabilities(), mx.getMediaConfig()]);
 | 
			
		||||
      const capabilities = promiseFulfilledResult(result[0]);
 | 
			
		||||
      const mediaConfig = promiseFulfilledResult(result[1]);
 | 
			
		||||
      return [capabilities, mediaConfig];
 | 
			
		||||
    }, [mx])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    load();
 | 
			
		||||
  }, [load]);
 | 
			
		||||
 | 
			
		||||
  const [capabilities, mediaConfig] =
 | 
			
		||||
    state.status === AsyncStatus.Success ? state.data : [undefined, undefined];
 | 
			
		||||
  return children(capabilities, mediaConfig);
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -17,12 +17,16 @@ import { JoinRule } from 'matrix-js-sdk';
 | 
			
		|||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { stopPropagation } from '../utils/keyboard';
 | 
			
		||||
 | 
			
		||||
type JoinRuleIcons = Record<JoinRule, IconSrc>;
 | 
			
		||||
export type ExtraJoinRules = 'knock_restricted';
 | 
			
		||||
export type ExtendedJoinRules = JoinRule | ExtraJoinRules;
 | 
			
		||||
 | 
			
		||||
type JoinRuleIcons = Record<ExtendedJoinRules, IconSrc>;
 | 
			
		||||
export const useRoomJoinRuleIcon = (): JoinRuleIcons =>
 | 
			
		||||
  useMemo(
 | 
			
		||||
    () => ({
 | 
			
		||||
      [JoinRule.Invite]: Icons.HashLock,
 | 
			
		||||
      [JoinRule.Knock]: Icons.HashLock,
 | 
			
		||||
      knock_restricted: Icons.Hash,
 | 
			
		||||
      [JoinRule.Restricted]: Icons.Hash,
 | 
			
		||||
      [JoinRule.Public]: Icons.HashGlobe,
 | 
			
		||||
      [JoinRule.Private]: Icons.HashLock,
 | 
			
		||||
| 
						 | 
				
			
			@ -34,6 +38,7 @@ export const useSpaceJoinRuleIcon = (): JoinRuleIcons =>
 | 
			
		|||
    () => ({
 | 
			
		||||
      [JoinRule.Invite]: Icons.SpaceLock,
 | 
			
		||||
      [JoinRule.Knock]: Icons.SpaceLock,
 | 
			
		||||
      knock_restricted: Icons.Space,
 | 
			
		||||
      [JoinRule.Restricted]: Icons.Space,
 | 
			
		||||
      [JoinRule.Public]: Icons.SpaceGlobe,
 | 
			
		||||
      [JoinRule.Private]: Icons.SpaceLock,
 | 
			
		||||
| 
						 | 
				
			
			@ -41,12 +46,13 @@ export const useSpaceJoinRuleIcon = (): JoinRuleIcons =>
 | 
			
		|||
    []
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
type JoinRuleLabels = Record<JoinRule, string>;
 | 
			
		||||
type JoinRuleLabels = Record<ExtendedJoinRules, string>;
 | 
			
		||||
export const useRoomJoinRuleLabel = (): JoinRuleLabels =>
 | 
			
		||||
  useMemo(
 | 
			
		||||
    () => ({
 | 
			
		||||
      [JoinRule.Invite]: 'Invite Only',
 | 
			
		||||
      [JoinRule.Knock]: 'Knock & Invite',
 | 
			
		||||
      knock_restricted: 'Space Members or Knock',
 | 
			
		||||
      [JoinRule.Restricted]: 'Space Members',
 | 
			
		||||
      [JoinRule.Public]: 'Public',
 | 
			
		||||
      [JoinRule.Private]: 'Invite Only',
 | 
			
		||||
| 
						 | 
				
			
			@ -54,7 +60,7 @@ export const useRoomJoinRuleLabel = (): JoinRuleLabels =>
 | 
			
		|||
    []
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
type JoinRulesSwitcherProps<T extends JoinRule[]> = {
 | 
			
		||||
type JoinRulesSwitcherProps<T extends ExtendedJoinRules[]> = {
 | 
			
		||||
  icons: JoinRuleIcons;
 | 
			
		||||
  labels: JoinRuleLabels;
 | 
			
		||||
  rules: T;
 | 
			
		||||
| 
						 | 
				
			
			@ -63,7 +69,7 @@ type JoinRulesSwitcherProps<T extends JoinRule[]> = {
 | 
			
		|||
  disabled?: boolean;
 | 
			
		||||
  changing?: boolean;
 | 
			
		||||
};
 | 
			
		||||
export function JoinRulesSwitcher<T extends JoinRule[]>({
 | 
			
		||||
export function JoinRulesSwitcher<T extends ExtendedJoinRules[]>({
 | 
			
		||||
  icons,
 | 
			
		||||
  labels,
 | 
			
		||||
  rules,
 | 
			
		||||
| 
						 | 
				
			
			@ -79,7 +85,7 @@ export function JoinRulesSwitcher<T extends JoinRule[]>({
 | 
			
		|||
  };
 | 
			
		||||
 | 
			
		||||
  const handleChange = useCallback(
 | 
			
		||||
    (selectedRule: JoinRule) => {
 | 
			
		||||
    (selectedRule: ExtendedJoinRules) => {
 | 
			
		||||
      setCords(undefined);
 | 
			
		||||
      onChange(selectedRule);
 | 
			
		||||
    },
 | 
			
		||||
| 
						 | 
				
			
			@ -131,7 +137,7 @@ export function JoinRulesSwitcher<T extends JoinRule[]>({
 | 
			
		|||
        fill="Soft"
 | 
			
		||||
        radii="300"
 | 
			
		||||
        outlined
 | 
			
		||||
        before={<Icon size="100" src={icons[value]} />}
 | 
			
		||||
        before={<Icon size="100" src={icons[value] ?? icons[JoinRule.Restricted]} />}
 | 
			
		||||
        after={
 | 
			
		||||
          changing ? (
 | 
			
		||||
            <Spinner size="100" variant="Secondary" fill="Soft" />
 | 
			
		||||
| 
						 | 
				
			
			@ -142,7 +148,7 @@ export function JoinRulesSwitcher<T extends JoinRule[]>({
 | 
			
		|||
        onClick={handleOpenMenu}
 | 
			
		||||
        disabled={disabled}
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300">{labels[value]}</Text>
 | 
			
		||||
        <Text size="B300">{labels[value] ?? 'Unsupported'}</Text>
 | 
			
		||||
      </Button>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,7 +1,6 @@
 | 
			
		|||
import React, { FormEventHandler, useCallback } from 'react';
 | 
			
		||||
import { Box, Text, Button, Spinner, color } from 'folds';
 | 
			
		||||
import { decodeRecoveryKey } from 'matrix-js-sdk/lib/crypto-api';
 | 
			
		||||
import { deriveKey } from 'matrix-js-sdk/lib/crypto/key_passphrase';
 | 
			
		||||
import { decodeRecoveryKey, deriveRecoveryKeyFromPassphrase } from 'matrix-js-sdk/lib/crypto-api';
 | 
			
		||||
import { PasswordInput } from './password-input';
 | 
			
		||||
import {
 | 
			
		||||
  SecretStorageKeyContent,
 | 
			
		||||
| 
						 | 
				
			
			@ -29,11 +28,16 @@ export function SecretStorageRecoveryPassphrase({
 | 
			
		|||
  const [driveKeyState, submitPassphrase] = useAsyncCallback<
 | 
			
		||||
    Uint8Array,
 | 
			
		||||
    Error,
 | 
			
		||||
    Parameters<typeof deriveKey>
 | 
			
		||||
    Parameters<typeof deriveRecoveryKeyFromPassphrase>
 | 
			
		||||
  >(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (passphrase, salt, iterations, bits) => {
 | 
			
		||||
        const decodedRecoveryKey = await deriveKey(passphrase, salt, iterations, bits);
 | 
			
		||||
        const decodedRecoveryKey = await deriveRecoveryKeyFromPassphrase(
 | 
			
		||||
          passphrase,
 | 
			
		||||
          salt,
 | 
			
		||||
          iterations,
 | 
			
		||||
          bits
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        const match = await mx.secretStorage.checkKey(decodedRecoveryKey, keyContent as any);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										52
									
								
								src/app/components/ServerConfigsLoader.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/app/components/ServerConfigsLoader.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,52 @@
 | 
			
		|||
import { ReactNode, useCallback, useMemo } from 'react';
 | 
			
		||||
import { Capabilities, validateAuthMetadata, ValidatedAuthMetadata } from 'matrix-js-sdk';
 | 
			
		||||
import { AsyncStatus, useAsyncCallbackValue } from '../hooks/useAsyncCallback';
 | 
			
		||||
import { useMatrixClient } from '../hooks/useMatrixClient';
 | 
			
		||||
import { MediaConfig } from '../hooks/useMediaConfig';
 | 
			
		||||
import { promiseFulfilledResult } from '../utils/common';
 | 
			
		||||
 | 
			
		||||
export type ServerConfigs = {
 | 
			
		||||
  capabilities?: Capabilities;
 | 
			
		||||
  mediaConfig?: MediaConfig;
 | 
			
		||||
  authMetadata?: ValidatedAuthMetadata;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
type ServerConfigsLoaderProps = {
 | 
			
		||||
  children: (configs: ServerConfigs) => ReactNode;
 | 
			
		||||
};
 | 
			
		||||
export function ServerConfigsLoader({ children }: ServerConfigsLoaderProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const fallbackConfigs = useMemo(() => ({}), []);
 | 
			
		||||
 | 
			
		||||
  const [configsState] = useAsyncCallbackValue<ServerConfigs, unknown>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      const result = await Promise.allSettled([
 | 
			
		||||
        mx.getCapabilities(),
 | 
			
		||||
        mx.getMediaConfig(),
 | 
			
		||||
        mx.getAuthMetadata(),
 | 
			
		||||
      ]);
 | 
			
		||||
 | 
			
		||||
      const capabilities = promiseFulfilledResult(result[0]);
 | 
			
		||||
      const mediaConfig = promiseFulfilledResult(result[1]);
 | 
			
		||||
      const authMetadata = promiseFulfilledResult(result[2]);
 | 
			
		||||
      let validatedAuthMetadata: ValidatedAuthMetadata | undefined;
 | 
			
		||||
 | 
			
		||||
      try {
 | 
			
		||||
        validatedAuthMetadata = validateAuthMetadata(authMetadata);
 | 
			
		||||
      } catch (e) {
 | 
			
		||||
        console.error(e);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return {
 | 
			
		||||
        capabilities,
 | 
			
		||||
        mediaConfig,
 | 
			
		||||
        authMetadata: validatedAuthMetadata,
 | 
			
		||||
      };
 | 
			
		||||
    }, [mx])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const configs: ServerConfigs =
 | 
			
		||||
    configsState.status === AsyncStatus.Success ? configsState.data : fallbackConfigs;
 | 
			
		||||
 | 
			
		||||
  return children(configs);
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										55
									
								
								src/app/components/UserRoomProfileRenderer.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								src/app/components/UserRoomProfileRenderer.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,55 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import { Menu, PopOut, toRem } from 'folds';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { useCloseUserRoomProfile, useUserRoomProfileState } from '../state/hooks/userRoomProfile';
 | 
			
		||||
import { UserRoomProfile } from './user-profile';
 | 
			
		||||
import { UserRoomProfileState } from '../state/userRoomProfile';
 | 
			
		||||
import { useAllJoinedRoomsSet, useGetRoom } from '../hooks/useGetRoom';
 | 
			
		||||
import { stopPropagation } from '../utils/keyboard';
 | 
			
		||||
import { SpaceProvider } from '../hooks/useSpace';
 | 
			
		||||
import { RoomProvider } from '../hooks/useRoom';
 | 
			
		||||
 | 
			
		||||
function UserRoomProfileContextMenu({ state }: { state: UserRoomProfileState }) {
 | 
			
		||||
  const { roomId, spaceId, userId, cords, position } = state;
 | 
			
		||||
  const allJoinedRooms = useAllJoinedRoomsSet();
 | 
			
		||||
  const getRoom = useGetRoom(allJoinedRooms);
 | 
			
		||||
  const room = getRoom(roomId);
 | 
			
		||||
  const space = spaceId ? getRoom(spaceId) : undefined;
 | 
			
		||||
 | 
			
		||||
  const close = useCloseUserRoomProfile();
 | 
			
		||||
 | 
			
		||||
  if (!room) return null;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position={position ?? 'Top'}
 | 
			
		||||
      align="Start"
 | 
			
		||||
      content={
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: close,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Menu style={{ width: toRem(340) }}>
 | 
			
		||||
            <SpaceProvider value={space ?? null}>
 | 
			
		||||
              <RoomProvider value={room}>
 | 
			
		||||
                <UserRoomProfile userId={userId} />
 | 
			
		||||
              </RoomProvider>
 | 
			
		||||
            </SpaceProvider>
 | 
			
		||||
          </Menu>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      }
 | 
			
		||||
    />
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function UserRoomProfileRenderer() {
 | 
			
		||||
  const state = useUserRoomProfileState();
 | 
			
		||||
 | 
			
		||||
  if (!state) return null;
 | 
			
		||||
  return <UserRoomProfileContextMenu state={state} />;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										294
									
								
								src/app/components/create-room/AdditionalCreatorInput.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										294
									
								
								src/app/components/create-room/AdditionalCreatorInput.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,294 @@
 | 
			
		|||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  Chip,
 | 
			
		||||
  config,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Input,
 | 
			
		||||
  Line,
 | 
			
		||||
  Menu,
 | 
			
		||||
  MenuItem,
 | 
			
		||||
  PopOut,
 | 
			
		||||
  RectCords,
 | 
			
		||||
  Scroll,
 | 
			
		||||
  Text,
 | 
			
		||||
  toRem,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import React, {
 | 
			
		||||
  ChangeEventHandler,
 | 
			
		||||
  KeyboardEventHandler,
 | 
			
		||||
  MouseEventHandler,
 | 
			
		||||
  useMemo,
 | 
			
		||||
  useState,
 | 
			
		||||
} from 'react';
 | 
			
		||||
import { getMxIdLocalPart, getMxIdServer, isUserId } from '../../utils/matrix';
 | 
			
		||||
import { useDirectUsers } from '../../hooks/useDirectUsers';
 | 
			
		||||
import { SettingTile } from '../setting-tile';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { useAsyncSearch, UseAsyncSearchOptions } from '../../hooks/useAsyncSearch';
 | 
			
		||||
import { highlightText, makeHighlightRegex } from '../../plugins/react-custom-html-parser';
 | 
			
		||||
 | 
			
		||||
export const useAdditionalCreators = (defaultCreators?: string[]) => {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const [additionalCreators, setAdditionalCreators] = useState<string[]>(
 | 
			
		||||
    () => defaultCreators?.filter((id) => id !== mx.getSafeUserId()) ?? []
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const addAdditionalCreator = (userId: string) => {
 | 
			
		||||
    if (userId === mx.getSafeUserId()) return;
 | 
			
		||||
 | 
			
		||||
    setAdditionalCreators((creators) => {
 | 
			
		||||
      const creatorsSet = new Set(creators);
 | 
			
		||||
      creatorsSet.add(userId);
 | 
			
		||||
      return Array.from(creatorsSet);
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const removeAdditionalCreator = (userId: string) => {
 | 
			
		||||
    setAdditionalCreators((creators) => {
 | 
			
		||||
      const creatorsSet = new Set(creators);
 | 
			
		||||
      creatorsSet.delete(userId);
 | 
			
		||||
      return Array.from(creatorsSet);
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    additionalCreators,
 | 
			
		||||
    addAdditionalCreator,
 | 
			
		||||
    removeAdditionalCreator,
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const SEARCH_OPTIONS: UseAsyncSearchOptions = {
 | 
			
		||||
  limit: 1000,
 | 
			
		||||
  matchOptions: {
 | 
			
		||||
    contain: true,
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
const getUserIdString = (userId: string) => getMxIdLocalPart(userId) ?? userId;
 | 
			
		||||
 | 
			
		||||
type AdditionalCreatorInputProps = {
 | 
			
		||||
  additionalCreators: string[];
 | 
			
		||||
  onSelect: (userId: string) => void;
 | 
			
		||||
  onRemove: (userId: string) => void;
 | 
			
		||||
  disabled?: boolean;
 | 
			
		||||
};
 | 
			
		||||
export function AdditionalCreatorInput({
 | 
			
		||||
  additionalCreators,
 | 
			
		||||
  onSelect,
 | 
			
		||||
  onRemove,
 | 
			
		||||
  disabled,
 | 
			
		||||
}: AdditionalCreatorInputProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const [menuCords, setMenuCords] = useState<RectCords>();
 | 
			
		||||
  const directUsers = useDirectUsers();
 | 
			
		||||
 | 
			
		||||
  const [validUserId, setValidUserId] = useState<string>();
 | 
			
		||||
  const filteredUsers = useMemo(
 | 
			
		||||
    () => directUsers.filter((userId) => !additionalCreators.includes(userId)),
 | 
			
		||||
    [directUsers, additionalCreators]
 | 
			
		||||
  );
 | 
			
		||||
  const [result, search, resetSearch] = useAsyncSearch(
 | 
			
		||||
    filteredUsers,
 | 
			
		||||
    getUserIdString,
 | 
			
		||||
    SEARCH_OPTIONS
 | 
			
		||||
  );
 | 
			
		||||
  const queryHighlighRegex = result?.query ? makeHighlightRegex([result.query]) : undefined;
 | 
			
		||||
 | 
			
		||||
  const suggestionUsers = result
 | 
			
		||||
    ? result.items
 | 
			
		||||
    : filteredUsers.sort((a, b) => (a.toLocaleLowerCase() >= b.toLocaleLowerCase() ? 1 : -1));
 | 
			
		||||
 | 
			
		||||
  const handleOpenMenu: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setMenuCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
  const handleCloseMenu = () => {
 | 
			
		||||
    setMenuCords(undefined);
 | 
			
		||||
    setValidUserId(undefined);
 | 
			
		||||
    resetSearch();
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleCreatorChange: ChangeEventHandler<HTMLInputElement> = (evt) => {
 | 
			
		||||
    const creatorInput = evt.currentTarget;
 | 
			
		||||
    const creator = creatorInput.value.trim();
 | 
			
		||||
    if (isUserId(creator)) {
 | 
			
		||||
      setValidUserId(creator);
 | 
			
		||||
    } else {
 | 
			
		||||
      setValidUserId(undefined);
 | 
			
		||||
      const term =
 | 
			
		||||
        getMxIdLocalPart(creator) ?? (creator.startsWith('@') ? creator.slice(1) : creator);
 | 
			
		||||
      if (term) {
 | 
			
		||||
        search(term);
 | 
			
		||||
      } else {
 | 
			
		||||
        resetSearch();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleSelectUserId = (userId?: string) => {
 | 
			
		||||
    if (userId && isUserId(userId)) {
 | 
			
		||||
      onSelect(userId);
 | 
			
		||||
      handleCloseMenu();
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleCreatorKeyDown: KeyboardEventHandler<HTMLInputElement> = (evt) => {
 | 
			
		||||
    if (isKeyHotkey('enter', evt)) {
 | 
			
		||||
      evt.preventDefault();
 | 
			
		||||
      const creator = evt.currentTarget.value.trim();
 | 
			
		||||
      handleSelectUserId(isUserId(creator) ? creator : suggestionUsers[0]);
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleEnterClick = () => {
 | 
			
		||||
    handleSelectUserId(validUserId);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SettingTile
 | 
			
		||||
      title="Founders"
 | 
			
		||||
      description="Special privileged users can be assigned during creation. These users have elevated control and can only be modified during a upgrade."
 | 
			
		||||
    >
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Box gap="200" wrap="Wrap">
 | 
			
		||||
          <Chip type="button" variant="Primary" radii="Pill" outlined>
 | 
			
		||||
            <Text size="B300">{mx.getSafeUserId()}</Text>
 | 
			
		||||
          </Chip>
 | 
			
		||||
          {additionalCreators.map((creator) => (
 | 
			
		||||
            <Chip
 | 
			
		||||
              type="button"
 | 
			
		||||
              key={creator}
 | 
			
		||||
              variant="Secondary"
 | 
			
		||||
              radii="Pill"
 | 
			
		||||
              after={<Icon size="50" src={Icons.Cross} />}
 | 
			
		||||
              onClick={() => onRemove(creator)}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">{creator}</Text>
 | 
			
		||||
            </Chip>
 | 
			
		||||
          ))}
 | 
			
		||||
          <PopOut
 | 
			
		||||
            anchor={menuCords}
 | 
			
		||||
            position="Bottom"
 | 
			
		||||
            align="Center"
 | 
			
		||||
            content={
 | 
			
		||||
              <FocusTrap
 | 
			
		||||
                focusTrapOptions={{
 | 
			
		||||
                  onDeactivate: handleCloseMenu,
 | 
			
		||||
                  clickOutsideDeactivates: true,
 | 
			
		||||
                  isKeyForward: (evt: KeyboardEvent) => evt.key === 'ArrowDown',
 | 
			
		||||
                  isKeyBackward: (evt: KeyboardEvent) => evt.key === 'ArrowUp',
 | 
			
		||||
                  escapeDeactivates: stopPropagation,
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Menu
 | 
			
		||||
                  style={{
 | 
			
		||||
                    width: '100vw',
 | 
			
		||||
                    maxWidth: toRem(300),
 | 
			
		||||
                    height: toRem(250),
 | 
			
		||||
                    display: 'flex',
 | 
			
		||||
                  }}
 | 
			
		||||
                >
 | 
			
		||||
                  <Box grow="Yes" direction="Column">
 | 
			
		||||
                    <Box shrink="No" gap="100" style={{ padding: config.space.S100 }}>
 | 
			
		||||
                      <Box grow="Yes" direction="Column" gap="100">
 | 
			
		||||
                        <Input
 | 
			
		||||
                          size="400"
 | 
			
		||||
                          variant="Background"
 | 
			
		||||
                          radii="300"
 | 
			
		||||
                          outlined
 | 
			
		||||
                          placeholder="@john:server"
 | 
			
		||||
                          onChange={handleCreatorChange}
 | 
			
		||||
                          onKeyDown={handleCreatorKeyDown}
 | 
			
		||||
                        />
 | 
			
		||||
                      </Box>
 | 
			
		||||
                      <Button
 | 
			
		||||
                        type="button"
 | 
			
		||||
                        variant="Success"
 | 
			
		||||
                        radii="300"
 | 
			
		||||
                        onClick={handleEnterClick}
 | 
			
		||||
                        disabled={!validUserId}
 | 
			
		||||
                      >
 | 
			
		||||
                        <Text size="B400">Enter</Text>
 | 
			
		||||
                      </Button>
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    <Line size="300" />
 | 
			
		||||
                    <Box grow="Yes" direction="Column">
 | 
			
		||||
                      {!validUserId && suggestionUsers.length > 0 ? (
 | 
			
		||||
                        <Scroll size="300" hideTrack>
 | 
			
		||||
                          <Box
 | 
			
		||||
                            grow="Yes"
 | 
			
		||||
                            direction="Column"
 | 
			
		||||
                            gap="100"
 | 
			
		||||
                            style={{ padding: config.space.S200, paddingRight: 0 }}
 | 
			
		||||
                          >
 | 
			
		||||
                            {suggestionUsers.map((userId) => (
 | 
			
		||||
                              <MenuItem
 | 
			
		||||
                                key={userId}
 | 
			
		||||
                                size="300"
 | 
			
		||||
                                variant="Surface"
 | 
			
		||||
                                radii="300"
 | 
			
		||||
                                onClick={() => handleSelectUserId(userId)}
 | 
			
		||||
                                after={
 | 
			
		||||
                                  <Text size="T200" truncate>
 | 
			
		||||
                                    {getMxIdServer(userId)}
 | 
			
		||||
                                  </Text>
 | 
			
		||||
                                }
 | 
			
		||||
                              >
 | 
			
		||||
                                <Box grow="Yes">
 | 
			
		||||
                                  <Text size="T200" truncate>
 | 
			
		||||
                                    <b>
 | 
			
		||||
                                      {queryHighlighRegex
 | 
			
		||||
                                        ? highlightText(queryHighlighRegex, [
 | 
			
		||||
                                            getMxIdLocalPart(userId) ?? userId,
 | 
			
		||||
                                          ])
 | 
			
		||||
                                        : getMxIdLocalPart(userId)}
 | 
			
		||||
                                    </b>
 | 
			
		||||
                                  </Text>
 | 
			
		||||
                                </Box>
 | 
			
		||||
                              </MenuItem>
 | 
			
		||||
                            ))}
 | 
			
		||||
                          </Box>
 | 
			
		||||
                        </Scroll>
 | 
			
		||||
                      ) : (
 | 
			
		||||
                        <Box
 | 
			
		||||
                          grow="Yes"
 | 
			
		||||
                          alignItems="Center"
 | 
			
		||||
                          justifyContent="Center"
 | 
			
		||||
                          direction="Column"
 | 
			
		||||
                          gap="100"
 | 
			
		||||
                        >
 | 
			
		||||
                          <Text size="H6" align="Center">
 | 
			
		||||
                            No Suggestions
 | 
			
		||||
                          </Text>
 | 
			
		||||
                          <Text size="T200" align="Center">
 | 
			
		||||
                            Please provide the user ID and hit Enter.
 | 
			
		||||
                          </Text>
 | 
			
		||||
                        </Box>
 | 
			
		||||
                      )}
 | 
			
		||||
                    </Box>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Menu>
 | 
			
		||||
              </FocusTrap>
 | 
			
		||||
            }
 | 
			
		||||
          >
 | 
			
		||||
            <Chip
 | 
			
		||||
              type="button"
 | 
			
		||||
              variant="Secondary"
 | 
			
		||||
              radii="Pill"
 | 
			
		||||
              onClick={handleOpenMenu}
 | 
			
		||||
              aria-pressed={!!menuCords}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Icon size="50" src={Icons.Plus} />
 | 
			
		||||
            </Chip>
 | 
			
		||||
          </PopOut>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </SettingTile>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										118
									
								
								src/app/components/create-room/CreateRoomAliasInput.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								src/app/components/create-room/CreateRoomAliasInput.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,118 @@
 | 
			
		|||
import React, {
 | 
			
		||||
  FormEventHandler,
 | 
			
		||||
  KeyboardEventHandler,
 | 
			
		||||
  useCallback,
 | 
			
		||||
  useEffect,
 | 
			
		||||
  useRef,
 | 
			
		||||
  useState,
 | 
			
		||||
} from 'react';
 | 
			
		||||
import { MatrixError } from 'matrix-js-sdk';
 | 
			
		||||
import { Box, color, Icon, Icons, Input, Spinner, Text, toRem } from 'folds';
 | 
			
		||||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import { getMxIdServer } from '../../utils/matrix';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { replaceSpaceWithDash } from '../../utils/common';
 | 
			
		||||
import { AsyncState, AsyncStatus, useAsync } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { useDebounce } from '../../hooks/useDebounce';
 | 
			
		||||
 | 
			
		||||
export function CreateRoomAliasInput({ disabled }: { disabled?: boolean }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const aliasInputRef = useRef<HTMLInputElement>(null);
 | 
			
		||||
  const [aliasAvail, setAliasAvail] = useState<AsyncState<boolean, Error>>({
 | 
			
		||||
    status: AsyncStatus.Idle,
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    if (aliasAvail.status === AsyncStatus.Success && aliasInputRef.current?.value === '') {
 | 
			
		||||
      setAliasAvail({ status: AsyncStatus.Idle });
 | 
			
		||||
    }
 | 
			
		||||
  }, [aliasAvail]);
 | 
			
		||||
 | 
			
		||||
  const checkAliasAvail = useAsync(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (aliasLocalPart: string) => {
 | 
			
		||||
        const roomAlias = `#${aliasLocalPart}:${getMxIdServer(mx.getSafeUserId())}`;
 | 
			
		||||
        try {
 | 
			
		||||
          const result = await mx.getRoomIdForAlias(roomAlias);
 | 
			
		||||
          return typeof result.room_id !== 'string';
 | 
			
		||||
        } catch (e) {
 | 
			
		||||
          if (e instanceof MatrixError && e.httpStatus === 404) {
 | 
			
		||||
            return true;
 | 
			
		||||
          }
 | 
			
		||||
          throw e;
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      [mx]
 | 
			
		||||
    ),
 | 
			
		||||
    setAliasAvail
 | 
			
		||||
  );
 | 
			
		||||
  const aliasAvailable: boolean | undefined =
 | 
			
		||||
    aliasAvail.status === AsyncStatus.Success ? aliasAvail.data : undefined;
 | 
			
		||||
 | 
			
		||||
  const debounceCheckAliasAvail = useDebounce(checkAliasAvail, { wait: 500 });
 | 
			
		||||
 | 
			
		||||
  const handleAliasChange: FormEventHandler<HTMLInputElement> = (evt) => {
 | 
			
		||||
    const aliasInput = evt.currentTarget;
 | 
			
		||||
    const aliasLocalPart = replaceSpaceWithDash(aliasInput.value);
 | 
			
		||||
    if (aliasLocalPart) {
 | 
			
		||||
      aliasInput.value = aliasLocalPart;
 | 
			
		||||
      debounceCheckAliasAvail(aliasLocalPart);
 | 
			
		||||
    } else {
 | 
			
		||||
      setAliasAvail({ status: AsyncStatus.Idle });
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleAliasKeyDown: KeyboardEventHandler<HTMLInputElement> = (evt) => {
 | 
			
		||||
    if (isKeyHotkey('enter', evt)) {
 | 
			
		||||
      evt.preventDefault();
 | 
			
		||||
 | 
			
		||||
      const aliasInput = evt.currentTarget;
 | 
			
		||||
      const aliasLocalPart = replaceSpaceWithDash(aliasInput.value);
 | 
			
		||||
      if (aliasLocalPart) {
 | 
			
		||||
        checkAliasAvail(aliasLocalPart);
 | 
			
		||||
      } else {
 | 
			
		||||
        setAliasAvail({ status: AsyncStatus.Idle });
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
      <Text size="L400">Address (Optional)</Text>
 | 
			
		||||
      <Text size="T200" priority="300">
 | 
			
		||||
        Pick an unique address to make it discoverable.
 | 
			
		||||
      </Text>
 | 
			
		||||
      <Input
 | 
			
		||||
        ref={aliasInputRef}
 | 
			
		||||
        onChange={handleAliasChange}
 | 
			
		||||
        before={
 | 
			
		||||
          aliasAvail.status === AsyncStatus.Loading ? (
 | 
			
		||||
            <Spinner size="100" variant="Secondary" />
 | 
			
		||||
          ) : (
 | 
			
		||||
            <Icon size="100" src={Icons.Hash} />
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        after={
 | 
			
		||||
          <Text style={{ maxWidth: toRem(150) }} truncate>
 | 
			
		||||
            :{getMxIdServer(mx.getSafeUserId())}
 | 
			
		||||
          </Text>
 | 
			
		||||
        }
 | 
			
		||||
        onKeyDown={handleAliasKeyDown}
 | 
			
		||||
        name="aliasInput"
 | 
			
		||||
        size="500"
 | 
			
		||||
        variant={aliasAvailable === true ? 'Success' : 'SurfaceVariant'}
 | 
			
		||||
        radii="400"
 | 
			
		||||
        autoComplete="off"
 | 
			
		||||
        disabled={disabled}
 | 
			
		||||
      />
 | 
			
		||||
      {aliasAvailable === false && (
 | 
			
		||||
        <Box style={{ color: color.Critical.Main }} alignItems="Center" gap="100">
 | 
			
		||||
          <Icon src={Icons.Warning} filled size="50" />
 | 
			
		||||
          <Text size="T200">
 | 
			
		||||
            <b>This address is already taken. Please select a different one.</b>
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Box>
 | 
			
		||||
      )}
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										94
									
								
								src/app/components/create-room/CreateRoomKindSelector.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								src/app/components/create-room/CreateRoomKindSelector.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,94 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import { Box, Text, Icon, Icons, config, IconSrc } from 'folds';
 | 
			
		||||
import { SequenceCard } from '../sequence-card';
 | 
			
		||||
import { SettingTile } from '../setting-tile';
 | 
			
		||||
 | 
			
		||||
export enum CreateRoomKind {
 | 
			
		||||
  Private = 'private',
 | 
			
		||||
  Restricted = 'restricted',
 | 
			
		||||
  Public = 'public',
 | 
			
		||||
}
 | 
			
		||||
type CreateRoomKindSelectorProps = {
 | 
			
		||||
  value?: CreateRoomKind;
 | 
			
		||||
  onSelect: (value: CreateRoomKind) => void;
 | 
			
		||||
  canRestrict?: boolean;
 | 
			
		||||
  disabled?: boolean;
 | 
			
		||||
  getIcon: (kind: CreateRoomKind) => IconSrc;
 | 
			
		||||
};
 | 
			
		||||
export function CreateRoomKindSelector({
 | 
			
		||||
  value,
 | 
			
		||||
  onSelect,
 | 
			
		||||
  canRestrict,
 | 
			
		||||
  disabled,
 | 
			
		||||
  getIcon,
 | 
			
		||||
}: CreateRoomKindSelectorProps) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
      {canRestrict && (
 | 
			
		||||
        <SequenceCard
 | 
			
		||||
          style={{ padding: config.space.S300 }}
 | 
			
		||||
          variant={value === CreateRoomKind.Restricted ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
          direction="Column"
 | 
			
		||||
          gap="100"
 | 
			
		||||
          as="button"
 | 
			
		||||
          type="button"
 | 
			
		||||
          aria-pressed={value === CreateRoomKind.Restricted}
 | 
			
		||||
          onClick={() => onSelect(CreateRoomKind.Restricted)}
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
        >
 | 
			
		||||
          <SettingTile
 | 
			
		||||
            before={<Icon size="400" src={getIcon(CreateRoomKind.Restricted)} />}
 | 
			
		||||
            after={value === CreateRoomKind.Restricted && <Icon src={Icons.Check} />}
 | 
			
		||||
          >
 | 
			
		||||
            <Text size="H6">Restricted</Text>
 | 
			
		||||
            <Text size="T300" priority="300">
 | 
			
		||||
              Only member of parent space can join.
 | 
			
		||||
            </Text>
 | 
			
		||||
          </SettingTile>
 | 
			
		||||
        </SequenceCard>
 | 
			
		||||
      )}
 | 
			
		||||
      <SequenceCard
 | 
			
		||||
        style={{ padding: config.space.S300 }}
 | 
			
		||||
        variant={value === CreateRoomKind.Private ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
        direction="Column"
 | 
			
		||||
        gap="100"
 | 
			
		||||
        as="button"
 | 
			
		||||
        type="button"
 | 
			
		||||
        aria-pressed={value === CreateRoomKind.Private}
 | 
			
		||||
        onClick={() => onSelect(CreateRoomKind.Private)}
 | 
			
		||||
        disabled={disabled}
 | 
			
		||||
      >
 | 
			
		||||
        <SettingTile
 | 
			
		||||
          before={<Icon size="400" src={getIcon(CreateRoomKind.Private)} />}
 | 
			
		||||
          after={value === CreateRoomKind.Private && <Icon src={Icons.Check} />}
 | 
			
		||||
        >
 | 
			
		||||
          <Text size="H6">Private</Text>
 | 
			
		||||
          <Text size="T300" priority="300">
 | 
			
		||||
            Only people with invite can join.
 | 
			
		||||
          </Text>
 | 
			
		||||
        </SettingTile>
 | 
			
		||||
      </SequenceCard>
 | 
			
		||||
      <SequenceCard
 | 
			
		||||
        style={{ padding: config.space.S300 }}
 | 
			
		||||
        variant={value === CreateRoomKind.Public ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
        direction="Column"
 | 
			
		||||
        gap="100"
 | 
			
		||||
        as="button"
 | 
			
		||||
        type="button"
 | 
			
		||||
        aria-pressed={value === CreateRoomKind.Public}
 | 
			
		||||
        onClick={() => onSelect(CreateRoomKind.Public)}
 | 
			
		||||
        disabled={disabled}
 | 
			
		||||
      >
 | 
			
		||||
        <SettingTile
 | 
			
		||||
          before={<Icon size="400" src={getIcon(CreateRoomKind.Public)} />}
 | 
			
		||||
          after={value === CreateRoomKind.Public && <Icon src={Icons.Check} />}
 | 
			
		||||
        >
 | 
			
		||||
          <Text size="H6">Public</Text>
 | 
			
		||||
          <Text size="T300" priority="300">
 | 
			
		||||
            Anyone with the address can join.
 | 
			
		||||
          </Text>
 | 
			
		||||
        </SettingTile>
 | 
			
		||||
      </SequenceCard>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										117
									
								
								src/app/components/create-room/RoomVersionSelector.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								src/app/components/create-room/RoomVersionSelector.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,117 @@
 | 
			
		|||
import React, { MouseEventHandler, useState } from 'react';
 | 
			
		||||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  Chip,
 | 
			
		||||
  config,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Menu,
 | 
			
		||||
  PopOut,
 | 
			
		||||
  RectCords,
 | 
			
		||||
  Text,
 | 
			
		||||
  toRem,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { SettingTile } from '../setting-tile';
 | 
			
		||||
import { SequenceCard } from '../sequence-card';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
 | 
			
		||||
export function RoomVersionSelector({
 | 
			
		||||
  versions,
 | 
			
		||||
  value,
 | 
			
		||||
  onChange,
 | 
			
		||||
  disabled,
 | 
			
		||||
}: {
 | 
			
		||||
  versions: string[];
 | 
			
		||||
  value: string;
 | 
			
		||||
  onChange: (value: string) => void;
 | 
			
		||||
  disabled?: boolean;
 | 
			
		||||
}) {
 | 
			
		||||
  const [menuCords, setMenuCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const handleMenu: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setMenuCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleSelect = (version: string) => {
 | 
			
		||||
    setMenuCords(undefined);
 | 
			
		||||
    onChange(version);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SequenceCard
 | 
			
		||||
      style={{ padding: config.space.S300 }}
 | 
			
		||||
      variant="SurfaceVariant"
 | 
			
		||||
      direction="Column"
 | 
			
		||||
      gap="500"
 | 
			
		||||
    >
 | 
			
		||||
      <SettingTile
 | 
			
		||||
        title="Version"
 | 
			
		||||
        after={
 | 
			
		||||
          <PopOut
 | 
			
		||||
            anchor={menuCords}
 | 
			
		||||
            offset={5}
 | 
			
		||||
            position="Bottom"
 | 
			
		||||
            align="End"
 | 
			
		||||
            content={
 | 
			
		||||
              <FocusTrap
 | 
			
		||||
                focusTrapOptions={{
 | 
			
		||||
                  initialFocus: false,
 | 
			
		||||
                  onDeactivate: () => setMenuCords(undefined),
 | 
			
		||||
                  clickOutsideDeactivates: true,
 | 
			
		||||
                  isKeyForward: (evt: KeyboardEvent) =>
 | 
			
		||||
                    evt.key === 'ArrowDown' || evt.key === 'ArrowRight',
 | 
			
		||||
                  isKeyBackward: (evt: KeyboardEvent) =>
 | 
			
		||||
                    evt.key === 'ArrowUp' || evt.key === 'ArrowLeft',
 | 
			
		||||
                  escapeDeactivates: stopPropagation,
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Menu>
 | 
			
		||||
                  <Box
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="200"
 | 
			
		||||
                    style={{ padding: config.space.S200, maxWidth: toRem(300) }}
 | 
			
		||||
                  >
 | 
			
		||||
                    <Text size="L400">Versions</Text>
 | 
			
		||||
                    <Box wrap="Wrap" gap="100">
 | 
			
		||||
                      {versions.map((version) => (
 | 
			
		||||
                        <Chip
 | 
			
		||||
                          key={version}
 | 
			
		||||
                          variant={value === version ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                          aria-pressed={value === version}
 | 
			
		||||
                          outlined={value === version}
 | 
			
		||||
                          radii="300"
 | 
			
		||||
                          onClick={() => handleSelect(version)}
 | 
			
		||||
                          type="button"
 | 
			
		||||
                        >
 | 
			
		||||
                          <Text truncate size="T300">
 | 
			
		||||
                            {version}
 | 
			
		||||
                          </Text>
 | 
			
		||||
                        </Chip>
 | 
			
		||||
                      ))}
 | 
			
		||||
                    </Box>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Menu>
 | 
			
		||||
              </FocusTrap>
 | 
			
		||||
            }
 | 
			
		||||
          >
 | 
			
		||||
            <Button
 | 
			
		||||
              type="button"
 | 
			
		||||
              onClick={handleMenu}
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Secondary"
 | 
			
		||||
              fill="Soft"
 | 
			
		||||
              radii="300"
 | 
			
		||||
              aria-pressed={!!menuCords}
 | 
			
		||||
              before={<Icon size="50" src={menuCords ? Icons.ChevronTop : Icons.ChevronBottom} />}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">{value}</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          </PopOut>
 | 
			
		||||
        }
 | 
			
		||||
      />
 | 
			
		||||
    </SequenceCard>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										5
									
								
								src/app/components/create-room/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/app/components/create-room/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,5 @@
 | 
			
		|||
export * from './CreateRoomKindSelector';
 | 
			
		||||
export * from './CreateRoomAliasInput';
 | 
			
		||||
export * from './RoomVersionSelector';
 | 
			
		||||
export * from './utils';
 | 
			
		||||
export * from './AdditionalCreatorInput';
 | 
			
		||||
							
								
								
									
										140
									
								
								src/app/components/create-room/utils.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								src/app/components/create-room/utils.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,140 @@
 | 
			
		|||
import {
 | 
			
		||||
  ICreateRoomOpts,
 | 
			
		||||
  ICreateRoomStateEvent,
 | 
			
		||||
  JoinRule,
 | 
			
		||||
  MatrixClient,
 | 
			
		||||
  RestrictedAllowType,
 | 
			
		||||
  Room,
 | 
			
		||||
} from 'matrix-js-sdk';
 | 
			
		||||
import { RoomJoinRulesEventContent } from 'matrix-js-sdk/lib/types';
 | 
			
		||||
import { CreateRoomKind } from './CreateRoomKindSelector';
 | 
			
		||||
import { RoomType, StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { getViaServers } from '../../plugins/via-servers';
 | 
			
		||||
import { getMxIdServer } from '../../utils/matrix';
 | 
			
		||||
 | 
			
		||||
export const createRoomCreationContent = (
 | 
			
		||||
  type: RoomType | undefined,
 | 
			
		||||
  allowFederation: boolean,
 | 
			
		||||
  additionalCreators: string[] | undefined
 | 
			
		||||
): object => {
 | 
			
		||||
  const content: Record<string, any> = {};
 | 
			
		||||
  if (typeof type === 'string') {
 | 
			
		||||
    content.type = type;
 | 
			
		||||
  }
 | 
			
		||||
  if (allowFederation === false) {
 | 
			
		||||
    content['m.federate'] = false;
 | 
			
		||||
  }
 | 
			
		||||
  if (Array.isArray(additionalCreators)) {
 | 
			
		||||
    content.additional_creators = additionalCreators;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return content;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const createRoomJoinRulesState = (
 | 
			
		||||
  kind: CreateRoomKind,
 | 
			
		||||
  parent: Room | undefined,
 | 
			
		||||
  knock: boolean
 | 
			
		||||
) => {
 | 
			
		||||
  let content: RoomJoinRulesEventContent = {
 | 
			
		||||
    join_rule: knock ? JoinRule.Knock : JoinRule.Invite,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  if (kind === CreateRoomKind.Public) {
 | 
			
		||||
    content = {
 | 
			
		||||
      join_rule: JoinRule.Public,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (kind === CreateRoomKind.Restricted && parent) {
 | 
			
		||||
    content = {
 | 
			
		||||
      join_rule: knock ? ('knock_restricted' as JoinRule) : JoinRule.Restricted,
 | 
			
		||||
      allow: [
 | 
			
		||||
        {
 | 
			
		||||
          type: RestrictedAllowType.RoomMembership,
 | 
			
		||||
          room_id: parent.roomId,
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    type: StateEvent.RoomJoinRules,
 | 
			
		||||
    state_key: '',
 | 
			
		||||
    content,
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const createRoomParentState = (parent: Room) => ({
 | 
			
		||||
  type: StateEvent.SpaceParent,
 | 
			
		||||
  state_key: parent.roomId,
 | 
			
		||||
  content: {
 | 
			
		||||
    canonical: true,
 | 
			
		||||
    via: getViaServers(parent),
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const createRoomEncryptionState = () => ({
 | 
			
		||||
  type: 'm.room.encryption',
 | 
			
		||||
  state_key: '',
 | 
			
		||||
  content: {
 | 
			
		||||
    algorithm: 'm.megolm.v1.aes-sha2',
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export type CreateRoomData = {
 | 
			
		||||
  version: string;
 | 
			
		||||
  type?: RoomType;
 | 
			
		||||
  parent?: Room;
 | 
			
		||||
  kind: CreateRoomKind;
 | 
			
		||||
  name: string;
 | 
			
		||||
  topic?: string;
 | 
			
		||||
  aliasLocalPart?: string;
 | 
			
		||||
  encryption?: boolean;
 | 
			
		||||
  knock: boolean;
 | 
			
		||||
  allowFederation: boolean;
 | 
			
		||||
  additionalCreators?: string[];
 | 
			
		||||
};
 | 
			
		||||
export const createRoom = async (mx: MatrixClient, data: CreateRoomData): Promise<string> => {
 | 
			
		||||
  const initialState: ICreateRoomStateEvent[] = [];
 | 
			
		||||
 | 
			
		||||
  if (data.encryption) {
 | 
			
		||||
    initialState.push(createRoomEncryptionState());
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (data.parent) {
 | 
			
		||||
    initialState.push(createRoomParentState(data.parent));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  initialState.push(createRoomJoinRulesState(data.kind, data.parent, data.knock));
 | 
			
		||||
 | 
			
		||||
  const options: ICreateRoomOpts = {
 | 
			
		||||
    room_version: data.version,
 | 
			
		||||
    name: data.name,
 | 
			
		||||
    topic: data.topic,
 | 
			
		||||
    room_alias_name: data.aliasLocalPart,
 | 
			
		||||
    creation_content: createRoomCreationContent(
 | 
			
		||||
      data.type,
 | 
			
		||||
      data.allowFederation,
 | 
			
		||||
      data.additionalCreators
 | 
			
		||||
    ),
 | 
			
		||||
    initial_state: initialState,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const result = await mx.createRoom(options);
 | 
			
		||||
 | 
			
		||||
  if (data.parent) {
 | 
			
		||||
    await mx.sendStateEvent(
 | 
			
		||||
      data.parent.roomId,
 | 
			
		||||
      StateEvent.SpaceChild as any,
 | 
			
		||||
      {
 | 
			
		||||
        auto_join: false,
 | 
			
		||||
        suggested: false,
 | 
			
		||||
        via: [getMxIdServer(mx.getUserId() ?? '') ?? ''],
 | 
			
		||||
      },
 | 
			
		||||
      result.room_id
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return result.room_id;
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -41,21 +41,21 @@ export const EditorTextarea = style([
 | 
			
		|||
  },
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
export const EditorPlaceholder = style([
 | 
			
		||||
export const EditorPlaceholderContainer = style([
 | 
			
		||||
  DefaultReset,
 | 
			
		||||
  {
 | 
			
		||||
    position: 'absolute',
 | 
			
		||||
    zIndex: 1,
 | 
			
		||||
    width: '100%',
 | 
			
		||||
    opacity: config.opacity.Placeholder,
 | 
			
		||||
    pointerEvents: 'none',
 | 
			
		||||
    userSelect: 'none',
 | 
			
		||||
  },
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
    selectors: {
 | 
			
		||||
      '&:not(:first-child)': {
 | 
			
		||||
        display: 'none',
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
export const EditorPlaceholderTextVisual = style([
 | 
			
		||||
  DefaultReset,
 | 
			
		||||
  {
 | 
			
		||||
    display: 'block',
 | 
			
		||||
    paddingTop: toRem(13),
 | 
			
		||||
    paddingLeft: toRem(1),
 | 
			
		||||
  },
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -106,22 +106,17 @@ export const CustomEditor = forwardRef<HTMLDivElement, CustomEditorProps>(
 | 
			
		|||
      [editor, onKeyDown]
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    const renderPlaceholder = useCallback(({ attributes, children }: RenderPlaceholderProps) => {
 | 
			
		||||
      // drop style attribute as we use our custom placeholder css.
 | 
			
		||||
      // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | 
			
		||||
      const { style, ...props } = attributes;
 | 
			
		||||
      return (
 | 
			
		||||
        <Text
 | 
			
		||||
          as="span"
 | 
			
		||||
          {...props}
 | 
			
		||||
          className={css.EditorPlaceholder}
 | 
			
		||||
          contentEditable={false}
 | 
			
		||||
          truncate
 | 
			
		||||
        >
 | 
			
		||||
          {children}
 | 
			
		||||
        </Text>
 | 
			
		||||
      );
 | 
			
		||||
    }, []);
 | 
			
		||||
    const renderPlaceholder = useCallback(
 | 
			
		||||
      ({ attributes, children }: RenderPlaceholderProps) => (
 | 
			
		||||
        <span {...attributes} className={css.EditorPlaceholderContainer}>
 | 
			
		||||
          {/* Inner component to style the actual text position and appearance */}
 | 
			
		||||
          <Text as="span" className={css.EditorPlaceholderTextVisual} truncate>
 | 
			
		||||
            {children}
 | 
			
		||||
          </Text>
 | 
			
		||||
        </span>
 | 
			
		||||
      ),
 | 
			
		||||
      []
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className={css.Editor} ref={ref}>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -157,7 +157,7 @@ export function RenderElement({ attributes, element, children }: RenderElementPr
 | 
			
		|||
        <Text as="pre" className={css.CodeBlock} {...attributes}>
 | 
			
		||||
          <Scroll
 | 
			
		||||
            direction="Horizontal"
 | 
			
		||||
            variant="Secondary"
 | 
			
		||||
            variant="SurfaceVariant"
 | 
			
		||||
            size="300"
 | 
			
		||||
            visibility="Hover"
 | 
			
		||||
            hideTrack
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -339,7 +339,7 @@ export function Toolbar() {
 | 
			
		|||
          <Box className={css.MarkdownBtnBox} shrink="No" grow="Yes" justifyContent="End">
 | 
			
		||||
            <TooltipProvider
 | 
			
		||||
              align="End"
 | 
			
		||||
              tooltip={<BtnTooltip text="Toggle Markdown" />}
 | 
			
		||||
              tooltip={<BtnTooltip text={isMarkdown ? 'Disable Markdown' : 'Enable Markdown'} />}
 | 
			
		||||
              delay={500}
 | 
			
		||||
            >
 | 
			
		||||
              {(triggerRef) => (
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -9,7 +9,7 @@ import { getDirectRoomAvatarUrl } from '../../../utils/room';
 | 
			
		|||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { AutocompleteQuery } from './autocompleteQuery';
 | 
			
		||||
import { AutocompleteMenu } from './AutocompleteMenu';
 | 
			
		||||
import { getMxIdServer, validMxId } from '../../../utils/matrix';
 | 
			
		||||
import { getMxIdServer, isRoomAlias } from '../../../utils/matrix';
 | 
			
		||||
import { UseAsyncSearchOptions, useAsyncSearch } from '../../../hooks/useAsyncSearch';
 | 
			
		||||
import { onTabPress } from '../../../utils/keyboard';
 | 
			
		||||
import { useKeyDown } from '../../../hooks/useKeyDown';
 | 
			
		||||
| 
						 | 
				
			
			@ -22,7 +22,7 @@ import { getViaServers } from '../../../plugins/via-servers';
 | 
			
		|||
type MentionAutoCompleteHandler = (roomAliasOrId: string, name: string) => void;
 | 
			
		||||
 | 
			
		||||
const roomAliasFromQueryText = (mx: MatrixClient, text: string) =>
 | 
			
		||||
  validMxId(`#${text}`)
 | 
			
		||||
  isRoomAlias(`#${text}`)
 | 
			
		||||
    ? `#${text}`
 | 
			
		||||
    : `#${text}${text.endsWith(':') ? '' : ':'}${getMxIdServer(mx.getUserId() ?? '')}`;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -15,7 +15,7 @@ import {
 | 
			
		|||
import { onTabPress } from '../../../utils/keyboard';
 | 
			
		||||
import { createMentionElement, moveCursor, replaceWithElement } from '../utils';
 | 
			
		||||
import { useKeyDown } from '../../../hooks/useKeyDown';
 | 
			
		||||
import { getMxIdLocalPart, getMxIdServer, validMxId } from '../../../utils/matrix';
 | 
			
		||||
import { getMxIdLocalPart, getMxIdServer, isUserId } from '../../../utils/matrix';
 | 
			
		||||
import { getMemberDisplayName, getMemberSearchStr } from '../../../utils/room';
 | 
			
		||||
import { UserAvatar } from '../../user-avatar';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
| 
						 | 
				
			
			@ -24,7 +24,7 @@ import { Membership } from '../../../../types/matrix/room';
 | 
			
		|||
type MentionAutoCompleteHandler = (userId: string, name: string) => void;
 | 
			
		||||
 | 
			
		||||
const userIdFromQueryText = (mx: MatrixClient, text: string) =>
 | 
			
		||||
  validMxId(`@${text}`)
 | 
			
		||||
  isUserId(`@${text}`)
 | 
			
		||||
    ? `@${text}`
 | 
			
		||||
    : `@${text}${text.endsWith(':') ? '' : ':'}${getMxIdServer(mx.getUserId() ?? '')}`;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -19,9 +19,11 @@ import { getMemberDisplayName } from '../../utils/room';
 | 
			
		|||
import { getMxIdLocalPart } from '../../utils/matrix';
 | 
			
		||||
import * as css from './EventReaders.css';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { openProfileViewer } from '../../../client/action/navigation';
 | 
			
		||||
import { UserAvatar } from '../user-avatar';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useOpenUserRoomProfile } from '../../state/hooks/userRoomProfile';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { getMouseEventCords } from '../../utils/dom';
 | 
			
		||||
 | 
			
		||||
export type EventReadersProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -33,6 +35,8 @@ export const EventReaders = as<'div', EventReadersProps>(
 | 
			
		|||
    const mx = useMatrixClient();
 | 
			
		||||
    const useAuthentication = useMediaAuthentication();
 | 
			
		||||
    const latestEventReaders = useRoomEventReaders(room, eventId);
 | 
			
		||||
    const openProfile = useOpenUserRoomProfile();
 | 
			
		||||
    const space = useSpaceOptionally();
 | 
			
		||||
 | 
			
		||||
    const getName = (userId: string) =>
 | 
			
		||||
      getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
 | 
			
		||||
| 
						 | 
				
			
			@ -57,19 +61,32 @@ export const EventReaders = as<'div', EventReadersProps>(
 | 
			
		|||
            <Box className={css.Content} direction="Column">
 | 
			
		||||
              {latestEventReaders.map((readerId) => {
 | 
			
		||||
                const name = getName(readerId);
 | 
			
		||||
                const avatarMxcUrl = room
 | 
			
		||||
                  .getMember(readerId)
 | 
			
		||||
                  ?.getMxcAvatarUrl();
 | 
			
		||||
                const avatarUrl = avatarMxcUrl ? mx.mxcUrlToHttp(avatarMxcUrl, 100, 100, 'crop', undefined, false, useAuthentication) : undefined;
 | 
			
		||||
                const avatarMxcUrl = room.getMember(readerId)?.getMxcAvatarUrl();
 | 
			
		||||
                const avatarUrl = avatarMxcUrl
 | 
			
		||||
                  ? mx.mxcUrlToHttp(
 | 
			
		||||
                      avatarMxcUrl,
 | 
			
		||||
                      100,
 | 
			
		||||
                      100,
 | 
			
		||||
                      'crop',
 | 
			
		||||
                      undefined,
 | 
			
		||||
                      false,
 | 
			
		||||
                      useAuthentication
 | 
			
		||||
                    )
 | 
			
		||||
                  : undefined;
 | 
			
		||||
 | 
			
		||||
                return (
 | 
			
		||||
                  <MenuItem
 | 
			
		||||
                    key={readerId}
 | 
			
		||||
                    style={{ padding: `0 ${config.space.S200}` }}
 | 
			
		||||
                    radii="400"
 | 
			
		||||
                    onClick={() => {
 | 
			
		||||
                      requestClose();
 | 
			
		||||
                      openProfileViewer(readerId, room.roomId);
 | 
			
		||||
                    onClick={(event) => {
 | 
			
		||||
                      openProfile(
 | 
			
		||||
                        room.roomId,
 | 
			
		||||
                        space?.roomId,
 | 
			
		||||
                        readerId,
 | 
			
		||||
                        getMouseEventCords(event.nativeEvent),
 | 
			
		||||
                        'Bottom'
 | 
			
		||||
                      );
 | 
			
		||||
                    }}
 | 
			
		||||
                    before={
 | 
			
		||||
                      <Avatar size="200">
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,12 +1,14 @@
 | 
			
		|||
import React, { useCallback, useMemo } from 'react';
 | 
			
		||||
import { Room } from 'matrix-js-sdk';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { ImagePackContent } from './ImagePackContent';
 | 
			
		||||
import { ImagePack, PackContent } from '../../plugins/custom-emoji';
 | 
			
		||||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { useRoomImagePack } from '../../hooks/useImagePacks';
 | 
			
		||||
import { randomStr } from '../../utils/common';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
 | 
			
		||||
type RoomImagePackProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -17,9 +19,10 @@ export function RoomImagePack({ room, stateKey }: RoomImagePackProps) {
 | 
			
		|||
  const mx = useMatrixClient();
 | 
			
		||||
  const userId = mx.getUserId()!;
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const { getPowerLevel, canSendStateEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canEditImagePack = canSendStateEvent(StateEvent.PoniesRoomEmotes, getPowerLevel(userId));
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const canEditImagePack = permissions.stateEvent(StateEvent.PoniesRoomEmotes, userId);
 | 
			
		||||
 | 
			
		||||
  const fallbackPack = useMemo(() => {
 | 
			
		||||
    const fakePackId = randomStr(4);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										131
									
								
								src/app/components/join-address-prompt/JoinAddressPrompt.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								src/app/components/join-address-prompt/JoinAddressPrompt.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,131 @@
 | 
			
		|||
import React, { FormEventHandler, useState } from 'react';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import {
 | 
			
		||||
  Dialog,
 | 
			
		||||
  Overlay,
 | 
			
		||||
  OverlayCenter,
 | 
			
		||||
  OverlayBackdrop,
 | 
			
		||||
  Header,
 | 
			
		||||
  config,
 | 
			
		||||
  Box,
 | 
			
		||||
  Text,
 | 
			
		||||
  IconButton,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Button,
 | 
			
		||||
  Input,
 | 
			
		||||
  color,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { isRoomAlias, isRoomId } from '../../utils/matrix';
 | 
			
		||||
import { parseMatrixToRoom, parseMatrixToRoomEvent, testMatrixTo } from '../../plugins/matrix-to';
 | 
			
		||||
import { tryDecodeURIComponent } from '../../utils/dom';
 | 
			
		||||
 | 
			
		||||
type JoinAddressProps = {
 | 
			
		||||
  onOpen: (roomIdOrAlias: string, via?: string[], eventId?: string) => void;
 | 
			
		||||
  onCancel: () => void;
 | 
			
		||||
};
 | 
			
		||||
export function JoinAddressPrompt({ onOpen, onCancel }: JoinAddressProps) {
 | 
			
		||||
  const [invalid, setInvalid] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const handleSubmit: FormEventHandler<HTMLFormElement> = (evt) => {
 | 
			
		||||
    evt.preventDefault();
 | 
			
		||||
    setInvalid(false);
 | 
			
		||||
 | 
			
		||||
    const target = evt.target as HTMLFormElement | undefined;
 | 
			
		||||
    const addressInput = target?.addressInput as HTMLInputElement | undefined;
 | 
			
		||||
    const address = addressInput?.value.trim();
 | 
			
		||||
    if (!address) return;
 | 
			
		||||
 | 
			
		||||
    if (isRoomId(address) || isRoomAlias(address)) {
 | 
			
		||||
      onOpen(address);
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (testMatrixTo(address)) {
 | 
			
		||||
      const decodedAddress = tryDecodeURIComponent(address);
 | 
			
		||||
      const toRoom = parseMatrixToRoom(decodedAddress);
 | 
			
		||||
      if (toRoom) {
 | 
			
		||||
        onOpen(toRoom.roomIdOrAlias, toRoom.viaServers);
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const toEvent = parseMatrixToRoomEvent(decodedAddress);
 | 
			
		||||
      if (toEvent) {
 | 
			
		||||
        onOpen(toEvent.roomIdOrAlias, toEvent.viaServers, toEvent.eventId);
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    setInvalid(true);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
      <OverlayCenter>
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: onCancel,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Dialog variant="Surface">
 | 
			
		||||
            <Header
 | 
			
		||||
              style={{
 | 
			
		||||
                padding: `0 ${config.space.S200} 0 ${config.space.S400}`,
 | 
			
		||||
              }}
 | 
			
		||||
              variant="Surface"
 | 
			
		||||
              size="500"
 | 
			
		||||
            >
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Text size="H4">Join with Address</Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <IconButton size="300" onClick={onCancel} radii="300">
 | 
			
		||||
                <Icon src={Icons.Cross} />
 | 
			
		||||
              </IconButton>
 | 
			
		||||
            </Header>
 | 
			
		||||
            <Box
 | 
			
		||||
              as="form"
 | 
			
		||||
              onSubmit={handleSubmit}
 | 
			
		||||
              style={{ padding: config.space.S400, paddingTop: 0 }}
 | 
			
		||||
              direction="Column"
 | 
			
		||||
              gap="400"
 | 
			
		||||
            >
 | 
			
		||||
              <Box direction="Column" gap="200">
 | 
			
		||||
                <Text priority="400" size="T300">
 | 
			
		||||
                  Enter public address to join the community. Addresses looks like:
 | 
			
		||||
                </Text>
 | 
			
		||||
                <Text as="ul" size="T200" priority="300" style={{ paddingLeft: config.space.S400 }}>
 | 
			
		||||
                  <li>#community:server</li>
 | 
			
		||||
                  <li>https://matrix.to/#/#community:server</li>
 | 
			
		||||
                  <li>https://matrix.to/#/!xYzAj?via=server</li>
 | 
			
		||||
                </Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Box direction="Column" gap="100">
 | 
			
		||||
                <Text size="L400">Address</Text>
 | 
			
		||||
                <Input
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  autoFocus
 | 
			
		||||
                  name="addressInput"
 | 
			
		||||
                  variant="Background"
 | 
			
		||||
                  placeholder="#community:server"
 | 
			
		||||
                  required
 | 
			
		||||
                />
 | 
			
		||||
                {invalid && (
 | 
			
		||||
                  <Text size="T200" style={{ color: color.Critical.Main }}>
 | 
			
		||||
                    <b>Invalid Address</b>
 | 
			
		||||
                  </Text>
 | 
			
		||||
                )}
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Button type="submit" variant="Primary">
 | 
			
		||||
                <Text size="B400">Open</Text>
 | 
			
		||||
              </Button>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Dialog>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      </OverlayCenter>
 | 
			
		||||
    </Overlay>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								src/app/components/join-address-prompt/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/app/components/join-address-prompt/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1 @@
 | 
			
		|||
export * from './JoinAddressPrompt';
 | 
			
		||||
| 
						 | 
				
			
			@ -7,7 +7,6 @@ export const ReplyBend = style({
 | 
			
		|||
 | 
			
		||||
export const ThreadIndicator = style({
 | 
			
		||||
  opacity: config.opacity.P300,
 | 
			
		||||
  gap: toRem(2),
 | 
			
		||||
 | 
			
		||||
  selectors: {
 | 
			
		||||
    'button&': {
 | 
			
		||||
| 
						 | 
				
			
			@ -19,11 +18,6 @@ export const ThreadIndicator = style({
 | 
			
		|||
  },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const ThreadIndicatorIcon = style({
 | 
			
		||||
  width: toRem(14),
 | 
			
		||||
  height: toRem(14),
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const Reply = style({
 | 
			
		||||
  marginBottom: toRem(1),
 | 
			
		||||
  minWidth: 0,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -10,8 +10,8 @@ import * as css from './Reply.css';
 | 
			
		|||
import { MessageBadEncryptedContent, MessageDeletedContent, MessageFailedContent } from './content';
 | 
			
		||||
import { scaleSystemEmoji } from '../../plugins/react-custom-html-parser';
 | 
			
		||||
import { useRoomEvent } from '../../hooks/useRoomEvent';
 | 
			
		||||
import { GetPowerLevelTag } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import colorMXID from '../../../util/colorMXID';
 | 
			
		||||
import { GetMemberPowerTag } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
 | 
			
		||||
type ReplyLayoutProps = {
 | 
			
		||||
  userColor?: string;
 | 
			
		||||
| 
						 | 
				
			
			@ -38,9 +38,16 @@ export const ReplyLayout = as<'div', ReplyLayoutProps>(
 | 
			
		|||
);
 | 
			
		||||
 | 
			
		||||
export const ThreadIndicator = as<'div'>(({ ...props }, ref) => (
 | 
			
		||||
  <Box className={css.ThreadIndicator} alignItems="Center" {...props} ref={ref}>
 | 
			
		||||
    <Icon className={css.ThreadIndicatorIcon} src={Icons.Message} />
 | 
			
		||||
    <Text size="T200">Threaded reply</Text>
 | 
			
		||||
  <Box
 | 
			
		||||
    shrink="No"
 | 
			
		||||
    className={css.ThreadIndicator}
 | 
			
		||||
    alignItems="Center"
 | 
			
		||||
    gap="100"
 | 
			
		||||
    {...props}
 | 
			
		||||
    ref={ref}
 | 
			
		||||
  >
 | 
			
		||||
    <Icon size="50" src={Icons.Thread} />
 | 
			
		||||
    <Text size="L400">Thread</Text>
 | 
			
		||||
  </Box>
 | 
			
		||||
));
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -50,8 +57,7 @@ type ReplyProps = {
 | 
			
		|||
  replyEventId: string;
 | 
			
		||||
  threadRootId?: string | undefined;
 | 
			
		||||
  onClick?: MouseEventHandler | undefined;
 | 
			
		||||
  getPowerLevel?: (userId: string) => number;
 | 
			
		||||
  getPowerLevelTag?: GetPowerLevelTag;
 | 
			
		||||
  getMemberPowerTag?: GetMemberPowerTag;
 | 
			
		||||
  accessibleTagColors?: Map<string, string>;
 | 
			
		||||
  legacyUsernameColor?: boolean;
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -64,8 +70,7 @@ export const Reply = as<'div', ReplyProps>(
 | 
			
		|||
      replyEventId,
 | 
			
		||||
      threadRootId,
 | 
			
		||||
      onClick,
 | 
			
		||||
      getPowerLevel,
 | 
			
		||||
      getPowerLevelTag,
 | 
			
		||||
      getMemberPowerTag,
 | 
			
		||||
      accessibleTagColors,
 | 
			
		||||
      legacyUsernameColor,
 | 
			
		||||
      ...props
 | 
			
		||||
| 
						 | 
				
			
			@ -81,8 +86,7 @@ export const Reply = as<'div', ReplyProps>(
 | 
			
		|||
 | 
			
		||||
    const { body } = replyEvent?.getContent() ?? {};
 | 
			
		||||
    const sender = replyEvent?.getSender();
 | 
			
		||||
    const senderPL = sender && getPowerLevel?.(sender);
 | 
			
		||||
    const powerTag = typeof senderPL === 'number' ? getPowerLevelTag?.(senderPL) : undefined;
 | 
			
		||||
    const powerTag = sender ? getMemberPowerTag?.(sender) : undefined;
 | 
			
		||||
    const tagColor = powerTag?.color ? accessibleTagColors?.get(powerTag.color) : undefined;
 | 
			
		||||
 | 
			
		||||
    const usernameColor = legacyUsernameColor ? colorMXID(sender ?? replyEventId) : tagColor;
 | 
			
		||||
| 
						 | 
				
			
			@ -97,7 +101,7 @@ export const Reply = as<'div', ReplyProps>(
 | 
			
		|||
    const bodyJSX = body ? scaleSystemEmoji(trimReplyFromBody(body)) : fallbackBody;
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <Box direction="Column" alignItems="Start" {...props} ref={ref}>
 | 
			
		||||
      <Box direction="Row" gap="200" alignItems="Center" {...props} ref={ref}>
 | 
			
		||||
        {threadRootId && (
 | 
			
		||||
          <ThreadIndicator as="button" data-event-id={threadRootId} onClick={onClick} />
 | 
			
		||||
        )}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,19 +5,35 @@ import { timeDayMonYear, timeHourMinute, today, yesterday } from '../../utils/ti
 | 
			
		|||
export type TimeProps = {
 | 
			
		||||
  compact?: boolean;
 | 
			
		||||
  ts: number;
 | 
			
		||||
  hour24Clock: boolean;
 | 
			
		||||
  dateFormatString: string;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Renders a formatted timestamp, supporting compact and full display modes.
 | 
			
		||||
 *
 | 
			
		||||
 * Displays the time in hour:minute format if the message is from today, yesterday, or if `compact` is true.
 | 
			
		||||
 * For older messages, it shows the date and time.
 | 
			
		||||
 *
 | 
			
		||||
 * @param {number} ts - The timestamp to display.
 | 
			
		||||
 * @param {boolean} [compact=false] - If true, always show only the time.
 | 
			
		||||
 * @param {boolean} hour24Clock - Whether to use 24-hour time format.
 | 
			
		||||
 * @param {string} dateFormatString - Format string for the date part.
 | 
			
		||||
 * @returns {React.ReactElement} A <Text as="time"> element with the formatted date/time.
 | 
			
		||||
 */
 | 
			
		||||
export const Time = as<'span', TimeProps & ComponentProps<typeof Text>>(
 | 
			
		||||
  ({ compact, ts, ...props }, ref) => {
 | 
			
		||||
  ({ compact, hour24Clock, dateFormatString, ts, ...props }, ref) => {
 | 
			
		||||
    const formattedTime = timeHourMinute(ts, hour24Clock);
 | 
			
		||||
 | 
			
		||||
    let time = '';
 | 
			
		||||
    if (compact) {
 | 
			
		||||
      time = timeHourMinute(ts);
 | 
			
		||||
      time = formattedTime;
 | 
			
		||||
    } else if (today(ts)) {
 | 
			
		||||
      time = timeHourMinute(ts);
 | 
			
		||||
      time = formattedTime;
 | 
			
		||||
    } else if (yesterday(ts)) {
 | 
			
		||||
      time = `Yesterday ${timeHourMinute(ts)}`;
 | 
			
		||||
      time = `Yesterday ${formattedTime}`;
 | 
			
		||||
    } else {
 | 
			
		||||
      time = `${timeDayMonYear(ts)} ${timeHourMinute(ts)}`;
 | 
			
		||||
      time = `${timeDayMonYear(ts, dateFormatString)} ${formattedTime}`;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,6 +30,7 @@ import { stopPropagation } from '../../../utils/keyboard';
 | 
			
		|||
import { decryptFile, downloadEncryptedMedia, mxcUrlToHttp } from '../../../utils/matrix';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { ModalWide } from '../../../styles/Modal.css';
 | 
			
		||||
import { validBlurHash } from '../../../utils/blurHash';
 | 
			
		||||
 | 
			
		||||
type RenderViewerProps = {
 | 
			
		||||
  src: string;
 | 
			
		||||
| 
						 | 
				
			
			@ -77,7 +78,7 @@ export const ImageContent = as<'div', ImageContentProps>(
 | 
			
		|||
  ) => {
 | 
			
		||||
    const mx = useMatrixClient();
 | 
			
		||||
    const useAuthentication = useMediaAuthentication();
 | 
			
		||||
    const blurHash = info?.[MATRIX_BLUR_HASH_PROPERTY_NAME];
 | 
			
		||||
    const blurHash = validBlurHash(info?.[MATRIX_BLUR_HASH_PROPERTY_NAME]);
 | 
			
		||||
 | 
			
		||||
    const [load, setLoad] = useState(false);
 | 
			
		||||
    const [error, setError] = useState(false);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,6 +32,7 @@ import {
 | 
			
		|||
  mxcUrlToHttp,
 | 
			
		||||
} from '../../../utils/matrix';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { validBlurHash } from '../../../utils/blurHash';
 | 
			
		||||
 | 
			
		||||
type RenderVideoProps = {
 | 
			
		||||
  title: string;
 | 
			
		||||
| 
						 | 
				
			
			@ -73,7 +74,7 @@ export const VideoContent = as<'div', VideoContentProps>(
 | 
			
		|||
  ) => {
 | 
			
		||||
    const mx = useMatrixClient();
 | 
			
		||||
    const useAuthentication = useMediaAuthentication();
 | 
			
		||||
    const blurHash = info.thumbnail_info?.[MATRIX_BLUR_HASH_PROPERTY_NAME];
 | 
			
		||||
    const blurHash = validBlurHash(info.thumbnail_info?.[MATRIX_BLUR_HASH_PROPERTY_NAME]);
 | 
			
		||||
 | 
			
		||||
    const [load, setLoad] = useState(false);
 | 
			
		||||
    const [error, setError] = useState(false);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -16,6 +16,7 @@ export const AbsoluteContainer = style([
 | 
			
		|||
    position: 'absolute',
 | 
			
		||||
    top: 0,
 | 
			
		||||
    left: 0,
 | 
			
		||||
    zIndex: 1,
 | 
			
		||||
    width: '100%',
 | 
			
		||||
    height: '100%',
 | 
			
		||||
  },
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -124,7 +124,7 @@ export const AvatarBase = style({
 | 
			
		|||
 | 
			
		||||
  selectors: {
 | 
			
		||||
    '&:hover': {
 | 
			
		||||
      transform: `translateY(${toRem(-4)})`,
 | 
			
		||||
      transform: `translateY(${toRem(-2)})`,
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -105,6 +105,20 @@ export const PageContent = as<'div'>(({ className, ...props }, ref) => (
 | 
			
		|||
  <div className={classNames(css.PageContent, className)} {...props} ref={ref} />
 | 
			
		||||
));
 | 
			
		||||
 | 
			
		||||
export function PageHeroEmpty({ children }: { children: ReactNode }) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Box
 | 
			
		||||
      className={classNames(ContainerColor({ variant: 'SurfaceVariant' }), css.PageHeroEmpty)}
 | 
			
		||||
      direction="Column"
 | 
			
		||||
      alignItems="Center"
 | 
			
		||||
      justifyContent="Center"
 | 
			
		||||
      gap="200"
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const PageHeroSection = as<'div', ComponentProps<typeof Box>>(
 | 
			
		||||
  ({ className, ...props }, ref) => (
 | 
			
		||||
    <Box
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -92,6 +92,15 @@ export const PageContent = style([
 | 
			
		|||
  },
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
export const PageHeroEmpty = style([
 | 
			
		||||
  DefaultReset,
 | 
			
		||||
  {
 | 
			
		||||
    padding: config.space.S400,
 | 
			
		||||
    borderRadius: config.radii.R400,
 | 
			
		||||
    minHeight: toRem(450),
 | 
			
		||||
  },
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
export const PageHeroSection = style([
 | 
			
		||||
  DefaultReset,
 | 
			
		||||
  {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										80
									
								
								src/app/components/presence/Presence.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								src/app/components/presence/Presence.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,80 @@
 | 
			
		|||
import {
 | 
			
		||||
  as,
 | 
			
		||||
  Badge,
 | 
			
		||||
  Box,
 | 
			
		||||
  color,
 | 
			
		||||
  ContainerColor,
 | 
			
		||||
  MainColor,
 | 
			
		||||
  Text,
 | 
			
		||||
  Tooltip,
 | 
			
		||||
  TooltipProvider,
 | 
			
		||||
  toRem,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import React, { ReactNode, useId } from 'react';
 | 
			
		||||
import * as css from './styles.css';
 | 
			
		||||
import { Presence, usePresenceLabel } from '../../hooks/useUserPresence';
 | 
			
		||||
 | 
			
		||||
const PresenceToColor: Record<Presence, MainColor> = {
 | 
			
		||||
  [Presence.Online]: 'Success',
 | 
			
		||||
  [Presence.Unavailable]: 'Warning',
 | 
			
		||||
  [Presence.Offline]: 'Secondary',
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
type PresenceBadgeProps = {
 | 
			
		||||
  presence: Presence;
 | 
			
		||||
  status?: string;
 | 
			
		||||
  size?: '200' | '300' | '400' | '500';
 | 
			
		||||
};
 | 
			
		||||
export function PresenceBadge({ presence, status, size }: PresenceBadgeProps) {
 | 
			
		||||
  const label = usePresenceLabel();
 | 
			
		||||
  const badgeLabelId = useId();
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <TooltipProvider
 | 
			
		||||
      position="Right"
 | 
			
		||||
      align="Center"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      delay={200}
 | 
			
		||||
      tooltip={
 | 
			
		||||
        <Tooltip id={badgeLabelId}>
 | 
			
		||||
          <Box style={{ maxWidth: toRem(250) }} alignItems="Baseline" gap="100">
 | 
			
		||||
            <Text size="L400">{label[presence]}</Text>
 | 
			
		||||
            {status && <Text size="T200">•</Text>}
 | 
			
		||||
            {status && <Text size="T200">{status}</Text>}
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Tooltip>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      {(triggerRef) => (
 | 
			
		||||
        <Badge
 | 
			
		||||
          aria-labelledby={badgeLabelId}
 | 
			
		||||
          ref={triggerRef}
 | 
			
		||||
          size={size}
 | 
			
		||||
          variant={PresenceToColor[presence]}
 | 
			
		||||
          fill={presence === Presence.Offline ? 'Soft' : 'Solid'}
 | 
			
		||||
          radii="Pill"
 | 
			
		||||
        />
 | 
			
		||||
      )}
 | 
			
		||||
    </TooltipProvider>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type AvatarPresenceProps = {
 | 
			
		||||
  badge: ReactNode;
 | 
			
		||||
  variant?: ContainerColor;
 | 
			
		||||
};
 | 
			
		||||
export const AvatarPresence = as<'div', AvatarPresenceProps>(
 | 
			
		||||
  ({ as: AsAvatarPresence, badge, variant = 'Surface', children, ...props }, ref) => (
 | 
			
		||||
    <Box as={AsAvatarPresence} className={css.AvatarPresence} {...props} ref={ref}>
 | 
			
		||||
      {badge && (
 | 
			
		||||
        <div
 | 
			
		||||
          className={css.AvatarPresenceBadge}
 | 
			
		||||
          style={{ backgroundColor: color[variant].Container }}
 | 
			
		||||
        >
 | 
			
		||||
          {badge}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
      {children}
 | 
			
		||||
    </Box>
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
							
								
								
									
										1
									
								
								src/app/components/presence/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/app/components/presence/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1 @@
 | 
			
		|||
export * from './Presence';
 | 
			
		||||
							
								
								
									
										22
									
								
								src/app/components/presence/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/app/components/presence/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,22 @@
 | 
			
		|||
import { style } from '@vanilla-extract/css';
 | 
			
		||||
import { config } from 'folds';
 | 
			
		||||
 | 
			
		||||
export const AvatarPresence = style({
 | 
			
		||||
  display: 'flex',
 | 
			
		||||
  position: 'relative',
 | 
			
		||||
  flexShrink: 0,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const AvatarPresenceBadge = style({
 | 
			
		||||
  position: 'absolute',
 | 
			
		||||
  bottom: 0,
 | 
			
		||||
  right: 0,
 | 
			
		||||
  transform: 'translate(25%, 25%)',
 | 
			
		||||
  zIndex: 1,
 | 
			
		||||
 | 
			
		||||
  display: 'flex',
 | 
			
		||||
  padding: config.borderWidth.B600,
 | 
			
		||||
  backgroundColor: 'inherit',
 | 
			
		||||
  borderRadius: config.radii.Pill,
 | 
			
		||||
  overflow: 'hidden',
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			@ -15,6 +15,8 @@ import { nameInitials } from '../../utils/common';
 | 
			
		|||
import { useRoomAvatar, useRoomName, useRoomTopic } from '../../hooks/useRoomMeta';
 | 
			
		||||
import { mDirectAtom } from '../../state/mDirectList';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { settingsAtom } from '../../state/settings';
 | 
			
		||||
 | 
			
		||||
export type RoomIntroProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -43,6 +45,8 @@ export const RoomIntro = as<'div', RoomIntroProps>(({ room, ...props }, ref) =>
 | 
			
		|||
    useCallback(async (roomId: string) => mx.joinRoom(roomId), [mx])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column" grow="Yes" gap="500" {...props} ref={ref}>
 | 
			
		||||
      <Box>
 | 
			
		||||
| 
						 | 
				
			
			@ -67,7 +71,7 @@ export const RoomIntro = as<'div', RoomIntroProps>(({ room, ...props }, ref) =>
 | 
			
		|||
            <Text size="T200" priority="300">
 | 
			
		||||
              {'Created by '}
 | 
			
		||||
              <b>@{creatorName}</b>
 | 
			
		||||
              {` on ${timeDayMonthYear(ts)} ${timeHourMinute(ts)}`}
 | 
			
		||||
              {` on ${timeDayMonthYear(ts)} ${timeHourMinute(ts, hour24Clock)}`}
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
| 
						 | 
				
			
			@ -83,7 +87,7 @@ export const RoomIntro = as<'div', RoomIntroProps>(({ room, ...props }, ref) =>
 | 
			
		|||
          {typeof prevRoomId === 'string' &&
 | 
			
		||||
            (mx.getRoom(prevRoomId)?.getMyMembership() === Membership.Join ? (
 | 
			
		||||
              <Button
 | 
			
		||||
                onClick={() => navigateRoom(prevRoomId)}
 | 
			
		||||
                onClick={() => navigateRoom(prevRoomId, createContent?.predecessor?.event_id)}
 | 
			
		||||
                variant="Success"
 | 
			
		||||
                size="300"
 | 
			
		||||
                fill="Soft"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -7,12 +7,31 @@ import * as css from './style.css';
 | 
			
		|||
export const SequenceCard = as<
 | 
			
		||||
  'div',
 | 
			
		||||
  ComponentProps<typeof Box> & ContainerColorVariants & css.SequenceCardVariants
 | 
			
		||||
>(({ className, variant, firstChild, lastChild, outlined, ...props }, ref) => (
 | 
			
		||||
  <Box
 | 
			
		||||
    className={classNames(css.SequenceCard({ outlined }), ContainerColor({ variant }), className)}
 | 
			
		||||
    data-first-child={firstChild}
 | 
			
		||||
    data-last-child={lastChild}
 | 
			
		||||
    {...props}
 | 
			
		||||
    ref={ref}
 | 
			
		||||
  />
 | 
			
		||||
));
 | 
			
		||||
>(
 | 
			
		||||
  (
 | 
			
		||||
    {
 | 
			
		||||
      as: AsSequenceCard = 'div',
 | 
			
		||||
      className,
 | 
			
		||||
      variant,
 | 
			
		||||
      radii,
 | 
			
		||||
      firstChild,
 | 
			
		||||
      lastChild,
 | 
			
		||||
      outlined,
 | 
			
		||||
      ...props
 | 
			
		||||
    },
 | 
			
		||||
    ref
 | 
			
		||||
  ) => (
 | 
			
		||||
    <Box
 | 
			
		||||
      as={AsSequenceCard}
 | 
			
		||||
      className={classNames(
 | 
			
		||||
        css.SequenceCard({ radii, outlined }),
 | 
			
		||||
        ContainerColor({ variant }),
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      data-first-child={firstChild}
 | 
			
		||||
      data-last-child={lastChild}
 | 
			
		||||
      {...props}
 | 
			
		||||
      ref={ref}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -3,6 +3,7 @@ import { RecipeVariants, recipe } from '@vanilla-extract/recipes';
 | 
			
		|||
import { config } from 'folds';
 | 
			
		||||
 | 
			
		||||
const outlinedWidth = createVar('0');
 | 
			
		||||
const radii = createVar(config.radii.R400);
 | 
			
		||||
export const SequenceCard = recipe({
 | 
			
		||||
  base: {
 | 
			
		||||
    vars: {
 | 
			
		||||
| 
						 | 
				
			
			@ -13,33 +14,59 @@ export const SequenceCard = recipe({
 | 
			
		|||
    borderBottomWidth: 0,
 | 
			
		||||
    selectors: {
 | 
			
		||||
      '&:first-child, :not(&) + &': {
 | 
			
		||||
        borderTopLeftRadius: config.radii.R400,
 | 
			
		||||
        borderTopRightRadius: config.radii.R400,
 | 
			
		||||
        borderTopLeftRadius: [radii],
 | 
			
		||||
        borderTopRightRadius: [radii],
 | 
			
		||||
      },
 | 
			
		||||
      '&:last-child, &:not(:has(+&))': {
 | 
			
		||||
        borderBottomLeftRadius: config.radii.R400,
 | 
			
		||||
        borderBottomRightRadius: config.radii.R400,
 | 
			
		||||
        borderBottomLeftRadius: [radii],
 | 
			
		||||
        borderBottomRightRadius: [radii],
 | 
			
		||||
        borderBottomWidth: outlinedWidth,
 | 
			
		||||
      },
 | 
			
		||||
      [`&[data-first-child="true"]`]: {
 | 
			
		||||
        borderTopLeftRadius: config.radii.R400,
 | 
			
		||||
        borderTopRightRadius: config.radii.R400,
 | 
			
		||||
        borderTopLeftRadius: [radii],
 | 
			
		||||
        borderTopRightRadius: [radii],
 | 
			
		||||
      },
 | 
			
		||||
      [`&[data-first-child="false"]`]: {
 | 
			
		||||
        borderTopLeftRadius: 0,
 | 
			
		||||
        borderTopRightRadius: 0,
 | 
			
		||||
      },
 | 
			
		||||
      [`&[data-last-child="true"]`]: {
 | 
			
		||||
        borderBottomLeftRadius: config.radii.R400,
 | 
			
		||||
        borderBottomRightRadius: config.radii.R400,
 | 
			
		||||
        borderBottomLeftRadius: [radii],
 | 
			
		||||
        borderBottomRightRadius: [radii],
 | 
			
		||||
      },
 | 
			
		||||
      [`&[data-last-child="false"]`]: {
 | 
			
		||||
        borderBottomLeftRadius: 0,
 | 
			
		||||
        borderBottomRightRadius: 0,
 | 
			
		||||
      },
 | 
			
		||||
 | 
			
		||||
      'button&': {
 | 
			
		||||
        cursor: 'pointer',
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  variants: {
 | 
			
		||||
    radii: {
 | 
			
		||||
      '0': {
 | 
			
		||||
        vars: {
 | 
			
		||||
          [radii]: config.radii.R0,
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      '300': {
 | 
			
		||||
        vars: {
 | 
			
		||||
          [radii]: config.radii.R300,
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      '400': {
 | 
			
		||||
        vars: {
 | 
			
		||||
          [radii]: config.radii.R400,
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      '500': {
 | 
			
		||||
        vars: {
 | 
			
		||||
          [radii]: config.radii.R500,
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    outlined: {
 | 
			
		||||
      true: {
 | 
			
		||||
        vars: {
 | 
			
		||||
| 
						 | 
				
			
			@ -48,5 +75,8 @@ export const SequenceCard = recipe({
 | 
			
		|||
      },
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  defaultVariants: {
 | 
			
		||||
    radii: '400',
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
export type SequenceCardVariants = RecipeVariants<typeof SequenceCard>;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -24,7 +24,7 @@ export const TextViewerContent = forwardRef<HTMLPreElement, TextViewerContentPro
 | 
			
		|||
    >
 | 
			
		||||
      <ErrorBoundary fallback={<code>{text}</code>}>
 | 
			
		||||
        <Suspense fallback={<code>{text}</code>}>
 | 
			
		||||
          <ReactPrism>{(codeRef) => <code ref={codeRef}>{text}</code>}</ReactPrism>
 | 
			
		||||
          <ReactPrism key={text}>{(codeRef) => <code ref={codeRef}>{text}</code>}</ReactPrism>
 | 
			
		||||
        </Suspense>
 | 
			
		||||
      </ErrorBoundary>
 | 
			
		||||
    </Text>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										129
									
								
								src/app/components/time-date/DatePicker.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								src/app/components/time-date/DatePicker.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,129 @@
 | 
			
		|||
import React, { forwardRef } from 'react';
 | 
			
		||||
import { Menu, Box, Text, Chip } from 'folds';
 | 
			
		||||
import dayjs from 'dayjs';
 | 
			
		||||
import * as css from './styles.css';
 | 
			
		||||
import { PickerColumn } from './PickerColumn';
 | 
			
		||||
import { dateFor, daysInMonth, daysToMs } from '../../utils/time';
 | 
			
		||||
 | 
			
		||||
type DatePickerProps = {
 | 
			
		||||
  min: number;
 | 
			
		||||
  max: number;
 | 
			
		||||
  value: number;
 | 
			
		||||
  onChange: (value: number) => void;
 | 
			
		||||
};
 | 
			
		||||
export const DatePicker = forwardRef<HTMLDivElement, DatePickerProps>(
 | 
			
		||||
  ({ min, max, value, onChange }, ref) => {
 | 
			
		||||
    const selectedYear = dayjs(value).year();
 | 
			
		||||
    const selectedMonth = dayjs(value).month() + 1;
 | 
			
		||||
    const selectedDay = dayjs(value).date();
 | 
			
		||||
 | 
			
		||||
    const handleSubmit = (newValue: number) => {
 | 
			
		||||
      onChange(Math.min(Math.max(min, newValue), max));
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleDay = (day: number) => {
 | 
			
		||||
      const seconds = daysToMs(day);
 | 
			
		||||
      const lastSeconds = daysToMs(selectedDay);
 | 
			
		||||
      const newValue = value + (seconds - lastSeconds);
 | 
			
		||||
      handleSubmit(newValue);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleMonthAndYear = (month: number, year: number) => {
 | 
			
		||||
      const mDays = daysInMonth(month, year);
 | 
			
		||||
      const currentDate = dateFor(selectedYear, selectedMonth, selectedDay);
 | 
			
		||||
      const time = value - currentDate;
 | 
			
		||||
 | 
			
		||||
      const newDate = dateFor(year, month, mDays < selectedDay ? mDays : selectedDay);
 | 
			
		||||
 | 
			
		||||
      const newValue = newDate + time;
 | 
			
		||||
      handleSubmit(newValue);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleMonth = (month: number) => {
 | 
			
		||||
      handleMonthAndYear(month, selectedYear);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleYear = (year: number) => {
 | 
			
		||||
      handleMonthAndYear(selectedMonth, year);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const minYear = dayjs(min).year();
 | 
			
		||||
    const maxYear = dayjs(max).year();
 | 
			
		||||
    const yearsRange = maxYear - minYear + 1;
 | 
			
		||||
 | 
			
		||||
    const minMonth = dayjs(min).month() + 1;
 | 
			
		||||
    const maxMonth = dayjs(max).month() + 1;
 | 
			
		||||
 | 
			
		||||
    const minDay = dayjs(min).date();
 | 
			
		||||
    const maxDay = dayjs(max).date();
 | 
			
		||||
    return (
 | 
			
		||||
      <Menu className={css.PickerMenu} ref={ref}>
 | 
			
		||||
        <Box direction="Row" gap="200" className={css.PickerContainer}>
 | 
			
		||||
          <PickerColumn title="Day">
 | 
			
		||||
            {Array.from(Array(daysInMonth(selectedMonth, selectedYear)).keys())
 | 
			
		||||
              .map((i) => i + 1)
 | 
			
		||||
              .map((day) => (
 | 
			
		||||
                <Chip
 | 
			
		||||
                  key={day}
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  variant={selectedDay === day ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                  fill="None"
 | 
			
		||||
                  radii="300"
 | 
			
		||||
                  aria-selected={selectedDay === day}
 | 
			
		||||
                  onClick={() => handleDay(day)}
 | 
			
		||||
                  disabled={
 | 
			
		||||
                    (selectedYear === minYear && selectedMonth === minMonth && day < minDay) ||
 | 
			
		||||
                    (selectedYear === maxYear && selectedMonth === maxMonth && day > maxDay)
 | 
			
		||||
                  }
 | 
			
		||||
                >
 | 
			
		||||
                  <Text size="T300">{day}</Text>
 | 
			
		||||
                </Chip>
 | 
			
		||||
              ))}
 | 
			
		||||
          </PickerColumn>
 | 
			
		||||
          <PickerColumn title="Month">
 | 
			
		||||
            {Array.from(Array(12).keys())
 | 
			
		||||
              .map((i) => i + 1)
 | 
			
		||||
              .map((month) => (
 | 
			
		||||
                <Chip
 | 
			
		||||
                  key={month}
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  variant={selectedMonth === month ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                  fill="None"
 | 
			
		||||
                  radii="300"
 | 
			
		||||
                  aria-selected={selectedMonth === month}
 | 
			
		||||
                  onClick={() => handleMonth(month)}
 | 
			
		||||
                  disabled={
 | 
			
		||||
                    (selectedYear === minYear && month < minMonth) ||
 | 
			
		||||
                    (selectedYear === maxYear && month > maxMonth)
 | 
			
		||||
                  }
 | 
			
		||||
                >
 | 
			
		||||
                  <Text size="T300">
 | 
			
		||||
                    {dayjs()
 | 
			
		||||
                      .month(month - 1)
 | 
			
		||||
                      .format('MMM')}
 | 
			
		||||
                  </Text>
 | 
			
		||||
                </Chip>
 | 
			
		||||
              ))}
 | 
			
		||||
          </PickerColumn>
 | 
			
		||||
          <PickerColumn title="Year">
 | 
			
		||||
            {Array.from(Array(yearsRange).keys())
 | 
			
		||||
              .map((i) => minYear + i)
 | 
			
		||||
              .map((year) => (
 | 
			
		||||
                <Chip
 | 
			
		||||
                  key={year}
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  variant={selectedYear === year ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                  fill="None"
 | 
			
		||||
                  radii="300"
 | 
			
		||||
                  aria-selected={selectedYear === year}
 | 
			
		||||
                  onClick={() => handleYear(year)}
 | 
			
		||||
                >
 | 
			
		||||
                  <Text size="T300">{year}</Text>
 | 
			
		||||
                </Chip>
 | 
			
		||||
              ))}
 | 
			
		||||
          </PickerColumn>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Menu>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
);
 | 
			
		||||
							
								
								
									
										23
									
								
								src/app/components/time-date/PickerColumn.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/app/components/time-date/PickerColumn.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,23 @@
 | 
			
		|||
import React, { ReactNode } from 'react';
 | 
			
		||||
import { Box, Text, Scroll } from 'folds';
 | 
			
		||||
import { CutoutCard } from '../cutout-card';
 | 
			
		||||
import * as css from './styles.css';
 | 
			
		||||
 | 
			
		||||
export function PickerColumn({ title, children }: { title: string; children: ReactNode }) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column" gap="100">
 | 
			
		||||
      <Text className={css.PickerColumnLabel} size="L400">
 | 
			
		||||
        {title}
 | 
			
		||||
      </Text>
 | 
			
		||||
      <Box grow="Yes">
 | 
			
		||||
        <CutoutCard variant="Background">
 | 
			
		||||
          <Scroll variant="Background" size="300" hideTrack>
 | 
			
		||||
            <Box className={css.PickerColumnContent} direction="Column" gap="100">
 | 
			
		||||
              {children}
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Scroll>
 | 
			
		||||
        </CutoutCard>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										153
									
								
								src/app/components/time-date/TimePicker.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								src/app/components/time-date/TimePicker.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,153 @@
 | 
			
		|||
import React, { forwardRef } from 'react';
 | 
			
		||||
import { Menu, Box, Text, Chip } from 'folds';
 | 
			
		||||
import dayjs from 'dayjs';
 | 
			
		||||
import * as css from './styles.css';
 | 
			
		||||
import { PickerColumn } from './PickerColumn';
 | 
			
		||||
import { hour12to24, hour24to12, hoursToMs, inSameDay, minutesToMs } from '../../utils/time';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { settingsAtom } from '../../state/settings';
 | 
			
		||||
 | 
			
		||||
type TimePickerProps = {
 | 
			
		||||
  min: number;
 | 
			
		||||
  max: number;
 | 
			
		||||
  value: number;
 | 
			
		||||
  onChange: (value: number) => void;
 | 
			
		||||
};
 | 
			
		||||
export const TimePicker = forwardRef<HTMLDivElement, TimePickerProps>(
 | 
			
		||||
  ({ min, max, value, onChange }, ref) => {
 | 
			
		||||
    const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
 | 
			
		||||
    const hour24 = dayjs(value).hour();
 | 
			
		||||
 | 
			
		||||
    const selectedHour = hour24Clock ? hour24 : hour24to12(hour24);
 | 
			
		||||
    const selectedMinute = dayjs(value).minute();
 | 
			
		||||
    const selectedPM = hour24 >= 12;
 | 
			
		||||
 | 
			
		||||
    const handleSubmit = (newValue: number) => {
 | 
			
		||||
      onChange(Math.min(Math.max(min, newValue), max));
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleHour = (hour: number) => {
 | 
			
		||||
      const seconds = hoursToMs(hour24Clock ? hour : hour12to24(hour, selectedPM));
 | 
			
		||||
      const lastSeconds = hoursToMs(hour24);
 | 
			
		||||
      const newValue = value + (seconds - lastSeconds);
 | 
			
		||||
      handleSubmit(newValue);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleMinute = (minute: number) => {
 | 
			
		||||
      const seconds = minutesToMs(minute);
 | 
			
		||||
      const lastSeconds = minutesToMs(selectedMinute);
 | 
			
		||||
      const newValue = value + (seconds - lastSeconds);
 | 
			
		||||
      handleSubmit(newValue);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handlePeriod = (pm: boolean) => {
 | 
			
		||||
      const seconds = hoursToMs(hour12to24(selectedHour, pm));
 | 
			
		||||
      const lastSeconds = hoursToMs(hour24);
 | 
			
		||||
      const newValue = value + (seconds - lastSeconds);
 | 
			
		||||
      handleSubmit(newValue);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const minHour24 = dayjs(min).hour();
 | 
			
		||||
    const maxHour24 = dayjs(max).hour();
 | 
			
		||||
 | 
			
		||||
    const minMinute = dayjs(min).minute();
 | 
			
		||||
    const maxMinute = dayjs(max).minute();
 | 
			
		||||
    const minPM = minHour24 >= 12;
 | 
			
		||||
    const maxPM = maxHour24 >= 12;
 | 
			
		||||
 | 
			
		||||
    const minDay = inSameDay(min, value);
 | 
			
		||||
    const maxDay = inSameDay(max, value);
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <Menu className={css.PickerMenu} ref={ref}>
 | 
			
		||||
        <Box direction="Row" gap="200" className={css.PickerContainer}>
 | 
			
		||||
          <PickerColumn title="Hour">
 | 
			
		||||
            {hour24Clock
 | 
			
		||||
              ? Array.from(Array(24).keys()).map((hour) => (
 | 
			
		||||
                  <Chip
 | 
			
		||||
                    key={hour}
 | 
			
		||||
                    size="500"
 | 
			
		||||
                    variant={hour === selectedHour ? 'Primary' : 'Background'}
 | 
			
		||||
                    fill="None"
 | 
			
		||||
                    radii="300"
 | 
			
		||||
                    aria-selected={hour === selectedHour}
 | 
			
		||||
                    onClick={() => handleHour(hour)}
 | 
			
		||||
                    disabled={(minDay && hour < minHour24) || (maxDay && hour > maxHour24)}
 | 
			
		||||
                  >
 | 
			
		||||
                    <Text size="T300">{hour < 10 ? `0${hour}` : hour}</Text>
 | 
			
		||||
                  </Chip>
 | 
			
		||||
                ))
 | 
			
		||||
              : Array.from(Array(12).keys())
 | 
			
		||||
                  .map((i) => {
 | 
			
		||||
                    if (i === 0) return 12;
 | 
			
		||||
                    return i;
 | 
			
		||||
                  })
 | 
			
		||||
                  .map((hour) => (
 | 
			
		||||
                    <Chip
 | 
			
		||||
                      key={hour}
 | 
			
		||||
                      size="500"
 | 
			
		||||
                      variant={hour === selectedHour ? 'Primary' : 'Background'}
 | 
			
		||||
                      fill="None"
 | 
			
		||||
                      radii="300"
 | 
			
		||||
                      aria-selected={hour === selectedHour}
 | 
			
		||||
                      onClick={() => handleHour(hour)}
 | 
			
		||||
                      disabled={
 | 
			
		||||
                        (minDay && hour12to24(hour, selectedPM) < minHour24) ||
 | 
			
		||||
                        (maxDay && hour12to24(hour, selectedPM) > maxHour24)
 | 
			
		||||
                      }
 | 
			
		||||
                    >
 | 
			
		||||
                      <Text size="T300">{hour < 10 ? `0${hour}` : hour}</Text>
 | 
			
		||||
                    </Chip>
 | 
			
		||||
                  ))}
 | 
			
		||||
          </PickerColumn>
 | 
			
		||||
          <PickerColumn title="Minutes">
 | 
			
		||||
            {Array.from(Array(60).keys()).map((minute) => (
 | 
			
		||||
              <Chip
 | 
			
		||||
                key={minute}
 | 
			
		||||
                size="500"
 | 
			
		||||
                variant={minute === selectedMinute ? 'Primary' : 'Background'}
 | 
			
		||||
                fill="None"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                aria-selected={minute === selectedMinute}
 | 
			
		||||
                onClick={() => handleMinute(minute)}
 | 
			
		||||
                disabled={
 | 
			
		||||
                  (minDay && hour24 === minHour24 && minute < minMinute) ||
 | 
			
		||||
                  (maxDay && hour24 === maxHour24 && minute > maxMinute)
 | 
			
		||||
                }
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="T300">{minute < 10 ? `0${minute}` : minute}</Text>
 | 
			
		||||
              </Chip>
 | 
			
		||||
            ))}
 | 
			
		||||
          </PickerColumn>
 | 
			
		||||
          {!hour24Clock && (
 | 
			
		||||
            <PickerColumn title="Period">
 | 
			
		||||
              <Chip
 | 
			
		||||
                size="500"
 | 
			
		||||
                variant={!selectedPM ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                fill="None"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                aria-selected={!selectedPM}
 | 
			
		||||
                onClick={() => handlePeriod(false)}
 | 
			
		||||
                disabled={minDay && minPM}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="T300">AM</Text>
 | 
			
		||||
              </Chip>
 | 
			
		||||
              <Chip
 | 
			
		||||
                size="500"
 | 
			
		||||
                variant={selectedPM ? 'Primary' : 'SurfaceVariant'}
 | 
			
		||||
                fill="None"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                aria-selected={selectedPM}
 | 
			
		||||
                onClick={() => handlePeriod(true)}
 | 
			
		||||
                disabled={maxDay && !maxPM}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="T300">PM</Text>
 | 
			
		||||
              </Chip>
 | 
			
		||||
            </PickerColumn>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Menu>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
);
 | 
			
		||||
							
								
								
									
										2
									
								
								src/app/components/time-date/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/app/components/time-date/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,2 @@
 | 
			
		|||
export * from './TimePicker';
 | 
			
		||||
export * from './DatePicker';
 | 
			
		||||
							
								
								
									
										16
									
								
								src/app/components/time-date/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/app/components/time-date/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,16 @@
 | 
			
		|||
import { style } from '@vanilla-extract/css';
 | 
			
		||||
import { config, toRem } from 'folds';
 | 
			
		||||
 | 
			
		||||
export const PickerMenu = style({
 | 
			
		||||
  padding: config.space.S200,
 | 
			
		||||
});
 | 
			
		||||
export const PickerContainer = style({
 | 
			
		||||
  maxHeight: toRem(250),
 | 
			
		||||
});
 | 
			
		||||
export const PickerColumnLabel = style({
 | 
			
		||||
  padding: config.space.S200,
 | 
			
		||||
});
 | 
			
		||||
export const PickerColumnContent = style({
 | 
			
		||||
  padding: config.space.S200,
 | 
			
		||||
  paddingRight: 0,
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			@ -4,7 +4,8 @@ import { UploadCard, UploadCardError, CompactUploadCardProgress } from './Upload
 | 
			
		|||
import { TUploadAtom, UploadStatus, UploadSuccess, useBindUploadAtom } from '../../state/upload';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { TUploadContent } from '../../utils/matrix';
 | 
			
		||||
import { getFileTypeIcon } from '../../utils/common';
 | 
			
		||||
import { bytesToSize, getFileTypeIcon } from '../../utils/common';
 | 
			
		||||
import { useMediaConfig } from '../../hooks/useMediaConfig';
 | 
			
		||||
 | 
			
		||||
type CompactUploadCardRendererProps = {
 | 
			
		||||
  isEncrypted?: boolean;
 | 
			
		||||
| 
						 | 
				
			
			@ -19,10 +20,16 @@ export function CompactUploadCardRenderer({
 | 
			
		|||
  onComplete,
 | 
			
		||||
}: CompactUploadCardRendererProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const mediaConfig = useMediaConfig();
 | 
			
		||||
  const allowSize = mediaConfig['m.upload.size'] || Infinity;
 | 
			
		||||
 | 
			
		||||
  const { upload, startUpload, cancelUpload } = useBindUploadAtom(mx, uploadAtom, isEncrypted);
 | 
			
		||||
  const { file } = upload;
 | 
			
		||||
  const fileSizeExceeded = file.size >= allowSize;
 | 
			
		||||
 | 
			
		||||
  if (upload.status === UploadStatus.Idle) startUpload();
 | 
			
		||||
  if (upload.status === UploadStatus.Idle && !fileSizeExceeded) {
 | 
			
		||||
    startUpload();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const removeUpload = () => {
 | 
			
		||||
    cancelUpload();
 | 
			
		||||
| 
						 | 
				
			
			@ -76,7 +83,7 @@ export function CompactUploadCardRenderer({
 | 
			
		|||
        </>
 | 
			
		||||
      ) : (
 | 
			
		||||
        <>
 | 
			
		||||
          {upload.status === UploadStatus.Idle && (
 | 
			
		||||
          {upload.status === UploadStatus.Idle && !fileSizeExceeded && (
 | 
			
		||||
            <CompactUploadCardProgress sentBytes={0} totalBytes={file.size} />
 | 
			
		||||
          )}
 | 
			
		||||
          {upload.status === UploadStatus.Loading && (
 | 
			
		||||
| 
						 | 
				
			
			@ -87,6 +94,15 @@ export function CompactUploadCardRenderer({
 | 
			
		|||
              <Text size="T200">{upload.error.message}</Text>
 | 
			
		||||
            </UploadCardError>
 | 
			
		||||
          )}
 | 
			
		||||
          {upload.status === UploadStatus.Idle && fileSizeExceeded && (
 | 
			
		||||
            <UploadCardError>
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                The file size exceeds the limit. Maximum allowed size is{' '}
 | 
			
		||||
                <b>{bytesToSize(allowSize)}</b>, but the uploaded file is{' '}
 | 
			
		||||
                <b>{bytesToSize(file.size)}</b>.
 | 
			
		||||
              </Text>
 | 
			
		||||
            </UploadCardError>
 | 
			
		||||
          )}
 | 
			
		||||
        </>
 | 
			
		||||
      )}
 | 
			
		||||
    </UploadCard>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,13 +4,14 @@ import { UploadCard, UploadCardError, UploadCardProgress } from './UploadCard';
 | 
			
		|||
import { UploadStatus, UploadSuccess, useBindUploadAtom } from '../../state/upload';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { TUploadContent } from '../../utils/matrix';
 | 
			
		||||
import { getFileTypeIcon } from '../../utils/common';
 | 
			
		||||
import { bytesToSize, getFileTypeIcon } from '../../utils/common';
 | 
			
		||||
import {
 | 
			
		||||
  roomUploadAtomFamily,
 | 
			
		||||
  TUploadItem,
 | 
			
		||||
  TUploadMetadata,
 | 
			
		||||
} from '../../state/room/roomInputDrafts';
 | 
			
		||||
import { useObjectURL } from '../../hooks/useObjectURL';
 | 
			
		||||
import { useMediaConfig } from '../../hooks/useMediaConfig';
 | 
			
		||||
 | 
			
		||||
type PreviewImageProps = {
 | 
			
		||||
  fileItem: TUploadItem;
 | 
			
		||||
| 
						 | 
				
			
			@ -112,12 +113,18 @@ export function UploadCardRenderer({
 | 
			
		|||
  onComplete,
 | 
			
		||||
}: UploadCardRendererProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const mediaConfig = useMediaConfig();
 | 
			
		||||
  const allowSize = mediaConfig['m.upload.size'] || Infinity;
 | 
			
		||||
 | 
			
		||||
  const uploadAtom = roomUploadAtomFamily(fileItem.file);
 | 
			
		||||
  const { metadata } = fileItem;
 | 
			
		||||
  const { upload, startUpload, cancelUpload } = useBindUploadAtom(mx, uploadAtom, isEncrypted);
 | 
			
		||||
  const { file } = upload;
 | 
			
		||||
  const fileSizeExceeded = file.size >= allowSize;
 | 
			
		||||
 | 
			
		||||
  if (upload.status === UploadStatus.Idle) startUpload();
 | 
			
		||||
  if (upload.status === UploadStatus.Idle && !fileSizeExceeded) {
 | 
			
		||||
    startUpload();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const handleSpoiler = (marked: boolean) => {
 | 
			
		||||
    setMetadata(fileItem, { ...metadata, markedAsSpoiler: marked });
 | 
			
		||||
| 
						 | 
				
			
			@ -175,7 +182,7 @@ export function UploadCardRenderer({
 | 
			
		|||
              <PreviewVideo fileItem={fileItem} />
 | 
			
		||||
            </MediaPreview>
 | 
			
		||||
          )}
 | 
			
		||||
          {upload.status === UploadStatus.Idle && (
 | 
			
		||||
          {upload.status === UploadStatus.Idle && !fileSizeExceeded && (
 | 
			
		||||
            <UploadCardProgress sentBytes={0} totalBytes={file.size} />
 | 
			
		||||
          )}
 | 
			
		||||
          {upload.status === UploadStatus.Loading && (
 | 
			
		||||
| 
						 | 
				
			
			@ -186,6 +193,15 @@ export function UploadCardRenderer({
 | 
			
		|||
              <Text size="T200">{upload.error.message}</Text>
 | 
			
		||||
            </UploadCardError>
 | 
			
		||||
          )}
 | 
			
		||||
          {upload.status === UploadStatus.Idle && fileSizeExceeded && (
 | 
			
		||||
            <UploadCardError>
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                The file size exceeds the limit. Maximum allowed size is{' '}
 | 
			
		||||
                <b>{bytesToSize(allowSize)}</b>, but the uploaded file is{' '}
 | 
			
		||||
                <b>{bytesToSize(file.size)}</b>.
 | 
			
		||||
              </Text>
 | 
			
		||||
            </UploadCardError>
 | 
			
		||||
          )}
 | 
			
		||||
        </>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										101
									
								
								src/app/components/user-profile/CreatorChip.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								src/app/components/user-profile/CreatorChip.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,101 @@
 | 
			
		|||
import { Chip, config, Icon, Icons, Menu, MenuItem, PopOut, RectCords, Text } from 'folds';
 | 
			
		||||
import React, { MouseEventHandler, useState } from 'react';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import { useRoomCreatorsTag } from '../../hooks/useRoomCreatorsTag';
 | 
			
		||||
import { PowerColorBadge, PowerIcon } from '../power';
 | 
			
		||||
import { getPowerTagIconSrc } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { useRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { useOpenRoomSettings } from '../../state/hooks/roomSettings';
 | 
			
		||||
import { useOpenSpaceSettings } from '../../state/hooks/spaceSettings';
 | 
			
		||||
import { SpaceSettingsPage } from '../../state/spaceSettings';
 | 
			
		||||
import { RoomSettingsPage } from '../../state/roomSettings';
 | 
			
		||||
 | 
			
		||||
export function CreatorChip() {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
  const openRoomSettings = useOpenRoomSettings();
 | 
			
		||||
  const openSpaceSettings = useOpenSpaceSettings();
 | 
			
		||||
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
  const tag = useRoomCreatorsTag();
 | 
			
		||||
  const tagIconSrc = tag.icon && getPowerTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position="Bottom"
 | 
			
		||||
      align="Start"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      content={
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: close,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
            isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
            isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Menu>
 | 
			
		||||
            <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Surface"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  if (room.isSpaceRoom()) {
 | 
			
		||||
                    openSpaceSettings(
 | 
			
		||||
                      room.roomId,
 | 
			
		||||
                      space?.roomId,
 | 
			
		||||
                      SpaceSettingsPage.PermissionsPage
 | 
			
		||||
                    );
 | 
			
		||||
                  } else {
 | 
			
		||||
                    openRoomSettings(room.roomId, space?.roomId, RoomSettingsPage.PermissionsPage);
 | 
			
		||||
                  }
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Manage Powers</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
            </div>
 | 
			
		||||
          </Menu>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Chip
 | 
			
		||||
        variant="Success"
 | 
			
		||||
        outlined
 | 
			
		||||
        radii="Pill"
 | 
			
		||||
        before={
 | 
			
		||||
          cords ? (
 | 
			
		||||
            <Icon size="50" src={Icons.ChevronBottom} />
 | 
			
		||||
          ) : (
 | 
			
		||||
            <PowerColorBadge color={tag.color} />
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        after={tagIconSrc ? <PowerIcon size="50" iconSrc={tagIconSrc} /> : undefined}
 | 
			
		||||
        onClick={open}
 | 
			
		||||
        aria-pressed={!!cords}
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300" truncate>
 | 
			
		||||
          {tag.name}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										357
									
								
								src/app/components/user-profile/PowerChip.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										357
									
								
								src/app/components/user-profile/PowerChip.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,357 @@
 | 
			
		|||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  Chip,
 | 
			
		||||
  config,
 | 
			
		||||
  Dialog,
 | 
			
		||||
  Header,
 | 
			
		||||
  Icon,
 | 
			
		||||
  IconButton,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Line,
 | 
			
		||||
  Menu,
 | 
			
		||||
  MenuItem,
 | 
			
		||||
  Overlay,
 | 
			
		||||
  OverlayBackdrop,
 | 
			
		||||
  OverlayCenter,
 | 
			
		||||
  PopOut,
 | 
			
		||||
  RectCords,
 | 
			
		||||
  Spinner,
 | 
			
		||||
  Text,
 | 
			
		||||
  toRem,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import React, { MouseEventHandler, useCallback, useState } from 'react';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { PowerColorBadge, PowerIcon } from '../power';
 | 
			
		||||
import { useGetMemberPowerLevel, usePowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { getPowers, usePowerLevelTags } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { useOpenRoomSettings } from '../../state/hooks/roomSettings';
 | 
			
		||||
import { RoomSettingsPage } from '../../state/roomSettings';
 | 
			
		||||
import { useRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { CutoutCard } from '../cutout-card';
 | 
			
		||||
import { useOpenSpaceSettings } from '../../state/hooks/spaceSettings';
 | 
			
		||||
import { SpaceSettingsPage } from '../../state/spaceSettings';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { BreakWord } from '../../styles/Text.css';
 | 
			
		||||
import { getPowerTagIconSrc, useGetMemberPowerTag } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { useMemberPowerCompare } from '../../hooks/useMemberPowerCompare';
 | 
			
		||||
 | 
			
		||||
type SelfDemoteAlertProps = {
 | 
			
		||||
  power: number;
 | 
			
		||||
  onCancel: () => void;
 | 
			
		||||
  onChange: (power: number) => void;
 | 
			
		||||
};
 | 
			
		||||
function SelfDemoteAlert({ power, onCancel, onChange }: SelfDemoteAlertProps) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
      <OverlayCenter>
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: onCancel,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Dialog variant="Surface">
 | 
			
		||||
            <Header
 | 
			
		||||
              style={{ padding: `0 ${config.space.S200} 0 ${config.space.S400}` }}
 | 
			
		||||
              variant="Surface"
 | 
			
		||||
              size="500"
 | 
			
		||||
            >
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Text size="H4">Self Demotion</Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <IconButton size="300" onClick={onCancel} radii="300">
 | 
			
		||||
                <Icon src={Icons.Cross} />
 | 
			
		||||
              </IconButton>
 | 
			
		||||
            </Header>
 | 
			
		||||
            <Box style={{ padding: config.space.S400, paddingTop: 0 }} direction="Column" gap="500">
 | 
			
		||||
              <Box direction="Column" gap="200">
 | 
			
		||||
                <Text priority="400">
 | 
			
		||||
                  You are about to demote yourself! You will not be able to regain this power
 | 
			
		||||
                  yourself. Are you sure?
 | 
			
		||||
                </Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Box direction="Column" gap="200">
 | 
			
		||||
                <Button type="submit" variant="Warning" onClick={() => onChange(power)}>
 | 
			
		||||
                  <Text size="B400">Demote</Text>
 | 
			
		||||
                </Button>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Dialog>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      </OverlayCenter>
 | 
			
		||||
    </Overlay>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type SharedPowerAlertProps = {
 | 
			
		||||
  power: number;
 | 
			
		||||
  onCancel: () => void;
 | 
			
		||||
  onChange: (power: number) => void;
 | 
			
		||||
};
 | 
			
		||||
function SharedPowerAlert({ power, onCancel, onChange }: SharedPowerAlertProps) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
      <OverlayCenter>
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: onCancel,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Dialog variant="Surface">
 | 
			
		||||
            <Header
 | 
			
		||||
              style={{ padding: `0 ${config.space.S200} 0 ${config.space.S400}` }}
 | 
			
		||||
              variant="Surface"
 | 
			
		||||
              size="500"
 | 
			
		||||
            >
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Text size="H4">Shared Power</Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <IconButton size="300" onClick={onCancel} radii="300">
 | 
			
		||||
                <Icon src={Icons.Cross} />
 | 
			
		||||
              </IconButton>
 | 
			
		||||
            </Header>
 | 
			
		||||
            <Box style={{ padding: config.space.S400, paddingTop: 0 }} direction="Column" gap="500">
 | 
			
		||||
              <Box direction="Column" gap="200">
 | 
			
		||||
                <Text priority="400">
 | 
			
		||||
                  You are promoting the user to have the same power as yourself! You will not be
 | 
			
		||||
                  able to change their power afterward. Are you sure?
 | 
			
		||||
                </Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Box direction="Column" gap="200">
 | 
			
		||||
                <Button type="submit" variant="Warning" onClick={() => onChange(power)}>
 | 
			
		||||
                  <Text size="B400">Promote</Text>
 | 
			
		||||
                </Button>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Dialog>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      </OverlayCenter>
 | 
			
		||||
    </Overlay>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function PowerChip({ userId }: { userId: string }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const openRoomSettings = useOpenRoomSettings();
 | 
			
		||||
  const openSpaceSettings = useOpenSpaceSettings();
 | 
			
		||||
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const getMemberPowerLevel = useGetMemberPowerLevel(powerLevels);
 | 
			
		||||
  const { hasMorePower } = useMemberPowerCompare(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const getMemberPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const myUserId = mx.getSafeUserId();
 | 
			
		||||
  const canChangePowers =
 | 
			
		||||
    permissions.stateEvent(StateEvent.RoomPowerLevels, myUserId) &&
 | 
			
		||||
    (myUserId === userId ? true : hasMorePower(myUserId, userId));
 | 
			
		||||
 | 
			
		||||
  const tag = getMemberPowerTag(userId);
 | 
			
		||||
  const tagIconSrc = tag.icon && getPowerTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  const [powerState, changePower] = useAsyncCallback<undefined, Error, [number]>(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (power: number) => {
 | 
			
		||||
        await mx.setPowerLevel(room.roomId, userId, power);
 | 
			
		||||
      },
 | 
			
		||||
      [mx, userId, room]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
  const changing = powerState.status === AsyncStatus.Loading;
 | 
			
		||||
  const error = powerState.status === AsyncStatus.Error;
 | 
			
		||||
  const [selfDemote, setSelfDemote] = useState<number>();
 | 
			
		||||
  const [sharedPower, setSharedPower] = useState<number>();
 | 
			
		||||
 | 
			
		||||
  const handlePowerSelect = (power: number): void => {
 | 
			
		||||
    close();
 | 
			
		||||
    if (!canChangePowers) return;
 | 
			
		||||
    if (power === getMemberPowerLevel(userId)) return;
 | 
			
		||||
 | 
			
		||||
    if (userId === mx.getSafeUserId()) {
 | 
			
		||||
      setSelfDemote(power);
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    if (!creators.has(myUserId) && power === getMemberPowerLevel(myUserId)) {
 | 
			
		||||
      setSharedPower(power);
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    changePower(power);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleSelfDemote = (power: number) => {
 | 
			
		||||
    setSelfDemote(undefined);
 | 
			
		||||
    changePower(power);
 | 
			
		||||
  };
 | 
			
		||||
  const handleSharedPower = (power: number) => {
 | 
			
		||||
    setSharedPower(undefined);
 | 
			
		||||
    changePower(power);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      <PopOut
 | 
			
		||||
        anchor={cords}
 | 
			
		||||
        position="Bottom"
 | 
			
		||||
        align="Start"
 | 
			
		||||
        offset={4}
 | 
			
		||||
        content={
 | 
			
		||||
          <FocusTrap
 | 
			
		||||
            focusTrapOptions={{
 | 
			
		||||
              initialFocus: false,
 | 
			
		||||
              onDeactivate: close,
 | 
			
		||||
              clickOutsideDeactivates: true,
 | 
			
		||||
              escapeDeactivates: stopPropagation,
 | 
			
		||||
              isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
              isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
            }}
 | 
			
		||||
          >
 | 
			
		||||
            <Menu>
 | 
			
		||||
              <Box
 | 
			
		||||
                direction="Column"
 | 
			
		||||
                gap="100"
 | 
			
		||||
                style={{ padding: config.space.S100, maxWidth: toRem(200) }}
 | 
			
		||||
              >
 | 
			
		||||
                {error && (
 | 
			
		||||
                  <CutoutCard style={{ padding: config.space.S200 }} variant="Critical">
 | 
			
		||||
                    <Text size="L400">Error: {powerState.error.name}</Text>
 | 
			
		||||
                    <Text className={BreakWord} size="T200">
 | 
			
		||||
                      {powerState.error.message}
 | 
			
		||||
                    </Text>
 | 
			
		||||
                  </CutoutCard>
 | 
			
		||||
                )}
 | 
			
		||||
                {getPowers(powerLevelTags).map((power) => {
 | 
			
		||||
                  const powerTag = powerLevelTags[power];
 | 
			
		||||
                  const powerTagIconSrc =
 | 
			
		||||
                    powerTag.icon && getPowerTagIconSrc(mx, useAuthentication, powerTag.icon);
 | 
			
		||||
 | 
			
		||||
                  const selected = getMemberPowerLevel(userId) === power;
 | 
			
		||||
                  const canAssignPower = creators.has(myUserId)
 | 
			
		||||
                    ? true
 | 
			
		||||
                    : power <= getMemberPowerLevel(myUserId);
 | 
			
		||||
 | 
			
		||||
                  return (
 | 
			
		||||
                    <MenuItem
 | 
			
		||||
                      key={power}
 | 
			
		||||
                      variant={selected ? 'Primary' : 'Surface'}
 | 
			
		||||
                      fill="None"
 | 
			
		||||
                      size="300"
 | 
			
		||||
                      radii="300"
 | 
			
		||||
                      aria-disabled={changing || !canChangePowers || !canAssignPower}
 | 
			
		||||
                      aria-pressed={selected}
 | 
			
		||||
                      before={<PowerColorBadge color={powerTag.color} />}
 | 
			
		||||
                      after={
 | 
			
		||||
                        powerTagIconSrc ? (
 | 
			
		||||
                          <PowerIcon size="50" iconSrc={powerTagIconSrc} />
 | 
			
		||||
                        ) : undefined
 | 
			
		||||
                      }
 | 
			
		||||
                      onClick={
 | 
			
		||||
                        canChangePowers && canAssignPower
 | 
			
		||||
                          ? () => handlePowerSelect(power)
 | 
			
		||||
                          : undefined
 | 
			
		||||
                      }
 | 
			
		||||
                    >
 | 
			
		||||
                      <Text size="B300">{powerTag.name}</Text>
 | 
			
		||||
                    </MenuItem>
 | 
			
		||||
                  );
 | 
			
		||||
                })}
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Line size="300" />
 | 
			
		||||
              <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
                <MenuItem
 | 
			
		||||
                  variant="Surface"
 | 
			
		||||
                  fill="None"
 | 
			
		||||
                  size="300"
 | 
			
		||||
                  radii="300"
 | 
			
		||||
                  onClick={() => {
 | 
			
		||||
                    if (room.isSpaceRoom()) {
 | 
			
		||||
                      openSpaceSettings(
 | 
			
		||||
                        room.roomId,
 | 
			
		||||
                        space?.roomId,
 | 
			
		||||
                        SpaceSettingsPage.PermissionsPage
 | 
			
		||||
                      );
 | 
			
		||||
                    } else {
 | 
			
		||||
                      openRoomSettings(
 | 
			
		||||
                        room.roomId,
 | 
			
		||||
                        space?.roomId,
 | 
			
		||||
                        RoomSettingsPage.PermissionsPage
 | 
			
		||||
                      );
 | 
			
		||||
                    }
 | 
			
		||||
                    close();
 | 
			
		||||
                  }}
 | 
			
		||||
                >
 | 
			
		||||
                  <Text size="B300">Manage Powers</Text>
 | 
			
		||||
                </MenuItem>
 | 
			
		||||
              </div>
 | 
			
		||||
            </Menu>
 | 
			
		||||
          </FocusTrap>
 | 
			
		||||
        }
 | 
			
		||||
      >
 | 
			
		||||
        <Chip
 | 
			
		||||
          variant={error ? 'Critical' : 'SurfaceVariant'}
 | 
			
		||||
          radii="Pill"
 | 
			
		||||
          before={
 | 
			
		||||
            cords ? (
 | 
			
		||||
              <Icon size="50" src={Icons.ChevronBottom} />
 | 
			
		||||
            ) : (
 | 
			
		||||
              <>
 | 
			
		||||
                {!changing && <PowerColorBadge color={tag.color} />}
 | 
			
		||||
                {changing && <Spinner size="50" variant="Secondary" fill="Soft" />}
 | 
			
		||||
              </>
 | 
			
		||||
            )
 | 
			
		||||
          }
 | 
			
		||||
          after={tagIconSrc ? <PowerIcon size="50" iconSrc={tagIconSrc} /> : undefined}
 | 
			
		||||
          onClick={open}
 | 
			
		||||
          aria-pressed={!!cords}
 | 
			
		||||
        >
 | 
			
		||||
          <Text size="B300" truncate>
 | 
			
		||||
            {tag.name}
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Chip>
 | 
			
		||||
      </PopOut>
 | 
			
		||||
      {typeof selfDemote === 'number' ? (
 | 
			
		||||
        <SelfDemoteAlert
 | 
			
		||||
          power={selfDemote}
 | 
			
		||||
          onCancel={() => setSelfDemote(undefined)}
 | 
			
		||||
          onChange={handleSelfDemote}
 | 
			
		||||
        />
 | 
			
		||||
      ) : null}
 | 
			
		||||
      {typeof sharedPower === 'number' ? (
 | 
			
		||||
        <SharedPowerAlert
 | 
			
		||||
          power={sharedPower}
 | 
			
		||||
          onCancel={() => setSharedPower(undefined)}
 | 
			
		||||
          onChange={handleSharedPower}
 | 
			
		||||
        />
 | 
			
		||||
      ) : null}
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										514
									
								
								src/app/components/user-profile/UserChips.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										514
									
								
								src/app/components/user-profile/UserChips.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,514 @@
 | 
			
		|||
import React, { MouseEventHandler, useCallback, useMemo, useState } from 'react';
 | 
			
		||||
import { useNavigate } from 'react-router-dom';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import { Room } from 'matrix-js-sdk';
 | 
			
		||||
import {
 | 
			
		||||
  PopOut,
 | 
			
		||||
  Menu,
 | 
			
		||||
  MenuItem,
 | 
			
		||||
  config,
 | 
			
		||||
  Text,
 | 
			
		||||
  Line,
 | 
			
		||||
  Chip,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  RectCords,
 | 
			
		||||
  Spinner,
 | 
			
		||||
  toRem,
 | 
			
		||||
  Box,
 | 
			
		||||
  Scroll,
 | 
			
		||||
  Avatar,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { getMxIdServer } from '../../utils/matrix';
 | 
			
		||||
import { useCloseUserRoomProfile } from '../../state/hooks/userRoomProfile';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { copyToClipboard } from '../../utils/dom';
 | 
			
		||||
import { getExploreServerPath } from '../../pages/pathUtils';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { factoryRoomIdByAtoZ } from '../../utils/sort';
 | 
			
		||||
import { useMutualRooms, useMutualRoomsSupport } from '../../hooks/useMutualRooms';
 | 
			
		||||
import { useRoomNavigate } from '../../hooks/useRoomNavigate';
 | 
			
		||||
import { useDirectRooms } from '../../pages/client/direct/useDirectRooms';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useAllJoinedRoomsSet, useGetRoom } from '../../hooks/useGetRoom';
 | 
			
		||||
import { RoomAvatar, RoomIcon } from '../room-avatar';
 | 
			
		||||
import { getDirectRoomAvatarUrl, getRoomAvatarUrl } from '../../utils/room';
 | 
			
		||||
import { nameInitials } from '../../utils/common';
 | 
			
		||||
import { getMatrixToUser } from '../../plugins/matrix-to';
 | 
			
		||||
import { useTimeoutToggle } from '../../hooks/useTimeoutToggle';
 | 
			
		||||
import { useIgnoredUsers } from '../../hooks/useIgnoredUsers';
 | 
			
		||||
import { CutoutCard } from '../cutout-card';
 | 
			
		||||
import { SettingTile } from '../setting-tile';
 | 
			
		||||
 | 
			
		||||
export function ServerChip({ server }: { server: string }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const myServer = getMxIdServer(mx.getSafeUserId());
 | 
			
		||||
  const navigate = useNavigate();
 | 
			
		||||
  const closeProfile = useCloseUserRoomProfile();
 | 
			
		||||
  const [copied, setCopied] = useTimeoutToggle();
 | 
			
		||||
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position="Bottom"
 | 
			
		||||
      align="Start"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      content={
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: close,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
            isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
            isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Menu>
 | 
			
		||||
            <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Surface"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  copyToClipboard(server);
 | 
			
		||||
                  setCopied();
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Copy Server</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Surface"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  navigate(getExploreServerPath(server));
 | 
			
		||||
                  closeProfile();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Explore Community</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
            </div>
 | 
			
		||||
            <Line size="300" />
 | 
			
		||||
            <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant={myServer === server ? 'Surface' : 'Critical'}
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  window.open(`https://${server}`, '_blank');
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Open in Browser</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
            </div>
 | 
			
		||||
          </Menu>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Chip
 | 
			
		||||
        variant={myServer === server ? 'SurfaceVariant' : 'Warning'}
 | 
			
		||||
        radii="Pill"
 | 
			
		||||
        before={
 | 
			
		||||
          cords ? (
 | 
			
		||||
            <Icon size="50" src={Icons.ChevronBottom} />
 | 
			
		||||
          ) : (
 | 
			
		||||
            <Icon size="50" src={copied ? Icons.Check : Icons.Server} />
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        onClick={open}
 | 
			
		||||
        aria-pressed={!!cords}
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300" truncate>
 | 
			
		||||
          {server}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function ShareChip({ userId }: { userId: string }) {
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const [copied, setCopied] = useTimeoutToggle();
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position="Bottom"
 | 
			
		||||
      align="Start"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      content={
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: close,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
            isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
            isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Menu>
 | 
			
		||||
            <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Surface"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  copyToClipboard(userId);
 | 
			
		||||
                  setCopied();
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Copy User ID</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Surface"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  copyToClipboard(getMatrixToUser(userId));
 | 
			
		||||
                  setCopied();
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Copy User Link</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
            </div>
 | 
			
		||||
          </Menu>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Chip
 | 
			
		||||
        variant={copied ? 'Success' : 'SurfaceVariant'}
 | 
			
		||||
        radii="Pill"
 | 
			
		||||
        before={
 | 
			
		||||
          cords ? (
 | 
			
		||||
            <Icon size="50" src={Icons.ChevronBottom} />
 | 
			
		||||
          ) : (
 | 
			
		||||
            <Icon size="50" src={copied ? Icons.Check : Icons.Link} />
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        onClick={open}
 | 
			
		||||
        aria-pressed={!!cords}
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300" truncate>
 | 
			
		||||
          Share
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type MutualRoomsData = {
 | 
			
		||||
  rooms: Room[];
 | 
			
		||||
  spaces: Room[];
 | 
			
		||||
  directs: Room[];
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function MutualRoomsChip({ userId }: { userId: string }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const mutualRoomSupported = useMutualRoomsSupport();
 | 
			
		||||
  const mutualRoomsState = useMutualRooms(userId);
 | 
			
		||||
  const { navigateRoom, navigateSpace } = useRoomNavigate();
 | 
			
		||||
  const closeUserRoomProfile = useCloseUserRoomProfile();
 | 
			
		||||
  const directs = useDirectRooms();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
 | 
			
		||||
  const allJoinedRooms = useAllJoinedRoomsSet();
 | 
			
		||||
  const getRoom = useGetRoom(allJoinedRooms);
 | 
			
		||||
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  const mutual: MutualRoomsData = useMemo(() => {
 | 
			
		||||
    const data: MutualRoomsData = {
 | 
			
		||||
      rooms: [],
 | 
			
		||||
      spaces: [],
 | 
			
		||||
      directs: [],
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    if (mutualRoomsState.status === AsyncStatus.Success) {
 | 
			
		||||
      const mutualRooms = mutualRoomsState.data
 | 
			
		||||
        .sort(factoryRoomIdByAtoZ(mx))
 | 
			
		||||
        .map(getRoom)
 | 
			
		||||
        .filter((room) => !!room);
 | 
			
		||||
      mutualRooms.forEach((room) => {
 | 
			
		||||
        if (room.isSpaceRoom()) {
 | 
			
		||||
          data.spaces.push(room);
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        if (directs.includes(room.roomId)) {
 | 
			
		||||
          data.directs.push(room);
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        data.rooms.push(room);
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    return data;
 | 
			
		||||
  }, [mutualRoomsState, getRoom, directs, mx]);
 | 
			
		||||
 | 
			
		||||
  if (
 | 
			
		||||
    userId === mx.getSafeUserId() ||
 | 
			
		||||
    !mutualRoomSupported ||
 | 
			
		||||
    mutualRoomsState.status === AsyncStatus.Error
 | 
			
		||||
  ) {
 | 
			
		||||
    return null;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const renderItem = (room: Room) => {
 | 
			
		||||
    const { roomId } = room;
 | 
			
		||||
    const dm = directs.includes(roomId);
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <MenuItem
 | 
			
		||||
        key={roomId}
 | 
			
		||||
        variant="Surface"
 | 
			
		||||
        fill="None"
 | 
			
		||||
        size="300"
 | 
			
		||||
        radii="300"
 | 
			
		||||
        style={{ paddingLeft: config.space.S100 }}
 | 
			
		||||
        onClick={() => {
 | 
			
		||||
          if (room.isSpaceRoom()) {
 | 
			
		||||
            navigateSpace(roomId);
 | 
			
		||||
          } else {
 | 
			
		||||
            navigateRoom(roomId);
 | 
			
		||||
          }
 | 
			
		||||
          closeUserRoomProfile();
 | 
			
		||||
        }}
 | 
			
		||||
        before={
 | 
			
		||||
          <Avatar size="200" radii={dm ? '400' : '300'}>
 | 
			
		||||
            {dm || room.isSpaceRoom() ? (
 | 
			
		||||
              <RoomAvatar
 | 
			
		||||
                roomId={room.roomId}
 | 
			
		||||
                src={
 | 
			
		||||
                  dm
 | 
			
		||||
                    ? getDirectRoomAvatarUrl(mx, room, 96, useAuthentication)
 | 
			
		||||
                    : getRoomAvatarUrl(mx, room, 96, useAuthentication)
 | 
			
		||||
                }
 | 
			
		||||
                alt={room.name}
 | 
			
		||||
                renderFallback={() => (
 | 
			
		||||
                  <Text as="span" size="H6">
 | 
			
		||||
                    {nameInitials(room.name)}
 | 
			
		||||
                  </Text>
 | 
			
		||||
                )}
 | 
			
		||||
              />
 | 
			
		||||
            ) : (
 | 
			
		||||
              <RoomIcon size="100" joinRule={room.getJoinRule()} />
 | 
			
		||||
            )}
 | 
			
		||||
          </Avatar>
 | 
			
		||||
        }
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300" truncate>
 | 
			
		||||
          {room.name}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </MenuItem>
 | 
			
		||||
    );
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position="Bottom"
 | 
			
		||||
      align="Start"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      content={
 | 
			
		||||
        mutualRoomsState.status === AsyncStatus.Success ? (
 | 
			
		||||
          <FocusTrap
 | 
			
		||||
            focusTrapOptions={{
 | 
			
		||||
              initialFocus: false,
 | 
			
		||||
              onDeactivate: close,
 | 
			
		||||
              clickOutsideDeactivates: true,
 | 
			
		||||
              escapeDeactivates: stopPropagation,
 | 
			
		||||
              isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
              isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
            }}
 | 
			
		||||
          >
 | 
			
		||||
            <Menu
 | 
			
		||||
              style={{
 | 
			
		||||
                display: 'flex',
 | 
			
		||||
                maxWidth: toRem(200),
 | 
			
		||||
                maxHeight: '80vh',
 | 
			
		||||
              }}
 | 
			
		||||
            >
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Scroll size="300" hideTrack>
 | 
			
		||||
                  <Box
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="400"
 | 
			
		||||
                    style={{ padding: config.space.S200, paddingRight: 0 }}
 | 
			
		||||
                  >
 | 
			
		||||
                    {mutual.spaces.length > 0 && (
 | 
			
		||||
                      <Box direction="Column" gap="100">
 | 
			
		||||
                        <Text style={{ paddingLeft: config.space.S100 }} size="L400">
 | 
			
		||||
                          Spaces
 | 
			
		||||
                        </Text>
 | 
			
		||||
                        {mutual.spaces.map(renderItem)}
 | 
			
		||||
                      </Box>
 | 
			
		||||
                    )}
 | 
			
		||||
                    {mutual.rooms.length > 0 && (
 | 
			
		||||
                      <Box direction="Column" gap="100">
 | 
			
		||||
                        <Text style={{ paddingLeft: config.space.S100 }} size="L400">
 | 
			
		||||
                          Rooms
 | 
			
		||||
                        </Text>
 | 
			
		||||
                        {mutual.rooms.map(renderItem)}
 | 
			
		||||
                      </Box>
 | 
			
		||||
                    )}
 | 
			
		||||
                    {mutual.directs.length > 0 && (
 | 
			
		||||
                      <Box direction="Column" gap="100">
 | 
			
		||||
                        <Text style={{ paddingLeft: config.space.S100 }} size="L400">
 | 
			
		||||
                          Direct Messages
 | 
			
		||||
                        </Text>
 | 
			
		||||
                        {mutual.directs.map(renderItem)}
 | 
			
		||||
                      </Box>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Scroll>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Menu>
 | 
			
		||||
          </FocusTrap>
 | 
			
		||||
        ) : null
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Chip
 | 
			
		||||
        variant="SurfaceVariant"
 | 
			
		||||
        radii="Pill"
 | 
			
		||||
        before={mutualRoomsState.status === AsyncStatus.Loading && <Spinner size="50" />}
 | 
			
		||||
        disabled={
 | 
			
		||||
          mutualRoomsState.status !== AsyncStatus.Success || mutualRoomsState.data.length === 0
 | 
			
		||||
        }
 | 
			
		||||
        onClick={open}
 | 
			
		||||
        aria-pressed={!!cords}
 | 
			
		||||
      >
 | 
			
		||||
        <Text size="B300">
 | 
			
		||||
          {mutualRoomsState.status === AsyncStatus.Success &&
 | 
			
		||||
            `${mutualRoomsState.data.length} Mutual Rooms`}
 | 
			
		||||
          {mutualRoomsState.status === AsyncStatus.Loading && 'Mutual Rooms'}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function IgnoredUserAlert() {
 | 
			
		||||
  return (
 | 
			
		||||
    <CutoutCard style={{ padding: config.space.S200 }} variant="Critical">
 | 
			
		||||
      <SettingTile>
 | 
			
		||||
        <Box direction="Column" gap="200">
 | 
			
		||||
          <Box gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
            <Text size="L400">Blocked User</Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
          <Box direction="Column">
 | 
			
		||||
            <Text size="T200">You do not receive any messages or invites from this user.</Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </SettingTile>
 | 
			
		||||
    </CutoutCard>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function OptionsChip({ userId }: { userId: string }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const open: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const close = () => setCords(undefined);
 | 
			
		||||
 | 
			
		||||
  const ignoredUsers = useIgnoredUsers();
 | 
			
		||||
  const ignored = ignoredUsers.includes(userId);
 | 
			
		||||
 | 
			
		||||
  const [ignoreState, toggleIgnore] = useAsyncCallback(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      const users = ignoredUsers.filter((u) => u !== userId);
 | 
			
		||||
      if (!ignored) users.push(userId);
 | 
			
		||||
      await mx.setIgnoredUsers(users);
 | 
			
		||||
    }, [mx, ignoredUsers, userId, ignored])
 | 
			
		||||
  );
 | 
			
		||||
  const ignoring = ignoreState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <PopOut
 | 
			
		||||
      anchor={cords}
 | 
			
		||||
      position="Bottom"
 | 
			
		||||
      align="Start"
 | 
			
		||||
      offset={4}
 | 
			
		||||
      content={
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: close,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
            isKeyForward: (evt: KeyboardEvent) => isKeyHotkey('arrowdown', evt),
 | 
			
		||||
            isKeyBackward: (evt: KeyboardEvent) => isKeyHotkey('arrowup', evt),
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Menu>
 | 
			
		||||
            <div style={{ padding: config.space.S100 }}>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                variant="Critical"
 | 
			
		||||
                fill="None"
 | 
			
		||||
                size="300"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                onClick={() => {
 | 
			
		||||
                  toggleIgnore();
 | 
			
		||||
                  close();
 | 
			
		||||
                }}
 | 
			
		||||
                before={
 | 
			
		||||
                  ignoring ? (
 | 
			
		||||
                    <Spinner variant="Critical" size="50" />
 | 
			
		||||
                  ) : (
 | 
			
		||||
                    <Icon size="50" src={Icons.Prohibited} />
 | 
			
		||||
                  )
 | 
			
		||||
                }
 | 
			
		||||
                disabled={ignoring}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">{ignored ? 'Unblock User' : 'Block User'}</Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
            </div>
 | 
			
		||||
          </Menu>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Chip variant="SurfaceVariant" radii="Pill" onClick={open} aria-pressed={!!cords}>
 | 
			
		||||
        {ignoring ? (
 | 
			
		||||
          <Spinner variant="Secondary" size="50" />
 | 
			
		||||
        ) : (
 | 
			
		||||
          <Icon size="50" src={Icons.HorizontalDots} />
 | 
			
		||||
        )}
 | 
			
		||||
      </Chip>
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										75
									
								
								src/app/components/user-profile/UserHero.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								src/app/components/user-profile/UserHero.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,75 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import { Avatar, Box, Icon, Icons, Text } from 'folds';
 | 
			
		||||
import classNames from 'classnames';
 | 
			
		||||
import * as css from './styles.css';
 | 
			
		||||
import { UserAvatar } from '../user-avatar';
 | 
			
		||||
import colorMXID from '../../../util/colorMXID';
 | 
			
		||||
import { getMxIdLocalPart } from '../../utils/matrix';
 | 
			
		||||
import { BreakWord, LineClamp3 } from '../../styles/Text.css';
 | 
			
		||||
import { UserPresence } from '../../hooks/useUserPresence';
 | 
			
		||||
import { AvatarPresence, PresenceBadge } from '../presence';
 | 
			
		||||
 | 
			
		||||
type UserHeroProps = {
 | 
			
		||||
  userId: string;
 | 
			
		||||
  avatarUrl?: string;
 | 
			
		||||
  presence?: UserPresence;
 | 
			
		||||
};
 | 
			
		||||
export function UserHero({ userId, avatarUrl, presence }: UserHeroProps) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column" className={css.UserHero}>
 | 
			
		||||
      <div
 | 
			
		||||
        className={css.UserHeroCoverContainer}
 | 
			
		||||
        style={{
 | 
			
		||||
          backgroundColor: colorMXID(userId),
 | 
			
		||||
          filter: avatarUrl ? undefined : 'brightness(50%)',
 | 
			
		||||
        }}
 | 
			
		||||
      >
 | 
			
		||||
        {avatarUrl && <img className={css.UserHeroCover} src={avatarUrl} alt={userId} />}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div className={css.UserHeroAvatarContainer}>
 | 
			
		||||
        <AvatarPresence
 | 
			
		||||
          className={css.UserAvatarContainer}
 | 
			
		||||
          badge={
 | 
			
		||||
            presence && <PresenceBadge presence={presence.presence} status={presence.status} />
 | 
			
		||||
          }
 | 
			
		||||
        >
 | 
			
		||||
          <Avatar className={css.UserHeroAvatar} size="500">
 | 
			
		||||
            <UserAvatar
 | 
			
		||||
              userId={userId}
 | 
			
		||||
              src={avatarUrl}
 | 
			
		||||
              alt={userId}
 | 
			
		||||
              renderFallback={() => <Icon size="500" src={Icons.User} filled />}
 | 
			
		||||
            />
 | 
			
		||||
          </Avatar>
 | 
			
		||||
        </AvatarPresence>
 | 
			
		||||
      </div>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserHeroNameProps = {
 | 
			
		||||
  displayName?: string;
 | 
			
		||||
  userId: string;
 | 
			
		||||
};
 | 
			
		||||
export function UserHeroName({ displayName, userId }: UserHeroNameProps) {
 | 
			
		||||
  const username = getMxIdLocalPart(userId);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box grow="Yes" direction="Column" gap="0">
 | 
			
		||||
      <Box alignItems="Baseline" gap="200" wrap="Wrap">
 | 
			
		||||
        <Text
 | 
			
		||||
          size="H4"
 | 
			
		||||
          className={classNames(BreakWord, LineClamp3)}
 | 
			
		||||
          title={displayName ?? username}
 | 
			
		||||
        >
 | 
			
		||||
          {displayName ?? username ?? userId}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Box alignItems="Center" gap="100" wrap="Wrap">
 | 
			
		||||
        <Text size="T200" className={classNames(BreakWord, LineClamp3)} title={username}>
 | 
			
		||||
          @{username}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										349
									
								
								src/app/components/user-profile/UserModeration.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										349
									
								
								src/app/components/user-profile/UserModeration.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,349 @@
 | 
			
		|||
import { Box, Button, color, config, Icon, Icons, Spinner, Text, Input } from 'folds';
 | 
			
		||||
import React, { useCallback, useRef } from 'react';
 | 
			
		||||
import { useRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { CutoutCard } from '../cutout-card';
 | 
			
		||||
import { SettingTile } from '../setting-tile';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { BreakWord } from '../../styles/Text.css';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { settingsAtom } from '../../state/settings';
 | 
			
		||||
import { timeDayMonYear, timeHourMinute } from '../../utils/time';
 | 
			
		||||
 | 
			
		||||
type UserKickAlertProps = {
 | 
			
		||||
  reason?: string;
 | 
			
		||||
  kickedBy?: string;
 | 
			
		||||
  ts?: number;
 | 
			
		||||
};
 | 
			
		||||
export function UserKickAlert({ reason, kickedBy, ts }: UserKickAlertProps) {
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
  const [dateFormatString] = useSetting(settingsAtom, 'dateFormatString');
 | 
			
		||||
 | 
			
		||||
  const time = ts ? timeHourMinute(ts, hour24Clock) : undefined;
 | 
			
		||||
  const date = ts ? timeDayMonYear(ts, dateFormatString) : undefined;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <CutoutCard style={{ padding: config.space.S200 }} variant="Critical">
 | 
			
		||||
      <SettingTile>
 | 
			
		||||
        <Box direction="Column" gap="200">
 | 
			
		||||
          <Box gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
            <Text size="L400">Kicked User</Text>
 | 
			
		||||
            {time && date && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                {date} {time}
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
          </Box>
 | 
			
		||||
          <Box direction="Column">
 | 
			
		||||
            {kickedBy && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                Kicked by: <b>{kickedBy}</b>
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
            <Text size="T200">
 | 
			
		||||
              {reason ? (
 | 
			
		||||
                <>
 | 
			
		||||
                  Reason: <b>{reason}</b>
 | 
			
		||||
                </>
 | 
			
		||||
              ) : (
 | 
			
		||||
                <i>No Reason Provided.</i>
 | 
			
		||||
              )}
 | 
			
		||||
            </Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </SettingTile>
 | 
			
		||||
    </CutoutCard>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserBanAlertProps = {
 | 
			
		||||
  userId: string;
 | 
			
		||||
  reason?: string;
 | 
			
		||||
  canUnban?: boolean;
 | 
			
		||||
  bannedBy?: string;
 | 
			
		||||
  ts?: number;
 | 
			
		||||
};
 | 
			
		||||
export function UserBanAlert({ userId, reason, canUnban, bannedBy, ts }: UserBanAlertProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
  const [dateFormatString] = useSetting(settingsAtom, 'dateFormatString');
 | 
			
		||||
 | 
			
		||||
  const time = ts ? timeHourMinute(ts, hour24Clock) : undefined;
 | 
			
		||||
  const date = ts ? timeDayMonYear(ts, dateFormatString) : undefined;
 | 
			
		||||
 | 
			
		||||
  const [unbanState, unban] = useAsyncCallback<undefined, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      await mx.unban(room.roomId, userId);
 | 
			
		||||
    }, [mx, room, userId])
 | 
			
		||||
  );
 | 
			
		||||
  const banning = unbanState.status === AsyncStatus.Loading;
 | 
			
		||||
  const error = unbanState.status === AsyncStatus.Error;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <CutoutCard style={{ padding: config.space.S200 }} variant="Critical">
 | 
			
		||||
      <SettingTile>
 | 
			
		||||
        <Box direction="Column" gap="200">
 | 
			
		||||
          <Box gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
            <Text size="L400">Banned User</Text>
 | 
			
		||||
            {time && date && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                {date} {time}
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
          </Box>
 | 
			
		||||
          <Box direction="Column">
 | 
			
		||||
            {bannedBy && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                Banned by: <b>{bannedBy}</b>
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
            <Text size="T200">
 | 
			
		||||
              {reason ? (
 | 
			
		||||
                <>
 | 
			
		||||
                  Reason: <b>{reason}</b>
 | 
			
		||||
                </>
 | 
			
		||||
              ) : (
 | 
			
		||||
                <i>No Reason Provided.</i>
 | 
			
		||||
              )}
 | 
			
		||||
            </Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
          {error && (
 | 
			
		||||
            <Text className={BreakWord} size="T200" style={{ color: color.Critical.Main }}>
 | 
			
		||||
              <b>{unbanState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
          {canUnban && (
 | 
			
		||||
            <Button
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Critical"
 | 
			
		||||
              radii="300"
 | 
			
		||||
              onClick={unban}
 | 
			
		||||
              before={banning && <Spinner size="100" variant="Critical" fill="Solid" />}
 | 
			
		||||
              disabled={banning}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">Unban</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
      </SettingTile>
 | 
			
		||||
    </CutoutCard>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserInviteAlertProps = {
 | 
			
		||||
  userId: string;
 | 
			
		||||
  reason?: string;
 | 
			
		||||
  canKick?: boolean;
 | 
			
		||||
  invitedBy?: string;
 | 
			
		||||
  ts?: number;
 | 
			
		||||
};
 | 
			
		||||
export function UserInviteAlert({ userId, reason, canKick, invitedBy, ts }: UserInviteAlertProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
  const [dateFormatString] = useSetting(settingsAtom, 'dateFormatString');
 | 
			
		||||
 | 
			
		||||
  const time = ts ? timeHourMinute(ts, hour24Clock) : undefined;
 | 
			
		||||
  const date = ts ? timeDayMonYear(ts, dateFormatString) : undefined;
 | 
			
		||||
 | 
			
		||||
  const [kickState, kick] = useAsyncCallback<undefined, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      await mx.kick(room.roomId, userId);
 | 
			
		||||
    }, [mx, room, userId])
 | 
			
		||||
  );
 | 
			
		||||
  const kicking = kickState.status === AsyncStatus.Loading;
 | 
			
		||||
  const error = kickState.status === AsyncStatus.Error;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <CutoutCard style={{ padding: config.space.S200 }} variant="Success">
 | 
			
		||||
      <SettingTile>
 | 
			
		||||
        <Box direction="Column" gap="200">
 | 
			
		||||
          <Box gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
            <Text size="L400">Invited User</Text>
 | 
			
		||||
            {time && date && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                {date} {time}
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
          </Box>
 | 
			
		||||
          <Box direction="Column">
 | 
			
		||||
            {invitedBy && (
 | 
			
		||||
              <Text size="T200">
 | 
			
		||||
                Invited by: <b>{invitedBy}</b>
 | 
			
		||||
              </Text>
 | 
			
		||||
            )}
 | 
			
		||||
            <Text size="T200">
 | 
			
		||||
              {reason ? (
 | 
			
		||||
                <>
 | 
			
		||||
                  Reason: <b>{reason}</b>
 | 
			
		||||
                </>
 | 
			
		||||
              ) : (
 | 
			
		||||
                <i>No Reason Provided.</i>
 | 
			
		||||
              )}
 | 
			
		||||
            </Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
          {error && (
 | 
			
		||||
            <Text className={BreakWord} size="T200" style={{ color: color.Critical.Main }}>
 | 
			
		||||
              <b>{kickState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
          {canKick && (
 | 
			
		||||
            <Button
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Success"
 | 
			
		||||
              fill="Soft"
 | 
			
		||||
              outlined
 | 
			
		||||
              radii="300"
 | 
			
		||||
              onClick={kick}
 | 
			
		||||
              before={kicking && <Spinner size="100" variant="Success" fill="Soft" />}
 | 
			
		||||
              disabled={kicking}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">Cancel Invite</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
      </SettingTile>
 | 
			
		||||
    </CutoutCard>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserModerationProps = {
 | 
			
		||||
  userId: string;
 | 
			
		||||
  canKick: boolean;
 | 
			
		||||
  canBan: boolean;
 | 
			
		||||
  canInvite: boolean;
 | 
			
		||||
};
 | 
			
		||||
export function UserModeration({ userId, canKick, canBan, canInvite }: UserModerationProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const reasonInputRef = useRef<HTMLInputElement>(null);
 | 
			
		||||
 | 
			
		||||
  const getReason = useCallback((): string | undefined => {
 | 
			
		||||
    const reason = reasonInputRef.current?.value.trim() || undefined;
 | 
			
		||||
    if (reasonInputRef.current) {
 | 
			
		||||
      reasonInputRef.current.value = '';
 | 
			
		||||
    }
 | 
			
		||||
    return reason;
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  const [kickState, kick] = useAsyncCallback<undefined, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      await mx.kick(room.roomId, userId, getReason());
 | 
			
		||||
    }, [mx, room, userId, getReason])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [banState, ban] = useAsyncCallback<undefined, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      await mx.ban(room.roomId, userId, getReason());
 | 
			
		||||
    }, [mx, room, userId, getReason])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [inviteState, invite] = useAsyncCallback<undefined, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      await mx.invite(room.roomId, userId, getReason());
 | 
			
		||||
    }, [mx, room, userId, getReason])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const disabled =
 | 
			
		||||
    kickState.status === AsyncStatus.Loading ||
 | 
			
		||||
    banState.status === AsyncStatus.Loading ||
 | 
			
		||||
    inviteState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  if (!canBan && !canKick && !canInvite) return null;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column" gap="400">
 | 
			
		||||
      <Box direction="Column" gap="200">
 | 
			
		||||
        <Box grow="Yes" direction="Column" gap="100">
 | 
			
		||||
          <Text size="L400">Moderation</Text>
 | 
			
		||||
          <Input
 | 
			
		||||
            ref={reasonInputRef}
 | 
			
		||||
            placeholder="Reason"
 | 
			
		||||
            size="300"
 | 
			
		||||
            variant="Background"
 | 
			
		||||
            radii="300"
 | 
			
		||||
            disabled={disabled}
 | 
			
		||||
          />
 | 
			
		||||
          {kickState.status === AsyncStatus.Error && (
 | 
			
		||||
            <Text style={{ color: color.Critical.Main }} className={BreakWord} size="T200">
 | 
			
		||||
              <b>{kickState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
          {banState.status === AsyncStatus.Error && (
 | 
			
		||||
            <Text style={{ color: color.Critical.Main }} className={BreakWord} size="T200">
 | 
			
		||||
              <b>{banState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
          {inviteState.status === AsyncStatus.Error && (
 | 
			
		||||
            <Text style={{ color: color.Critical.Main }} className={BreakWord} size="T200">
 | 
			
		||||
              <b>{inviteState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
        <Box shrink="No" gap="200">
 | 
			
		||||
          {canInvite && (
 | 
			
		||||
            <Button
 | 
			
		||||
              style={{ flexGrow: 1 }}
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Secondary"
 | 
			
		||||
              fill="Soft"
 | 
			
		||||
              radii="300"
 | 
			
		||||
              before={
 | 
			
		||||
                inviteState.status === AsyncStatus.Loading ? (
 | 
			
		||||
                  <Spinner size="50" variant="Secondary" fill="Soft" />
 | 
			
		||||
                ) : (
 | 
			
		||||
                  <Icon size="50" src={Icons.ArrowRight} />
 | 
			
		||||
                )
 | 
			
		||||
              }
 | 
			
		||||
              onClick={invite}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">Invite</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          )}
 | 
			
		||||
          {canKick && (
 | 
			
		||||
            <Button
 | 
			
		||||
              style={{ flexGrow: 1 }}
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Critical"
 | 
			
		||||
              fill="Soft"
 | 
			
		||||
              radii="300"
 | 
			
		||||
              before={
 | 
			
		||||
                kickState.status === AsyncStatus.Loading ? (
 | 
			
		||||
                  <Spinner size="50" variant="Critical" fill="Soft" />
 | 
			
		||||
                ) : (
 | 
			
		||||
                  <Icon size="50" src={Icons.ArrowLeft} />
 | 
			
		||||
                )
 | 
			
		||||
              }
 | 
			
		||||
              onClick={kick}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">Kick</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          )}
 | 
			
		||||
          {canBan && (
 | 
			
		||||
            <Button
 | 
			
		||||
              style={{ flexGrow: 1 }}
 | 
			
		||||
              size="300"
 | 
			
		||||
              variant="Critical"
 | 
			
		||||
              fill="Solid"
 | 
			
		||||
              radii="300"
 | 
			
		||||
              before={
 | 
			
		||||
                banState.status === AsyncStatus.Loading ? (
 | 
			
		||||
                  <Spinner size="50" variant="Critical" fill="Solid" />
 | 
			
		||||
                ) : (
 | 
			
		||||
                  <Icon size="50" src={Icons.Prohibited} />
 | 
			
		||||
                )
 | 
			
		||||
              }
 | 
			
		||||
              onClick={ban}
 | 
			
		||||
              disabled={disabled}
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="B300">Ban</Text>
 | 
			
		||||
            </Button>
 | 
			
		||||
          )}
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										169
									
								
								src/app/components/user-profile/UserRoomProfile.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								src/app/components/user-profile/UserRoomProfile.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,169 @@
 | 
			
		|||
import { Box, Button, color, config, Icon, Icons, Spinner, Text } from 'folds';
 | 
			
		||||
import React, { useCallback } from 'react';
 | 
			
		||||
import { UserHero, UserHeroName } from './UserHero';
 | 
			
		||||
import { getDMRoomFor, getMxIdServer, mxcUrlToHttp } from '../../utils/matrix';
 | 
			
		||||
import { getMemberAvatarMxc, getMemberDisplayName } from '../../utils/room';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { usePowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { useRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useUserPresence } from '../../hooks/useUserPresence';
 | 
			
		||||
import { IgnoredUserAlert, MutualRoomsChip, OptionsChip, ServerChip, ShareChip } from './UserChips';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { createDM } from '../../../client/action/room';
 | 
			
		||||
import { hasDevices } from '../../../util/matrixUtil';
 | 
			
		||||
import { useRoomNavigate } from '../../hooks/useRoomNavigate';
 | 
			
		||||
import { useAlive } from '../../hooks/useAlive';
 | 
			
		||||
import { useCloseUserRoomProfile } from '../../state/hooks/userRoomProfile';
 | 
			
		||||
import { PowerChip } from './PowerChip';
 | 
			
		||||
import { UserInviteAlert, UserBanAlert, UserModeration, UserKickAlert } from './UserModeration';
 | 
			
		||||
import { useIgnoredUsers } from '../../hooks/useIgnoredUsers';
 | 
			
		||||
import { useMembership } from '../../hooks/useMembership';
 | 
			
		||||
import { Membership } from '../../../types/matrix/room';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { useMemberPowerCompare } from '../../hooks/useMemberPowerCompare';
 | 
			
		||||
import { CreatorChip } from './CreatorChip';
 | 
			
		||||
 | 
			
		||||
type UserRoomProfileProps = {
 | 
			
		||||
  userId: string;
 | 
			
		||||
};
 | 
			
		||||
export function UserRoomProfile({ userId }: UserRoomProfileProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const { navigateRoom } = useRoomNavigate();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
  const closeUserRoomProfile = useCloseUserRoomProfile();
 | 
			
		||||
  const ignoredUsers = useIgnoredUsers();
 | 
			
		||||
  const ignored = ignoredUsers.includes(userId);
 | 
			
		||||
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const { hasMorePower } = useMemberPowerCompare(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const myUserId = mx.getSafeUserId();
 | 
			
		||||
  const creator = creators.has(userId);
 | 
			
		||||
 | 
			
		||||
  const canKickUser = permissions.action('kick', myUserId) && hasMorePower(myUserId, userId);
 | 
			
		||||
  const canBanUser = permissions.action('ban', myUserId) && hasMorePower(myUserId, userId);
 | 
			
		||||
  const canUnban = permissions.action('ban', myUserId);
 | 
			
		||||
  const canInvite = permissions.action('invite', myUserId);
 | 
			
		||||
 | 
			
		||||
  const member = room.getMember(userId);
 | 
			
		||||
  const membership = useMembership(room, userId);
 | 
			
		||||
 | 
			
		||||
  const server = getMxIdServer(userId);
 | 
			
		||||
  const displayName = getMemberDisplayName(room, userId);
 | 
			
		||||
  const avatarMxc = getMemberAvatarMxc(room, userId);
 | 
			
		||||
  const avatarUrl = (avatarMxc && mxcUrlToHttp(mx, avatarMxc, useAuthentication)) ?? undefined;
 | 
			
		||||
 | 
			
		||||
  const presence = useUserPresence(userId);
 | 
			
		||||
 | 
			
		||||
  const [directMessageState, directMessage] = useAsyncCallback<string, Error, []>(
 | 
			
		||||
    useCallback(async () => {
 | 
			
		||||
      const result = await createDM(mx, userId, await hasDevices(mx, userId));
 | 
			
		||||
      return result.room_id as string;
 | 
			
		||||
    }, [userId, mx])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const handleMessage = () => {
 | 
			
		||||
    const dmRoomId = getDMRoomFor(mx, userId)?.roomId;
 | 
			
		||||
    if (dmRoomId) {
 | 
			
		||||
      navigateRoom(dmRoomId);
 | 
			
		||||
      closeUserRoomProfile();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    directMessage().then((rId) => {
 | 
			
		||||
      if (alive()) {
 | 
			
		||||
        navigateRoom(rId);
 | 
			
		||||
        closeUserRoomProfile();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column">
 | 
			
		||||
      <UserHero
 | 
			
		||||
        userId={userId}
 | 
			
		||||
        avatarUrl={avatarUrl}
 | 
			
		||||
        presence={presence && presence.lastActiveTs !== 0 ? presence : undefined}
 | 
			
		||||
      />
 | 
			
		||||
      <Box direction="Column" gap="500" style={{ padding: config.space.S400 }}>
 | 
			
		||||
        <Box direction="Column" gap="400">
 | 
			
		||||
          <Box gap="400" alignItems="Start">
 | 
			
		||||
            <UserHeroName displayName={displayName} userId={userId} />
 | 
			
		||||
            <Box shrink="No">
 | 
			
		||||
              <Button
 | 
			
		||||
                size="300"
 | 
			
		||||
                variant="Primary"
 | 
			
		||||
                fill="Solid"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                disabled={directMessageState.status === AsyncStatus.Loading}
 | 
			
		||||
                before={
 | 
			
		||||
                  directMessageState.status === AsyncStatus.Loading ? (
 | 
			
		||||
                    <Spinner size="50" variant="Primary" fill="Solid" />
 | 
			
		||||
                  ) : (
 | 
			
		||||
                    <Icon size="50" src={Icons.Message} filled />
 | 
			
		||||
                  )
 | 
			
		||||
                }
 | 
			
		||||
                onClick={handleMessage}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Message</Text>
 | 
			
		||||
              </Button>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Box>
 | 
			
		||||
          {directMessageState.status === AsyncStatus.Error && (
 | 
			
		||||
            <Text style={{ color: color.Critical.Main }}>
 | 
			
		||||
              <b>{directMessageState.error.message}</b>
 | 
			
		||||
            </Text>
 | 
			
		||||
          )}
 | 
			
		||||
          <Box alignItems="Center" gap="200" wrap="Wrap">
 | 
			
		||||
            {server && <ServerChip server={server} />}
 | 
			
		||||
            <ShareChip userId={userId} />
 | 
			
		||||
            {creator ? <CreatorChip /> : <PowerChip userId={userId} />}
 | 
			
		||||
            {userId !== myUserId && <MutualRoomsChip userId={userId} />}
 | 
			
		||||
            {userId !== myUserId && <OptionsChip userId={userId} />}
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
        {ignored && <IgnoredUserAlert />}
 | 
			
		||||
        {member && membership === Membership.Ban && (
 | 
			
		||||
          <UserBanAlert
 | 
			
		||||
            userId={userId}
 | 
			
		||||
            reason={member.events.member?.getContent().reason}
 | 
			
		||||
            canUnban={canUnban}
 | 
			
		||||
            bannedBy={member.events.member?.getSender()}
 | 
			
		||||
            ts={member.events.member?.getTs()}
 | 
			
		||||
          />
 | 
			
		||||
        )}
 | 
			
		||||
        {member &&
 | 
			
		||||
          membership === Membership.Leave &&
 | 
			
		||||
          member.events.member &&
 | 
			
		||||
          member.events.member.getSender() !== userId && (
 | 
			
		||||
            <UserKickAlert
 | 
			
		||||
              reason={member.events.member?.getContent().reason}
 | 
			
		||||
              kickedBy={member.events.member?.getSender()}
 | 
			
		||||
              ts={member.events.member?.getTs()}
 | 
			
		||||
            />
 | 
			
		||||
          )}
 | 
			
		||||
        {member && membership === Membership.Invite && (
 | 
			
		||||
          <UserInviteAlert
 | 
			
		||||
            userId={userId}
 | 
			
		||||
            reason={member.events.member?.getContent().reason}
 | 
			
		||||
            canKick={canKickUser}
 | 
			
		||||
            invitedBy={member.events.member?.getSender()}
 | 
			
		||||
            ts={member.events.member?.getTs()}
 | 
			
		||||
          />
 | 
			
		||||
        )}
 | 
			
		||||
        <UserModeration
 | 
			
		||||
          userId={userId}
 | 
			
		||||
          canInvite={canInvite && membership === Membership.Leave}
 | 
			
		||||
          canKick={canKickUser && membership === Membership.Join}
 | 
			
		||||
          canBan={canBanUser && membership !== Membership.Ban}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								src/app/components/user-profile/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/app/components/user-profile/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1 @@
 | 
			
		|||
export * from './UserRoomProfile';
 | 
			
		||||
							
								
								
									
										42
									
								
								src/app/components/user-profile/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/app/components/user-profile/styles.css.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,42 @@
 | 
			
		|||
import { style } from '@vanilla-extract/css';
 | 
			
		||||
import { color, config, toRem } from 'folds';
 | 
			
		||||
 | 
			
		||||
export const UserHeader = style({
 | 
			
		||||
  position: 'absolute',
 | 
			
		||||
  top: 0,
 | 
			
		||||
  left: 0,
 | 
			
		||||
  right: 0,
 | 
			
		||||
  zIndex: 1,
 | 
			
		||||
  padding: config.space.S200,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const UserHero = style({
 | 
			
		||||
  position: 'relative',
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const UserHeroCoverContainer = style({
 | 
			
		||||
  height: toRem(96),
 | 
			
		||||
  overflow: 'hidden',
 | 
			
		||||
});
 | 
			
		||||
export const UserHeroCover = style({
 | 
			
		||||
  height: '100%',
 | 
			
		||||
  width: '100%',
 | 
			
		||||
  objectFit: 'cover',
 | 
			
		||||
  filter: 'blur(16px)',
 | 
			
		||||
  transform: 'scale(2)',
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const UserHeroAvatarContainer = style({
 | 
			
		||||
  position: 'relative',
 | 
			
		||||
  height: toRem(29),
 | 
			
		||||
});
 | 
			
		||||
export const UserAvatarContainer = style({
 | 
			
		||||
  position: 'absolute',
 | 
			
		||||
  left: config.space.S400,
 | 
			
		||||
  top: 0,
 | 
			
		||||
  transform: 'translateY(-50%)',
 | 
			
		||||
  backgroundColor: color.Surface.Container,
 | 
			
		||||
});
 | 
			
		||||
export const UserHeroAvatar = style({
 | 
			
		||||
  outline: `${config.borderWidth.B600} solid ${color.Surface.Container}`,
 | 
			
		||||
});
 | 
			
		||||
							
								
								
									
										375
									
								
								src/app/features/add-existing/AddExisting.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										375
									
								
								src/app/features/add-existing/AddExisting.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,375 @@
 | 
			
		|||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import {
 | 
			
		||||
  Avatar,
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  config,
 | 
			
		||||
  Header,
 | 
			
		||||
  Icon,
 | 
			
		||||
  IconButton,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Input,
 | 
			
		||||
  Menu,
 | 
			
		||||
  MenuItem,
 | 
			
		||||
  Modal,
 | 
			
		||||
  Overlay,
 | 
			
		||||
  OverlayBackdrop,
 | 
			
		||||
  OverlayCenter,
 | 
			
		||||
  Scroll,
 | 
			
		||||
  Spinner,
 | 
			
		||||
  Text,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import React, {
 | 
			
		||||
  ChangeEventHandler,
 | 
			
		||||
  MouseEventHandler,
 | 
			
		||||
  useCallback,
 | 
			
		||||
  useMemo,
 | 
			
		||||
  useRef,
 | 
			
		||||
  useState,
 | 
			
		||||
} from 'react';
 | 
			
		||||
import { useAtomValue } from 'jotai';
 | 
			
		||||
import { useVirtualizer } from '@tanstack/react-virtual';
 | 
			
		||||
import { Room } from 'matrix-js-sdk';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { useDirects, useRooms, useSpaces } from '../../state/hooks/roomList';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { allRoomsAtom } from '../../state/room-list/roomList';
 | 
			
		||||
import { mDirectAtom } from '../../state/mDirectList';
 | 
			
		||||
import { roomToParentsAtom } from '../../state/room/roomToParents';
 | 
			
		||||
import { useAllJoinedRoomsSet, useGetRoom } from '../../hooks/useGetRoom';
 | 
			
		||||
import { VirtualTile } from '../../components/virtualizer';
 | 
			
		||||
import { getDirectRoomAvatarUrl, getRoomAvatarUrl } from '../../utils/room';
 | 
			
		||||
import { RoomAvatar, RoomIcon } from '../../components/room-avatar';
 | 
			
		||||
import { nameInitials } from '../../utils/common';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { factoryRoomIdByAtoZ } from '../../utils/sort';
 | 
			
		||||
import {
 | 
			
		||||
  SearchItemStrGetter,
 | 
			
		||||
  useAsyncSearch,
 | 
			
		||||
  UseAsyncSearchOptions,
 | 
			
		||||
} from '../../hooks/useAsyncSearch';
 | 
			
		||||
import { highlightText, makeHighlightRegex } from '../../plugins/react-custom-html-parser';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { getViaServers } from '../../plugins/via-servers';
 | 
			
		||||
import { rateLimitedActions } from '../../utils/matrix';
 | 
			
		||||
import { useAlive } from '../../hooks/useAlive';
 | 
			
		||||
 | 
			
		||||
const SEARCH_OPTS: UseAsyncSearchOptions = {
 | 
			
		||||
  limit: 500,
 | 
			
		||||
  matchOptions: {
 | 
			
		||||
    contain: true,
 | 
			
		||||
  },
 | 
			
		||||
  normalizeOptions: {
 | 
			
		||||
    ignoreWhitespace: false,
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
type AddExistingModalProps = {
 | 
			
		||||
  parentId: string;
 | 
			
		||||
  space?: boolean;
 | 
			
		||||
  requestClose: () => void;
 | 
			
		||||
};
 | 
			
		||||
export function AddExistingModal({ parentId, space, requestClose }: AddExistingModalProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
 | 
			
		||||
  const mDirects = useAtomValue(mDirectAtom);
 | 
			
		||||
  const spaces = useSpaces(mx, allRoomsAtom);
 | 
			
		||||
  const rooms = useRooms(mx, allRoomsAtom, mDirects);
 | 
			
		||||
  const directs = useDirects(mx, allRoomsAtom, mDirects);
 | 
			
		||||
  const roomIdToParents = useAtomValue(roomToParentsAtom);
 | 
			
		||||
  const scrollRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
 | 
			
		||||
  const [selected, setSelected] = useState<string[]>([]);
 | 
			
		||||
 | 
			
		||||
  const allRoomsSet = useAllJoinedRoomsSet();
 | 
			
		||||
  const getRoom = useGetRoom(allRoomsSet);
 | 
			
		||||
 | 
			
		||||
  const allItems: string[] = useMemo(() => {
 | 
			
		||||
    const rIds = space ? [...spaces] : [...rooms, ...directs];
 | 
			
		||||
 | 
			
		||||
    return rIds
 | 
			
		||||
      .filter((rId) => rId !== parentId && !roomIdToParents.get(rId)?.has(parentId))
 | 
			
		||||
      .sort(factoryRoomIdByAtoZ(mx));
 | 
			
		||||
  }, [spaces, rooms, directs, space, parentId, roomIdToParents, mx]);
 | 
			
		||||
 | 
			
		||||
  const getRoomNameStr: SearchItemStrGetter<string> = useCallback(
 | 
			
		||||
    (rId) => getRoom(rId)?.name ?? rId,
 | 
			
		||||
    [getRoom]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [searchResult, searchRoom, resetSearch] = useAsyncSearch(
 | 
			
		||||
    allItems,
 | 
			
		||||
    getRoomNameStr,
 | 
			
		||||
    SEARCH_OPTS
 | 
			
		||||
  );
 | 
			
		||||
  const queryHighlighRegex = searchResult?.query
 | 
			
		||||
    ? makeHighlightRegex(searchResult.query.split(' '))
 | 
			
		||||
    : undefined;
 | 
			
		||||
 | 
			
		||||
  const items = searchResult ? searchResult.items : allItems;
 | 
			
		||||
 | 
			
		||||
  const virtualizer = useVirtualizer({
 | 
			
		||||
    count: items.length,
 | 
			
		||||
    getScrollElement: () => scrollRef.current,
 | 
			
		||||
    estimateSize: () => 32,
 | 
			
		||||
    overscan: 5,
 | 
			
		||||
  });
 | 
			
		||||
  const vItems = virtualizer.getVirtualItems();
 | 
			
		||||
 | 
			
		||||
  const handleSearchChange: ChangeEventHandler<HTMLInputElement> = (evt) => {
 | 
			
		||||
    const value = evt.currentTarget.value.trim();
 | 
			
		||||
    if (!value) {
 | 
			
		||||
      resetSearch();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    searchRoom(value);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const [applyState, applyChanges] = useAsyncCallback<undefined, Error, [Room[]]>(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (selectedRooms) => {
 | 
			
		||||
        await rateLimitedActions(selectedRooms, async (room) => {
 | 
			
		||||
          const via = getViaServers(room);
 | 
			
		||||
 | 
			
		||||
          await mx.sendStateEvent(
 | 
			
		||||
            parentId,
 | 
			
		||||
            StateEvent.SpaceChild as any,
 | 
			
		||||
            {
 | 
			
		||||
              auto_join: false,
 | 
			
		||||
              suggested: false,
 | 
			
		||||
              via,
 | 
			
		||||
            },
 | 
			
		||||
            room.roomId
 | 
			
		||||
          );
 | 
			
		||||
        });
 | 
			
		||||
      },
 | 
			
		||||
      [mx, parentId]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
  const applyingChanges = applyState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const handleRoomClick: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    const roomId = evt.currentTarget.getAttribute('data-room-id');
 | 
			
		||||
    if (!roomId) return;
 | 
			
		||||
    if (selected?.includes(roomId)) {
 | 
			
		||||
      setSelected(selected?.filter((rId) => rId !== roomId));
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    const addedRooms = [...(selected ?? [])];
 | 
			
		||||
    addedRooms.push(roomId);
 | 
			
		||||
    setSelected(addedRooms);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleApplyChanges = () => {
 | 
			
		||||
    const selectedRooms = selected.map((rId) => getRoom(rId)).filter((room) => room !== undefined);
 | 
			
		||||
    applyChanges(selectedRooms).then(() => {
 | 
			
		||||
      if (alive()) {
 | 
			
		||||
        setSelected([]);
 | 
			
		||||
        requestClose();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const resetChanges = () => {
 | 
			
		||||
    setSelected([]);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
      <OverlayCenter>
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            onDeactivate: requestClose,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Modal size="300">
 | 
			
		||||
            <Box grow="Yes" direction="Column">
 | 
			
		||||
              <Header
 | 
			
		||||
                size="500"
 | 
			
		||||
                style={{
 | 
			
		||||
                  padding: config.space.S200,
 | 
			
		||||
                  paddingLeft: config.space.S400,
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Box grow="Yes">
 | 
			
		||||
                  <Text size="H4">Add Existing</Text>
 | 
			
		||||
                </Box>
 | 
			
		||||
                <Box shrink="No">
 | 
			
		||||
                  <IconButton size="300" radii="300" onClick={requestClose}>
 | 
			
		||||
                    <Icon src={Icons.Cross} />
 | 
			
		||||
                  </IconButton>
 | 
			
		||||
                </Box>
 | 
			
		||||
              </Header>
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Scroll ref={scrollRef} size="300" hideTrack>
 | 
			
		||||
                  <Box
 | 
			
		||||
                    style={{ padding: config.space.S300, paddingRight: 0 }}
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="500"
 | 
			
		||||
                  >
 | 
			
		||||
                    <Box
 | 
			
		||||
                      direction="Column"
 | 
			
		||||
                      style={{ position: 'sticky', top: config.space.S300, zIndex: 1 }}
 | 
			
		||||
                    >
 | 
			
		||||
                      <Input
 | 
			
		||||
                        onChange={handleSearchChange}
 | 
			
		||||
                        before={<Icon size="200" src={Icons.Search} />}
 | 
			
		||||
                        placeholder="Search"
 | 
			
		||||
                        size="400"
 | 
			
		||||
                        variant="Background"
 | 
			
		||||
                        outlined
 | 
			
		||||
                      />
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    {vItems.length === 0 && (
 | 
			
		||||
                      <Box
 | 
			
		||||
                        style={{ paddingTop: config.space.S700 }}
 | 
			
		||||
                        grow="Yes"
 | 
			
		||||
                        alignItems="Center"
 | 
			
		||||
                        justifyContent="Center"
 | 
			
		||||
                        direction="Column"
 | 
			
		||||
                        gap="100"
 | 
			
		||||
                      >
 | 
			
		||||
                        <Text size="H6" align="Center">
 | 
			
		||||
                          {searchResult ? 'No Match Found' : `No ${space ? 'Spaces' : 'Rooms'}`}
 | 
			
		||||
                        </Text>
 | 
			
		||||
                        <Text size="T200" align="Center">
 | 
			
		||||
                          {searchResult
 | 
			
		||||
                            ? `No match found for "${searchResult.query}".`
 | 
			
		||||
                            : `You do not have any ${space ? 'Spaces' : 'Rooms'} to display yet.`}
 | 
			
		||||
                        </Text>
 | 
			
		||||
                      </Box>
 | 
			
		||||
                    )}
 | 
			
		||||
                    <Box
 | 
			
		||||
                      style={{
 | 
			
		||||
                        position: 'relative',
 | 
			
		||||
                        height: virtualizer.getTotalSize(),
 | 
			
		||||
                      }}
 | 
			
		||||
                    >
 | 
			
		||||
                      {vItems.map((vItem) => {
 | 
			
		||||
                        const roomId = items[vItem.index];
 | 
			
		||||
                        const room = getRoom(roomId);
 | 
			
		||||
                        if (!room) return null;
 | 
			
		||||
                        const selectedItem = selected?.includes(roomId);
 | 
			
		||||
                        const dm = mDirects.has(room.roomId);
 | 
			
		||||
 | 
			
		||||
                        return (
 | 
			
		||||
                          <VirtualTile
 | 
			
		||||
                            virtualItem={vItem}
 | 
			
		||||
                            style={{ paddingBottom: config.space.S100 }}
 | 
			
		||||
                            ref={virtualizer.measureElement}
 | 
			
		||||
                            key={vItem.index}
 | 
			
		||||
                          >
 | 
			
		||||
                            <MenuItem
 | 
			
		||||
                              data-room-id={roomId}
 | 
			
		||||
                              onClick={handleRoomClick}
 | 
			
		||||
                              variant={selectedItem ? 'Success' : 'Surface'}
 | 
			
		||||
                              size="400"
 | 
			
		||||
                              radii="400"
 | 
			
		||||
                              disabled={applyingChanges}
 | 
			
		||||
                              aria-pressed={selectedItem}
 | 
			
		||||
                              before={
 | 
			
		||||
                                <Avatar size="200" radii={dm ? '400' : '300'}>
 | 
			
		||||
                                  {dm || room.isSpaceRoom() ? (
 | 
			
		||||
                                    <RoomAvatar
 | 
			
		||||
                                      roomId={room.roomId}
 | 
			
		||||
                                      src={
 | 
			
		||||
                                        dm
 | 
			
		||||
                                          ? getDirectRoomAvatarUrl(mx, room, 96, useAuthentication)
 | 
			
		||||
                                          : getRoomAvatarUrl(mx, room, 96, useAuthentication)
 | 
			
		||||
                                      }
 | 
			
		||||
                                      alt={room.name}
 | 
			
		||||
                                      renderFallback={() => (
 | 
			
		||||
                                        <Text as="span" size="H6">
 | 
			
		||||
                                          {nameInitials(room.name)}
 | 
			
		||||
                                        </Text>
 | 
			
		||||
                                      )}
 | 
			
		||||
                                    />
 | 
			
		||||
                                  ) : (
 | 
			
		||||
                                    <RoomIcon size="200" joinRule={room.getJoinRule()} />
 | 
			
		||||
                                  )}
 | 
			
		||||
                                </Avatar>
 | 
			
		||||
                              }
 | 
			
		||||
                              after={selectedItem && <Icon size="200" src={Icons.Check} />}
 | 
			
		||||
                            >
 | 
			
		||||
                              <Box grow="Yes">
 | 
			
		||||
                                <Text truncate size="T400">
 | 
			
		||||
                                  {queryHighlighRegex
 | 
			
		||||
                                    ? highlightText(queryHighlighRegex, [room.name])
 | 
			
		||||
                                    : room.name}
 | 
			
		||||
                                </Text>
 | 
			
		||||
                              </Box>
 | 
			
		||||
                            </MenuItem>
 | 
			
		||||
                          </VirtualTile>
 | 
			
		||||
                        );
 | 
			
		||||
                      })}
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    {selected.length > 0 && (
 | 
			
		||||
                      <Menu
 | 
			
		||||
                        style={{
 | 
			
		||||
                          position: 'sticky',
 | 
			
		||||
                          padding: config.space.S200,
 | 
			
		||||
                          paddingLeft: config.space.S400,
 | 
			
		||||
                          bottom: config.space.S400,
 | 
			
		||||
                          left: config.space.S400,
 | 
			
		||||
                          right: 0,
 | 
			
		||||
                          zIndex: 1,
 | 
			
		||||
                        }}
 | 
			
		||||
                        variant="Success"
 | 
			
		||||
                      >
 | 
			
		||||
                        <Box alignItems="Center" gap="400">
 | 
			
		||||
                          <Box grow="Yes" direction="Column">
 | 
			
		||||
                            {applyState.status === AsyncStatus.Error ? (
 | 
			
		||||
                              <Text size="T200">
 | 
			
		||||
                                <b>Failed to apply changes! Please try again.</b>
 | 
			
		||||
                              </Text>
 | 
			
		||||
                            ) : (
 | 
			
		||||
                              <Text size="T200">
 | 
			
		||||
                                <b>Apply when ready. ({selected.length} Selected)</b>
 | 
			
		||||
                              </Text>
 | 
			
		||||
                            )}
 | 
			
		||||
                          </Box>
 | 
			
		||||
                          <Box shrink="No" gap="200">
 | 
			
		||||
                            <Button
 | 
			
		||||
                              size="300"
 | 
			
		||||
                              variant="Success"
 | 
			
		||||
                              fill="None"
 | 
			
		||||
                              radii="300"
 | 
			
		||||
                              disabled={applyingChanges}
 | 
			
		||||
                              onClick={resetChanges}
 | 
			
		||||
                            >
 | 
			
		||||
                              <Text size="B300">Reset</Text>
 | 
			
		||||
                            </Button>
 | 
			
		||||
                            <Button
 | 
			
		||||
                              size="300"
 | 
			
		||||
                              variant="Success"
 | 
			
		||||
                              radii="300"
 | 
			
		||||
                              disabled={applyingChanges}
 | 
			
		||||
                              before={
 | 
			
		||||
                                applyingChanges && (
 | 
			
		||||
                                  <Spinner variant="Success" fill="Solid" size="100" />
 | 
			
		||||
                                )
 | 
			
		||||
                              }
 | 
			
		||||
                              onClick={handleApplyChanges}
 | 
			
		||||
                            >
 | 
			
		||||
                              <Text size="B300">Apply Changes</Text>
 | 
			
		||||
                            </Button>
 | 
			
		||||
                          </Box>
 | 
			
		||||
                        </Box>
 | 
			
		||||
                      </Menu>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Scroll>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Modal>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      </OverlayCenter>
 | 
			
		||||
    </Overlay>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								src/app/features/add-existing/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/app/features/add-existing/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1 @@
 | 
			
		|||
export * from './AddExisting';
 | 
			
		||||
| 
						 | 
				
			
			@ -27,8 +27,10 @@ import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		|||
import { syntaxErrorPosition } from '../../../utils/dom';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { SequenceCardStyle } from '../styles.css';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevels } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useTextAreaCodeEditor } from '../../../hooks/useTextAreaCodeEditor';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
const EDITOR_INTENT_SPACE_COUNT = 2;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -244,8 +246,10 @@ export function StateEventEditor({ type, stateKey, requestClose }: StateEventEdi
 | 
			
		|||
  const stateEvent = useStateEvent(room, type as unknown as StateEvent, stateKey);
 | 
			
		||||
  const [editContent, setEditContent] = useState<object>();
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const { getPowerLevel, canSendStateEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canEdit = canSendStateEvent(type, getPowerLevel(mx.getSafeUserId()));
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const canEdit = permissions.stateEvent(type, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const eventJSONStr = useMemo(() => {
 | 
			
		||||
    if (!stateEvent) return '';
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,11 +33,13 @@ import { SequenceCardStyle } from '../styles.css';
 | 
			
		|||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { mxcUrlToHttp } from '../../../utils/matrix';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevels } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { suffixRename } from '../../../utils/common';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { useAlive } from '../../../hooks/useAlive';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type CreatePackTileProps = {
 | 
			
		||||
  packs: ImagePack[];
 | 
			
		||||
| 
						 | 
				
			
			@ -146,8 +148,10 @@ export function RoomPacks({ onViewPack }: RoomPacksProps) {
 | 
			
		|||
  const alive = useAlive();
 | 
			
		||||
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const { canSendStateEvent, getPowerLevel } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canEdit = canSendStateEvent(StateEvent.PoniesRoomEmotes, getPowerLevel(mx.getSafeUserId()));
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const canEdit = permissions.stateEvent(StateEvent.PoniesRoomEmotes, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const unfilteredPacks = useRoomImagePacks(room);
 | 
			
		||||
  const packs = useMemo(() => unfilteredPacks.filter((pack) => !pack.deleted), [unfilteredPacks]);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -15,7 +15,6 @@ import {
 | 
			
		|||
  toRem,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { MatrixError } from 'matrix-js-sdk';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../../room-settings/styles.css';
 | 
			
		||||
| 
						 | 
				
			
			@ -33,19 +32,19 @@ import { getIdServer } from '../../../../util/matrixUtil';
 | 
			
		|||
import { replaceSpaceWithDash } from '../../../utils/common';
 | 
			
		||||
import { useAlive } from '../../../hooks/useAlive';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RoomPublishedAddressesProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function RoomPublishedAddresses({ powerLevels }: RoomPublishedAddressesProps) {
 | 
			
		||||
export function RoomPublishedAddresses({ permissions }: RoomPublishedAddressesProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEditCanonical = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
 | 
			
		||||
  const canEditCanonical = permissions.stateEvent(
 | 
			
		||||
    StateEvent.RoomCanonicalAlias,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
    mx.getSafeUserId()
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [canonicalAlias, publishedAliases] = usePublishedAliases(room);
 | 
			
		||||
| 
						 | 
				
			
			@ -360,14 +359,13 @@ function LocalAddressesList({
 | 
			
		|||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function RoomLocalAddresses({ powerLevels }: { powerLevels: IPowerLevels }) {
 | 
			
		||||
export function RoomLocalAddresses({ permissions }: { permissions: RoomPermissionsAPI }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEditCanonical = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
 | 
			
		||||
  const canEditCanonical = permissions.stateEvent(
 | 
			
		||||
    StateEvent.RoomCanonicalAlias,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
    mx.getSafeUserId()
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [expand, setExpand] = useState(false);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -21,28 +21,24 @@ import FocusTrap from 'focus-trap-react';
 | 
			
		|||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../../room-settings/styles.css';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { useStateEvent } from '../../../hooks/useStateEvent';
 | 
			
		||||
import { stopPropagation } from '../../../utils/keyboard';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
const ROOM_ENC_ALGO = 'm.megolm.v1.aes-sha2';
 | 
			
		||||
 | 
			
		||||
type RoomEncryptionProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
export function RoomEncryption({ powerLevels }: RoomEncryptionProps) {
 | 
			
		||||
export function RoomEncryption({ permissions }: RoomEncryptionProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEnable = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
    StateEvent.RoomEncryption,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canEnable = permissions.stateEvent(StateEvent.RoomEncryption, mx.getSafeUserId());
 | 
			
		||||
  const content = useStateEvent(room, StateEvent.RoomEncryption)?.getContent<{
 | 
			
		||||
    algorithm: string;
 | 
			
		||||
  }>();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -18,13 +18,13 @@ import FocusTrap from 'focus-trap-react';
 | 
			
		|||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../../room-settings/styles.css';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { useStateEvent } from '../../../hooks/useStateEvent';
 | 
			
		||||
import { stopPropagation } from '../../../utils/keyboard';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
const useVisibilityStr = () =>
 | 
			
		||||
  useMemo(
 | 
			
		||||
| 
						 | 
				
			
			@ -49,17 +49,13 @@ const useVisibilityMenu = () =>
 | 
			
		|||
  );
 | 
			
		||||
 | 
			
		||||
type RoomHistoryVisibilityProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
export function RoomHistoryVisibility({ powerLevels }: RoomHistoryVisibilityProps) {
 | 
			
		||||
export function RoomHistoryVisibility({ permissions }: RoomHistoryVisibilityProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEdit = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
    StateEvent.RoomHistoryVisibility,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canEdit = permissions.stateEvent(StateEvent.RoomHistoryVisibility, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const visibilityEvent = useStateEvent(room, StateEvent.RoomHistoryVisibility);
 | 
			
		||||
  const historyVisibility: HistoryVisibility =
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -2,8 +2,9 @@ import React, { useCallback, useMemo } from 'react';
 | 
			
		|||
import { color, Text } from 'folds';
 | 
			
		||||
import { JoinRule, MatrixError, RestrictedAllowType } from 'matrix-js-sdk';
 | 
			
		||||
import { RoomJoinRulesEventContent } from 'matrix-js-sdk/lib/types';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useAtomValue } from 'jotai';
 | 
			
		||||
import {
 | 
			
		||||
  ExtendedJoinRules,
 | 
			
		||||
  JoinRulesSwitcher,
 | 
			
		||||
  useRoomJoinRuleIcon,
 | 
			
		||||
  useRoomJoinRuleLabel,
 | 
			
		||||
| 
						 | 
				
			
			@ -19,6 +20,18 @@ import { useStateEvent } from '../../../hooks/useStateEvent';
 | 
			
		|||
import { useSpaceOptionally } from '../../../hooks/useSpace';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { getStateEvents } from '../../../utils/room';
 | 
			
		||||
import {
 | 
			
		||||
  useRecursiveChildSpaceScopeFactory,
 | 
			
		||||
  useSpaceChildren,
 | 
			
		||||
} from '../../../state/hooks/roomList';
 | 
			
		||||
import { allRoomsAtom } from '../../../state/room-list/roomList';
 | 
			
		||||
import { roomToParentsAtom } from '../../../state/room/roomToParents';
 | 
			
		||||
import {
 | 
			
		||||
  knockRestrictedSupported,
 | 
			
		||||
  knockSupported,
 | 
			
		||||
  restrictedSupported,
 | 
			
		||||
} from '../../../utils/matrix';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RestrictedRoomAllowContent = {
 | 
			
		||||
  room_id: string;
 | 
			
		||||
| 
						 | 
				
			
			@ -26,39 +39,41 @@ type RestrictedRoomAllowContent = {
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
type RoomJoinRulesProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
export function RoomJoinRules({ powerLevels }: RoomJoinRulesProps) {
 | 
			
		||||
export function RoomJoinRules({ permissions }: RoomJoinRulesProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const roomVersion = parseInt(room.getVersion(), 10);
 | 
			
		||||
  const allowRestricted = roomVersion >= 8;
 | 
			
		||||
  const allowKnock = roomVersion >= 7;
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
  const allowKnockRestricted = knockRestrictedSupported(room.getVersion());
 | 
			
		||||
  const allowRestricted = restrictedSupported(room.getVersion());
 | 
			
		||||
  const allowKnock = knockSupported(room.getVersion());
 | 
			
		||||
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEdit = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
    StateEvent.RoomHistoryVisibility,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
  );
 | 
			
		||||
  const roomIdToParents = useAtomValue(roomToParentsAtom);
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
  const subspacesScope = useRecursiveChildSpaceScopeFactory(mx, roomIdToParents);
 | 
			
		||||
  const subspaces = useSpaceChildren(allRoomsAtom, space?.roomId ?? '', subspacesScope);
 | 
			
		||||
 | 
			
		||||
  const canEdit = permissions.stateEvent(StateEvent.RoomHistoryVisibility, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const joinRuleEvent = useStateEvent(room, StateEvent.RoomJoinRules);
 | 
			
		||||
  const content = joinRuleEvent?.getContent<RoomJoinRulesEventContent>();
 | 
			
		||||
  const rule: JoinRule = content?.join_rule ?? JoinRule.Invite;
 | 
			
		||||
 | 
			
		||||
  const joinRules: Array<JoinRule> = useMemo(() => {
 | 
			
		||||
    const r: JoinRule[] = [JoinRule.Invite];
 | 
			
		||||
  const joinRules: Array<ExtendedJoinRules> = useMemo(() => {
 | 
			
		||||
    const r: ExtendedJoinRules[] = [JoinRule.Invite];
 | 
			
		||||
    if (allowKnock) {
 | 
			
		||||
      r.push(JoinRule.Knock);
 | 
			
		||||
    }
 | 
			
		||||
    if (allowRestricted && space) {
 | 
			
		||||
      r.push(JoinRule.Restricted);
 | 
			
		||||
    }
 | 
			
		||||
    if (allowKnockRestricted && space) {
 | 
			
		||||
      r.push('knock_restricted');
 | 
			
		||||
    }
 | 
			
		||||
    r.push(JoinRule.Public);
 | 
			
		||||
 | 
			
		||||
    return r;
 | 
			
		||||
  }, [allowRestricted, allowKnock, space]);
 | 
			
		||||
  }, [allowKnockRestricted, allowRestricted, allowKnock, space]);
 | 
			
		||||
 | 
			
		||||
  const icons = useRoomJoinRuleIcon();
 | 
			
		||||
  const spaceIcons = useSpaceJoinRuleIcon();
 | 
			
		||||
| 
						 | 
				
			
			@ -66,12 +81,25 @@ export function RoomJoinRules({ powerLevels }: RoomJoinRulesProps) {
 | 
			
		|||
 | 
			
		||||
  const [submitState, submit] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (joinRule: JoinRule) => {
 | 
			
		||||
      async (joinRule: ExtendedJoinRules) => {
 | 
			
		||||
        const allow: RestrictedRoomAllowContent[] = [];
 | 
			
		||||
        if (joinRule === JoinRule.Restricted) {
 | 
			
		||||
          const parents = getStateEvents(room, StateEvent.SpaceParent).map((event) =>
 | 
			
		||||
            event.getStateKey()
 | 
			
		||||
          );
 | 
			
		||||
        if (joinRule === JoinRule.Restricted || joinRule === 'knock_restricted') {
 | 
			
		||||
          const roomParents = roomIdToParents.get(room.roomId);
 | 
			
		||||
 | 
			
		||||
          const parents = getStateEvents(room, StateEvent.SpaceParent)
 | 
			
		||||
            .map((event) => event.getStateKey())
 | 
			
		||||
            .filter((parentId) => typeof parentId === 'string')
 | 
			
		||||
            .filter((parentId) => roomParents?.has(parentId));
 | 
			
		||||
 | 
			
		||||
          if (parents.length === 0 && space && roomParents) {
 | 
			
		||||
            // if no m.space.parent found
 | 
			
		||||
            // find parent in current space
 | 
			
		||||
            const selectedParents = subspaces.filter((rId) => roomParents.has(rId));
 | 
			
		||||
            if (roomParents.has(space.roomId)) {
 | 
			
		||||
              selectedParents.push(space.roomId);
 | 
			
		||||
            }
 | 
			
		||||
            selectedParents.forEach((pId) => parents.push(pId));
 | 
			
		||||
          }
 | 
			
		||||
          parents.forEach((parentRoomId) => {
 | 
			
		||||
            if (!parentRoomId) return;
 | 
			
		||||
            allow.push({
 | 
			
		||||
| 
						 | 
				
			
			@ -82,12 +110,12 @@ export function RoomJoinRules({ powerLevels }: RoomJoinRulesProps) {
 | 
			
		|||
        }
 | 
			
		||||
 | 
			
		||||
        const c: RoomJoinRulesEventContent = {
 | 
			
		||||
          join_rule: joinRule,
 | 
			
		||||
          join_rule: joinRule as JoinRule,
 | 
			
		||||
        };
 | 
			
		||||
        if (allow.length > 0) c.allow = allow;
 | 
			
		||||
        await mx.sendStateEvent(room.roomId, StateEvent.RoomJoinRules as any, c);
 | 
			
		||||
      },
 | 
			
		||||
      [mx, room]
 | 
			
		||||
      [mx, room, space, subspaces, roomIdToParents]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,6 @@ import { RoomAvatar, RoomIcon } from '../../../components/room-avatar';
 | 
			
		|||
import { mxcUrlToHttp } from '../../../utils/matrix';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { IPowerLevels, usePowerLevelsAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { CompactUploadCardRenderer } from '../../../components/upload-card';
 | 
			
		||||
import { useObjectURL } from '../../../hooks/useObjectURL';
 | 
			
		||||
| 
						 | 
				
			
			@ -40,6 +39,7 @@ import { createUploadAtom, UploadSuccess } from '../../../state/upload';
 | 
			
		|||
import { useFilePicker } from '../../../hooks/useFilePicker';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { useAlive } from '../../../hooks/useAlive';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RoomProfileEditProps = {
 | 
			
		||||
  canEditAvatar: boolean;
 | 
			
		||||
| 
						 | 
				
			
			@ -261,24 +261,22 @@ export function RoomProfileEdit({
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
type RoomProfileProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
export function RoomProfile({ powerLevels }: RoomProfileProps) {
 | 
			
		||||
export function RoomProfile({ permissions }: RoomProfileProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const directs = useAtomValue(mDirectAtom);
 | 
			
		||||
  const { getPowerLevel, canSendStateEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const userPowerLevel = getPowerLevel(mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const avatar = useRoomAvatar(room, directs.has(room.roomId));
 | 
			
		||||
  const name = useRoomName(room);
 | 
			
		||||
  const topic = useRoomTopic(room);
 | 
			
		||||
  const joinRule = useRoomJoinRule(room);
 | 
			
		||||
 | 
			
		||||
  const canEditAvatar = canSendStateEvent(StateEvent.RoomAvatar, userPowerLevel);
 | 
			
		||||
  const canEditName = canSendStateEvent(StateEvent.RoomName, userPowerLevel);
 | 
			
		||||
  const canEditTopic = canSendStateEvent(StateEvent.RoomTopic, userPowerLevel);
 | 
			
		||||
  const canEditAvatar = permissions.stateEvent(StateEvent.RoomAvatar, mx.getSafeUserId());
 | 
			
		||||
  const canEditName = permissions.stateEvent(StateEvent.RoomName, mx.getSafeUserId());
 | 
			
		||||
  const canEditTopic = permissions.stateEvent(StateEvent.RoomTopic, mx.getSafeUserId());
 | 
			
		||||
  const canEdit = canEditAvatar || canEditName || canEditTopic;
 | 
			
		||||
 | 
			
		||||
  const avatarUrl = avatar
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,28 +1,33 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import { Box, color, Spinner, Switch, Text } from 'folds';
 | 
			
		||||
import { MatrixError } from 'matrix-js-sdk';
 | 
			
		||||
import { JoinRule, MatrixError } from 'matrix-js-sdk';
 | 
			
		||||
import { RoomJoinRulesEventContent } from 'matrix-js-sdk/lib/types';
 | 
			
		||||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../../room-settings/styles.css';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { useRoomDirectoryVisibility } from '../../../hooks/useRoomDirectoryVisibility';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { useStateEvent } from '../../../hooks/useStateEvent';
 | 
			
		||||
import { ExtendedJoinRules } from '../../../components/JoinRulesSwitcher';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RoomPublishProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
};
 | 
			
		||||
export function RoomPublish({ powerLevels }: RoomPublishProps) {
 | 
			
		||||
export function RoomPublish({ permissions }: RoomPublishProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canEditCanonical = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
 | 
			
		||||
  const canEditCanonical = permissions.stateEvent(
 | 
			
		||||
    StateEvent.RoomCanonicalAlias,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
    mx.getSafeUserId()
 | 
			
		||||
  );
 | 
			
		||||
  const joinRuleEvent = useStateEvent(room, StateEvent.RoomJoinRules);
 | 
			
		||||
  const content = joinRuleEvent?.getContent<RoomJoinRulesEventContent>();
 | 
			
		||||
  const rule: ExtendedJoinRules = (content?.join_rule as ExtendedJoinRules) ?? JoinRule.Invite;
 | 
			
		||||
 | 
			
		||||
  const { visibilityState, setVisibility } = useRoomDirectoryVisibility(room.roomId);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -30,6 +35,8 @@ export function RoomPublish({ powerLevels }: RoomPublishProps) {
 | 
			
		|||
 | 
			
		||||
  const loading =
 | 
			
		||||
    visibilityState.status === AsyncStatus.Loading || toggleState.status === AsyncStatus.Loading;
 | 
			
		||||
  const validRule =
 | 
			
		||||
    rule === JoinRule.Public || rule === JoinRule.Knock || rule === 'knock_restricted';
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SequenceCard
 | 
			
		||||
| 
						 | 
				
			
			@ -39,7 +46,12 @@ export function RoomPublish({ powerLevels }: RoomPublishProps) {
 | 
			
		|||
      gap="400"
 | 
			
		||||
    >
 | 
			
		||||
      <SettingTile
 | 
			
		||||
        title="Publish To Directory"
 | 
			
		||||
        title="Publish to Directory"
 | 
			
		||||
        description={
 | 
			
		||||
          room.isSpaceRoom()
 | 
			
		||||
            ? 'List the space in the public directory to make it discoverable by others.'
 | 
			
		||||
            : 'List the room in the public directory to make it discoverable by others.'
 | 
			
		||||
        }
 | 
			
		||||
        after={
 | 
			
		||||
          <Box gap="200" alignItems="Center">
 | 
			
		||||
            {loading && <Spinner variant="Secondary" />}
 | 
			
		||||
| 
						 | 
				
			
			@ -47,7 +59,7 @@ export function RoomPublish({ powerLevels }: RoomPublishProps) {
 | 
			
		|||
              <Switch
 | 
			
		||||
                value={visibilityState.data}
 | 
			
		||||
                onChange={toggleVisibility}
 | 
			
		||||
                disabled={!canEditCanonical}
 | 
			
		||||
                disabled={!canEditCanonical || !validRule}
 | 
			
		||||
              />
 | 
			
		||||
            )}
 | 
			
		||||
          </Box>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,4 +1,4 @@
 | 
			
		|||
import React, { FormEventHandler, useCallback, useState } from 'react';
 | 
			
		||||
import React, { useCallback, useEffect, useState } from 'react';
 | 
			
		||||
import {
 | 
			
		||||
  Button,
 | 
			
		||||
  color,
 | 
			
		||||
| 
						 | 
				
			
			@ -14,54 +14,172 @@ import {
 | 
			
		|||
  IconButton,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Input,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { MatrixError } from 'matrix-js-sdk';
 | 
			
		||||
import { RoomCreateEventContent, RoomTombstoneEventContent } from 'matrix-js-sdk/lib/types';
 | 
			
		||||
import { MatrixError, Method } from 'matrix-js-sdk';
 | 
			
		||||
import { RoomTombstoneEventContent } from 'matrix-js-sdk/lib/types';
 | 
			
		||||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../../room-settings/styles.css';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { IRoomCreateContent, StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { useStateEvent } from '../../../hooks/useStateEvent';
 | 
			
		||||
import { useRoomNavigate } from '../../../hooks/useRoomNavigate';
 | 
			
		||||
import { useCapabilities } from '../../../hooks/useCapabilities';
 | 
			
		||||
import { stopPropagation } from '../../../utils/keyboard';
 | 
			
		||||
import { RoomPermissionsAPI } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
import {
 | 
			
		||||
  AdditionalCreatorInput,
 | 
			
		||||
  RoomVersionSelector,
 | 
			
		||||
  useAdditionalCreators,
 | 
			
		||||
} from '../../../components/create-room';
 | 
			
		||||
import { useAlive } from '../../../hooks/useAlive';
 | 
			
		||||
import { creatorsSupported } from '../../../utils/matrix';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { BreakWord } from '../../../styles/Text.css';
 | 
			
		||||
 | 
			
		||||
function RoomUpgradeDialog({ requestClose }: { requestClose: () => void }) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const capabilities = useCapabilities();
 | 
			
		||||
  const roomVersions = capabilities['m.room_versions'];
 | 
			
		||||
  const [selectedRoomVersion, selectRoomVersion] = useState(roomVersions?.default ?? '1');
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    // capabilities load async
 | 
			
		||||
    selectRoomVersion(roomVersions?.default ?? '1');
 | 
			
		||||
  }, [roomVersions?.default]);
 | 
			
		||||
 | 
			
		||||
  const allowAdditionalCreators = creatorsSupported(selectedRoomVersion);
 | 
			
		||||
  const { additionalCreators, addAdditionalCreator, removeAdditionalCreator } =
 | 
			
		||||
    useAdditionalCreators(Array.from(creators));
 | 
			
		||||
 | 
			
		||||
  const [upgradeState, upgrade] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (version: string, newAdditionalCreators?: string[]) => {
 | 
			
		||||
        await mx.http.authedRequest(Method.Post, `/rooms/${room.roomId}/upgrade`, undefined, {
 | 
			
		||||
          new_version: version,
 | 
			
		||||
          additional_creators: newAdditionalCreators,
 | 
			
		||||
        });
 | 
			
		||||
      },
 | 
			
		||||
      [mx, room]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const upgrading = upgradeState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const handleUpgradeRoom = () => {
 | 
			
		||||
    const version = selectedRoomVersion;
 | 
			
		||||
 | 
			
		||||
    upgrade(version, allowAdditionalCreators ? additionalCreators : undefined).then(() => {
 | 
			
		||||
      if (alive()) {
 | 
			
		||||
        requestClose();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
      <OverlayCenter>
 | 
			
		||||
        <FocusTrap
 | 
			
		||||
          focusTrapOptions={{
 | 
			
		||||
            initialFocus: false,
 | 
			
		||||
            onDeactivate: requestClose,
 | 
			
		||||
            clickOutsideDeactivates: true,
 | 
			
		||||
            escapeDeactivates: stopPropagation,
 | 
			
		||||
          }}
 | 
			
		||||
        >
 | 
			
		||||
          <Dialog variant="Surface">
 | 
			
		||||
            <Header
 | 
			
		||||
              style={{
 | 
			
		||||
                padding: `0 ${config.space.S200} 0 ${config.space.S400}`,
 | 
			
		||||
                borderBottomWidth: config.borderWidth.B300,
 | 
			
		||||
              }}
 | 
			
		||||
              variant="Surface"
 | 
			
		||||
              size="500"
 | 
			
		||||
            >
 | 
			
		||||
              <Box grow="Yes">
 | 
			
		||||
                <Text size="H4">{room.isSpaceRoom() ? 'Space Upgrade' : 'Room Upgrade'}</Text>
 | 
			
		||||
              </Box>
 | 
			
		||||
              <IconButton size="300" onClick={requestClose} radii="300">
 | 
			
		||||
                <Icon src={Icons.Cross} />
 | 
			
		||||
              </IconButton>
 | 
			
		||||
            </Header>
 | 
			
		||||
            <Box style={{ padding: config.space.S400 }} direction="Column" gap="400">
 | 
			
		||||
              <Text priority="400" style={{ color: color.Critical.Main }}>
 | 
			
		||||
                <b>This action is irreversible!</b>
 | 
			
		||||
              </Text>
 | 
			
		||||
              <Box direction="Column" gap="100">
 | 
			
		||||
                <Text size="L400">Options</Text>
 | 
			
		||||
                <RoomVersionSelector
 | 
			
		||||
                  versions={roomVersions?.available ? Object.keys(roomVersions.available) : ['1']}
 | 
			
		||||
                  value={selectedRoomVersion}
 | 
			
		||||
                  onChange={selectRoomVersion}
 | 
			
		||||
                  disabled={upgrading}
 | 
			
		||||
                />
 | 
			
		||||
                {allowAdditionalCreators && (
 | 
			
		||||
                  <SequenceCard
 | 
			
		||||
                    style={{ padding: config.space.S300 }}
 | 
			
		||||
                    variant="SurfaceVariant"
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="500"
 | 
			
		||||
                  >
 | 
			
		||||
                    <AdditionalCreatorInput
 | 
			
		||||
                      additionalCreators={additionalCreators}
 | 
			
		||||
                      onSelect={addAdditionalCreator}
 | 
			
		||||
                      onRemove={removeAdditionalCreator}
 | 
			
		||||
                      disabled={upgrading}
 | 
			
		||||
                    />
 | 
			
		||||
                  </SequenceCard>
 | 
			
		||||
                )}
 | 
			
		||||
              </Box>
 | 
			
		||||
              {upgradeState.status === AsyncStatus.Error && (
 | 
			
		||||
                <Text className={BreakWord} style={{ color: color.Critical.Main }} size="T200">
 | 
			
		||||
                  {(upgradeState.error as MatrixError).message}
 | 
			
		||||
                </Text>
 | 
			
		||||
              )}
 | 
			
		||||
              <Button
 | 
			
		||||
                onClick={handleUpgradeRoom}
 | 
			
		||||
                variant="Secondary"
 | 
			
		||||
                disabled={upgrading}
 | 
			
		||||
                before={upgrading && <Spinner size="200" variant="Secondary" fill="Solid" />}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B400">{room.isSpaceRoom() ? 'Upgrade Space' : 'Upgrade Room'}</Text>
 | 
			
		||||
              </Button>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </Dialog>
 | 
			
		||||
        </FocusTrap>
 | 
			
		||||
      </OverlayCenter>
 | 
			
		||||
    </Overlay>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type RoomUpgradeProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissions: RoomPermissionsAPI;
 | 
			
		||||
  requestClose: () => void;
 | 
			
		||||
};
 | 
			
		||||
export function RoomUpgrade({ powerLevels, requestClose }: RoomUpgradeProps) {
 | 
			
		||||
export function RoomUpgrade({ permissions, requestClose }: RoomUpgradeProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const { navigateRoom, navigateSpace } = useRoomNavigate();
 | 
			
		||||
  const createContent = useStateEvent(
 | 
			
		||||
    room,
 | 
			
		||||
    StateEvent.RoomCreate
 | 
			
		||||
  )?.getContent<RoomCreateEventContent>();
 | 
			
		||||
  const roomVersion = createContent?.room_version ?? 1;
 | 
			
		||||
  )?.getContent<IRoomCreateContent>();
 | 
			
		||||
  const roomVersion = createContent?.room_version ?? '1';
 | 
			
		||||
  const predecessorRoomId = createContent?.predecessor?.room_id;
 | 
			
		||||
 | 
			
		||||
  const capabilities = useCapabilities();
 | 
			
		||||
  const defaultRoomVersion = capabilities['m.room_versions']?.default;
 | 
			
		||||
 | 
			
		||||
  const tombstoneContent = useStateEvent(
 | 
			
		||||
    room,
 | 
			
		||||
    StateEvent.RoomTombstone
 | 
			
		||||
  )?.getContent<RoomTombstoneEventContent>();
 | 
			
		||||
  const replacementRoom = tombstoneContent?.replacement_room;
 | 
			
		||||
 | 
			
		||||
  const userPowerLevel = powerLevelAPI.getPowerLevel(powerLevels, mx.getSafeUserId());
 | 
			
		||||
  const canUpgrade = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
    powerLevels,
 | 
			
		||||
    StateEvent.RoomTombstone,
 | 
			
		||||
    userPowerLevel
 | 
			
		||||
  );
 | 
			
		||||
  const canUpgrade = permissions.stateEvent(StateEvent.RoomTombstone, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  const handleOpenRoom = () => {
 | 
			
		||||
    if (replacementRoom) {
 | 
			
		||||
| 
						 | 
				
			
			@ -85,31 +203,8 @@ export function RoomUpgrade({ powerLevels, requestClose }: RoomUpgradeProps) {
 | 
			
		|||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const [upgradeState, upgrade] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (version: string) => {
 | 
			
		||||
        await mx.upgradeRoom(room.roomId, version);
 | 
			
		||||
      },
 | 
			
		||||
      [mx, room]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const upgrading = upgradeState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const [prompt, setPrompt] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const handleSubmitUpgrade: FormEventHandler<HTMLFormElement> = (evt) => {
 | 
			
		||||
    evt.preventDefault();
 | 
			
		||||
 | 
			
		||||
    const target = evt.target as HTMLFormElement | undefined;
 | 
			
		||||
    const versionInput = target?.versionInput as HTMLInputElement | undefined;
 | 
			
		||||
    const version = versionInput?.value.trim();
 | 
			
		||||
    if (!version) return;
 | 
			
		||||
 | 
			
		||||
    upgrade(version);
 | 
			
		||||
    setPrompt(false);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SequenceCard
 | 
			
		||||
      className={SequenceCardStyle}
 | 
			
		||||
| 
						 | 
				
			
			@ -123,7 +218,7 @@ export function RoomUpgrade({ powerLevels, requestClose }: RoomUpgradeProps) {
 | 
			
		|||
          replacementRoom
 | 
			
		||||
            ? tombstoneContent.body ||
 | 
			
		||||
              `This ${room.isSpaceRoom() ? 'space' : 'room'} has been replaced!`
 | 
			
		||||
            : `Current room version: ${roomVersion}.`
 | 
			
		||||
            : `Current version: ${roomVersion}.`
 | 
			
		||||
        }
 | 
			
		||||
        after={
 | 
			
		||||
          <Box alignItems="Center" gap="200">
 | 
			
		||||
| 
						 | 
				
			
			@ -155,8 +250,7 @@ export function RoomUpgrade({ powerLevels, requestClose }: RoomUpgradeProps) {
 | 
			
		|||
                variant="Secondary"
 | 
			
		||||
                fill="Solid"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                disabled={upgrading || !canUpgrade}
 | 
			
		||||
                before={upgrading && <Spinner size="100" variant="Secondary" fill="Solid" />}
 | 
			
		||||
                disabled={!canUpgrade}
 | 
			
		||||
                onClick={() => setPrompt(true)}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="B300">Upgrade</Text>
 | 
			
		||||
| 
						 | 
				
			
			@ -165,63 +259,7 @@ export function RoomUpgrade({ powerLevels, requestClose }: RoomUpgradeProps) {
 | 
			
		|||
          </Box>
 | 
			
		||||
        }
 | 
			
		||||
      >
 | 
			
		||||
        {upgradeState.status === AsyncStatus.Error && (
 | 
			
		||||
          <Text style={{ color: color.Critical.Main }} size="T200">
 | 
			
		||||
            {(upgradeState.error as MatrixError).message}
 | 
			
		||||
          </Text>
 | 
			
		||||
        )}
 | 
			
		||||
 | 
			
		||||
        {prompt && (
 | 
			
		||||
          <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
            <OverlayCenter>
 | 
			
		||||
              <FocusTrap
 | 
			
		||||
                focusTrapOptions={{
 | 
			
		||||
                  initialFocus: false,
 | 
			
		||||
                  onDeactivate: () => setPrompt(false),
 | 
			
		||||
                  clickOutsideDeactivates: true,
 | 
			
		||||
                  escapeDeactivates: stopPropagation,
 | 
			
		||||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <Dialog variant="Surface" as="form" onSubmit={handleSubmitUpgrade}>
 | 
			
		||||
                  <Header
 | 
			
		||||
                    style={{
 | 
			
		||||
                      padding: `0 ${config.space.S200} 0 ${config.space.S400}`,
 | 
			
		||||
                      borderBottomWidth: config.borderWidth.B300,
 | 
			
		||||
                    }}
 | 
			
		||||
                    variant="Surface"
 | 
			
		||||
                    size="500"
 | 
			
		||||
                  >
 | 
			
		||||
                    <Box grow="Yes">
 | 
			
		||||
                      <Text size="H4">{room.isSpaceRoom() ? 'Space Upgrade' : 'Room Upgrade'}</Text>
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    <IconButton size="300" onClick={() => setPrompt(false)} radii="300">
 | 
			
		||||
                      <Icon src={Icons.Cross} />
 | 
			
		||||
                    </IconButton>
 | 
			
		||||
                  </Header>
 | 
			
		||||
                  <Box style={{ padding: config.space.S400 }} direction="Column" gap="400">
 | 
			
		||||
                    <Text priority="400" style={{ color: color.Critical.Main }}>
 | 
			
		||||
                      <b>This action is irreversible!</b>
 | 
			
		||||
                    </Text>
 | 
			
		||||
                    <Box direction="Column" gap="100">
 | 
			
		||||
                      <Text size="L400">Version</Text>
 | 
			
		||||
                      <Input
 | 
			
		||||
                        defaultValue={defaultRoomVersion}
 | 
			
		||||
                        name="versionInput"
 | 
			
		||||
                        variant="Background"
 | 
			
		||||
                        required
 | 
			
		||||
                      />
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    <Button type="submit" variant="Secondary">
 | 
			
		||||
                      <Text size="B400">
 | 
			
		||||
                        {room.isSpaceRoom() ? 'Upgrade Space' : 'Upgrade Room'}
 | 
			
		||||
                      </Text>
 | 
			
		||||
                    </Button>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Dialog>
 | 
			
		||||
              </FocusTrap>
 | 
			
		||||
            </OverlayCenter>
 | 
			
		||||
          </Overlay>
 | 
			
		||||
        )}
 | 
			
		||||
        {prompt && <RoomUpgradeDialog requestClose={() => setPrompt(false)} />}
 | 
			
		||||
      </SettingTile>
 | 
			
		||||
    </SequenceCard>
 | 
			
		||||
  );
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -27,17 +27,12 @@ import { Page, PageContent, PageHeader } from '../../../components/page';
 | 
			
		|||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { useRoomMembers } from '../../../hooks/useRoomMembers';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import {
 | 
			
		||||
  useFlattenPowerLevelTagMembers,
 | 
			
		||||
  usePowerLevelTags,
 | 
			
		||||
} from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
import { usePowerLevels } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { VirtualTile } from '../../../components/virtualizer';
 | 
			
		||||
import { MemberTile } from '../../../components/member-tile';
 | 
			
		||||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { getMxIdLocalPart, getMxIdServer } from '../../../utils/matrix';
 | 
			
		||||
import { ServerBadge } from '../../../components/server-badge';
 | 
			
		||||
import { openProfileViewer } from '../../../../client/action/navigation';
 | 
			
		||||
import { useDebounce } from '../../../hooks/useDebounce';
 | 
			
		||||
import {
 | 
			
		||||
  SearchItemStrGetter,
 | 
			
		||||
| 
						 | 
				
			
			@ -46,13 +41,21 @@ import {
 | 
			
		|||
} from '../../../hooks/useAsyncSearch';
 | 
			
		||||
import { getMemberSearchStr } from '../../../utils/room';
 | 
			
		||||
import { useMembershipFilter, useMembershipFilterMenu } from '../../../hooks/useMemberFilter';
 | 
			
		||||
import { useMemberSort, useMemberSortMenu } from '../../../hooks/useMemberSort';
 | 
			
		||||
import { useMemberPowerSort, useMemberSort, useMemberSortMenu } from '../../../hooks/useMemberSort';
 | 
			
		||||
import { settingsAtom } from '../../../state/settings';
 | 
			
		||||
import { useSetting } from '../../../state/hooks/settings';
 | 
			
		||||
import { UseStateProvider } from '../../../components/UseStateProvider';
 | 
			
		||||
import { MembershipFilterMenu } from '../../../components/MembershipFilterMenu';
 | 
			
		||||
import { MemberSortMenu } from '../../../components/MemberSortMenu';
 | 
			
		||||
import { ScrollTopContainer } from '../../../components/scroll-top-container';
 | 
			
		||||
import {
 | 
			
		||||
  useOpenUserRoomProfile,
 | 
			
		||||
  useUserRoomProfileState,
 | 
			
		||||
} from '../../../state/hooks/userRoomProfile';
 | 
			
		||||
import { useSpaceOptionally } from '../../../hooks/useSpace';
 | 
			
		||||
import { useFlattenPowerTagMembers, useGetMemberPowerTag } from '../../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { getMouseEventCords } from '../../../utils/dom';
 | 
			
		||||
 | 
			
		||||
const SEARCH_OPTIONS: UseAsyncSearchOptions = {
 | 
			
		||||
  limit: 1000,
 | 
			
		||||
| 
						 | 
				
			
			@ -77,15 +80,19 @@ export function Members({ requestClose }: MembersProps) {
 | 
			
		|||
  const room = useRoom();
 | 
			
		||||
  const members = useRoomMembers(mx, room.roomId);
 | 
			
		||||
  const fetchingMembers = members.length < room.getJoinedMemberCount();
 | 
			
		||||
  const openProfile = useOpenUserRoomProfile();
 | 
			
		||||
  const profileUser = useUserRoomProfileState();
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const { getPowerLevel } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const [, getPowerLevelTag] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
  const getPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const [membershipFilterIndex, setMembershipFilterIndex] = useState(0);
 | 
			
		||||
  const [sortFilterIndex, setSortFilterIndex] = useSetting(settingsAtom, 'memberSortFilterIndex');
 | 
			
		||||
  const membershipFilter = useMembershipFilter(membershipFilterIndex, useMembershipFilterMenu());
 | 
			
		||||
  const memberSort = useMemberSort(sortFilterIndex, useMemberSortMenu());
 | 
			
		||||
  const memberPowerSort = useMemberPowerSort(creators);
 | 
			
		||||
 | 
			
		||||
  const scrollRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const searchInputRef = useRef<HTMLInputElement>(null);
 | 
			
		||||
| 
						 | 
				
			
			@ -96,8 +103,8 @@ export function Members({ requestClose }: MembersProps) {
 | 
			
		|||
      Array.from(members)
 | 
			
		||||
        .filter(membershipFilter.filterFn)
 | 
			
		||||
        .sort(memberSort.sortFn)
 | 
			
		||||
        .sort((a, b) => b.powerLevel - a.powerLevel),
 | 
			
		||||
    [members, membershipFilter, memberSort]
 | 
			
		||||
        .sort(memberPowerSort),
 | 
			
		||||
    [members, membershipFilter, memberSort, memberPowerSort]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [result, search, resetSearch] = useAsyncSearch(
 | 
			
		||||
| 
						 | 
				
			
			@ -107,11 +114,7 @@ export function Members({ requestClose }: MembersProps) {
 | 
			
		|||
  );
 | 
			
		||||
  if (!result && searchInputRef.current?.value) search(searchInputRef.current.value);
 | 
			
		||||
 | 
			
		||||
  const flattenTagMembers = useFlattenPowerLevelTagMembers(
 | 
			
		||||
    result?.items ?? sortedMembers,
 | 
			
		||||
    getPowerLevel,
 | 
			
		||||
    getPowerLevelTag
 | 
			
		||||
  );
 | 
			
		||||
  const flattenTagMembers = useFlattenPowerTagMembers(result?.items ?? sortedMembers, getPowerTag);
 | 
			
		||||
 | 
			
		||||
  const virtualizer = useVirtualizer({
 | 
			
		||||
    count: flattenTagMembers.length,
 | 
			
		||||
| 
						 | 
				
			
			@ -142,8 +145,9 @@ export function Members({ requestClose }: MembersProps) {
 | 
			
		|||
  const handleMemberClick: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    const btn = evt.currentTarget as HTMLButtonElement;
 | 
			
		||||
    const userId = btn.getAttribute('data-user-id');
 | 
			
		||||
    openProfileViewer(userId, room.roomId);
 | 
			
		||||
    requestClose();
 | 
			
		||||
    if (userId) {
 | 
			
		||||
      openProfile(room.roomId, space?.roomId, userId, getMouseEventCords(evt.nativeEvent));
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
| 
						 | 
				
			
			@ -317,6 +321,7 @@ export function Members({ requestClose }: MembersProps) {
 | 
			
		|||
                          <MemberTile
 | 
			
		||||
                            data-user-id={tagOrMember.userId}
 | 
			
		||||
                            onClick={handleMemberClick}
 | 
			
		||||
                            aria-pressed={profileUser?.userId === tagOrMember.userId}
 | 
			
		||||
                            mx={mx}
 | 
			
		||||
                            room={room}
 | 
			
		||||
                            member={tagOrMember}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -10,10 +10,9 @@ import {
 | 
			
		|||
  getPermissionPower,
 | 
			
		||||
  IPowerLevels,
 | 
			
		||||
  PermissionLocation,
 | 
			
		||||
  usePowerLevelsAPI,
 | 
			
		||||
} from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { PermissionGroup } from './types';
 | 
			
		||||
import { getPowers, usePowerLevelTags } from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
import { getPowerLevelTag, getPowers, usePowerLevelTags } from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
| 
						 | 
				
			
			@ -26,19 +25,20 @@ const USER_DEFAULT_LOCATION: PermissionLocation = {
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
type PermissionGroupsProps = {
 | 
			
		||||
  canEdit: boolean;
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  permissionGroups: PermissionGroup[];
 | 
			
		||||
};
 | 
			
		||||
export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGroupsProps) {
 | 
			
		||||
export function PermissionGroups({
 | 
			
		||||
  powerLevels,
 | 
			
		||||
  permissionGroups,
 | 
			
		||||
  canEdit,
 | 
			
		||||
}: PermissionGroupsProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
  const { getPowerLevel, canSendStateEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canChangePermission = canSendStateEvent(
 | 
			
		||||
    StateEvent.RoomPowerLevels,
 | 
			
		||||
    getPowerLevel(mx.getSafeUserId())
 | 
			
		||||
  );
 | 
			
		||||
  const [powerLevelTags, getPowerLevelTag] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const maxPower = useMemo(() => Math.max(...getPowers(powerLevelTags)), [powerLevelTags]);
 | 
			
		||||
 | 
			
		||||
  const [permissionUpdate, setPermissionUpdate] = useState<Map<PermissionLocation, number>>(
 | 
			
		||||
| 
						 | 
				
			
			@ -82,6 +82,7 @@ export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGr
 | 
			
		|||
        permissionUpdate.forEach((power, location) =>
 | 
			
		||||
          applyPermissionPower(draftPowerLevels, location, power)
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return draftPowerLevels;
 | 
			
		||||
      });
 | 
			
		||||
      await mx.sendStateEvent(room.roomId, StateEvent.RoomPowerLevels as any, editedPowerLevels);
 | 
			
		||||
| 
						 | 
				
			
			@ -108,7 +109,7 @@ export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGr
 | 
			
		|||
    const powerUpdate = permissionUpdate.get(USER_DEFAULT_LOCATION);
 | 
			
		||||
    const value = powerUpdate ?? power;
 | 
			
		||||
 | 
			
		||||
    const tag = getPowerLevelTag(value);
 | 
			
		||||
    const tag = getPowerLevelTag(powerLevelTags, value);
 | 
			
		||||
    const powerChanges = value !== power;
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
| 
						 | 
				
			
			@ -136,14 +137,14 @@ export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGr
 | 
			
		|||
                    fill="Soft"
 | 
			
		||||
                    radii="Pill"
 | 
			
		||||
                    aria-selected={opened}
 | 
			
		||||
                    disabled={!canChangePermission || applyingChanges}
 | 
			
		||||
                    disabled={!canEdit || applyingChanges}
 | 
			
		||||
                    after={
 | 
			
		||||
                      powerChanges && (
 | 
			
		||||
                        <Badge size="200" variant="Success" fill="Solid" radii="Pill" />
 | 
			
		||||
                      )
 | 
			
		||||
                    }
 | 
			
		||||
                    before={
 | 
			
		||||
                      canChangePermission && (
 | 
			
		||||
                      canEdit && (
 | 
			
		||||
                        <Icon size="50" src={opened ? Icons.ChevronTop : Icons.ChevronBottom} />
 | 
			
		||||
                      )
 | 
			
		||||
                    }
 | 
			
		||||
| 
						 | 
				
			
			@ -173,7 +174,7 @@ export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGr
 | 
			
		|||
            const powerUpdate = permissionUpdate.get(item.location);
 | 
			
		||||
            const value = powerUpdate ?? power;
 | 
			
		||||
 | 
			
		||||
            const tag = getPowerLevelTag(value);
 | 
			
		||||
            const tag = getPowerLevelTag(powerLevelTags, value);
 | 
			
		||||
            const powerChanges = value !== power;
 | 
			
		||||
 | 
			
		||||
            return (
 | 
			
		||||
| 
						 | 
				
			
			@ -200,14 +201,14 @@ export function PermissionGroups({ powerLevels, permissionGroups }: PermissionGr
 | 
			
		|||
                          fill="Soft"
 | 
			
		||||
                          radii="Pill"
 | 
			
		||||
                          aria-selected={opened}
 | 
			
		||||
                          disabled={!canChangePermission || applyingChanges}
 | 
			
		||||
                          disabled={!canEdit || applyingChanges}
 | 
			
		||||
                          after={
 | 
			
		||||
                            powerChanges && (
 | 
			
		||||
                              <Badge size="200" variant="Success" fill="Solid" radii="Pill" />
 | 
			
		||||
                            )
 | 
			
		||||
                          }
 | 
			
		||||
                          before={
 | 
			
		||||
                            canChangePermission && (
 | 
			
		||||
                            canEdit && (
 | 
			
		||||
                              <Icon
 | 
			
		||||
                                size="50"
 | 
			
		||||
                                src={opened ? Icons.ChevronTop : Icons.ChevronBottom}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -16,7 +16,7 @@ import {
 | 
			
		|||
} from 'folds';
 | 
			
		||||
import { SequenceCard } from '../../../components/sequence-card';
 | 
			
		||||
import { SequenceCardStyle } from '../styles.css';
 | 
			
		||||
import { getPowers, getTagIconSrc, usePowerLevelTags } from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
import { getPowers, usePowerLevelTags } from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import { getPermissionPower, IPowerLevels } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
| 
						 | 
				
			
			@ -25,6 +25,9 @@ import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		|||
import { useMediaAuthentication } from '../../../hooks/useMediaAuthentication';
 | 
			
		||||
import { stopPropagation } from '../../../utils/keyboard';
 | 
			
		||||
import { PermissionGroup } from './types';
 | 
			
		||||
import { getPowerTagIconSrc } from '../../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreatorsTag } from '../../../hooks/useRoomCreatorsTag';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
 | 
			
		||||
type PeekPermissionsProps = {
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
| 
						 | 
				
			
			@ -108,10 +111,43 @@ export function Powers({ powerLevels, permissionGroups, onEdit }: PowersProps) {
 | 
			
		|||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const [powerLevelTags] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
  const creatorsTag = useRoomCreatorsTag();
 | 
			
		||||
  const creatorTagIconSrc =
 | 
			
		||||
    creatorsTag.icon && getPowerTagIconSrc(mx, useAuthentication, creatorsTag.icon);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box direction="Column" gap="100">
 | 
			
		||||
      {creators.size > 0 && (
 | 
			
		||||
        <SequenceCard
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          className={SequenceCardStyle}
 | 
			
		||||
          direction="Column"
 | 
			
		||||
          gap="400"
 | 
			
		||||
        >
 | 
			
		||||
          <SettingTile
 | 
			
		||||
            title="Founders"
 | 
			
		||||
            description="Founding members has all permissions and can only be changed during upgrade."
 | 
			
		||||
          />
 | 
			
		||||
 | 
			
		||||
          <SettingTile>
 | 
			
		||||
            <Box gap="200" wrap="Wrap">
 | 
			
		||||
              <Chip
 | 
			
		||||
                disabled
 | 
			
		||||
                variant="Secondary"
 | 
			
		||||
                radii="300"
 | 
			
		||||
                before={<PowerColorBadge color={creatorsTag.color} />}
 | 
			
		||||
                after={creatorTagIconSrc && <PowerIcon size="50" iconSrc={creatorTagIconSrc} />}
 | 
			
		||||
              >
 | 
			
		||||
                <Text size="T300" truncate>
 | 
			
		||||
                  <b>{creatorsTag.name}</b>
 | 
			
		||||
                </Text>
 | 
			
		||||
              </Chip>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </SettingTile>
 | 
			
		||||
        </SequenceCard>
 | 
			
		||||
      )}
 | 
			
		||||
      <SequenceCard
 | 
			
		||||
        variant="SurfaceVariant"
 | 
			
		||||
        className={SequenceCardStyle}
 | 
			
		||||
| 
						 | 
				
			
			@ -142,7 +178,7 @@ export function Powers({ powerLevels, permissionGroups, onEdit }: PowersProps) {
 | 
			
		|||
          <Box gap="200" wrap="Wrap">
 | 
			
		||||
            {getPowers(powerLevelTags).map((power) => {
 | 
			
		||||
              const tag = powerLevelTags[power];
 | 
			
		||||
              const tagIconSrc = tag.icon && getTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
              const tagIconSrc = tag.icon && getPowerTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
 | 
			
		||||
              return (
 | 
			
		||||
                <PeekPermissions
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -27,10 +27,7 @@ import { SequenceCardStyle } from '../styles.css';
 | 
			
		|||
import { SettingTile } from '../../../components/setting-tile';
 | 
			
		||||
import {
 | 
			
		||||
  getPowers,
 | 
			
		||||
  getTagIconSrc,
 | 
			
		||||
  getUsedPowers,
 | 
			
		||||
  PowerLevelTag,
 | 
			
		||||
  PowerLevelTagIcon,
 | 
			
		||||
  PowerLevelTags,
 | 
			
		||||
  usePowerLevelTags,
 | 
			
		||||
} from '../../../hooks/usePowerLevelTags';
 | 
			
		||||
| 
						 | 
				
			
			@ -47,15 +44,17 @@ import { useFilePicker } from '../../../hooks/useFilePicker';
 | 
			
		|||
import { CompactUploadCardRenderer } from '../../../components/upload-card';
 | 
			
		||||
import { createUploadAtom, UploadSuccess } from '../../../state/upload';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../../hooks/useAsyncCallback';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { MemberPowerTag, MemberPowerTagIcon, StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { useAlive } from '../../../hooks/useAlive';
 | 
			
		||||
import { BetaNoticeBadge } from '../../../components/BetaNoticeBadge';
 | 
			
		||||
import { getPowerTagIconSrc } from '../../../hooks/useMemberPowerTag';
 | 
			
		||||
import { creatorsSupported } from '../../../utils/matrix';
 | 
			
		||||
 | 
			
		||||
type EditPowerProps = {
 | 
			
		||||
  maxPower: number;
 | 
			
		||||
  power?: number;
 | 
			
		||||
  tag?: PowerLevelTag;
 | 
			
		||||
  onSave: (power: number, tag: PowerLevelTag) => void;
 | 
			
		||||
  tag?: MemberPowerTag;
 | 
			
		||||
  onSave: (power: number, tag: MemberPowerTag) => void;
 | 
			
		||||
  onClose: () => void;
 | 
			
		||||
};
 | 
			
		||||
function EditPower({ maxPower, power, tag, onSave, onClose }: EditPowerProps) {
 | 
			
		||||
| 
						 | 
				
			
			@ -63,6 +62,7 @@ function EditPower({ maxPower, power, tag, onSave, onClose }: EditPowerProps) {
 | 
			
		|||
  const room = useRoom();
 | 
			
		||||
  const roomToParents = useAtomValue(roomToParentsAtom);
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const supportCreators = creatorsSupported(room.getVersion());
 | 
			
		||||
 | 
			
		||||
  const imagePackRooms = useImagePackRooms(room.roomId, roomToParents);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -70,9 +70,9 @@ function EditPower({ maxPower, power, tag, onSave, onClose }: EditPowerProps) {
 | 
			
		|||
  const pickFile = useFilePicker(setIconFile, false);
 | 
			
		||||
 | 
			
		||||
  const [tagColor, setTagColor] = useState<string | undefined>(tag?.color);
 | 
			
		||||
  const [tagIcon, setTagIcon] = useState<PowerLevelTagIcon | undefined>(tag?.icon);
 | 
			
		||||
  const [tagIcon, setTagIcon] = useState<MemberPowerTagIcon | undefined>(tag?.icon);
 | 
			
		||||
  const uploadingIcon = iconFile && !tagIcon;
 | 
			
		||||
  const tagIconSrc = tagIcon && getTagIconSrc(mx, useAuthentication, tagIcon);
 | 
			
		||||
  const tagIconSrc = tagIcon && getPowerTagIconSrc(mx, useAuthentication, tagIcon);
 | 
			
		||||
 | 
			
		||||
  const iconUploadAtom = useMemo(() => {
 | 
			
		||||
    if (iconFile) return createUploadAtom(iconFile);
 | 
			
		||||
| 
						 | 
				
			
			@ -101,11 +101,11 @@ function EditPower({ maxPower, power, tag, onSave, onClose }: EditPowerProps) {
 | 
			
		|||
 | 
			
		||||
    const tagPower = parseInt(powerInput.value, 10);
 | 
			
		||||
    if (Number.isNaN(tagPower)) return;
 | 
			
		||||
    if (tagPower > maxPower) return;
 | 
			
		||||
 | 
			
		||||
    const tagName = nameInput.value.trim();
 | 
			
		||||
    if (!tagName) return;
 | 
			
		||||
 | 
			
		||||
    const editedTag: PowerLevelTag = {
 | 
			
		||||
    const editedTag: MemberPowerTag = {
 | 
			
		||||
      name: tagName,
 | 
			
		||||
      color: tagColor,
 | 
			
		||||
      icon: tagIcon,
 | 
			
		||||
| 
						 | 
				
			
			@ -165,7 +165,7 @@ function EditPower({ maxPower, power, tag, onSave, onClose }: EditPowerProps) {
 | 
			
		|||
              radii="300"
 | 
			
		||||
              type="number"
 | 
			
		||||
              placeholder="75"
 | 
			
		||||
              max={maxPower}
 | 
			
		||||
              max={supportCreators ? undefined : maxPower}
 | 
			
		||||
              outlined={typeof power === 'number'}
 | 
			
		||||
              readOnly={typeof power === 'number'}
 | 
			
		||||
              required
 | 
			
		||||
| 
						 | 
				
			
			@ -298,7 +298,7 @@ export function PowersEditor({ powerLevels, requestClose }: PowersEditorProps) {
 | 
			
		|||
    return [up, Math.max(...Array.from(up))];
 | 
			
		||||
  }, [powerLevels]);
 | 
			
		||||
 | 
			
		||||
  const [powerLevelTags] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const [editedPowerTags, setEditedPowerTags] = useState<PowerLevelTags>();
 | 
			
		||||
  const [deleted, setDeleted] = useState<Set<number>>(new Set());
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -317,7 +317,7 @@ export function PowersEditor({ powerLevels, requestClose }: PowersEditorProps) {
 | 
			
		|||
  }, []);
 | 
			
		||||
 | 
			
		||||
  const handleSaveTag = useCallback(
 | 
			
		||||
    (power: number, tag: PowerLevelTag) => {
 | 
			
		||||
    (power: number, tag: MemberPowerTag) => {
 | 
			
		||||
      setEditedPowerTags((tags) => {
 | 
			
		||||
        const editedTags = { ...(tags ?? powerLevelTags) };
 | 
			
		||||
        editedTags[power] = tag;
 | 
			
		||||
| 
						 | 
				
			
			@ -419,7 +419,8 @@ export function PowersEditor({ powerLevels, requestClose }: PowersEditorProps) {
 | 
			
		|||
                </SequenceCard>
 | 
			
		||||
                {getPowers(powerTags).map((power) => {
 | 
			
		||||
                  const tag = powerTags[power];
 | 
			
		||||
                  const tagIconSrc = tag.icon && getTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
                  const tagIconSrc =
 | 
			
		||||
                    tag.icon && getPowerTagIconSrc(mx, useAuthentication, tag.icon);
 | 
			
		||||
 | 
			
		||||
                  return (
 | 
			
		||||
                    <SequenceCard
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										306
									
								
								src/app/features/create-room/CreateRoom.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										306
									
								
								src/app/features/create-room/CreateRoom.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,306 @@
 | 
			
		|||
import React, { FormEventHandler, useCallback, useEffect, useState } from 'react';
 | 
			
		||||
import { MatrixError, Room } from 'matrix-js-sdk';
 | 
			
		||||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  Chip,
 | 
			
		||||
  color,
 | 
			
		||||
  config,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Input,
 | 
			
		||||
  Spinner,
 | 
			
		||||
  Switch,
 | 
			
		||||
  Text,
 | 
			
		||||
  TextArea,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { SettingTile } from '../../components/setting-tile';
 | 
			
		||||
import { SequenceCard } from '../../components/sequence-card';
 | 
			
		||||
import {
 | 
			
		||||
  creatorsSupported,
 | 
			
		||||
  knockRestrictedSupported,
 | 
			
		||||
  knockSupported,
 | 
			
		||||
  restrictedSupported,
 | 
			
		||||
} from '../../utils/matrix';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { millisecondsToMinutes, replaceSpaceWithDash } from '../../utils/common';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { useCapabilities } from '../../hooks/useCapabilities';
 | 
			
		||||
import { useAlive } from '../../hooks/useAlive';
 | 
			
		||||
import { ErrorCode } from '../../cs-errorcode';
 | 
			
		||||
import {
 | 
			
		||||
  AdditionalCreatorInput,
 | 
			
		||||
  createRoom,
 | 
			
		||||
  CreateRoomAliasInput,
 | 
			
		||||
  CreateRoomData,
 | 
			
		||||
  CreateRoomKind,
 | 
			
		||||
  CreateRoomKindSelector,
 | 
			
		||||
  RoomVersionSelector,
 | 
			
		||||
  useAdditionalCreators,
 | 
			
		||||
} from '../../components/create-room';
 | 
			
		||||
 | 
			
		||||
const getCreateRoomKindToIcon = (kind: CreateRoomKind) => {
 | 
			
		||||
  if (kind === CreateRoomKind.Private) return Icons.HashLock;
 | 
			
		||||
  if (kind === CreateRoomKind.Restricted) return Icons.Hash;
 | 
			
		||||
  return Icons.HashGlobe;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
type CreateRoomFormProps = {
 | 
			
		||||
  defaultKind?: CreateRoomKind;
 | 
			
		||||
  space?: Room;
 | 
			
		||||
  onCreate?: (roomId: string) => void;
 | 
			
		||||
};
 | 
			
		||||
export function CreateRoomForm({ defaultKind, space, onCreate }: CreateRoomFormProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
 | 
			
		||||
  const capabilities = useCapabilities();
 | 
			
		||||
  const roomVersions = capabilities['m.room_versions'];
 | 
			
		||||
  const [selectedRoomVersion, selectRoomVersion] = useState(roomVersions?.default ?? '1');
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    // capabilities load async
 | 
			
		||||
    selectRoomVersion(roomVersions?.default ?? '1');
 | 
			
		||||
  }, [roomVersions?.default]);
 | 
			
		||||
 | 
			
		||||
  const allowRestricted = space && restrictedSupported(selectedRoomVersion);
 | 
			
		||||
 | 
			
		||||
  const [kind, setKind] = useState(
 | 
			
		||||
    defaultKind ?? allowRestricted ? CreateRoomKind.Restricted : CreateRoomKind.Private
 | 
			
		||||
  );
 | 
			
		||||
  const allowAdditionalCreators = creatorsSupported(selectedRoomVersion);
 | 
			
		||||
  const { additionalCreators, addAdditionalCreator, removeAdditionalCreator } =
 | 
			
		||||
    useAdditionalCreators();
 | 
			
		||||
  const [federation, setFederation] = useState(true);
 | 
			
		||||
  const [encryption, setEncryption] = useState(false);
 | 
			
		||||
  const [knock, setKnock] = useState(false);
 | 
			
		||||
  const [advance, setAdvance] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const allowKnock = kind === CreateRoomKind.Private && knockSupported(selectedRoomVersion);
 | 
			
		||||
  const allowKnockRestricted =
 | 
			
		||||
    kind === CreateRoomKind.Restricted && knockRestrictedSupported(selectedRoomVersion);
 | 
			
		||||
 | 
			
		||||
  const handleRoomVersionChange = (version: string) => {
 | 
			
		||||
    if (!restrictedSupported(version)) {
 | 
			
		||||
      setKind(CreateRoomKind.Private);
 | 
			
		||||
    }
 | 
			
		||||
    selectRoomVersion(version);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const [createState, create] = useAsyncCallback<string, Error | MatrixError, [CreateRoomData]>(
 | 
			
		||||
    useCallback((data) => createRoom(mx, data), [mx])
 | 
			
		||||
  );
 | 
			
		||||
  const loading = createState.status === AsyncStatus.Loading;
 | 
			
		||||
  const error = createState.status === AsyncStatus.Error ? createState.error : undefined;
 | 
			
		||||
  const disabled = createState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const handleSubmit: FormEventHandler<HTMLFormElement> = (evt) => {
 | 
			
		||||
    evt.preventDefault();
 | 
			
		||||
    if (disabled) return;
 | 
			
		||||
    const form = evt.currentTarget;
 | 
			
		||||
 | 
			
		||||
    const nameInput = form.nameInput as HTMLInputElement | undefined;
 | 
			
		||||
    const topicTextArea = form.topicTextAria as HTMLTextAreaElement | undefined;
 | 
			
		||||
    const aliasInput = form.aliasInput as HTMLInputElement | undefined;
 | 
			
		||||
    const roomName = nameInput?.value.trim();
 | 
			
		||||
    const roomTopic = topicTextArea?.value.trim();
 | 
			
		||||
    const aliasLocalPart =
 | 
			
		||||
      aliasInput && aliasInput.value ? replaceSpaceWithDash(aliasInput.value) : undefined;
 | 
			
		||||
 | 
			
		||||
    if (!roomName) return;
 | 
			
		||||
    const publicRoom = kind === CreateRoomKind.Public;
 | 
			
		||||
    let roomKnock = false;
 | 
			
		||||
    if (allowKnock && kind === CreateRoomKind.Private) {
 | 
			
		||||
      roomKnock = knock;
 | 
			
		||||
    }
 | 
			
		||||
    if (allowKnockRestricted && kind === CreateRoomKind.Restricted) {
 | 
			
		||||
      roomKnock = knock;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    create({
 | 
			
		||||
      version: selectedRoomVersion,
 | 
			
		||||
      parent: space,
 | 
			
		||||
      kind,
 | 
			
		||||
      name: roomName,
 | 
			
		||||
      topic: roomTopic || undefined,
 | 
			
		||||
      aliasLocalPart: publicRoom ? aliasLocalPart : undefined,
 | 
			
		||||
      encryption: publicRoom ? false : encryption,
 | 
			
		||||
      knock: roomKnock,
 | 
			
		||||
      allowFederation: federation,
 | 
			
		||||
      additionalCreators: allowAdditionalCreators ? additionalCreators : undefined,
 | 
			
		||||
    }).then((roomId) => {
 | 
			
		||||
      if (alive()) {
 | 
			
		||||
        onCreate?.(roomId);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box as="form" onSubmit={handleSubmit} grow="Yes" direction="Column" gap="500">
 | 
			
		||||
      <Box direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Access</Text>
 | 
			
		||||
        <CreateRoomKindSelector
 | 
			
		||||
          value={kind}
 | 
			
		||||
          onSelect={setKind}
 | 
			
		||||
          canRestrict={allowRestricted}
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
          getIcon={getCreateRoomKindToIcon}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Name</Text>
 | 
			
		||||
        <Input
 | 
			
		||||
          required
 | 
			
		||||
          before={<Icon size="100" src={getCreateRoomKindToIcon(kind)} />}
 | 
			
		||||
          name="nameInput"
 | 
			
		||||
          autoFocus
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          autoComplete="off"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Topic (Optional)</Text>
 | 
			
		||||
        <TextArea
 | 
			
		||||
          name="topicTextAria"
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
 | 
			
		||||
      {kind === CreateRoomKind.Public && <CreateRoomAliasInput disabled={disabled} />}
 | 
			
		||||
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Box gap="200" alignItems="End">
 | 
			
		||||
          <Text size="L400">Options</Text>
 | 
			
		||||
          <Box grow="Yes" justifyContent="End">
 | 
			
		||||
            <Chip
 | 
			
		||||
              radii="Pill"
 | 
			
		||||
              before={<Icon src={advance ? Icons.ChevronTop : Icons.ChevronBottom} size="50" />}
 | 
			
		||||
              onClick={() => setAdvance(!advance)}
 | 
			
		||||
              type="button"
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="T200">Advance Options</Text>
 | 
			
		||||
            </Chip>
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
        {allowAdditionalCreators && (
 | 
			
		||||
          <SequenceCard
 | 
			
		||||
            style={{ padding: config.space.S300 }}
 | 
			
		||||
            variant="SurfaceVariant"
 | 
			
		||||
            direction="Column"
 | 
			
		||||
            gap="500"
 | 
			
		||||
          >
 | 
			
		||||
            <AdditionalCreatorInput
 | 
			
		||||
              additionalCreators={additionalCreators}
 | 
			
		||||
              onSelect={addAdditionalCreator}
 | 
			
		||||
              onRemove={removeAdditionalCreator}
 | 
			
		||||
            />
 | 
			
		||||
          </SequenceCard>
 | 
			
		||||
        )}
 | 
			
		||||
        {kind !== CreateRoomKind.Public && (
 | 
			
		||||
          <>
 | 
			
		||||
            <SequenceCard
 | 
			
		||||
              style={{ padding: config.space.S300 }}
 | 
			
		||||
              variant="SurfaceVariant"
 | 
			
		||||
              direction="Column"
 | 
			
		||||
              gap="500"
 | 
			
		||||
            >
 | 
			
		||||
              <SettingTile
 | 
			
		||||
                title="End-to-End Encryption"
 | 
			
		||||
                description="Once this feature is enabled, it can't be disabled after the room is created."
 | 
			
		||||
                after={
 | 
			
		||||
                  <Switch
 | 
			
		||||
                    variant="Primary"
 | 
			
		||||
                    value={encryption}
 | 
			
		||||
                    onChange={setEncryption}
 | 
			
		||||
                    disabled={disabled}
 | 
			
		||||
                  />
 | 
			
		||||
                }
 | 
			
		||||
              />
 | 
			
		||||
            </SequenceCard>
 | 
			
		||||
            {advance && (allowKnock || allowKnockRestricted) && (
 | 
			
		||||
              <SequenceCard
 | 
			
		||||
                style={{ padding: config.space.S300 }}
 | 
			
		||||
                variant="SurfaceVariant"
 | 
			
		||||
                direction="Column"
 | 
			
		||||
                gap="500"
 | 
			
		||||
              >
 | 
			
		||||
                <SettingTile
 | 
			
		||||
                  title="Knock to Join"
 | 
			
		||||
                  description="Anyone can send request to join this room."
 | 
			
		||||
                  after={
 | 
			
		||||
                    <Switch
 | 
			
		||||
                      variant="Primary"
 | 
			
		||||
                      value={knock}
 | 
			
		||||
                      onChange={setKnock}
 | 
			
		||||
                      disabled={disabled}
 | 
			
		||||
                    />
 | 
			
		||||
                  }
 | 
			
		||||
                />
 | 
			
		||||
              </SequenceCard>
 | 
			
		||||
            )}
 | 
			
		||||
          </>
 | 
			
		||||
        )}
 | 
			
		||||
 | 
			
		||||
        <SequenceCard
 | 
			
		||||
          style={{ padding: config.space.S300 }}
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          direction="Column"
 | 
			
		||||
          gap="500"
 | 
			
		||||
        >
 | 
			
		||||
          <SettingTile
 | 
			
		||||
            title="Allow Federation"
 | 
			
		||||
            description="Users from other servers can join."
 | 
			
		||||
            after={
 | 
			
		||||
              <Switch
 | 
			
		||||
                variant="Primary"
 | 
			
		||||
                value={federation}
 | 
			
		||||
                onChange={setFederation}
 | 
			
		||||
                disabled={disabled}
 | 
			
		||||
              />
 | 
			
		||||
            }
 | 
			
		||||
          />
 | 
			
		||||
        </SequenceCard>
 | 
			
		||||
        {advance && (
 | 
			
		||||
          <RoomVersionSelector
 | 
			
		||||
            versions={roomVersions?.available ? Object.keys(roomVersions.available) : ['1']}
 | 
			
		||||
            value={selectedRoomVersion}
 | 
			
		||||
            onChange={handleRoomVersionChange}
 | 
			
		||||
            disabled={disabled}
 | 
			
		||||
          />
 | 
			
		||||
        )}
 | 
			
		||||
      </Box>
 | 
			
		||||
 | 
			
		||||
      {error && (
 | 
			
		||||
        <Box style={{ color: color.Critical.Main }} alignItems="Center" gap="200">
 | 
			
		||||
          <Icon src={Icons.Warning} filled size="100" />
 | 
			
		||||
          <Text size="T300" style={{ color: color.Critical.Main }}>
 | 
			
		||||
            <b>
 | 
			
		||||
              {error instanceof MatrixError && error.name === ErrorCode.M_LIMIT_EXCEEDED
 | 
			
		||||
                ? `Server rate-limited your request for ${millisecondsToMinutes(
 | 
			
		||||
                    (error.data.retry_after_ms as number | undefined) ?? 0
 | 
			
		||||
                  )} minutes!`
 | 
			
		||||
                : error.message}
 | 
			
		||||
            </b>
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Box>
 | 
			
		||||
      )}
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="200">
 | 
			
		||||
        <Button
 | 
			
		||||
          type="submit"
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="Primary"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
          before={loading && <Spinner variant="Primary" fill="Solid" size="200" />}
 | 
			
		||||
        >
 | 
			
		||||
          <Text size="B500">Create</Text>
 | 
			
		||||
        </Button>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										94
									
								
								src/app/features/create-room/CreateRoomModal.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								src/app/features/create-room/CreateRoomModal.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,94 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  config,
 | 
			
		||||
  Header,
 | 
			
		||||
  Icon,
 | 
			
		||||
  IconButton,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Modal,
 | 
			
		||||
  Overlay,
 | 
			
		||||
  OverlayBackdrop,
 | 
			
		||||
  OverlayCenter,
 | 
			
		||||
  Scroll,
 | 
			
		||||
  Text,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { useAllJoinedRoomsSet, useGetRoom } from '../../hooks/useGetRoom';
 | 
			
		||||
import { SpaceProvider } from '../../hooks/useSpace';
 | 
			
		||||
import { CreateRoomForm } from './CreateRoom';
 | 
			
		||||
import {
 | 
			
		||||
  useCloseCreateRoomModal,
 | 
			
		||||
  useCreateRoomModalState,
 | 
			
		||||
} from '../../state/hooks/createRoomModal';
 | 
			
		||||
import { CreateRoomModalState } from '../../state/createRoomModal';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
 | 
			
		||||
type CreateRoomModalProps = {
 | 
			
		||||
  state: CreateRoomModalState;
 | 
			
		||||
};
 | 
			
		||||
function CreateRoomModal({ state }: CreateRoomModalProps) {
 | 
			
		||||
  const { spaceId } = state;
 | 
			
		||||
  const closeDialog = useCloseCreateRoomModal();
 | 
			
		||||
 | 
			
		||||
  const allJoinedRooms = useAllJoinedRoomsSet();
 | 
			
		||||
  const getRoom = useGetRoom(allJoinedRooms);
 | 
			
		||||
  const space = spaceId ? getRoom(spaceId) : undefined;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SpaceProvider value={space ?? null}>
 | 
			
		||||
      <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
        <OverlayCenter>
 | 
			
		||||
          <FocusTrap
 | 
			
		||||
            focusTrapOptions={{
 | 
			
		||||
              initialFocus: false,
 | 
			
		||||
              clickOutsideDeactivates: true,
 | 
			
		||||
              onDeactivate: closeDialog,
 | 
			
		||||
              escapeDeactivates: stopPropagation,
 | 
			
		||||
            }}
 | 
			
		||||
          >
 | 
			
		||||
            <Modal size="300" flexHeight>
 | 
			
		||||
              <Box direction="Column">
 | 
			
		||||
                <Header
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  style={{
 | 
			
		||||
                    padding: config.space.S200,
 | 
			
		||||
                    paddingLeft: config.space.S400,
 | 
			
		||||
                  }}
 | 
			
		||||
                >
 | 
			
		||||
                  <Box grow="Yes">
 | 
			
		||||
                    <Text size="H4">New Room</Text>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                  <Box shrink="No">
 | 
			
		||||
                    <IconButton size="300" radii="300" onClick={closeDialog}>
 | 
			
		||||
                      <Icon src={Icons.Cross} />
 | 
			
		||||
                    </IconButton>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Header>
 | 
			
		||||
                <Scroll size="300" hideTrack>
 | 
			
		||||
                  <Box
 | 
			
		||||
                    style={{
 | 
			
		||||
                      padding: config.space.S400,
 | 
			
		||||
                      paddingRight: config.space.S200,
 | 
			
		||||
                    }}
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="500"
 | 
			
		||||
                  >
 | 
			
		||||
                    <CreateRoomForm space={space} onCreate={closeDialog} />
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Scroll>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Modal>
 | 
			
		||||
          </FocusTrap>
 | 
			
		||||
        </OverlayCenter>
 | 
			
		||||
      </Overlay>
 | 
			
		||||
    </SpaceProvider>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function CreateRoomModalRenderer() {
 | 
			
		||||
  const state = useCreateRoomModalState();
 | 
			
		||||
 | 
			
		||||
  if (!state) return null;
 | 
			
		||||
  return <CreateRoomModal state={state} />;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										2
									
								
								src/app/features/create-room/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/app/features/create-room/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,2 @@
 | 
			
		|||
export * from './CreateRoom';
 | 
			
		||||
export * from './CreateRoomModal';
 | 
			
		||||
							
								
								
									
										279
									
								
								src/app/features/create-space/CreateSpace.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										279
									
								
								src/app/features/create-space/CreateSpace.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,279 @@
 | 
			
		|||
import React, { FormEventHandler, useCallback, useEffect, useState } from 'react';
 | 
			
		||||
import { MatrixError, Room } from 'matrix-js-sdk';
 | 
			
		||||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  Button,
 | 
			
		||||
  Chip,
 | 
			
		||||
  color,
 | 
			
		||||
  config,
 | 
			
		||||
  Icon,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Input,
 | 
			
		||||
  Spinner,
 | 
			
		||||
  Switch,
 | 
			
		||||
  Text,
 | 
			
		||||
  TextArea,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { SettingTile } from '../../components/setting-tile';
 | 
			
		||||
import { SequenceCard } from '../../components/sequence-card';
 | 
			
		||||
import {
 | 
			
		||||
  creatorsSupported,
 | 
			
		||||
  knockRestrictedSupported,
 | 
			
		||||
  knockSupported,
 | 
			
		||||
  restrictedSupported,
 | 
			
		||||
} from '../../utils/matrix';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { millisecondsToMinutes, replaceSpaceWithDash } from '../../utils/common';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { useCapabilities } from '../../hooks/useCapabilities';
 | 
			
		||||
import { useAlive } from '../../hooks/useAlive';
 | 
			
		||||
import { ErrorCode } from '../../cs-errorcode';
 | 
			
		||||
import {
 | 
			
		||||
  AdditionalCreatorInput,
 | 
			
		||||
  createRoom,
 | 
			
		||||
  CreateRoomAliasInput,
 | 
			
		||||
  CreateRoomData,
 | 
			
		||||
  CreateRoomKind,
 | 
			
		||||
  CreateRoomKindSelector,
 | 
			
		||||
  RoomVersionSelector,
 | 
			
		||||
  useAdditionalCreators,
 | 
			
		||||
} from '../../components/create-room';
 | 
			
		||||
import { RoomType } from '../../../types/matrix/room';
 | 
			
		||||
 | 
			
		||||
const getCreateSpaceKindToIcon = (kind: CreateRoomKind) => {
 | 
			
		||||
  if (kind === CreateRoomKind.Private) return Icons.SpaceLock;
 | 
			
		||||
  if (kind === CreateRoomKind.Restricted) return Icons.Space;
 | 
			
		||||
  return Icons.SpaceGlobe;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
type CreateSpaceFormProps = {
 | 
			
		||||
  defaultKind?: CreateRoomKind;
 | 
			
		||||
  space?: Room;
 | 
			
		||||
  onCreate?: (roomId: string) => void;
 | 
			
		||||
};
 | 
			
		||||
export function CreateSpaceForm({ defaultKind, space, onCreate }: CreateSpaceFormProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const alive = useAlive();
 | 
			
		||||
 | 
			
		||||
  const capabilities = useCapabilities();
 | 
			
		||||
  const roomVersions = capabilities['m.room_versions'];
 | 
			
		||||
  const [selectedRoomVersion, selectRoomVersion] = useState(roomVersions?.default ?? '1');
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    // capabilities load async
 | 
			
		||||
    selectRoomVersion(roomVersions?.default ?? '1');
 | 
			
		||||
  }, [roomVersions?.default]);
 | 
			
		||||
 | 
			
		||||
  const allowRestricted = space && restrictedSupported(selectedRoomVersion);
 | 
			
		||||
 | 
			
		||||
  const [kind, setKind] = useState(
 | 
			
		||||
    defaultKind ?? allowRestricted ? CreateRoomKind.Restricted : CreateRoomKind.Private
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const allowAdditionalCreators = creatorsSupported(selectedRoomVersion);
 | 
			
		||||
  const { additionalCreators, addAdditionalCreator, removeAdditionalCreator } =
 | 
			
		||||
    useAdditionalCreators();
 | 
			
		||||
  const [federation, setFederation] = useState(true);
 | 
			
		||||
  const [knock, setKnock] = useState(false);
 | 
			
		||||
  const [advance, setAdvance] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const allowKnock = kind === CreateRoomKind.Private && knockSupported(selectedRoomVersion);
 | 
			
		||||
  const allowKnockRestricted =
 | 
			
		||||
    kind === CreateRoomKind.Restricted && knockRestrictedSupported(selectedRoomVersion);
 | 
			
		||||
 | 
			
		||||
  const handleRoomVersionChange = (version: string) => {
 | 
			
		||||
    if (!restrictedSupported(version)) {
 | 
			
		||||
      setKind(CreateRoomKind.Private);
 | 
			
		||||
    }
 | 
			
		||||
    selectRoomVersion(version);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const [createState, create] = useAsyncCallback<string, Error | MatrixError, [CreateRoomData]>(
 | 
			
		||||
    useCallback((data) => createRoom(mx, data), [mx])
 | 
			
		||||
  );
 | 
			
		||||
  const loading = createState.status === AsyncStatus.Loading;
 | 
			
		||||
  const error = createState.status === AsyncStatus.Error ? createState.error : undefined;
 | 
			
		||||
  const disabled = createState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const handleSubmit: FormEventHandler<HTMLFormElement> = (evt) => {
 | 
			
		||||
    evt.preventDefault();
 | 
			
		||||
    if (disabled) return;
 | 
			
		||||
    const form = evt.currentTarget;
 | 
			
		||||
 | 
			
		||||
    const nameInput = form.nameInput as HTMLInputElement | undefined;
 | 
			
		||||
    const topicTextArea = form.topicTextAria as HTMLTextAreaElement | undefined;
 | 
			
		||||
    const aliasInput = form.aliasInput as HTMLInputElement | undefined;
 | 
			
		||||
    const roomName = nameInput?.value.trim();
 | 
			
		||||
    const roomTopic = topicTextArea?.value.trim();
 | 
			
		||||
    const aliasLocalPart =
 | 
			
		||||
      aliasInput && aliasInput.value ? replaceSpaceWithDash(aliasInput.value) : undefined;
 | 
			
		||||
 | 
			
		||||
    if (!roomName) return;
 | 
			
		||||
    const publicRoom = kind === CreateRoomKind.Public;
 | 
			
		||||
    let roomKnock = false;
 | 
			
		||||
    if (allowKnock && kind === CreateRoomKind.Private) {
 | 
			
		||||
      roomKnock = knock;
 | 
			
		||||
    }
 | 
			
		||||
    if (allowKnockRestricted && kind === CreateRoomKind.Restricted) {
 | 
			
		||||
      roomKnock = knock;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    create({
 | 
			
		||||
      version: selectedRoomVersion,
 | 
			
		||||
      type: RoomType.Space,
 | 
			
		||||
      parent: space,
 | 
			
		||||
      kind,
 | 
			
		||||
      name: roomName,
 | 
			
		||||
      topic: roomTopic || undefined,
 | 
			
		||||
      aliasLocalPart: publicRoom ? aliasLocalPart : undefined,
 | 
			
		||||
      knock: roomKnock,
 | 
			
		||||
      allowFederation: federation,
 | 
			
		||||
      additionalCreators: allowAdditionalCreators ? additionalCreators : undefined,
 | 
			
		||||
    }).then((roomId) => {
 | 
			
		||||
      if (alive()) {
 | 
			
		||||
        onCreate?.(roomId);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box as="form" onSubmit={handleSubmit} grow="Yes" direction="Column" gap="500">
 | 
			
		||||
      <Box direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Access</Text>
 | 
			
		||||
        <CreateRoomKindSelector
 | 
			
		||||
          value={kind}
 | 
			
		||||
          onSelect={setKind}
 | 
			
		||||
          canRestrict={allowRestricted}
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
          getIcon={getCreateSpaceKindToIcon}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Name</Text>
 | 
			
		||||
        <Input
 | 
			
		||||
          required
 | 
			
		||||
          before={<Icon size="100" src={getCreateSpaceKindToIcon(kind)} />}
 | 
			
		||||
          name="nameInput"
 | 
			
		||||
          autoFocus
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          autoComplete="off"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Text size="L400">Topic (Optional)</Text>
 | 
			
		||||
        <TextArea
 | 
			
		||||
          name="topicTextAria"
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
        />
 | 
			
		||||
      </Box>
 | 
			
		||||
 | 
			
		||||
      {kind === CreateRoomKind.Public && <CreateRoomAliasInput disabled={disabled} />}
 | 
			
		||||
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="100">
 | 
			
		||||
        <Box gap="200" alignItems="End">
 | 
			
		||||
          <Text size="L400">Options</Text>
 | 
			
		||||
          <Box grow="Yes" justifyContent="End">
 | 
			
		||||
            <Chip
 | 
			
		||||
              radii="Pill"
 | 
			
		||||
              before={<Icon src={advance ? Icons.ChevronTop : Icons.ChevronBottom} size="50" />}
 | 
			
		||||
              onClick={() => setAdvance(!advance)}
 | 
			
		||||
              type="button"
 | 
			
		||||
            >
 | 
			
		||||
              <Text size="T200">Advance Options</Text>
 | 
			
		||||
            </Chip>
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
        {allowAdditionalCreators && (
 | 
			
		||||
          <SequenceCard
 | 
			
		||||
            style={{ padding: config.space.S300 }}
 | 
			
		||||
            variant="SurfaceVariant"
 | 
			
		||||
            direction="Column"
 | 
			
		||||
            gap="500"
 | 
			
		||||
          >
 | 
			
		||||
            <AdditionalCreatorInput
 | 
			
		||||
              additionalCreators={additionalCreators}
 | 
			
		||||
              onSelect={addAdditionalCreator}
 | 
			
		||||
              onRemove={removeAdditionalCreator}
 | 
			
		||||
            />
 | 
			
		||||
          </SequenceCard>
 | 
			
		||||
        )}
 | 
			
		||||
        {kind !== CreateRoomKind.Public && advance && (allowKnock || allowKnockRestricted) && (
 | 
			
		||||
          <SequenceCard
 | 
			
		||||
            style={{ padding: config.space.S300 }}
 | 
			
		||||
            variant="SurfaceVariant"
 | 
			
		||||
            direction="Column"
 | 
			
		||||
            gap="500"
 | 
			
		||||
          >
 | 
			
		||||
            <SettingTile
 | 
			
		||||
              title="Knock to Join"
 | 
			
		||||
              description="Anyone can send request to join this space."
 | 
			
		||||
              after={
 | 
			
		||||
                <Switch variant="Primary" value={knock} onChange={setKnock} disabled={disabled} />
 | 
			
		||||
              }
 | 
			
		||||
            />
 | 
			
		||||
          </SequenceCard>
 | 
			
		||||
        )}
 | 
			
		||||
 | 
			
		||||
        <SequenceCard
 | 
			
		||||
          style={{ padding: config.space.S300 }}
 | 
			
		||||
          variant="SurfaceVariant"
 | 
			
		||||
          direction="Column"
 | 
			
		||||
          gap="500"
 | 
			
		||||
        >
 | 
			
		||||
          <SettingTile
 | 
			
		||||
            title="Allow Federation"
 | 
			
		||||
            description="Users from other servers can join."
 | 
			
		||||
            after={
 | 
			
		||||
              <Switch
 | 
			
		||||
                variant="Primary"
 | 
			
		||||
                value={federation}
 | 
			
		||||
                onChange={setFederation}
 | 
			
		||||
                disabled={disabled}
 | 
			
		||||
              />
 | 
			
		||||
            }
 | 
			
		||||
          />
 | 
			
		||||
        </SequenceCard>
 | 
			
		||||
        {advance && (
 | 
			
		||||
          <RoomVersionSelector
 | 
			
		||||
            versions={roomVersions?.available ? Object.keys(roomVersions.available) : ['1']}
 | 
			
		||||
            value={selectedRoomVersion}
 | 
			
		||||
            onChange={handleRoomVersionChange}
 | 
			
		||||
            disabled={disabled}
 | 
			
		||||
          />
 | 
			
		||||
        )}
 | 
			
		||||
      </Box>
 | 
			
		||||
 | 
			
		||||
      {error && (
 | 
			
		||||
        <Box style={{ color: color.Critical.Main }} alignItems="Center" gap="200">
 | 
			
		||||
          <Icon src={Icons.Warning} filled size="100" />
 | 
			
		||||
          <Text size="T300" style={{ color: color.Critical.Main }}>
 | 
			
		||||
            <b>
 | 
			
		||||
              {error instanceof MatrixError && error.name === ErrorCode.M_LIMIT_EXCEEDED
 | 
			
		||||
                ? `Server rate-limited your request for ${millisecondsToMinutes(
 | 
			
		||||
                    (error.data.retry_after_ms as number | undefined) ?? 0
 | 
			
		||||
                  )} minutes!`
 | 
			
		||||
                : error.message}
 | 
			
		||||
            </b>
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Box>
 | 
			
		||||
      )}
 | 
			
		||||
      <Box shrink="No" direction="Column" gap="200">
 | 
			
		||||
        <Button
 | 
			
		||||
          type="submit"
 | 
			
		||||
          size="500"
 | 
			
		||||
          variant="Primary"
 | 
			
		||||
          radii="400"
 | 
			
		||||
          disabled={disabled}
 | 
			
		||||
          before={loading && <Spinner variant="Primary" fill="Solid" size="200" />}
 | 
			
		||||
        >
 | 
			
		||||
          <Text size="B500">Create</Text>
 | 
			
		||||
        </Button>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Box>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										95
									
								
								src/app/features/create-space/CreateSpaceModal.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/app/features/create-space/CreateSpaceModal.tsx
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,95 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import {
 | 
			
		||||
  Box,
 | 
			
		||||
  config,
 | 
			
		||||
  Header,
 | 
			
		||||
  Icon,
 | 
			
		||||
  IconButton,
 | 
			
		||||
  Icons,
 | 
			
		||||
  Modal,
 | 
			
		||||
  Overlay,
 | 
			
		||||
  OverlayBackdrop,
 | 
			
		||||
  OverlayCenter,
 | 
			
		||||
  Scroll,
 | 
			
		||||
  Text,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import FocusTrap from 'focus-trap-react';
 | 
			
		||||
import { useAllJoinedRoomsSet, useGetRoom } from '../../hooks/useGetRoom';
 | 
			
		||||
import { SpaceProvider } from '../../hooks/useSpace';
 | 
			
		||||
import { CreateSpaceForm } from './CreateSpace';
 | 
			
		||||
import {
 | 
			
		||||
  useCloseCreateSpaceModal,
 | 
			
		||||
  useCreateSpaceModalState,
 | 
			
		||||
} from '../../state/hooks/createSpaceModal';
 | 
			
		||||
import { CreateSpaceModalState } from '../../state/createSpaceModal';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
 | 
			
		||||
type CreateSpaceModalProps = {
 | 
			
		||||
  state: CreateSpaceModalState;
 | 
			
		||||
};
 | 
			
		||||
function CreateSpaceModal({ state }: CreateSpaceModalProps) {
 | 
			
		||||
  const { spaceId } = state;
 | 
			
		||||
  const closeDialog = useCloseCreateSpaceModal();
 | 
			
		||||
 | 
			
		||||
  const allJoinedRooms = useAllJoinedRoomsSet();
 | 
			
		||||
  const getRoom = useGetRoom(allJoinedRooms);
 | 
			
		||||
  const space = spaceId ? getRoom(spaceId) : undefined;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SpaceProvider value={space ?? null}>
 | 
			
		||||
      <Overlay open backdrop={<OverlayBackdrop />}>
 | 
			
		||||
        <OverlayCenter>
 | 
			
		||||
          <FocusTrap
 | 
			
		||||
            focusTrapOptions={{
 | 
			
		||||
              initialFocus: false,
 | 
			
		||||
              clickOutsideDeactivates: true,
 | 
			
		||||
              onDeactivate: closeDialog,
 | 
			
		||||
              escapeDeactivates: stopPropagation,
 | 
			
		||||
            }}
 | 
			
		||||
          >
 | 
			
		||||
            <Modal size="300" flexHeight>
 | 
			
		||||
              <Box direction="Column">
 | 
			
		||||
                <Header
 | 
			
		||||
                  size="500"
 | 
			
		||||
                  style={{
 | 
			
		||||
                    padding: config.space.S200,
 | 
			
		||||
                    paddingLeft: config.space.S400,
 | 
			
		||||
                    borderBottomWidth: config.borderWidth.B300,
 | 
			
		||||
                  }}
 | 
			
		||||
                >
 | 
			
		||||
                  <Box grow="Yes">
 | 
			
		||||
                    <Text size="H4">New Space</Text>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                  <Box shrink="No">
 | 
			
		||||
                    <IconButton size="300" radii="300" onClick={closeDialog}>
 | 
			
		||||
                      <Icon src={Icons.Cross} />
 | 
			
		||||
                    </IconButton>
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Header>
 | 
			
		||||
                <Scroll size="300" hideTrack>
 | 
			
		||||
                  <Box
 | 
			
		||||
                    style={{
 | 
			
		||||
                      padding: config.space.S400,
 | 
			
		||||
                      paddingRight: config.space.S200,
 | 
			
		||||
                    }}
 | 
			
		||||
                    direction="Column"
 | 
			
		||||
                    gap="500"
 | 
			
		||||
                  >
 | 
			
		||||
                    <CreateSpaceForm space={space} onCreate={closeDialog} />
 | 
			
		||||
                  </Box>
 | 
			
		||||
                </Scroll>
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Modal>
 | 
			
		||||
          </FocusTrap>
 | 
			
		||||
        </OverlayCenter>
 | 
			
		||||
      </Overlay>
 | 
			
		||||
    </SpaceProvider>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function CreateSpaceModalRenderer() {
 | 
			
		||||
  const state = useCreateSpaceModalState();
 | 
			
		||||
 | 
			
		||||
  if (!state) return null;
 | 
			
		||||
  return <CreateSpaceModal state={state} />;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										2
									
								
								src/app/features/create-space/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/app/features/create-space/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,2 @@
 | 
			
		|||
export * from './CreateSpace';
 | 
			
		||||
export * from './CreateSpaceModal';
 | 
			
		||||
| 
						 | 
				
			
			@ -27,6 +27,9 @@ import { stopPropagation } from '../../utils/keyboard';
 | 
			
		|||
import { useOpenRoomSettings } from '../../state/hooks/roomSettings';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { useOpenSpaceSettings } from '../../state/hooks/spaceSettings';
 | 
			
		||||
import { IPowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { getRoomCreatorsForRoomId } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { getRoomPermissionsAPI } from '../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type HierarchyItemWithParent = HierarchyItem & {
 | 
			
		||||
  parentId: string;
 | 
			
		||||
| 
						 | 
				
			
			@ -45,7 +48,7 @@ function SuggestMenuItem({
 | 
			
		|||
  const [toggleState, handleToggleSuggested] = useAsyncCallback(
 | 
			
		||||
    useCallback(() => {
 | 
			
		||||
      const newContent: MSpaceChildContent = { ...content, suggested: !content.suggested };
 | 
			
		||||
      return mx.sendStateEvent(parentId, StateEvent.SpaceChild, newContent, roomId);
 | 
			
		||||
      return mx.sendStateEvent(parentId, StateEvent.SpaceChild as any, newContent, roomId);
 | 
			
		||||
    }, [mx, parentId, roomId, content])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -82,7 +85,7 @@ function RemoveMenuItem({
 | 
			
		|||
 | 
			
		||||
  const [removeState, handleRemove] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      () => mx.sendStateEvent(parentId, StateEvent.SpaceChild, {}, roomId),
 | 
			
		||||
      () => mx.sendStateEvent(parentId, StateEvent.SpaceChild as any, {}, roomId),
 | 
			
		||||
      [mx, parentId, roomId]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
| 
						 | 
				
			
			@ -180,7 +183,7 @@ type HierarchyItemMenuProps = {
 | 
			
		|||
    parentId: string;
 | 
			
		||||
  };
 | 
			
		||||
  joined: boolean;
 | 
			
		||||
  canInvite: boolean;
 | 
			
		||||
  powerLevels?: IPowerLevels;
 | 
			
		||||
  canEditChild: boolean;
 | 
			
		||||
  pinned?: boolean;
 | 
			
		||||
  onTogglePin?: (roomId: string) => void;
 | 
			
		||||
| 
						 | 
				
			
			@ -188,13 +191,22 @@ type HierarchyItemMenuProps = {
 | 
			
		|||
export function HierarchyItemMenu({
 | 
			
		||||
  item,
 | 
			
		||||
  joined,
 | 
			
		||||
  canInvite,
 | 
			
		||||
  powerLevels,
 | 
			
		||||
  canEditChild,
 | 
			
		||||
  pinned,
 | 
			
		||||
  onTogglePin,
 | 
			
		||||
}: HierarchyItemMenuProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const [menuAnchor, setMenuAnchor] = useState<RectCords>();
 | 
			
		||||
 | 
			
		||||
  const canInvite = (): boolean => {
 | 
			
		||||
    if (!powerLevels) return false;
 | 
			
		||||
    const creators = getRoomCreatorsForRoomId(mx, item.roomId);
 | 
			
		||||
    const permissions = getRoomPermissionsAPI(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
    return permissions.action('invite', mx.getSafeUserId());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleOpenMenu: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setMenuAnchor(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
| 
						 | 
				
			
			@ -254,7 +266,7 @@ export function HierarchyItemMenu({
 | 
			
		|||
                    <InviteMenuItem
 | 
			
		||||
                      item={item}
 | 
			
		||||
                      requestClose={handleRequestClose}
 | 
			
		||||
                      disabled={!canInvite}
 | 
			
		||||
                      disabled={!canInvite()}
 | 
			
		||||
                    />
 | 
			
		||||
                    <SettingsMenuItem item={item} requestClose={handleRequestClose} />
 | 
			
		||||
                    <UseStateProvider initial={false}>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
import React, { MouseEventHandler, useCallback, useMemo, useRef, useState } from 'react';
 | 
			
		||||
import { Box, Icon, IconButton, Icons, Line, Scroll, config } from 'folds';
 | 
			
		||||
import { Box, Chip, Icon, IconButton, Icons, Line, Scroll, Spinner, Text, config } from 'folds';
 | 
			
		||||
import { useVirtualizer } from '@tanstack/react-virtual';
 | 
			
		||||
import { useAtom, useAtomValue } from 'jotai';
 | 
			
		||||
import { useNavigate } from 'react-router-dom';
 | 
			
		||||
| 
						 | 
				
			
			@ -27,7 +27,6 @@ import { useElementSizeObserver } from '../../hooks/useElementSizeObserver';
 | 
			
		|||
import {
 | 
			
		||||
  IPowerLevels,
 | 
			
		||||
  PowerLevelsContextProvider,
 | 
			
		||||
  powerLevelAPI,
 | 
			
		||||
  usePowerLevels,
 | 
			
		||||
  useRoomsPowerLevels,
 | 
			
		||||
} from '../../hooks/usePowerLevels';
 | 
			
		||||
| 
						 | 
				
			
			@ -36,7 +35,7 @@ import { makeLobbyCategoryId } from '../../state/closedLobbyCategories';
 | 
			
		|||
import { useCategoryHandler } from '../../hooks/useCategoryHandler';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { allRoomsAtom } from '../../state/room-list/roomList';
 | 
			
		||||
import { getCanonicalAliasOrRoomId } from '../../utils/matrix';
 | 
			
		||||
import { getCanonicalAliasOrRoomId, rateLimitedActions } from '../../utils/matrix';
 | 
			
		||||
import { getSpaceRoomPath } from '../../pages/pathUtils';
 | 
			
		||||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { CanDropCallback, useDnDMonitor } from './DnD';
 | 
			
		||||
| 
						 | 
				
			
			@ -53,6 +52,101 @@ import { roomToParentsAtom } from '../../state/room/roomToParents';
 | 
			
		|||
import { AccountDataEvent } from '../../../types/matrix/accountData';
 | 
			
		||||
import { useRoomMembers } from '../../hooks/useRoomMembers';
 | 
			
		||||
import { SpaceHierarchy } from './SpaceHierarchy';
 | 
			
		||||
import { useGetRoom } from '../../hooks/useGetRoom';
 | 
			
		||||
import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		||||
import { getRoomPermissionsAPI } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { getRoomCreatorsForRoomId } from '../../hooks/useRoomCreators';
 | 
			
		||||
 | 
			
		||||
const useCanDropLobbyItem = (
 | 
			
		||||
  space: Room,
 | 
			
		||||
  roomsPowerLevels: Map<string, IPowerLevels>,
 | 
			
		||||
  getRoom: (roomId: string) => Room | undefined
 | 
			
		||||
): CanDropCallback => {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
 | 
			
		||||
  const canDropSpace: CanDropCallback = useCallback(
 | 
			
		||||
    (item, container) => {
 | 
			
		||||
      if (!('space' in container.item)) {
 | 
			
		||||
        // can not drop around rooms.
 | 
			
		||||
        // space can only be drop around other spaces
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const containerSpaceId = space.roomId;
 | 
			
		||||
 | 
			
		||||
      const powerLevels = roomsPowerLevels.get(containerSpaceId) ?? {};
 | 
			
		||||
      const creators = getRoomCreatorsForRoomId(mx, containerSpaceId);
 | 
			
		||||
      const permissions = getRoomPermissionsAPI(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
      if (
 | 
			
		||||
        getRoom(containerSpaceId) === undefined ||
 | 
			
		||||
        !permissions.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())
 | 
			
		||||
      ) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return true;
 | 
			
		||||
    },
 | 
			
		||||
    [space, roomsPowerLevels, getRoom, mx]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canDropRoom: CanDropCallback = useCallback(
 | 
			
		||||
    (item, container) => {
 | 
			
		||||
      const containerSpaceId =
 | 
			
		||||
        'space' in container.item ? container.item.roomId : container.item.parentId;
 | 
			
		||||
 | 
			
		||||
      const draggingOutsideSpace = item.parentId !== containerSpaceId;
 | 
			
		||||
      const restrictedItem = mx.getRoom(item.roomId)?.getJoinRule() === JoinRule.Restricted;
 | 
			
		||||
 | 
			
		||||
      // check and do not allow restricted room to be dragged outside
 | 
			
		||||
      // current space if can't change `m.room.join_rules` `content.allow`
 | 
			
		||||
      if (draggingOutsideSpace && restrictedItem) {
 | 
			
		||||
        const itemPowerLevels = roomsPowerLevels.get(item.roomId) ?? {};
 | 
			
		||||
        const itemCreators = getRoomCreatorsForRoomId(mx, item.roomId);
 | 
			
		||||
        const itemPermissions = getRoomPermissionsAPI(itemCreators, itemPowerLevels);
 | 
			
		||||
 | 
			
		||||
        const canChangeJoinRuleAllow = itemPermissions.stateEvent(
 | 
			
		||||
          StateEvent.RoomJoinRules,
 | 
			
		||||
          mx.getSafeUserId()
 | 
			
		||||
        );
 | 
			
		||||
        if (!canChangeJoinRuleAllow) {
 | 
			
		||||
          return false;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const powerLevels = roomsPowerLevels.get(containerSpaceId) ?? {};
 | 
			
		||||
      const creators = getRoomCreatorsForRoomId(mx, containerSpaceId);
 | 
			
		||||
      const permissions = getRoomPermissionsAPI(creators, powerLevels);
 | 
			
		||||
      if (
 | 
			
		||||
        getRoom(containerSpaceId) === undefined ||
 | 
			
		||||
        !permissions.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())
 | 
			
		||||
      ) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      return true;
 | 
			
		||||
    },
 | 
			
		||||
    [mx, getRoom, roomsPowerLevels]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canDrop: CanDropCallback = useCallback(
 | 
			
		||||
    (item, container): boolean => {
 | 
			
		||||
      if (item.roomId === container.item.roomId || item.roomId === container.nextRoomId) {
 | 
			
		||||
        // can not drop before or after itself
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // if we are dragging a space
 | 
			
		||||
      if ('space' in item) {
 | 
			
		||||
        return canDropSpace(item, container);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return canDropRoom(item, container);
 | 
			
		||||
    },
 | 
			
		||||
    [canDropSpace, canDropRoom]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  return canDrop;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function Lobby() {
 | 
			
		||||
  const navigate = useNavigate();
 | 
			
		||||
| 
						 | 
				
			
			@ -92,25 +186,7 @@ export function Lobby() {
 | 
			
		|||
    useCallback((w, height) => setHeroSectionHeight(height), [])
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const getRoom = useCallback(
 | 
			
		||||
    (rId: string) => {
 | 
			
		||||
      if (allJoinedRooms.has(rId)) {
 | 
			
		||||
        return mx.getRoom(rId) ?? undefined;
 | 
			
		||||
      }
 | 
			
		||||
      return undefined;
 | 
			
		||||
    },
 | 
			
		||||
    [mx, allJoinedRooms]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canEditSpaceChild = useCallback(
 | 
			
		||||
    (powerLevels: IPowerLevels) =>
 | 
			
		||||
      powerLevelAPI.canSendStateEvent(
 | 
			
		||||
        powerLevels,
 | 
			
		||||
        StateEvent.SpaceChild,
 | 
			
		||||
        powerLevelAPI.getPowerLevel(powerLevels, mx.getUserId() ?? undefined)
 | 
			
		||||
      ),
 | 
			
		||||
    [mx]
 | 
			
		||||
  );
 | 
			
		||||
  const getRoom = useGetRoom(allJoinedRooms);
 | 
			
		||||
 | 
			
		||||
  const [draggingItem, setDraggingItem] = useState<HierarchyItem>();
 | 
			
		||||
  const hierarchy = useSpaceHierarchy(
 | 
			
		||||
| 
						 | 
				
			
			@ -139,191 +215,163 @@ export function Lobby() {
 | 
			
		|||
      () =>
 | 
			
		||||
        hierarchy
 | 
			
		||||
          .flatMap((i) => {
 | 
			
		||||
            const childRooms = Array.isArray(i.rooms)
 | 
			
		||||
              ? i.rooms.map((r) => mx.getRoom(r.roomId))
 | 
			
		||||
              : [];
 | 
			
		||||
            const childRooms = Array.isArray(i.rooms) ? i.rooms.map((r) => getRoom(r.roomId)) : [];
 | 
			
		||||
 | 
			
		||||
            return [mx.getRoom(i.space.roomId), ...childRooms];
 | 
			
		||||
            return [getRoom(i.space.roomId), ...childRooms];
 | 
			
		||||
          })
 | 
			
		||||
          .filter((r) => !!r) as Room[],
 | 
			
		||||
      [mx, hierarchy]
 | 
			
		||||
      [hierarchy, getRoom]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const canDrop: CanDropCallback = useCallback(
 | 
			
		||||
    (item, container): boolean => {
 | 
			
		||||
      const restrictedItem = mx.getRoom(item.roomId)?.getJoinRule() === JoinRule.Restricted;
 | 
			
		||||
      if (item.roomId === container.item.roomId || item.roomId === container.nextRoomId) {
 | 
			
		||||
        // can not drop before or after itself
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
  const canDrop: CanDropCallback = useCanDropLobbyItem(space, roomsPowerLevels, getRoom);
 | 
			
		||||
 | 
			
		||||
      if ('space' in item) {
 | 
			
		||||
        if (!('space' in container.item)) return false;
 | 
			
		||||
        const containerSpaceId = space.roomId;
 | 
			
		||||
  const [reorderSpaceState, reorderSpace] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (item: HierarchyItemSpace, containerItem: HierarchyItem) => {
 | 
			
		||||
        if (!item.parentId) return;
 | 
			
		||||
 | 
			
		||||
        if (
 | 
			
		||||
          getRoom(containerSpaceId) === undefined ||
 | 
			
		||||
          !canEditSpaceChild(roomsPowerLevels.get(containerSpaceId) ?? {})
 | 
			
		||||
        ) {
 | 
			
		||||
          return false;
 | 
			
		||||
        }
 | 
			
		||||
        const itemSpaces: HierarchyItemSpace[] = hierarchy
 | 
			
		||||
          .map((i) => i.space)
 | 
			
		||||
          .filter((i) => i.roomId !== item.roomId);
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
        const beforeIndex = itemSpaces.findIndex((i) => i.roomId === containerItem.roomId);
 | 
			
		||||
        const insertIndex = beforeIndex + 1;
 | 
			
		||||
 | 
			
		||||
      const containerSpaceId =
 | 
			
		||||
        'space' in container.item ? container.item.roomId : container.item.parentId;
 | 
			
		||||
        itemSpaces.splice(insertIndex, 0, {
 | 
			
		||||
          ...item,
 | 
			
		||||
          content: { ...item.content, order: undefined },
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
      const dropOutsideSpace = item.parentId !== containerSpaceId;
 | 
			
		||||
        const currentOrders = itemSpaces.map((i) => {
 | 
			
		||||
          if (typeof i.content.order === 'string' && lex.has(i.content.order)) {
 | 
			
		||||
            return i.content.order;
 | 
			
		||||
          }
 | 
			
		||||
          return undefined;
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
      if (dropOutsideSpace && restrictedItem) {
 | 
			
		||||
        // do not allow restricted room to drop outside
 | 
			
		||||
        // current space if can't change join rule allow
 | 
			
		||||
        const itemPowerLevel = roomsPowerLevels.get(item.roomId) ?? {};
 | 
			
		||||
        const userPLInItem = powerLevelAPI.getPowerLevel(
 | 
			
		||||
          itemPowerLevel,
 | 
			
		||||
          mx.getUserId() ?? undefined
 | 
			
		||||
        );
 | 
			
		||||
        const canChangeJoinRuleAllow = powerLevelAPI.canSendStateEvent(
 | 
			
		||||
          itemPowerLevel,
 | 
			
		||||
          StateEvent.RoomJoinRules,
 | 
			
		||||
          userPLInItem
 | 
			
		||||
        );
 | 
			
		||||
        if (!canChangeJoinRuleAllow) {
 | 
			
		||||
          return false;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
        const newOrders = orderKeys(lex, currentOrders);
 | 
			
		||||
 | 
			
		||||
      if (
 | 
			
		||||
        getRoom(containerSpaceId) === undefined ||
 | 
			
		||||
        !canEditSpaceChild(roomsPowerLevels.get(containerSpaceId) ?? {})
 | 
			
		||||
      ) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      return true;
 | 
			
		||||
    },
 | 
			
		||||
    [getRoom, space.roomId, roomsPowerLevels, canEditSpaceChild, mx]
 | 
			
		||||
  );
 | 
			
		||||
        const reorders = newOrders
 | 
			
		||||
          ?.map((orderKey, index) => ({
 | 
			
		||||
            item: itemSpaces[index],
 | 
			
		||||
            orderKey,
 | 
			
		||||
          }))
 | 
			
		||||
          .filter((reorder, index) => {
 | 
			
		||||
            if (!reorder.item.parentId) return false;
 | 
			
		||||
            const parentPL = roomsPowerLevels.get(reorder.item.parentId);
 | 
			
		||||
            if (!parentPL) return false;
 | 
			
		||||
 | 
			
		||||
  const reorderSpace = useCallback(
 | 
			
		||||
    (item: HierarchyItemSpace, containerItem: HierarchyItem) => {
 | 
			
		||||
      if (!item.parentId) return;
 | 
			
		||||
            const creators = getRoomCreatorsForRoomId(mx, reorder.item.parentId);
 | 
			
		||||
            const permissions = getRoomPermissionsAPI(creators, parentPL);
 | 
			
		||||
            const canEdit = permissions.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId());
 | 
			
		||||
            return canEdit && reorder.orderKey !== currentOrders[index];
 | 
			
		||||
          });
 | 
			
		||||
 | 
			
		||||
      const itemSpaces: HierarchyItemSpace[] = hierarchy
 | 
			
		||||
        .map((i) => i.space)
 | 
			
		||||
        .filter((i) => i.roomId !== item.roomId);
 | 
			
		||||
 | 
			
		||||
      const beforeIndex = itemSpaces.findIndex((i) => i.roomId === containerItem.roomId);
 | 
			
		||||
      const insertIndex = beforeIndex + 1;
 | 
			
		||||
 | 
			
		||||
      itemSpaces.splice(insertIndex, 0, {
 | 
			
		||||
        ...item,
 | 
			
		||||
        content: { ...item.content, order: undefined },
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const currentOrders = itemSpaces.map((i) => {
 | 
			
		||||
        if (typeof i.content.order === 'string' && lex.has(i.content.order)) {
 | 
			
		||||
          return i.content.order;
 | 
			
		||||
        }
 | 
			
		||||
        return undefined;
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const newOrders = orderKeys(lex, currentOrders);
 | 
			
		||||
 | 
			
		||||
      newOrders?.forEach((orderKey, index) => {
 | 
			
		||||
        const itm = itemSpaces[index];
 | 
			
		||||
        if (!itm || !itm.parentId) return;
 | 
			
		||||
        const parentPL = roomsPowerLevels.get(itm.parentId);
 | 
			
		||||
        const canEdit = parentPL && canEditSpaceChild(parentPL);
 | 
			
		||||
        if (canEdit && orderKey !== currentOrders[index]) {
 | 
			
		||||
          mx.sendStateEvent(
 | 
			
		||||
            itm.parentId,
 | 
			
		||||
            StateEvent.SpaceChild as any,
 | 
			
		||||
            { ...itm.content, order: orderKey },
 | 
			
		||||
            itm.roomId
 | 
			
		||||
          );
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    },
 | 
			
		||||
    [mx, hierarchy, lex, roomsPowerLevels, canEditSpaceChild]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const reorderRoom = useCallback(
 | 
			
		||||
    (item: HierarchyItem, containerItem: HierarchyItem): void => {
 | 
			
		||||
      const itemRoom = mx.getRoom(item.roomId);
 | 
			
		||||
      if (!item.parentId) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      const containerParentId: string =
 | 
			
		||||
        'space' in containerItem ? containerItem.roomId : containerItem.parentId;
 | 
			
		||||
      const itemContent = item.content;
 | 
			
		||||
 | 
			
		||||
      if (item.parentId !== containerParentId) {
 | 
			
		||||
        mx.sendStateEvent(item.parentId, StateEvent.SpaceChild as any, {}, item.roomId);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (
 | 
			
		||||
        itemRoom &&
 | 
			
		||||
        itemRoom.getJoinRule() === JoinRule.Restricted &&
 | 
			
		||||
        item.parentId !== containerParentId
 | 
			
		||||
      ) {
 | 
			
		||||
        // change join rule allow parameter when dragging
 | 
			
		||||
        // restricted room from one space to another
 | 
			
		||||
        const joinRuleContent = getStateEvent(
 | 
			
		||||
          itemRoom,
 | 
			
		||||
          StateEvent.RoomJoinRules
 | 
			
		||||
        )?.getContent<RoomJoinRulesEventContent>();
 | 
			
		||||
 | 
			
		||||
        if (joinRuleContent) {
 | 
			
		||||
          const allow =
 | 
			
		||||
            joinRuleContent.allow?.filter((allowRule) => allowRule.room_id !== item.parentId) ?? [];
 | 
			
		||||
          allow.push({ type: RestrictedAllowType.RoomMembership, room_id: containerParentId });
 | 
			
		||||
          mx.sendStateEvent(itemRoom.roomId, StateEvent.RoomJoinRules as any, {
 | 
			
		||||
            ...joinRuleContent,
 | 
			
		||||
            allow,
 | 
			
		||||
        if (reorders) {
 | 
			
		||||
          await rateLimitedActions(reorders, async (reorder) => {
 | 
			
		||||
            if (!reorder.item.parentId) return;
 | 
			
		||||
            await mx.sendStateEvent(
 | 
			
		||||
              reorder.item.parentId,
 | 
			
		||||
              StateEvent.SpaceChild as any,
 | 
			
		||||
              { ...reorder.item.content, order: reorder.orderKey },
 | 
			
		||||
              reorder.item.roomId
 | 
			
		||||
            );
 | 
			
		||||
          });
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const itemSpaces = Array.from(
 | 
			
		||||
        hierarchy?.find((i) => i.space.roomId === containerParentId)?.rooms ?? []
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      const beforeItem: HierarchyItem | undefined =
 | 
			
		||||
        'space' in containerItem ? undefined : containerItem;
 | 
			
		||||
      const beforeIndex = itemSpaces.findIndex((i) => i.roomId === beforeItem?.roomId);
 | 
			
		||||
      const insertIndex = beforeIndex + 1;
 | 
			
		||||
 | 
			
		||||
      itemSpaces.splice(insertIndex, 0, {
 | 
			
		||||
        ...item,
 | 
			
		||||
        parentId: containerParentId,
 | 
			
		||||
        content: { ...itemContent, order: undefined },
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const currentOrders = itemSpaces.map((i) => {
 | 
			
		||||
        if (typeof i.content.order === 'string' && lex.has(i.content.order)) {
 | 
			
		||||
          return i.content.order;
 | 
			
		||||
        }
 | 
			
		||||
        return undefined;
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const newOrders = orderKeys(lex, currentOrders);
 | 
			
		||||
 | 
			
		||||
      newOrders?.forEach((orderKey, index) => {
 | 
			
		||||
        const itm = itemSpaces[index];
 | 
			
		||||
        if (itm && orderKey !== currentOrders[index]) {
 | 
			
		||||
          mx.sendStateEvent(
 | 
			
		||||
            containerParentId,
 | 
			
		||||
            StateEvent.SpaceChild as any,
 | 
			
		||||
            { ...itm.content, order: orderKey },
 | 
			
		||||
            itm.roomId
 | 
			
		||||
          );
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    },
 | 
			
		||||
    [mx, hierarchy, lex]
 | 
			
		||||
      },
 | 
			
		||||
      [mx, hierarchy, lex, roomsPowerLevels]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
  const reorderingSpace = reorderSpaceState.status === AsyncStatus.Loading;
 | 
			
		||||
 | 
			
		||||
  const [reorderRoomState, reorderRoom] = useAsyncCallback(
 | 
			
		||||
    useCallback(
 | 
			
		||||
      async (item: HierarchyItem, containerItem: HierarchyItem) => {
 | 
			
		||||
        const itemRoom = mx.getRoom(item.roomId);
 | 
			
		||||
        if (!item.parentId) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        const containerParentId: string =
 | 
			
		||||
          'space' in containerItem ? containerItem.roomId : containerItem.parentId;
 | 
			
		||||
        const itemContent = item.content;
 | 
			
		||||
 | 
			
		||||
        // remove from current space
 | 
			
		||||
        if (item.parentId !== containerParentId) {
 | 
			
		||||
          mx.sendStateEvent(item.parentId, StateEvent.SpaceChild as any, {}, item.roomId);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (
 | 
			
		||||
          itemRoom &&
 | 
			
		||||
          itemRoom.getJoinRule() === JoinRule.Restricted &&
 | 
			
		||||
          item.parentId !== containerParentId
 | 
			
		||||
        ) {
 | 
			
		||||
          // change join rule allow parameter when dragging
 | 
			
		||||
          // restricted room from one space to another
 | 
			
		||||
          const joinRuleContent = getStateEvent(
 | 
			
		||||
            itemRoom,
 | 
			
		||||
            StateEvent.RoomJoinRules
 | 
			
		||||
          )?.getContent<RoomJoinRulesEventContent>();
 | 
			
		||||
 | 
			
		||||
          if (joinRuleContent) {
 | 
			
		||||
            const allow =
 | 
			
		||||
              joinRuleContent.allow?.filter((allowRule) => allowRule.room_id !== item.parentId) ??
 | 
			
		||||
              [];
 | 
			
		||||
            allow.push({ type: RestrictedAllowType.RoomMembership, room_id: containerParentId });
 | 
			
		||||
            mx.sendStateEvent(itemRoom.roomId, StateEvent.RoomJoinRules as any, {
 | 
			
		||||
              ...joinRuleContent,
 | 
			
		||||
              allow,
 | 
			
		||||
            });
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        const itemSpaces = Array.from(
 | 
			
		||||
          hierarchy?.find((i) => i.space.roomId === containerParentId)?.rooms ?? []
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        const beforeItem: HierarchyItem | undefined =
 | 
			
		||||
          'space' in containerItem ? undefined : containerItem;
 | 
			
		||||
        const beforeIndex = itemSpaces.findIndex((i) => i.roomId === beforeItem?.roomId);
 | 
			
		||||
        const insertIndex = beforeIndex + 1;
 | 
			
		||||
 | 
			
		||||
        itemSpaces.splice(insertIndex, 0, {
 | 
			
		||||
          ...item,
 | 
			
		||||
          parentId: containerParentId,
 | 
			
		||||
          content: { ...itemContent, order: undefined },
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        const currentOrders = itemSpaces.map((i) => {
 | 
			
		||||
          if (typeof i.content.order === 'string' && lex.has(i.content.order)) {
 | 
			
		||||
            return i.content.order;
 | 
			
		||||
          }
 | 
			
		||||
          return undefined;
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        const newOrders = orderKeys(lex, currentOrders);
 | 
			
		||||
 | 
			
		||||
        const reorders = newOrders
 | 
			
		||||
          ?.map((orderKey, index) => ({
 | 
			
		||||
            item: itemSpaces[index],
 | 
			
		||||
            orderKey,
 | 
			
		||||
          }))
 | 
			
		||||
          .filter((reorder, index) => reorder.item && reorder.orderKey !== currentOrders[index]);
 | 
			
		||||
 | 
			
		||||
        if (reorders) {
 | 
			
		||||
          await rateLimitedActions(reorders, async (reorder) => {
 | 
			
		||||
            await mx.sendStateEvent(
 | 
			
		||||
              containerParentId,
 | 
			
		||||
              StateEvent.SpaceChild as any,
 | 
			
		||||
              { ...reorder.item.content, order: reorder.orderKey },
 | 
			
		||||
              reorder.item.roomId
 | 
			
		||||
            );
 | 
			
		||||
          });
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      [mx, hierarchy, lex]
 | 
			
		||||
    )
 | 
			
		||||
  );
 | 
			
		||||
  const reorderingRoom = reorderRoomState.status === AsyncStatus.Loading;
 | 
			
		||||
  const reordering = reorderingRoom || reorderingSpace;
 | 
			
		||||
 | 
			
		||||
  useDnDMonitor(
 | 
			
		||||
    scrollRef,
 | 
			
		||||
| 
						 | 
				
			
			@ -374,7 +422,7 @@ export function Lobby() {
 | 
			
		|||
        newItems.push(rId);
 | 
			
		||||
      }
 | 
			
		||||
      const newSpacesContent = makeCinnySpacesContent(mx, newItems);
 | 
			
		||||
      mx.setAccountData(AccountDataEvent.CinnySpaces, newSpacesContent);
 | 
			
		||||
      mx.setAccountData(AccountDataEvent.CinnySpaces as any, newSpacesContent as any);
 | 
			
		||||
    },
 | 
			
		||||
    [mx, sidebarItems, sidebarSpaces]
 | 
			
		||||
  );
 | 
			
		||||
| 
						 | 
				
			
			@ -439,7 +487,6 @@ export function Lobby() {
 | 
			
		|||
                            allJoinedRooms={allJoinedRooms}
 | 
			
		||||
                            mDirects={mDirects}
 | 
			
		||||
                            roomsPowerLevels={roomsPowerLevels}
 | 
			
		||||
                            canEditSpaceChild={canEditSpaceChild}
 | 
			
		||||
                            categoryId={categoryId}
 | 
			
		||||
                            closed={
 | 
			
		||||
                              closedCategories.has(categoryId) ||
 | 
			
		||||
| 
						 | 
				
			
			@ -449,6 +496,7 @@ export function Lobby() {
 | 
			
		|||
                            draggingItem={draggingItem}
 | 
			
		||||
                            onDragging={setDraggingItem}
 | 
			
		||||
                            canDrop={canDrop}
 | 
			
		||||
                            disabledReorder={reordering}
 | 
			
		||||
                            nextSpaceId={nextSpaceId}
 | 
			
		||||
                            getRoom={getRoom}
 | 
			
		||||
                            pinned={sidebarSpaces.has(item.space.roomId)}
 | 
			
		||||
| 
						 | 
				
			
			@ -460,6 +508,28 @@ export function Lobby() {
 | 
			
		|||
                      );
 | 
			
		||||
                    })}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  {reordering && (
 | 
			
		||||
                    <Box
 | 
			
		||||
                      style={{
 | 
			
		||||
                        position: 'absolute',
 | 
			
		||||
                        bottom: config.space.S400,
 | 
			
		||||
                        left: 0,
 | 
			
		||||
                        right: 0,
 | 
			
		||||
                        zIndex: 2,
 | 
			
		||||
                        pointerEvents: 'none',
 | 
			
		||||
                      }}
 | 
			
		||||
                      justifyContent="Center"
 | 
			
		||||
                    >
 | 
			
		||||
                      <Chip
 | 
			
		||||
                        variant="Secondary"
 | 
			
		||||
                        outlined
 | 
			
		||||
                        radii="Pill"
 | 
			
		||||
                        before={<Spinner variant="Secondary" fill="Soft" size="100" />}
 | 
			
		||||
                      >
 | 
			
		||||
                        <Text size="L400">Reordering</Text>
 | 
			
		||||
                      </Chip>
 | 
			
		||||
                    </Box>
 | 
			
		||||
                  )}
 | 
			
		||||
                </PageContentCenter>
 | 
			
		||||
              </PageContent>
 | 
			
		||||
            </Scroll>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -27,7 +27,7 @@ import { RoomAvatar } from '../../components/room-avatar';
 | 
			
		|||
import { nameInitials } from '../../utils/common';
 | 
			
		||||
import * as css from './LobbyHeader.css';
 | 
			
		||||
import { openInviteUser } from '../../../client/action/navigation';
 | 
			
		||||
import { IPowerLevels, usePowerLevelsAPI } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { IPowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { UseStateProvider } from '../../components/UseStateProvider';
 | 
			
		||||
import { LeaveSpacePrompt } from '../../components/leave-space-prompt';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
| 
						 | 
				
			
			@ -36,26 +36,30 @@ import { BackRouteHandler } from '../../components/BackRouteHandler';
 | 
			
		|||
import { mxcUrlToHttp } from '../../utils/matrix';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useOpenSpaceSettings } from '../../state/hooks/spaceSettings';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type LobbyMenuProps = {
 | 
			
		||||
  roomId: string;
 | 
			
		||||
  powerLevels: IPowerLevels;
 | 
			
		||||
  requestClose: () => void;
 | 
			
		||||
};
 | 
			
		||||
const LobbyMenu = forwardRef<HTMLDivElement, LobbyMenuProps>(
 | 
			
		||||
  ({ roomId, powerLevels, requestClose }, ref) => {
 | 
			
		||||
  ({ powerLevels, requestClose }, ref) => {
 | 
			
		||||
    const mx = useMatrixClient();
 | 
			
		||||
    const { getPowerLevel, canDoAction } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
    const canInvite = canDoAction('invite', getPowerLevel(mx.getUserId() ?? ''));
 | 
			
		||||
    const space = useSpace();
 | 
			
		||||
    const creators = useRoomCreators(space);
 | 
			
		||||
 | 
			
		||||
    const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
    const canInvite = permissions.action('invite', mx.getSafeUserId());
 | 
			
		||||
    const openSpaceSettings = useOpenSpaceSettings();
 | 
			
		||||
 | 
			
		||||
    const handleInvite = () => {
 | 
			
		||||
      openInviteUser(roomId);
 | 
			
		||||
      openInviteUser(space.roomId);
 | 
			
		||||
      requestClose();
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const handleRoomSettings = () => {
 | 
			
		||||
      openSpaceSettings(roomId);
 | 
			
		||||
      openSpaceSettings(space.roomId);
 | 
			
		||||
      requestClose();
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -106,7 +110,7 @@ const LobbyMenu = forwardRef<HTMLDivElement, LobbyMenuProps>(
 | 
			
		|||
                </MenuItem>
 | 
			
		||||
                {promptLeave && (
 | 
			
		||||
                  <LeaveSpacePrompt
 | 
			
		||||
                    roomId={roomId}
 | 
			
		||||
                    roomId={space.roomId}
 | 
			
		||||
                    onDone={requestClose}
 | 
			
		||||
                    onCancel={() => setPromptLeave(false)}
 | 
			
		||||
                  />
 | 
			
		||||
| 
						 | 
				
			
			@ -242,7 +246,6 @@ export function LobbyHeader({ showProfile, powerLevels }: LobbyHeaderProps) {
 | 
			
		|||
                }}
 | 
			
		||||
              >
 | 
			
		||||
                <LobbyMenu
 | 
			
		||||
                  roomId={space.roomId}
 | 
			
		||||
                  powerLevels={powerLevels}
 | 
			
		||||
                  requestClose={() => setMenuAnchor(undefined)}
 | 
			
		||||
                />
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -8,14 +8,16 @@ import {
 | 
			
		|||
  HierarchyItemSpace,
 | 
			
		||||
  useFetchSpaceHierarchyLevel,
 | 
			
		||||
} from '../../hooks/useSpaceHierarchy';
 | 
			
		||||
import { IPowerLevels, powerLevelAPI } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { IPowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { SpaceItemCard } from './SpaceItem';
 | 
			
		||||
import { AfterItemDropTarget, CanDropCallback } from './DnD';
 | 
			
		||||
import { HierarchyItemMenu } from './HierarchyItemMenu';
 | 
			
		||||
import { RoomItemCard } from './RoomItem';
 | 
			
		||||
import { RoomType } from '../../../types/matrix/room';
 | 
			
		||||
import { RoomType, StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { SequenceCard } from '../../components/sequence-card';
 | 
			
		||||
import { getRoomCreatorsForRoomId } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { getRoomPermissionsAPI } from '../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type SpaceHierarchyProps = {
 | 
			
		||||
  summary: IHierarchyRoom | undefined;
 | 
			
		||||
| 
						 | 
				
			
			@ -24,13 +26,13 @@ type SpaceHierarchyProps = {
 | 
			
		|||
  allJoinedRooms: Set<string>;
 | 
			
		||||
  mDirects: Set<string>;
 | 
			
		||||
  roomsPowerLevels: Map<string, IPowerLevels>;
 | 
			
		||||
  canEditSpaceChild: (powerLevels: IPowerLevels) => boolean;
 | 
			
		||||
  categoryId: string;
 | 
			
		||||
  closed: boolean;
 | 
			
		||||
  handleClose: MouseEventHandler<HTMLButtonElement>;
 | 
			
		||||
  draggingItem?: HierarchyItem;
 | 
			
		||||
  onDragging: (item?: HierarchyItem) => void;
 | 
			
		||||
  canDrop: CanDropCallback;
 | 
			
		||||
  disabledReorder?: boolean;
 | 
			
		||||
  nextSpaceId?: string;
 | 
			
		||||
  getRoom: (roomId: string) => Room | undefined;
 | 
			
		||||
  pinned: boolean;
 | 
			
		||||
| 
						 | 
				
			
			@ -47,13 +49,13 @@ export const SpaceHierarchy = forwardRef<HTMLDivElement, SpaceHierarchyProps>(
 | 
			
		|||
      allJoinedRooms,
 | 
			
		||||
      mDirects,
 | 
			
		||||
      roomsPowerLevels,
 | 
			
		||||
      canEditSpaceChild,
 | 
			
		||||
      categoryId,
 | 
			
		||||
      closed,
 | 
			
		||||
      handleClose,
 | 
			
		||||
      draggingItem,
 | 
			
		||||
      onDragging,
 | 
			
		||||
      canDrop,
 | 
			
		||||
      disabledReorder,
 | 
			
		||||
      nextSpaceId,
 | 
			
		||||
      getRoom,
 | 
			
		||||
      pinned,
 | 
			
		||||
| 
						 | 
				
			
			@ -77,25 +79,28 @@ export const SpaceHierarchy = forwardRef<HTMLDivElement, SpaceHierarchyProps>(
 | 
			
		|||
      return s;
 | 
			
		||||
    }, [rooms]);
 | 
			
		||||
 | 
			
		||||
    const spacePowerLevels = roomsPowerLevels.get(spaceItem.roomId) ?? {};
 | 
			
		||||
    const userPLInSpace = powerLevelAPI.getPowerLevel(
 | 
			
		||||
      spacePowerLevels,
 | 
			
		||||
      mx.getUserId() ?? undefined
 | 
			
		||||
    );
 | 
			
		||||
    const canInviteInSpace = powerLevelAPI.canDoAction(spacePowerLevels, 'invite', userPLInSpace);
 | 
			
		||||
    const spacePowerLevels = roomsPowerLevels.get(spaceItem.roomId);
 | 
			
		||||
    const spaceCreators = getRoomCreatorsForRoomId(mx, spaceItem.roomId);
 | 
			
		||||
    const spacePermissions =
 | 
			
		||||
      spacePowerLevels && getRoomPermissionsAPI(spaceCreators, spacePowerLevels);
 | 
			
		||||
 | 
			
		||||
    const draggingSpace =
 | 
			
		||||
      draggingItem?.roomId === spaceItem.roomId && draggingItem.parentId === spaceItem.parentId;
 | 
			
		||||
 | 
			
		||||
    const { parentId } = spaceItem;
 | 
			
		||||
    const parentPowerLevels = parentId ? roomsPowerLevels.get(parentId) ?? {} : undefined;
 | 
			
		||||
    const parentPowerLevels = parentId ? roomsPowerLevels.get(parentId) : undefined;
 | 
			
		||||
    const parentCreators = parentId ? getRoomCreatorsForRoomId(mx, parentId) : undefined;
 | 
			
		||||
    const parentPermissions =
 | 
			
		||||
      parentCreators &&
 | 
			
		||||
      parentPowerLevels &&
 | 
			
		||||
      getRoomPermissionsAPI(parentCreators, parentPowerLevels);
 | 
			
		||||
 | 
			
		||||
    useEffect(() => {
 | 
			
		||||
      onSpacesFound(Array.from(subspaces.values()));
 | 
			
		||||
    }, [subspaces, onSpacesFound]);
 | 
			
		||||
 | 
			
		||||
    let childItems = roomItems?.filter((i) => !subspaces.has(i.roomId));
 | 
			
		||||
    if (!canEditSpaceChild(spacePowerLevels)) {
 | 
			
		||||
    if (!spacePermissions?.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())) {
 | 
			
		||||
      // hide unknown rooms for normal user
 | 
			
		||||
      childItems = childItems?.filter((i) => {
 | 
			
		||||
        const forbidden = error instanceof MatrixError ? error.errcode === 'M_FORBIDDEN' : false;
 | 
			
		||||
| 
						 | 
				
			
			@ -115,16 +120,22 @@ export const SpaceHierarchy = forwardRef<HTMLDivElement, SpaceHierarchyProps>(
 | 
			
		|||
          closed={closed}
 | 
			
		||||
          handleClose={handleClose}
 | 
			
		||||
          getRoom={getRoom}
 | 
			
		||||
          canEditChild={canEditSpaceChild(spacePowerLevels)}
 | 
			
		||||
          canReorder={parentPowerLevels ? canEditSpaceChild(parentPowerLevels) : false}
 | 
			
		||||
          canEditChild={!!spacePermissions?.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())}
 | 
			
		||||
          canReorder={
 | 
			
		||||
            parentPowerLevels && !disabledReorder && parentPermissions
 | 
			
		||||
              ? parentPermissions.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())
 | 
			
		||||
              : false
 | 
			
		||||
          }
 | 
			
		||||
          options={
 | 
			
		||||
            parentId &&
 | 
			
		||||
            parentPowerLevels && (
 | 
			
		||||
              <HierarchyItemMenu
 | 
			
		||||
                item={{ ...spaceItem, parentId }}
 | 
			
		||||
                canInvite={canInviteInSpace}
 | 
			
		||||
                powerLevels={spacePowerLevels}
 | 
			
		||||
                joined={allJoinedRooms.has(spaceItem.roomId)}
 | 
			
		||||
                canEditChild={canEditSpaceChild(parentPowerLevels)}
 | 
			
		||||
                canEditChild={
 | 
			
		||||
                  !!parentPermissions?.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())
 | 
			
		||||
                }
 | 
			
		||||
                pinned={pinned}
 | 
			
		||||
                onTogglePin={togglePinToSidebar}
 | 
			
		||||
              />
 | 
			
		||||
| 
						 | 
				
			
			@ -147,15 +158,6 @@ export const SpaceHierarchy = forwardRef<HTMLDivElement, SpaceHierarchyProps>(
 | 
			
		|||
              const roomSummary = rooms.get(roomItem.roomId);
 | 
			
		||||
 | 
			
		||||
              const roomPowerLevels = roomsPowerLevels.get(roomItem.roomId) ?? {};
 | 
			
		||||
              const userPLInRoom = powerLevelAPI.getPowerLevel(
 | 
			
		||||
                roomPowerLevels,
 | 
			
		||||
                mx.getUserId() ?? undefined
 | 
			
		||||
              );
 | 
			
		||||
              const canInviteInRoom = powerLevelAPI.canDoAction(
 | 
			
		||||
                roomPowerLevels,
 | 
			
		||||
                'invite',
 | 
			
		||||
                userPLInRoom
 | 
			
		||||
              );
 | 
			
		||||
 | 
			
		||||
              const lastItem = index === childItems.length;
 | 
			
		||||
              const nextRoomId = lastItem ? nextSpaceId : childItems[index + 1]?.roomId;
 | 
			
		||||
| 
						 | 
				
			
			@ -174,13 +176,18 @@ export const SpaceHierarchy = forwardRef<HTMLDivElement, SpaceHierarchyProps>(
 | 
			
		|||
                  dm={mDirects.has(roomItem.roomId)}
 | 
			
		||||
                  onOpen={onOpenRoom}
 | 
			
		||||
                  getRoom={getRoom}
 | 
			
		||||
                  canReorder={canEditSpaceChild(spacePowerLevels)}
 | 
			
		||||
                  canReorder={
 | 
			
		||||
                    !!spacePermissions?.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId()) &&
 | 
			
		||||
                    !disabledReorder
 | 
			
		||||
                  }
 | 
			
		||||
                  options={
 | 
			
		||||
                    <HierarchyItemMenu
 | 
			
		||||
                      item={roomItem}
 | 
			
		||||
                      canInvite={canInviteInRoom}
 | 
			
		||||
                      powerLevels={roomPowerLevels}
 | 
			
		||||
                      joined={allJoinedRooms.has(roomItem.roomId)}
 | 
			
		||||
                      canEditChild={canEditSpaceChild(spacePowerLevels)}
 | 
			
		||||
                      canEditChild={
 | 
			
		||||
                        !!spacePermissions?.stateEvent(StateEvent.SpaceChild, mx.getSafeUserId())
 | 
			
		||||
                      }
 | 
			
		||||
                    />
 | 
			
		||||
                  }
 | 
			
		||||
                  after={
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,10 +30,12 @@ import { AsyncStatus, useAsyncCallback } from '../../hooks/useAsyncCallback';
 | 
			
		|||
import * as css from './SpaceItem.css';
 | 
			
		||||
import * as styleCss from './style.css';
 | 
			
		||||
import { useDraggableItem } from './DnD';
 | 
			
		||||
import { openCreateRoom, openSpaceAddExisting } from '../../../client/action/navigation';
 | 
			
		||||
import { stopPropagation } from '../../utils/keyboard';
 | 
			
		||||
import { mxcUrlToHttp } from '../../utils/matrix';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useOpenCreateRoomModal } from '../../state/hooks/createRoomModal';
 | 
			
		||||
import { useOpenCreateSpaceModal } from '../../state/hooks/createSpaceModal';
 | 
			
		||||
import { AddExistingModal } from '../add-existing';
 | 
			
		||||
 | 
			
		||||
function SpaceProfileLoading() {
 | 
			
		||||
  return (
 | 
			
		||||
| 
						 | 
				
			
			@ -240,18 +242,20 @@ function RootSpaceProfile({ closed, categoryId, handleClose }: RootSpaceProfileP
 | 
			
		|||
 | 
			
		||||
function AddRoomButton({ item }: { item: HierarchyItem }) {
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
  const openCreateRoomModal = useOpenCreateRoomModal();
 | 
			
		||||
  const [addExisting, setAddExisting] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const handleAddRoom: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleCreateRoom = () => {
 | 
			
		||||
    openCreateRoom(false, item.roomId as any);
 | 
			
		||||
    openCreateRoomModal(item.roomId);
 | 
			
		||||
    setCords(undefined);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleAddExisting = () => {
 | 
			
		||||
    openSpaceAddExisting(item.roomId);
 | 
			
		||||
    setAddExisting(true);
 | 
			
		||||
    setCords(undefined);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -297,24 +301,29 @@ function AddRoomButton({ item }: { item: HierarchyItem }) {
 | 
			
		|||
      >
 | 
			
		||||
        <Text size="B300">Add Room</Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
      {addExisting && (
 | 
			
		||||
        <AddExistingModal parentId={item.roomId} requestClose={() => setAddExisting(false)} />
 | 
			
		||||
      )}
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AddSpaceButton({ item }: { item: HierarchyItem }) {
 | 
			
		||||
  const [cords, setCords] = useState<RectCords>();
 | 
			
		||||
  const openCreateSpaceModal = useOpenCreateSpaceModal();
 | 
			
		||||
  const [addExisting, setAddExisting] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const handleAddSpace: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    setCords(evt.currentTarget.getBoundingClientRect());
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleCreateSpace = () => {
 | 
			
		||||
    openCreateRoom(true, item.roomId as any);
 | 
			
		||||
    openCreateSpaceModal(item.roomId as any);
 | 
			
		||||
    setCords(undefined);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const handleAddExisting = () => {
 | 
			
		||||
    openSpaceAddExisting(item.roomId, true);
 | 
			
		||||
    setAddExisting(true);
 | 
			
		||||
    setCords(undefined);
 | 
			
		||||
  };
 | 
			
		||||
  return (
 | 
			
		||||
| 
						 | 
				
			
			@ -359,6 +368,9 @@ function AddSpaceButton({ item }: { item: HierarchyItem }) {
 | 
			
		|||
      >
 | 
			
		||||
        <Text size="B300">Add Space</Text>
 | 
			
		||||
      </Chip>
 | 
			
		||||
      {addExisting && (
 | 
			
		||||
        <AddExistingModal space parentId={item.roomId} requestClose={() => setAddExisting(false)} />
 | 
			
		||||
      )}
 | 
			
		||||
    </PopOut>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -470,7 +482,7 @@ export const SpaceItemCard = as<'div', SpaceItemCardProps>(
 | 
			
		|||
              </>
 | 
			
		||||
            )}
 | 
			
		||||
          </Box>
 | 
			
		||||
          {canEditChild && (
 | 
			
		||||
          {space && canEditChild && (
 | 
			
		||||
            <Box shrink="No" alignItems="Inherit" gap="200">
 | 
			
		||||
              <AddRoomButton item={item} />
 | 
			
		||||
              {item.parentId === undefined && <AddSpaceButton item={item} />}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,7 +5,7 @@ import { useVirtualizer } from '@tanstack/react-virtual';
 | 
			
		|||
import { useInfiniteQuery } from '@tanstack/react-query';
 | 
			
		||||
import { useSearchParams } from 'react-router-dom';
 | 
			
		||||
import { SearchOrderBy } from 'matrix-js-sdk';
 | 
			
		||||
import { PageHero, PageHeroSection } from '../../components/page';
 | 
			
		||||
import { PageHero, PageHeroEmpty, PageHeroSection } from '../../components/page';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { _SearchPathSearchParams } from '../../pages/paths';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
| 
						 | 
				
			
			@ -57,6 +57,9 @@ export function MessageSearch({
 | 
			
		|||
  const [urlPreview] = useSetting(settingsAtom, 'urlPreview');
 | 
			
		||||
  const [legacyUsernameColor] = useSetting(settingsAtom, 'legacyUsernameColor');
 | 
			
		||||
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
  const [dateFormatString] = useSetting(settingsAtom, 'dateFormatString');
 | 
			
		||||
 | 
			
		||||
  const searchInputRef = useRef<HTMLInputElement>(null);
 | 
			
		||||
  const scrollTopAnchorRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const [searchParams, setSearchParams] = useSearchParams();
 | 
			
		||||
| 
						 | 
				
			
			@ -222,18 +225,7 @@ export function MessageSearch({
 | 
			
		|||
      </Box>
 | 
			
		||||
 | 
			
		||||
      {!msgSearchParams.term && status === 'pending' && (
 | 
			
		||||
        <Box
 | 
			
		||||
          className={ContainerColor({ variant: 'SurfaceVariant' })}
 | 
			
		||||
          style={{
 | 
			
		||||
            padding: config.space.S400,
 | 
			
		||||
            borderRadius: config.radii.R400,
 | 
			
		||||
            minHeight: toRem(450),
 | 
			
		||||
          }}
 | 
			
		||||
          direction="Column"
 | 
			
		||||
          alignItems="Center"
 | 
			
		||||
          justifyContent="Center"
 | 
			
		||||
          gap="200"
 | 
			
		||||
        >
 | 
			
		||||
        <PageHeroEmpty>
 | 
			
		||||
          <PageHeroSection>
 | 
			
		||||
            <PageHero
 | 
			
		||||
              icon={<Icon size="600" src={Icons.Message} />}
 | 
			
		||||
| 
						 | 
				
			
			@ -241,7 +233,7 @@ export function MessageSearch({
 | 
			
		|||
              subTitle="Find helpful messages in your community by searching with related keywords."
 | 
			
		||||
            />
 | 
			
		||||
          </PageHeroSection>
 | 
			
		||||
        </Box>
 | 
			
		||||
        </PageHeroEmpty>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      {msgSearchParams.term && groups.length === 0 && status === 'success' && (
 | 
			
		||||
| 
						 | 
				
			
			@ -300,6 +292,8 @@ export function MessageSearch({
 | 
			
		|||
                    urlPreview={urlPreview}
 | 
			
		||||
                    onOpen={navigateRoom}
 | 
			
		||||
                    legacyUsernameColor={legacyUsernameColor || mDirects.has(groupRoom.roomId)}
 | 
			
		||||
                    hour24Clock={hour24Clock}
 | 
			
		||||
                    dateFormatString={dateFormatString}
 | 
			
		||||
                  />
 | 
			
		||||
                </VirtualTile>
 | 
			
		||||
              );
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,6 +29,7 @@ export function SearchInput({ active, loading, searchInputRef, onSearch, onReset
 | 
			
		|||
        ref={searchInputRef}
 | 
			
		||||
        style={{ paddingRight: config.space.S300 }}
 | 
			
		||||
        name="searchInput"
 | 
			
		||||
        autoFocus
 | 
			
		||||
        size="500"
 | 
			
		||||
        variant="Background"
 | 
			
		||||
        placeholder="Search for keyword"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -39,15 +39,18 @@ import { UserAvatar } from '../../components/user-avatar';
 | 
			
		|||
import { useMentionClickHandler } from '../../hooks/useMentionClickHandler';
 | 
			
		||||
import { useSpoilerClickHandler } from '../../hooks/useSpoilerClickHandler';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../hooks/usePowerLevels';
 | 
			
		||||
import {
 | 
			
		||||
  getTagIconSrc,
 | 
			
		||||
  useAccessibleTagColors,
 | 
			
		||||
  usePowerLevelTags,
 | 
			
		||||
} from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { usePowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevelTags } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { useTheme } from '../../hooks/useTheme';
 | 
			
		||||
import { PowerIcon } from '../../components/power';
 | 
			
		||||
import colorMXID from '../../../util/colorMXID';
 | 
			
		||||
import {
 | 
			
		||||
  getPowerTagIconSrc,
 | 
			
		||||
  useAccessiblePowerTagColors,
 | 
			
		||||
  useGetMemberPowerTag,
 | 
			
		||||
} from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomCreatorsTag } from '../../hooks/useRoomCreatorsTag';
 | 
			
		||||
 | 
			
		||||
type SearchResultGroupProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -57,6 +60,8 @@ type SearchResultGroupProps = {
 | 
			
		|||
  urlPreview?: boolean;
 | 
			
		||||
  onOpen: (roomId: string, eventId: string) => void;
 | 
			
		||||
  legacyUsernameColor?: boolean;
 | 
			
		||||
  hour24Clock: boolean;
 | 
			
		||||
  dateFormatString: string;
 | 
			
		||||
};
 | 
			
		||||
export function SearchResultGroup({
 | 
			
		||||
  room,
 | 
			
		||||
| 
						 | 
				
			
			@ -66,16 +71,22 @@ export function SearchResultGroup({
 | 
			
		|||
  urlPreview,
 | 
			
		||||
  onOpen,
 | 
			
		||||
  legacyUsernameColor,
 | 
			
		||||
  hour24Clock,
 | 
			
		||||
  dateFormatString,
 | 
			
		||||
}: SearchResultGroupProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const highlightRegex = useMemo(() => makeHighlightRegex(highlights), [highlights]);
 | 
			
		||||
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const { getPowerLevel } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const [powerLevelTags, getPowerLevelTag] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const creatorsTag = useRoomCreatorsTag();
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const getMemberPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const theme = useTheme();
 | 
			
		||||
  const accessibleTagColors = useAccessibleTagColors(theme.kind, powerLevelTags);
 | 
			
		||||
  const accessibleTagColors = useAccessiblePowerTagColors(theme.kind, creatorsTag, powerLevelTags);
 | 
			
		||||
 | 
			
		||||
  const mentionClickHandler = useMentionClickHandler(room.roomId);
 | 
			
		||||
  const spoilerClickHandler = useSpoilerClickHandler();
 | 
			
		||||
| 
						 | 
				
			
			@ -222,13 +233,12 @@ export function SearchResultGroup({
 | 
			
		|||
          const threadRootId =
 | 
			
		||||
            relation?.rel_type === RelationType.Thread ? relation.event_id : undefined;
 | 
			
		||||
 | 
			
		||||
          const senderPowerLevel = getPowerLevel(event.sender);
 | 
			
		||||
          const powerLevelTag = getPowerLevelTag(senderPowerLevel);
 | 
			
		||||
          const tagColor = powerLevelTag?.color
 | 
			
		||||
            ? accessibleTagColors?.get(powerLevelTag.color)
 | 
			
		||||
          const memberPowerTag = getMemberPowerTag(event.sender);
 | 
			
		||||
          const tagColor = memberPowerTag?.color
 | 
			
		||||
            ? accessibleTagColors?.get(memberPowerTag.color)
 | 
			
		||||
            : undefined;
 | 
			
		||||
          const tagIconSrc = powerLevelTag?.icon
 | 
			
		||||
            ? getTagIconSrc(mx, useAuthentication, powerLevelTag.icon)
 | 
			
		||||
          const tagIconSrc = memberPowerTag?.icon
 | 
			
		||||
            ? getPowerTagIconSrc(mx, useAuthentication, memberPowerTag.icon)
 | 
			
		||||
            : undefined;
 | 
			
		||||
 | 
			
		||||
          const usernameColor = legacyUsernameColor ? colorMXID(event.sender) : tagColor;
 | 
			
		||||
| 
						 | 
				
			
			@ -275,7 +285,11 @@ export function SearchResultGroup({
 | 
			
		|||
                      </Username>
 | 
			
		||||
                      {tagIconSrc && <PowerIcon size="100" iconSrc={tagIconSrc} />}
 | 
			
		||||
                    </Box>
 | 
			
		||||
                    <Time ts={event.origin_server_ts} />
 | 
			
		||||
                    <Time
 | 
			
		||||
                      ts={event.origin_server_ts}
 | 
			
		||||
                      hour24Clock={hour24Clock}
 | 
			
		||||
                      dateFormatString={dateFormatString}
 | 
			
		||||
                    />
 | 
			
		||||
                  </Box>
 | 
			
		||||
                  <Box shrink="No" gap="200" alignItems="Center">
 | 
			
		||||
                    <Chip
 | 
			
		||||
| 
						 | 
				
			
			@ -294,8 +308,7 @@ export function SearchResultGroup({
 | 
			
		|||
                    replyEventId={replyEventId}
 | 
			
		||||
                    threadRootId={threadRootId}
 | 
			
		||||
                    onClick={handleOpenClick}
 | 
			
		||||
                    getPowerLevel={getPowerLevel}
 | 
			
		||||
                    getPowerLevelTag={getPowerLevelTag}
 | 
			
		||||
                    getMemberPowerTag={getMemberPowerTag}
 | 
			
		||||
                    accessibleTagColors={accessibleTagColors}
 | 
			
		||||
                    legacyUsernameColor={legacyUsernameColor}
 | 
			
		||||
                  />
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -27,7 +27,7 @@ import { nameInitials } from '../../utils/common';
 | 
			
		|||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useRoomUnread } from '../../state/hooks/unread';
 | 
			
		||||
import { roomToUnreadAtom } from '../../state/room/roomToUnread';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevels } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { copyToClipboard } from '../../utils/dom';
 | 
			
		||||
import { markAsRead } from '../../../client/action/notifications';
 | 
			
		||||
import { openInviteUser } from '../../../client/action/navigation';
 | 
			
		||||
| 
						 | 
				
			
			@ -49,6 +49,8 @@ import {
 | 
			
		|||
  RoomNotificationMode,
 | 
			
		||||
} from '../../hooks/useRoomsNotificationPreferences';
 | 
			
		||||
import { RoomNotificationModeSwitcher } from '../../components/RoomNotificationSwitcher';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RoomNavItemMenuProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -61,8 +63,10 @@ const RoomNavItemMenu = forwardRef<HTMLDivElement, RoomNavItemMenuProps>(
 | 
			
		|||
    const [hideActivity] = useSetting(settingsAtom, 'hideActivity');
 | 
			
		||||
    const unread = useRoomUnread(room.roomId, roomToUnreadAtom);
 | 
			
		||||
    const powerLevels = usePowerLevels(room);
 | 
			
		||||
    const { getPowerLevel, canDoAction } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
    const canInvite = canDoAction('invite', getPowerLevel(mx.getUserId() ?? ''));
 | 
			
		||||
    const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
    const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
    const canInvite = permissions.action('invite', mx.getSafeUserId());
 | 
			
		||||
    const openRoomSettings = useOpenRoomSettings();
 | 
			
		||||
    const space = useSpaceOptionally();
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -13,6 +13,8 @@ import {
 | 
			
		|||
  RoomPublish,
 | 
			
		||||
  RoomUpgrade,
 | 
			
		||||
} from '../../common-settings/general';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type GeneralProps = {
 | 
			
		||||
  requestClose: () => void;
 | 
			
		||||
| 
						 | 
				
			
			@ -20,6 +22,8 @@ type GeneralProps = {
 | 
			
		|||
export function General({ requestClose }: GeneralProps) {
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Page>
 | 
			
		||||
| 
						 | 
				
			
			@ -41,22 +45,22 @@ export function General({ requestClose }: GeneralProps) {
 | 
			
		|||
        <Scroll hideTrack visibility="Hover">
 | 
			
		||||
          <PageContent>
 | 
			
		||||
            <Box direction="Column" gap="700">
 | 
			
		||||
              <RoomProfile powerLevels={powerLevels} />
 | 
			
		||||
              <RoomProfile permissions={permissions} />
 | 
			
		||||
              <Box direction="Column" gap="100">
 | 
			
		||||
                <Text size="L400">Options</Text>
 | 
			
		||||
                <RoomJoinRules powerLevels={powerLevels} />
 | 
			
		||||
                <RoomHistoryVisibility powerLevels={powerLevels} />
 | 
			
		||||
                <RoomEncryption powerLevels={powerLevels} />
 | 
			
		||||
                <RoomPublish powerLevels={powerLevels} />
 | 
			
		||||
                <RoomJoinRules permissions={permissions} />
 | 
			
		||||
                <RoomHistoryVisibility permissions={permissions} />
 | 
			
		||||
                <RoomEncryption permissions={permissions} />
 | 
			
		||||
                <RoomPublish permissions={permissions} />
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Box direction="Column" gap="100">
 | 
			
		||||
                <Text size="L400">Addresses</Text>
 | 
			
		||||
                <RoomPublishedAddresses powerLevels={powerLevels} />
 | 
			
		||||
                <RoomLocalAddresses powerLevels={powerLevels} />
 | 
			
		||||
                <RoomPublishedAddresses permissions={permissions} />
 | 
			
		||||
                <RoomLocalAddresses permissions={permissions} />
 | 
			
		||||
              </Box>
 | 
			
		||||
              <Box direction="Column" gap="100">
 | 
			
		||||
                <Text size="L400">Advance Options</Text>
 | 
			
		||||
                <RoomUpgrade powerLevels={powerLevels} requestClose={requestClose} />
 | 
			
		||||
                <RoomUpgrade permissions={permissions} requestClose={requestClose} />
 | 
			
		||||
              </Box>
 | 
			
		||||
            </Box>
 | 
			
		||||
          </PageContent>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -2,11 +2,13 @@ import React, { useState } from 'react';
 | 
			
		|||
import { Box, Icon, IconButton, Icons, Scroll, Text } from 'folds';
 | 
			
		||||
import { Page, PageContent, PageHeader } from '../../../components/page';
 | 
			
		||||
import { useRoom } from '../../../hooks/useRoom';
 | 
			
		||||
import { usePowerLevels, usePowerLevelsAPI } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevels } from '../../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../../hooks/useMatrixClient';
 | 
			
		||||
import { StateEvent } from '../../../../types/matrix/room';
 | 
			
		||||
import { usePermissionGroups } from './usePermissionItems';
 | 
			
		||||
import { PermissionGroups, Powers, PowersEditor } from '../../common-settings/permissions';
 | 
			
		||||
import { useRoomCreators } from '../../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type PermissionsProps = {
 | 
			
		||||
  requestClose: () => void;
 | 
			
		||||
| 
						 | 
				
			
			@ -15,11 +17,12 @@ export function Permissions({ requestClose }: PermissionsProps) {
 | 
			
		|||
  const mx = useMatrixClient();
 | 
			
		||||
  const room = useRoom();
 | 
			
		||||
  const powerLevels = usePowerLevels(room);
 | 
			
		||||
  const { getPowerLevel, canSendStateEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canEditPowers = canSendStateEvent(
 | 
			
		||||
    StateEvent.PowerLevelTags,
 | 
			
		||||
    getPowerLevel(mx.getSafeUserId())
 | 
			
		||||
  );
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const canEditPowers = permissions.stateEvent(StateEvent.PowerLevelTags, mx.getSafeUserId());
 | 
			
		||||
  const canEditPermissions = permissions.stateEvent(StateEvent.RoomPowerLevels, mx.getSafeUserId());
 | 
			
		||||
  const permissionGroups = usePermissionGroups();
 | 
			
		||||
 | 
			
		||||
  const [powerEditor, setPowerEditor] = useState(false);
 | 
			
		||||
| 
						 | 
				
			
			@ -57,7 +60,11 @@ export function Permissions({ requestClose }: PermissionsProps) {
 | 
			
		|||
                onEdit={canEditPowers ? handleEditPowers : undefined}
 | 
			
		||||
                permissionGroups={permissionGroups}
 | 
			
		||||
              />
 | 
			
		||||
              <PermissionGroups powerLevels={powerLevels} permissionGroups={permissionGroups} />
 | 
			
		||||
              <PermissionGroups
 | 
			
		||||
                canEdit={canEditPermissions}
 | 
			
		||||
                powerLevels={powerLevels}
 | 
			
		||||
                permissionGroups={permissionGroups}
 | 
			
		||||
              />
 | 
			
		||||
            </Box>
 | 
			
		||||
          </PageContent>
 | 
			
		||||
        </Scroll>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,6 +1,6 @@
 | 
			
		|||
import React, { KeyboardEvent as ReactKeyboardEvent, useCallback, useEffect, useMemo } from 'react';
 | 
			
		||||
import { Editor } from 'slate';
 | 
			
		||||
import { Box, MenuItem, Text } from 'folds';
 | 
			
		||||
import { Box, config, MenuItem, Text } from 'folds';
 | 
			
		||||
import { Room } from 'matrix-js-sdk';
 | 
			
		||||
import { Command, useCommands } from '../../hooks/useCommands';
 | 
			
		||||
import {
 | 
			
		||||
| 
						 | 
				
			
			@ -75,9 +75,6 @@ export function CommandAutocomplete({
 | 
			
		|||
      headerContent={
 | 
			
		||||
        <Box grow="Yes" direction="Row" gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
          <Text size="L400">Commands</Text>
 | 
			
		||||
          <Text size="T200" priority="300" truncate>
 | 
			
		||||
            Begin your message with command
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Box>
 | 
			
		||||
      }
 | 
			
		||||
      requestClose={requestClose}
 | 
			
		||||
| 
						 | 
				
			
			@ -87,17 +84,22 @@ export function CommandAutocomplete({
 | 
			
		|||
          key={commandName}
 | 
			
		||||
          as="button"
 | 
			
		||||
          radii="300"
 | 
			
		||||
          style={{ height: 'unset' }}
 | 
			
		||||
          onKeyDown={(evt: ReactKeyboardEvent<HTMLButtonElement>) =>
 | 
			
		||||
            onTabPress(evt, () => handleAutocomplete(commandName))
 | 
			
		||||
          }
 | 
			
		||||
          onClick={() => handleAutocomplete(commandName)}
 | 
			
		||||
        >
 | 
			
		||||
          <Box grow="Yes" direction="Row" gap="200" justifyContent="SpaceBetween">
 | 
			
		||||
            <Box shrink="No">
 | 
			
		||||
              <Text style={{ flexGrow: 1 }} size="B400" truncate>
 | 
			
		||||
                {`/${commandName}`}
 | 
			
		||||
              </Text>
 | 
			
		||||
            </Box>
 | 
			
		||||
          <Box
 | 
			
		||||
            style={{ padding: `${config.space.S300} 0` }}
 | 
			
		||||
            grow="Yes"
 | 
			
		||||
            direction="Column"
 | 
			
		||||
            gap="100"
 | 
			
		||||
            justifyContent="SpaceBetween"
 | 
			
		||||
          >
 | 
			
		||||
            <Text style={{ flexGrow: 1 }} size="B400" truncate>
 | 
			
		||||
              {`/${commandName}`}
 | 
			
		||||
            </Text>
 | 
			
		||||
            <Text truncate priority="300" size="T200">
 | 
			
		||||
              {commands[commandName].description}
 | 
			
		||||
            </Text>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,10 +1,8 @@
 | 
			
		|||
import { keyframes, style } from '@vanilla-extract/css';
 | 
			
		||||
import { color, config, toRem } from 'folds';
 | 
			
		||||
import { config, toRem } from 'folds';
 | 
			
		||||
 | 
			
		||||
export const MembersDrawer = style({
 | 
			
		||||
  width: toRem(266),
 | 
			
		||||
  backgroundColor: color.Background.Container,
 | 
			
		||||
  color: color.Background.OnContainer,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export const MembersDrawerHeader = style({
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -26,11 +26,10 @@ import {
 | 
			
		|||
  TooltipProvider,
 | 
			
		||||
  config,
 | 
			
		||||
} from 'folds';
 | 
			
		||||
import { Room, RoomMember } from 'matrix-js-sdk';
 | 
			
		||||
import { MatrixClient, Room, RoomMember } from 'matrix-js-sdk';
 | 
			
		||||
import { useVirtualizer } from '@tanstack/react-virtual';
 | 
			
		||||
import classNames from 'classnames';
 | 
			
		||||
 | 
			
		||||
import { openProfileViewer } from '../../../client/action/navigation';
 | 
			
		||||
import * as css from './MembersDrawer.css';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { UseStateProvider } from '../../components/UseStateProvider';
 | 
			
		||||
| 
						 | 
				
			
			@ -40,7 +39,6 @@ import {
 | 
			
		|||
  useAsyncSearch,
 | 
			
		||||
} from '../../hooks/useAsyncSearch';
 | 
			
		||||
import { useDebounce } from '../../hooks/useDebounce';
 | 
			
		||||
import { usePowerLevelTags, useFlattenPowerLevelTagMembers } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { TypingIndicator } from '../../components/typing-indicator';
 | 
			
		||||
import { getMemberDisplayName, getMemberSearchStr } from '../../utils/room';
 | 
			
		||||
import { getMxIdLocalPart } from '../../utils/matrix';
 | 
			
		||||
| 
						 | 
				
			
			@ -52,10 +50,116 @@ import { UserAvatar } from '../../components/user-avatar';
 | 
			
		|||
import { useRoomTypingMember } from '../../hooks/useRoomTypingMembers';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useMembershipFilter, useMembershipFilterMenu } from '../../hooks/useMemberFilter';
 | 
			
		||||
import { useMemberSort, useMemberSortMenu } from '../../hooks/useMemberSort';
 | 
			
		||||
import { usePowerLevelsAPI, usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { useMemberPowerSort, useMemberSort, useMemberSortMenu } from '../../hooks/useMemberSort';
 | 
			
		||||
import { usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { MembershipFilterMenu } from '../../components/MembershipFilterMenu';
 | 
			
		||||
import { MemberSortMenu } from '../../components/MemberSortMenu';
 | 
			
		||||
import { useOpenUserRoomProfile, useUserRoomProfileState } from '../../state/hooks/userRoomProfile';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { ContainerColor } from '../../styles/ContainerColor.css';
 | 
			
		||||
import { useFlattenPowerTagMembers, useGetMemberPowerTag } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
 | 
			
		||||
type MemberDrawerHeaderProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
};
 | 
			
		||||
function MemberDrawerHeader({ room }: MemberDrawerHeaderProps) {
 | 
			
		||||
  const setPeopleDrawer = useSetSetting(settingsAtom, 'isPeopleDrawer');
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Header className={css.MembersDrawerHeader} variant="Background" size="600">
 | 
			
		||||
      <Box grow="Yes" alignItems="Center" gap="200">
 | 
			
		||||
        <Box grow="Yes" alignItems="Center" gap="200">
 | 
			
		||||
          <Text title={`${room.getJoinedMemberCount()} Members`} size="H5" truncate>
 | 
			
		||||
            {`${millify(room.getJoinedMemberCount())} Members`}
 | 
			
		||||
          </Text>
 | 
			
		||||
        </Box>
 | 
			
		||||
        <Box shrink="No" alignItems="Center">
 | 
			
		||||
          <TooltipProvider
 | 
			
		||||
            position="Bottom"
 | 
			
		||||
            align="End"
 | 
			
		||||
            offset={4}
 | 
			
		||||
            tooltip={
 | 
			
		||||
              <Tooltip>
 | 
			
		||||
                <Text>Close</Text>
 | 
			
		||||
              </Tooltip>
 | 
			
		||||
            }
 | 
			
		||||
          >
 | 
			
		||||
            {(triggerRef) => (
 | 
			
		||||
              <IconButton
 | 
			
		||||
                ref={triggerRef}
 | 
			
		||||
                variant="Background"
 | 
			
		||||
                onClick={() => setPeopleDrawer(false)}
 | 
			
		||||
              >
 | 
			
		||||
                <Icon src={Icons.Cross} />
 | 
			
		||||
              </IconButton>
 | 
			
		||||
            )}
 | 
			
		||||
          </TooltipProvider>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </Header>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type MemberItemProps = {
 | 
			
		||||
  mx: MatrixClient;
 | 
			
		||||
  useAuthentication: boolean;
 | 
			
		||||
  room: Room;
 | 
			
		||||
  member: RoomMember;
 | 
			
		||||
  onClick: MouseEventHandler<HTMLButtonElement>;
 | 
			
		||||
  pressed?: boolean;
 | 
			
		||||
  typing?: boolean;
 | 
			
		||||
};
 | 
			
		||||
function MemberItem({
 | 
			
		||||
  mx,
 | 
			
		||||
  useAuthentication,
 | 
			
		||||
  room,
 | 
			
		||||
  member,
 | 
			
		||||
  onClick,
 | 
			
		||||
  pressed,
 | 
			
		||||
  typing,
 | 
			
		||||
}: MemberItemProps) {
 | 
			
		||||
  const name =
 | 
			
		||||
    getMemberDisplayName(room, member.userId) ?? getMxIdLocalPart(member.userId) ?? member.userId;
 | 
			
		||||
  const avatarMxcUrl = member.getMxcAvatarUrl();
 | 
			
		||||
  const avatarUrl = avatarMxcUrl
 | 
			
		||||
    ? mx.mxcUrlToHttp(avatarMxcUrl, 100, 100, 'crop', undefined, false, useAuthentication)
 | 
			
		||||
    : undefined;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <MenuItem
 | 
			
		||||
      style={{ padding: `0 ${config.space.S200}` }}
 | 
			
		||||
      aria-pressed={pressed}
 | 
			
		||||
      data-user-id={member.userId}
 | 
			
		||||
      variant="Background"
 | 
			
		||||
      radii="400"
 | 
			
		||||
      onClick={onClick}
 | 
			
		||||
      before={
 | 
			
		||||
        <Avatar size="200">
 | 
			
		||||
          <UserAvatar
 | 
			
		||||
            userId={member.userId}
 | 
			
		||||
            src={avatarUrl ?? undefined}
 | 
			
		||||
            alt={name}
 | 
			
		||||
            renderFallback={() => <Icon size="50" src={Icons.User} filled />}
 | 
			
		||||
          />
 | 
			
		||||
        </Avatar>
 | 
			
		||||
      }
 | 
			
		||||
      after={
 | 
			
		||||
        typing && (
 | 
			
		||||
          <Badge size="300" variant="Secondary" fill="Soft" radii="Pill" outlined>
 | 
			
		||||
            <TypingIndicator size="300" />
 | 
			
		||||
          </Badge>
 | 
			
		||||
        )
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <Box grow="Yes">
 | 
			
		||||
        <Text size="T400" truncate>
 | 
			
		||||
          {name}
 | 
			
		||||
        </Text>
 | 
			
		||||
      </Box>
 | 
			
		||||
    </MenuItem>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const SEARCH_OPTIONS: UseAsyncSearchOptions = {
 | 
			
		||||
  limit: 1000,
 | 
			
		||||
| 
						 | 
				
			
			@ -79,28 +183,28 @@ export function MembersDrawer({ room, members }: MembersDrawerProps) {
 | 
			
		|||
  const searchInputRef = useRef<HTMLInputElement>(null);
 | 
			
		||||
  const scrollTopAnchorRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const powerLevels = usePowerLevelsContext();
 | 
			
		||||
  const [, getPowerLevelTag] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
  const getPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const fetchingMembers = members.length < room.getJoinedMemberCount();
 | 
			
		||||
  const setPeopleDrawer = useSetSetting(settingsAtom, 'isPeopleDrawer');
 | 
			
		||||
  const openUserRoomProfile = useOpenUserRoomProfile();
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
  const openProfileUserId = useUserRoomProfileState()?.userId;
 | 
			
		||||
 | 
			
		||||
  const membershipFilterMenu = useMembershipFilterMenu();
 | 
			
		||||
  const sortFilterMenu = useMemberSortMenu();
 | 
			
		||||
  const [sortFilterIndex, setSortFilterIndex] = useSetting(settingsAtom, 'memberSortFilterIndex');
 | 
			
		||||
  const [membershipFilterIndex, setMembershipFilterIndex] = useState(0);
 | 
			
		||||
  const { getPowerLevel } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
 | 
			
		||||
  const membershipFilter = useMembershipFilter(membershipFilterIndex, membershipFilterMenu);
 | 
			
		||||
  const memberSort = useMemberSort(sortFilterIndex, sortFilterMenu);
 | 
			
		||||
  const memberPowerSort = useMemberPowerSort(creators);
 | 
			
		||||
 | 
			
		||||
  const typingMembers = useRoomTypingMember(room.roomId);
 | 
			
		||||
 | 
			
		||||
  const filteredMembers = useMemo(
 | 
			
		||||
    () =>
 | 
			
		||||
      members
 | 
			
		||||
        .filter(membershipFilter.filterFn)
 | 
			
		||||
        .sort(memberSort.sortFn)
 | 
			
		||||
        .sort((a, b) => b.powerLevel - a.powerLevel),
 | 
			
		||||
    [members, membershipFilter, memberSort]
 | 
			
		||||
    () => members.filter(membershipFilter.filterFn).sort(memberSort.sortFn).sort(memberPowerSort),
 | 
			
		||||
    [members, membershipFilter, memberSort, memberPowerSort]
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const [result, search, resetSearch] = useAsyncSearch(
 | 
			
		||||
| 
						 | 
				
			
			@ -112,11 +216,7 @@ export function MembersDrawer({ room, members }: MembersDrawerProps) {
 | 
			
		|||
 | 
			
		||||
  const processMembers = result ? result.items : filteredMembers;
 | 
			
		||||
 | 
			
		||||
  const PLTagOrRoomMember = useFlattenPowerLevelTagMembers(
 | 
			
		||||
    processMembers,
 | 
			
		||||
    getPowerLevel,
 | 
			
		||||
    getPowerLevelTag
 | 
			
		||||
  );
 | 
			
		||||
  const PLTagOrRoomMember = useFlattenPowerTagMembers(processMembers, getPowerTag);
 | 
			
		||||
 | 
			
		||||
  const virtualizer = useVirtualizer({
 | 
			
		||||
    count: PLTagOrRoomMember.length,
 | 
			
		||||
| 
						 | 
				
			
			@ -136,48 +236,20 @@ export function MembersDrawer({ room, members }: MembersDrawerProps) {
 | 
			
		|||
    { wait: 200 }
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const getName = (member: RoomMember) =>
 | 
			
		||||
    getMemberDisplayName(room, member.userId) ?? getMxIdLocalPart(member.userId) ?? member.userId;
 | 
			
		||||
 | 
			
		||||
  const handleMemberClick: MouseEventHandler<HTMLButtonElement> = (evt) => {
 | 
			
		||||
    const btn = evt.currentTarget as HTMLButtonElement;
 | 
			
		||||
    const userId = btn.getAttribute('data-user-id');
 | 
			
		||||
    openProfileViewer(userId, room.roomId);
 | 
			
		||||
    if (!userId) return;
 | 
			
		||||
    openUserRoomProfile(room.roomId, space?.roomId, userId, btn.getBoundingClientRect(), 'Left');
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Box className={css.MembersDrawer} shrink="No" direction="Column">
 | 
			
		||||
      <Header className={css.MembersDrawerHeader} variant="Background" size="600">
 | 
			
		||||
        <Box grow="Yes" alignItems="Center" gap="200">
 | 
			
		||||
          <Box grow="Yes" alignItems="Center" gap="200">
 | 
			
		||||
            <Text title={`${room.getJoinedMemberCount()} Members`} size="H5" truncate>
 | 
			
		||||
              {`${millify(room.getJoinedMemberCount())} Members`}
 | 
			
		||||
            </Text>
 | 
			
		||||
          </Box>
 | 
			
		||||
          <Box shrink="No" alignItems="Center">
 | 
			
		||||
            <TooltipProvider
 | 
			
		||||
              position="Bottom"
 | 
			
		||||
              align="End"
 | 
			
		||||
              offset={4}
 | 
			
		||||
              tooltip={
 | 
			
		||||
                <Tooltip>
 | 
			
		||||
                  <Text>Close</Text>
 | 
			
		||||
                </Tooltip>
 | 
			
		||||
              }
 | 
			
		||||
            >
 | 
			
		||||
              {(triggerRef) => (
 | 
			
		||||
                <IconButton
 | 
			
		||||
                  ref={triggerRef}
 | 
			
		||||
                  variant="Background"
 | 
			
		||||
                  onClick={() => setPeopleDrawer(false)}
 | 
			
		||||
                >
 | 
			
		||||
                  <Icon src={Icons.Cross} />
 | 
			
		||||
                </IconButton>
 | 
			
		||||
              )}
 | 
			
		||||
            </TooltipProvider>
 | 
			
		||||
          </Box>
 | 
			
		||||
        </Box>
 | 
			
		||||
      </Header>
 | 
			
		||||
    <Box
 | 
			
		||||
      className={classNames(css.MembersDrawer, ContainerColor({ variant: 'Background' }))}
 | 
			
		||||
      shrink="No"
 | 
			
		||||
      direction="Column"
 | 
			
		||||
    >
 | 
			
		||||
      <MemberDrawerHeader room={room} />
 | 
			
		||||
      <Box className={css.MemberDrawerContentBase} grow="Yes">
 | 
			
		||||
        <Scroll ref={scrollRef} variant="Background" size="300" visibility="Hover" hideTrack>
 | 
			
		||||
          <Box className={css.MemberDrawerContent} direction="Column" gap="200">
 | 
			
		||||
| 
						 | 
				
			
			@ -329,59 +401,28 @@ export function MembersDrawer({ room, members }: MembersDrawerProps) {
 | 
			
		|||
                    );
 | 
			
		||||
                  }
 | 
			
		||||
 | 
			
		||||
                  const member = tagOrMember;
 | 
			
		||||
                  const name = getName(member);
 | 
			
		||||
                  const avatarMxcUrl = member.getMxcAvatarUrl();
 | 
			
		||||
                  const avatarUrl = avatarMxcUrl
 | 
			
		||||
                    ? mx.mxcUrlToHttp(
 | 
			
		||||
                        avatarMxcUrl,
 | 
			
		||||
                        100,
 | 
			
		||||
                        100,
 | 
			
		||||
                        'crop',
 | 
			
		||||
                        undefined,
 | 
			
		||||
                        false,
 | 
			
		||||
                        useAuthentication
 | 
			
		||||
                      )
 | 
			
		||||
                    : undefined;
 | 
			
		||||
 | 
			
		||||
                  return (
 | 
			
		||||
                    <MenuItem
 | 
			
		||||
                    <div
 | 
			
		||||
                      style={{
 | 
			
		||||
                        padding: `0 ${config.space.S200}`,
 | 
			
		||||
                        transform: `translateY(${vItem.start}px)`,
 | 
			
		||||
                      }}
 | 
			
		||||
                      data-index={vItem.index}
 | 
			
		||||
                      data-user-id={member.userId}
 | 
			
		||||
                      ref={virtualizer.measureElement}
 | 
			
		||||
                      key={`${room.roomId}-${member.userId}`}
 | 
			
		||||
                      className={css.DrawerVirtualItem}
 | 
			
		||||
                      variant="Background"
 | 
			
		||||
                      radii="400"
 | 
			
		||||
                      onClick={handleMemberClick}
 | 
			
		||||
                      before={
 | 
			
		||||
                        <Avatar size="200">
 | 
			
		||||
                          <UserAvatar
 | 
			
		||||
                            userId={member.userId}
 | 
			
		||||
                            src={avatarUrl ?? undefined}
 | 
			
		||||
                            alt={name}
 | 
			
		||||
                            renderFallback={() => <Icon size="50" src={Icons.User} filled />}
 | 
			
		||||
                          />
 | 
			
		||||
                        </Avatar>
 | 
			
		||||
                      }
 | 
			
		||||
                      after={
 | 
			
		||||
                        typingMembers.find((receipt) => receipt.userId === member.userId) && (
 | 
			
		||||
                          <Badge size="300" variant="Secondary" fill="Soft" radii="Pill" outlined>
 | 
			
		||||
                            <TypingIndicator size="300" />
 | 
			
		||||
                          </Badge>
 | 
			
		||||
                        )
 | 
			
		||||
                      }
 | 
			
		||||
                      data-index={vItem.index}
 | 
			
		||||
                      key={`${room.roomId}-${tagOrMember.userId}`}
 | 
			
		||||
                      ref={virtualizer.measureElement}
 | 
			
		||||
                    >
 | 
			
		||||
                      <Box grow="Yes">
 | 
			
		||||
                        <Text size="T400" truncate>
 | 
			
		||||
                          {name}
 | 
			
		||||
                        </Text>
 | 
			
		||||
                      </Box>
 | 
			
		||||
                    </MenuItem>
 | 
			
		||||
                      <MemberItem
 | 
			
		||||
                        mx={mx}
 | 
			
		||||
                        useAuthentication={useAuthentication}
 | 
			
		||||
                        room={room}
 | 
			
		||||
                        member={tagOrMember}
 | 
			
		||||
                        onClick={handleMemberClick}
 | 
			
		||||
                        pressed={openProfileUserId === tagOrMember.userId}
 | 
			
		||||
                        typing={typingMembers.some(
 | 
			
		||||
                          (receipt) => receipt.userId === tagOrMember.userId
 | 
			
		||||
                        )}
 | 
			
		||||
                      />
 | 
			
		||||
                    </div>
 | 
			
		||||
                  );
 | 
			
		||||
                })}
 | 
			
		||||
              </div>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -108,21 +108,23 @@ import { ReplyLayout, ThreadIndicator } from '../../components/message';
 | 
			
		|||
import { roomToParentsAtom } from '../../state/room/roomToParents';
 | 
			
		||||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useImagePackRooms } from '../../hooks/useImagePackRooms';
 | 
			
		||||
import { GetPowerLevelTag } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { powerLevelAPI, usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import colorMXID from '../../../util/colorMXID';
 | 
			
		||||
import { useIsDirectRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useAccessiblePowerTagColors, useGetMemberPowerTag } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useTheme } from '../../hooks/useTheme';
 | 
			
		||||
import { useRoomCreatorsTag } from '../../hooks/useRoomCreatorsTag';
 | 
			
		||||
import { usePowerLevelTags } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
 | 
			
		||||
interface RoomInputProps {
 | 
			
		||||
  editor: Editor;
 | 
			
		||||
  fileDropContainerRef: RefObject<HTMLElement>;
 | 
			
		||||
  roomId: string;
 | 
			
		||||
  room: Room;
 | 
			
		||||
  getPowerLevelTag: GetPowerLevelTag;
 | 
			
		||||
  accessibleTagColors: Map<string, string>;
 | 
			
		||||
}
 | 
			
		||||
export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
 | 
			
		||||
  ({ editor, fileDropContainerRef, roomId, room, getPowerLevelTag, accessibleTagColors }, ref) => {
 | 
			
		||||
  ({ editor, fileDropContainerRef, roomId, room }, ref) => {
 | 
			
		||||
    const mx = useMatrixClient();
 | 
			
		||||
    const useAuthentication = useMediaAuthentication();
 | 
			
		||||
    const [enterForNewline] = useSetting(settingsAtom, 'enterForNewline');
 | 
			
		||||
| 
						 | 
				
			
			@ -134,13 +136,24 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
 | 
			
		|||
    const emojiBtnRef = useRef<HTMLButtonElement>(null);
 | 
			
		||||
    const roomToParents = useAtomValue(roomToParentsAtom);
 | 
			
		||||
    const powerLevels = usePowerLevelsContext();
 | 
			
		||||
    const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
    const [msgDraft, setMsgDraft] = useAtom(roomIdToMsgDraftAtomFamily(roomId));
 | 
			
		||||
    const [replyDraft, setReplyDraft] = useAtom(roomIdToReplyDraftAtomFamily(roomId));
 | 
			
		||||
    const replyUserID = replyDraft?.userId;
 | 
			
		||||
 | 
			
		||||
    const replyPowerTag = getPowerLevelTag(powerLevelAPI.getPowerLevel(powerLevels, replyUserID));
 | 
			
		||||
    const replyPowerColor = replyPowerTag.color
 | 
			
		||||
    const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
    const creatorsTag = useRoomCreatorsTag();
 | 
			
		||||
    const getMemberPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
    const theme = useTheme();
 | 
			
		||||
    const accessibleTagColors = useAccessiblePowerTagColors(
 | 
			
		||||
      theme.kind,
 | 
			
		||||
      creatorsTag,
 | 
			
		||||
      powerLevelTags
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    const replyPowerTag = replyUserID ? getMemberPowerTag(replyUserID) : undefined;
 | 
			
		||||
    const replyPowerColor = replyPowerTag?.color
 | 
			
		||||
      ? accessibleTagColors.get(replyPowerTag.color)
 | 
			
		||||
      : undefined;
 | 
			
		||||
    const replyUsernameColor =
 | 
			
		||||
| 
						 | 
				
			
			@ -277,7 +290,7 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
 | 
			
		|||
      });
 | 
			
		||||
      handleCancelUpload(uploads);
 | 
			
		||||
      const contents = fulfilledPromiseSettledResult(await Promise.allSettled(contentsPromises));
 | 
			
		||||
      contents.forEach((content) => mx.sendMessage(roomId, content));
 | 
			
		||||
      contents.forEach((content) => mx.sendMessage(roomId, content as any));
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const submit = useCallback(() => {
 | 
			
		||||
| 
						 | 
				
			
			@ -356,7 +369,7 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
 | 
			
		|||
          content['m.relates_to'].is_falling_back = false;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      mx.sendMessage(roomId, content);
 | 
			
		||||
      mx.sendMessage(roomId, content as any);
 | 
			
		||||
      resetEditor(editor);
 | 
			
		||||
      resetEditorHistory(editor);
 | 
			
		||||
      setReplyDraft(undefined);
 | 
			
		||||
| 
						 | 
				
			
			@ -543,7 +556,7 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
 | 
			
		|||
                  >
 | 
			
		||||
                    <Icon src={Icons.Cross} size="50" />
 | 
			
		||||
                  </IconButton>
 | 
			
		||||
                  <Box direction="Column">
 | 
			
		||||
                  <Box direction="Row" gap="200" alignItems="Center">
 | 
			
		||||
                    {replyDraft.relation?.rel_type === RelationType.Thread && <ThreadIndicator />}
 | 
			
		||||
                    <ReplyLayout
 | 
			
		||||
                      userColor={replyUsernameColor}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -85,7 +85,6 @@ import {
 | 
			
		|||
} from '../../utils/room';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { MessageLayout, settingsAtom } from '../../state/settings';
 | 
			
		||||
import { openProfileViewer } from '../../../client/action/navigation';
 | 
			
		||||
import { useMatrixEventRenderer } from '../../hooks/useMatrixEventRenderer';
 | 
			
		||||
import { Reactions, Message, Event, EncryptedContent } from './message';
 | 
			
		||||
import { useMemberEventParser } from '../../hooks/useMemberEventParser';
 | 
			
		||||
| 
						 | 
				
			
			@ -102,7 +101,7 @@ import * as css from './RoomTimeline.css';
 | 
			
		|||
import { inSameDay, minuteDifference, timeDayMonthYear, today, yesterday } from '../../utils/time';
 | 
			
		||||
import { createMentionElement, isEmptyEditor, moveCursor } from '../../components/editor';
 | 
			
		||||
import { roomIdToReplyDraftAtomFamily } from '../../state/room/roomInputDrafts';
 | 
			
		||||
import { usePowerLevelsAPI, usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { GetContentCallback, MessageEvent, StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { useKeyDown } from '../../hooks/useKeyDown';
 | 
			
		||||
import { useDocumentFocusChange } from '../../hooks/useDocumentFocusChange';
 | 
			
		||||
| 
						 | 
				
			
			@ -118,8 +117,15 @@ import { useRoomNavigate } from '../../hooks/useRoomNavigate';
 | 
			
		|||
import { useMediaAuthentication } from '../../hooks/useMediaAuthentication';
 | 
			
		||||
import { useIgnoredUsers } from '../../hooks/useIgnoredUsers';
 | 
			
		||||
import { useImagePackRooms } from '../../hooks/useImagePackRooms';
 | 
			
		||||
import { GetPowerLevelTag } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { useIsDirectRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useOpenUserRoomProfile } from '../../state/hooks/userRoomProfile';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { useAccessiblePowerTagColors, useGetMemberPowerTag } from '../../hooks/useMemberPowerTag';
 | 
			
		||||
import { useTheme } from '../../hooks/useTheme';
 | 
			
		||||
import { useRoomCreatorsTag } from '../../hooks/useRoomCreatorsTag';
 | 
			
		||||
import { usePowerLevelTags } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
 | 
			
		||||
const TimelineFloat = as<'div', css.TimelineFloatVariants>(
 | 
			
		||||
  ({ position, className, ...props }, ref) => (
 | 
			
		||||
| 
						 | 
				
			
			@ -222,8 +228,6 @@ type RoomTimelineProps = {
 | 
			
		|||
  eventId?: string;
 | 
			
		||||
  roomInputRef: RefObject<HTMLElement>;
 | 
			
		||||
  editor: Editor;
 | 
			
		||||
  getPowerLevelTag: GetPowerLevelTag;
 | 
			
		||||
  accessibleTagColors: Map<string, string>;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const PAGINATION_LIMIT = 80;
 | 
			
		||||
| 
						 | 
				
			
			@ -426,14 +430,7 @@ const getRoomUnreadInfo = (room: Room, scrollTo = false) => {
 | 
			
		|||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function RoomTimeline({
 | 
			
		||||
  room,
 | 
			
		||||
  eventId,
 | 
			
		||||
  roomInputRef,
 | 
			
		||||
  editor,
 | 
			
		||||
  getPowerLevelTag,
 | 
			
		||||
  accessibleTagColors,
 | 
			
		||||
}: RoomTimelineProps) {
 | 
			
		||||
export function RoomTimeline({ room, eventId, roomInputRef, editor }: RoomTimelineProps) {
 | 
			
		||||
  const mx = useMatrixClient();
 | 
			
		||||
  const useAuthentication = useMediaAuthentication();
 | 
			
		||||
  const [hideActivity] = useSetting(settingsAtom, 'hideActivity');
 | 
			
		||||
| 
						 | 
				
			
			@ -448,19 +445,34 @@ export function RoomTimeline({
 | 
			
		|||
  const [encUrlPreview] = useSetting(settingsAtom, 'encUrlPreview');
 | 
			
		||||
  const showUrlPreview = room.hasEncryptionStateEvent() ? encUrlPreview : urlPreview;
 | 
			
		||||
  const [showHiddenEvents] = useSetting(settingsAtom, 'showHiddenEvents');
 | 
			
		||||
  const [showDeveloperTools] = useSetting(settingsAtom, 'developerTools');
 | 
			
		||||
 | 
			
		||||
  const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
 | 
			
		||||
  const [dateFormatString] = useSetting(settingsAtom, 'dateFormatString');
 | 
			
		||||
 | 
			
		||||
  const ignoredUsersList = useIgnoredUsers();
 | 
			
		||||
  const ignoredUsersSet = useMemo(() => new Set(ignoredUsersList), [ignoredUsersList]);
 | 
			
		||||
 | 
			
		||||
  const setReplyDraft = useSetAtom(roomIdToReplyDraftAtomFamily(room.roomId));
 | 
			
		||||
  const powerLevels = usePowerLevelsContext();
 | 
			
		||||
  const { canDoAction, canSendEvent, canSendStateEvent, getPowerLevel } =
 | 
			
		||||
    usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const myPowerLevel = getPowerLevel(mx.getUserId() ?? '');
 | 
			
		||||
  const canRedact = canDoAction('redact', myPowerLevel);
 | 
			
		||||
  const canSendReaction = canSendEvent(MessageEvent.Reaction, myPowerLevel);
 | 
			
		||||
  const canPinEvent = canSendStateEvent(StateEvent.RoomPinnedEvents, myPowerLevel);
 | 
			
		||||
  const creatorsTag = useRoomCreatorsTag();
 | 
			
		||||
  const powerLevelTags = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const getMemberPowerTag = useGetMemberPowerTag(room, creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const theme = useTheme();
 | 
			
		||||
  const accessiblePowerTagColors = useAccessiblePowerTagColors(
 | 
			
		||||
    theme.kind,
 | 
			
		||||
    creatorsTag,
 | 
			
		||||
    powerLevelTags
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
 | 
			
		||||
  const canRedact = permissions.action('redact', mx.getSafeUserId());
 | 
			
		||||
  const canSendReaction = permissions.event(MessageEvent.Reaction, mx.getSafeUserId());
 | 
			
		||||
  const canPinEvent = permissions.stateEvent(StateEvent.RoomPinnedEvents, mx.getSafeUserId());
 | 
			
		||||
  const [editId, setEditId] = useState<string>();
 | 
			
		||||
 | 
			
		||||
  const roomToParents = useAtomValue(roomToParentsAtom);
 | 
			
		||||
| 
						 | 
				
			
			@ -468,6 +480,8 @@ export function RoomTimeline({
 | 
			
		|||
  const { navigateRoom } = useRoomNavigate();
 | 
			
		||||
  const mentionClickHandler = useMentionClickHandler(room.roomId);
 | 
			
		||||
  const spoilerClickHandler = useSpoilerClickHandler();
 | 
			
		||||
  const openUserRoomProfile = useOpenUserRoomProfile();
 | 
			
		||||
  const space = useSpaceOptionally();
 | 
			
		||||
 | 
			
		||||
  const imagePackRooms: Room[] = useImagePackRooms(room.roomId, roomToParents);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -905,9 +919,14 @@ export function RoomTimeline({
 | 
			
		|||
        console.warn('Button should have "data-user-id" attribute!');
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      openProfileViewer(userId, room.roomId);
 | 
			
		||||
      openUserRoomProfile(
 | 
			
		||||
        room.roomId,
 | 
			
		||||
        space?.roomId,
 | 
			
		||||
        userId,
 | 
			
		||||
        evt.currentTarget.getBoundingClientRect()
 | 
			
		||||
      );
 | 
			
		||||
    },
 | 
			
		||||
    [room]
 | 
			
		||||
    [room, space, openUserRoomProfile]
 | 
			
		||||
  );
 | 
			
		||||
  const handleUsernameClick: MouseEventHandler<HTMLButtonElement> = useCallback(
 | 
			
		||||
    (evt) => {
 | 
			
		||||
| 
						 | 
				
			
			@ -932,7 +951,7 @@ export function RoomTimeline({
 | 
			
		|||
  );
 | 
			
		||||
 | 
			
		||||
  const handleReplyClick: MouseEventHandler<HTMLButtonElement> = useCallback(
 | 
			
		||||
    (evt) => {
 | 
			
		||||
    (evt, startThread = false) => {
 | 
			
		||||
      const replyId = evt.currentTarget.getAttribute('data-event-id');
 | 
			
		||||
      if (!replyId) {
 | 
			
		||||
        console.warn('Button should have "data-event-id" attribute!');
 | 
			
		||||
| 
						 | 
				
			
			@ -943,7 +962,9 @@ export function RoomTimeline({
 | 
			
		|||
      const editedReply = getEditedEvent(replyId, replyEvt, room.getUnfilteredTimelineSet());
 | 
			
		||||
      const content: IContent = editedReply?.getContent()['m.new_content'] ?? replyEvt.getContent();
 | 
			
		||||
      const { body, formatted_body: formattedBody } = content;
 | 
			
		||||
      const { 'm.relates_to': relation } = replyEvt.getWireContent();
 | 
			
		||||
      const { 'm.relates_to': relation } = startThread
 | 
			
		||||
        ? { 'm.relates_to': { rel_type: 'm.thread', event_id: replyId } }
 | 
			
		||||
        : replyEvt.getWireContent();
 | 
			
		||||
      const senderId = replyEvt.getSender();
 | 
			
		||||
      if (senderId && typeof body === 'string') {
 | 
			
		||||
        setReplyDraft({
 | 
			
		||||
| 
						 | 
				
			
			@ -976,7 +997,7 @@ export function RoomTimeline({
 | 
			
		|||
        (reactions.find(eventWithShortcode)?.getContent().shortcode as string | undefined);
 | 
			
		||||
      mx.sendEvent(
 | 
			
		||||
        room.roomId,
 | 
			
		||||
        MessageEvent.Reaction,
 | 
			
		||||
        MessageEvent.Reaction as any,
 | 
			
		||||
        getReactionContent(targetEventId, key, rShortcode)
 | 
			
		||||
      );
 | 
			
		||||
    },
 | 
			
		||||
| 
						 | 
				
			
			@ -1011,7 +1032,6 @@ export function RoomTimeline({
 | 
			
		|||
          editedEvent?.getContent()['m.new_content'] ?? mEvent.getContent()) as GetContentCallback;
 | 
			
		||||
 | 
			
		||||
        const senderId = mEvent.getSender() ?? '';
 | 
			
		||||
        const senderPowerLevel = getPowerLevel(mEvent.getSender());
 | 
			
		||||
        const senderDisplayName =
 | 
			
		||||
          getMemberDisplayName(room, senderId) ?? getMxIdLocalPart(senderId) ?? senderId;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1045,9 +1065,8 @@ export function RoomTimeline({
 | 
			
		|||
                  replyEventId={replyEventId}
 | 
			
		||||
                  threadRootId={threadRootId}
 | 
			
		||||
                  onClick={handleOpenReply}
 | 
			
		||||
                  getPowerLevel={getPowerLevel}
 | 
			
		||||
                  getPowerLevelTag={getPowerLevelTag}
 | 
			
		||||
                  accessibleTagColors={accessibleTagColors}
 | 
			
		||||
                  getMemberPowerTag={getMemberPowerTag}
 | 
			
		||||
                  accessibleTagColors={accessiblePowerTagColors}
 | 
			
		||||
                  legacyUsernameColor={legacyUsernameColor || direct}
 | 
			
		||||
                />
 | 
			
		||||
              )
 | 
			
		||||
| 
						 | 
				
			
			@ -1065,9 +1084,12 @@ export function RoomTimeline({
 | 
			
		|||
              )
 | 
			
		||||
            }
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            powerLevelTag={getPowerLevelTag(senderPowerLevel)}
 | 
			
		||||
            accessibleTagColors={accessibleTagColors}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
            memberPowerTag={getMemberPowerTag(senderId)}
 | 
			
		||||
            accessibleTagColors={accessiblePowerTagColors}
 | 
			
		||||
            legacyUsernameColor={legacyUsernameColor || direct}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          >
 | 
			
		||||
            {mEvent.isRedacted() ? (
 | 
			
		||||
              <RedactedContent reason={mEvent.getUnsigned().redacted_because?.content.reason} />
 | 
			
		||||
| 
						 | 
				
			
			@ -1094,7 +1116,6 @@ export function RoomTimeline({
 | 
			
		|||
        const hasReactions = reactions && reactions.length > 0;
 | 
			
		||||
        const { replyEventId, threadRootId } = mEvent;
 | 
			
		||||
        const highlighted = focusItem?.index === item && focusItem.highlight;
 | 
			
		||||
        const senderPowerLevel = getPowerLevel(mEvent.getSender());
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
          <Message
 | 
			
		||||
| 
						 | 
				
			
			@ -1126,9 +1147,8 @@ export function RoomTimeline({
 | 
			
		|||
                  replyEventId={replyEventId}
 | 
			
		||||
                  threadRootId={threadRootId}
 | 
			
		||||
                  onClick={handleOpenReply}
 | 
			
		||||
                  getPowerLevel={getPowerLevel}
 | 
			
		||||
                  getPowerLevelTag={getPowerLevelTag}
 | 
			
		||||
                  accessibleTagColors={accessibleTagColors}
 | 
			
		||||
                  getMemberPowerTag={getMemberPowerTag}
 | 
			
		||||
                  accessibleTagColors={accessiblePowerTagColors}
 | 
			
		||||
                  legacyUsernameColor={legacyUsernameColor || direct}
 | 
			
		||||
                />
 | 
			
		||||
              )
 | 
			
		||||
| 
						 | 
				
			
			@ -1146,9 +1166,12 @@ export function RoomTimeline({
 | 
			
		|||
              )
 | 
			
		||||
            }
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            powerLevelTag={getPowerLevelTag(senderPowerLevel)}
 | 
			
		||||
            accessibleTagColors={accessibleTagColors}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
            memberPowerTag={getMemberPowerTag(mEvent.getSender() ?? '')}
 | 
			
		||||
            accessibleTagColors={accessiblePowerTagColors}
 | 
			
		||||
            legacyUsernameColor={legacyUsernameColor || direct}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          >
 | 
			
		||||
            <EncryptedContent mEvent={mEvent}>
 | 
			
		||||
              {() => {
 | 
			
		||||
| 
						 | 
				
			
			@ -1212,7 +1235,6 @@ export function RoomTimeline({
 | 
			
		|||
        const reactions = reactionRelations && reactionRelations.getSortedAnnotationsByKey();
 | 
			
		||||
        const hasReactions = reactions && reactions.length > 0;
 | 
			
		||||
        const highlighted = focusItem?.index === item && focusItem.highlight;
 | 
			
		||||
        const senderPowerLevel = getPowerLevel(mEvent.getSender());
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
          <Message
 | 
			
		||||
| 
						 | 
				
			
			@ -1247,9 +1269,12 @@ export function RoomTimeline({
 | 
			
		|||
              )
 | 
			
		||||
            }
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            powerLevelTag={getPowerLevelTag(senderPowerLevel)}
 | 
			
		||||
            accessibleTagColors={accessibleTagColors}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
            memberPowerTag={getMemberPowerTag(mEvent.getSender() ?? '')}
 | 
			
		||||
            accessibleTagColors={accessiblePowerTagColors}
 | 
			
		||||
            legacyUsernameColor={legacyUsernameColor || direct}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          >
 | 
			
		||||
            {mEvent.isRedacted() ? (
 | 
			
		||||
              <RedactedContent reason={mEvent.getUnsigned().redacted_because?.content.reason} />
 | 
			
		||||
| 
						 | 
				
			
			@ -1278,7 +1303,12 @@ export function RoomTimeline({
 | 
			
		|||
        const parsed = parseMemberEvent(mEvent);
 | 
			
		||||
 | 
			
		||||
        const timeJSX = (
 | 
			
		||||
          <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
          <Time
 | 
			
		||||
            ts={mEvent.getTs()}
 | 
			
		||||
            compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1292,6 +1322,7 @@ export function RoomTimeline({
 | 
			
		|||
            messageSpacing={messageSpacing}
 | 
			
		||||
            canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
          >
 | 
			
		||||
            <EventContent
 | 
			
		||||
              messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			@ -1314,7 +1345,12 @@ export function RoomTimeline({
 | 
			
		|||
        const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
 | 
			
		||||
 | 
			
		||||
        const timeJSX = (
 | 
			
		||||
          <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
          <Time
 | 
			
		||||
            ts={mEvent.getTs()}
 | 
			
		||||
            compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1328,6 +1364,7 @@ export function RoomTimeline({
 | 
			
		|||
            messageSpacing={messageSpacing}
 | 
			
		||||
            canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
          >
 | 
			
		||||
            <EventContent
 | 
			
		||||
              messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			@ -1351,7 +1388,12 @@ export function RoomTimeline({
 | 
			
		|||
        const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
 | 
			
		||||
 | 
			
		||||
        const timeJSX = (
 | 
			
		||||
          <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
          <Time
 | 
			
		||||
            ts={mEvent.getTs()}
 | 
			
		||||
            compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1365,6 +1407,7 @@ export function RoomTimeline({
 | 
			
		|||
            messageSpacing={messageSpacing}
 | 
			
		||||
            canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
          >
 | 
			
		||||
            <EventContent
 | 
			
		||||
              messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			@ -1388,7 +1431,12 @@ export function RoomTimeline({
 | 
			
		|||
        const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
 | 
			
		||||
 | 
			
		||||
        const timeJSX = (
 | 
			
		||||
          <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
          <Time
 | 
			
		||||
            ts={mEvent.getTs()}
 | 
			
		||||
            compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
            hour24Clock={hour24Clock}
 | 
			
		||||
            dateFormatString={dateFormatString}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1402,6 +1450,7 @@ export function RoomTimeline({
 | 
			
		|||
            messageSpacing={messageSpacing}
 | 
			
		||||
            canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
            hideReadReceipts={hideActivity}
 | 
			
		||||
            showDeveloperTools={showDeveloperTools}
 | 
			
		||||
          >
 | 
			
		||||
            <EventContent
 | 
			
		||||
              messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			@ -1427,7 +1476,12 @@ export function RoomTimeline({
 | 
			
		|||
      const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
 | 
			
		||||
 | 
			
		||||
      const timeJSX = (
 | 
			
		||||
        <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
        <Time
 | 
			
		||||
          ts={mEvent.getTs()}
 | 
			
		||||
          compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
          hour24Clock={hour24Clock}
 | 
			
		||||
          dateFormatString={dateFormatString}
 | 
			
		||||
        />
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1441,6 +1495,7 @@ export function RoomTimeline({
 | 
			
		|||
          messageSpacing={messageSpacing}
 | 
			
		||||
          canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
          hideReadReceipts={hideActivity}
 | 
			
		||||
          showDeveloperTools={showDeveloperTools}
 | 
			
		||||
        >
 | 
			
		||||
          <EventContent
 | 
			
		||||
            messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			@ -1471,7 +1526,12 @@ export function RoomTimeline({
 | 
			
		|||
      const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
 | 
			
		||||
 | 
			
		||||
      const timeJSX = (
 | 
			
		||||
        <Time ts={mEvent.getTs()} compact={messageLayout === MessageLayout.Compact} />
 | 
			
		||||
        <Time
 | 
			
		||||
          ts={mEvent.getTs()}
 | 
			
		||||
          compact={messageLayout === MessageLayout.Compact}
 | 
			
		||||
          hour24Clock={hour24Clock}
 | 
			
		||||
          dateFormatString={dateFormatString}
 | 
			
		||||
        />
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      return (
 | 
			
		||||
| 
						 | 
				
			
			@ -1485,6 +1545,7 @@ export function RoomTimeline({
 | 
			
		|||
          messageSpacing={messageSpacing}
 | 
			
		||||
          canDelete={canRedact || mEvent.getSender() === mx.getUserId()}
 | 
			
		||||
          hideReadReceipts={hideActivity}
 | 
			
		||||
          showDeveloperTools={showDeveloperTools}
 | 
			
		||||
        >
 | 
			
		||||
          <EventContent
 | 
			
		||||
            messageLayout={messageLayout}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,7 +5,7 @@ import { ReactEditor } from 'slate-react';
 | 
			
		|||
import { isKeyHotkey } from 'is-hotkey';
 | 
			
		||||
import { useStateEvent } from '../../hooks/useStateEvent';
 | 
			
		||||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { usePowerLevelsAPI, usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useEditor } from '../../components/editor';
 | 
			
		||||
import { RoomInputPlaceholder } from './RoomInputPlaceholder';
 | 
			
		||||
| 
						 | 
				
			
			@ -21,8 +21,8 @@ import { editableActiveElement } from '../../utils/dom';
 | 
			
		|||
import navigation from '../../../client/state/navigation';
 | 
			
		||||
import { settingsAtom } from '../../state/settings';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { useAccessibleTagColors, usePowerLevelTags } from '../../hooks/usePowerLevelTags';
 | 
			
		||||
import { useTheme } from '../../hooks/useTheme';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
 | 
			
		||||
const FN_KEYS_REGEX = /^F\d+$/;
 | 
			
		||||
const shouldFocusMessageField = (evt: KeyboardEvent): boolean => {
 | 
			
		||||
| 
						 | 
				
			
			@ -70,15 +70,10 @@ export function RoomView({ room, eventId }: { room: Room; eventId?: string }) {
 | 
			
		|||
 | 
			
		||||
  const tombstoneEvent = useStateEvent(room, StateEvent.RoomTombstone);
 | 
			
		||||
  const powerLevels = usePowerLevelsContext();
 | 
			
		||||
  const { getPowerLevel, canSendEvent } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const myUserId = mx.getUserId();
 | 
			
		||||
  const canMessage = myUserId
 | 
			
		||||
    ? canSendEvent(EventType.RoomMessage, getPowerLevel(myUserId))
 | 
			
		||||
    : false;
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const [powerLevelTags, getPowerLevelTag] = usePowerLevelTags(room, powerLevels);
 | 
			
		||||
  const theme = useTheme();
 | 
			
		||||
  const accessibleTagColors = useAccessibleTagColors(theme.kind, powerLevelTags);
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const canMessage = permissions.event(EventType.RoomMessage, mx.getSafeUserId());
 | 
			
		||||
 | 
			
		||||
  useKeyDown(
 | 
			
		||||
    window,
 | 
			
		||||
| 
						 | 
				
			
			@ -109,8 +104,6 @@ export function RoomView({ room, eventId }: { room: Room; eventId?: string }) {
 | 
			
		|||
          eventId={eventId}
 | 
			
		||||
          roomInputRef={roomInputRef}
 | 
			
		||||
          editor={editor}
 | 
			
		||||
          getPowerLevelTag={getPowerLevelTag}
 | 
			
		||||
          accessibleTagColors={accessibleTagColors}
 | 
			
		||||
        />
 | 
			
		||||
        <RoomViewTyping room={room} />
 | 
			
		||||
      </Box>
 | 
			
		||||
| 
						 | 
				
			
			@ -131,8 +124,6 @@ export function RoomView({ room, eventId }: { room: Room; eventId?: string }) {
 | 
			
		|||
                  roomId={roomId}
 | 
			
		||||
                  fileDropContainerRef={roomViewRef}
 | 
			
		||||
                  ref={roomInputRef}
 | 
			
		||||
                  getPowerLevelTag={getPowerLevelTag}
 | 
			
		||||
                  accessibleTagColors={accessibleTagColors}
 | 
			
		||||
                />
 | 
			
		||||
              )}
 | 
			
		||||
              {!canMessage && (
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,7 +34,7 @@ import { RoomTopicViewer } from '../../components/room-topic-viewer';
 | 
			
		|||
import { StateEvent } from '../../../types/matrix/room';
 | 
			
		||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
 | 
			
		||||
import { useRoom } from '../../hooks/useRoom';
 | 
			
		||||
import { useSetSetting, useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { useSetting } from '../../state/hooks/settings';
 | 
			
		||||
import { settingsAtom } from '../../state/settings';
 | 
			
		||||
import { useSpaceOptionally } from '../../hooks/useSpace';
 | 
			
		||||
import { getHomeSearchPath, getSpaceSearchPath, withSearchParam } from '../../pages/pathUtils';
 | 
			
		||||
| 
						 | 
				
			
			@ -42,7 +42,7 @@ import { getCanonicalAliasOrRoomId, isRoomAlias, mxcUrlToHttp } from '../../util
 | 
			
		|||
import { _SearchPathSearchParams } from '../../pages/paths';
 | 
			
		||||
import * as css from './RoomViewHeader.css';
 | 
			
		||||
import { useRoomUnread } from '../../state/hooks/unread';
 | 
			
		||||
import { usePowerLevelsAPI, usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { usePowerLevelsContext } from '../../hooks/usePowerLevels';
 | 
			
		||||
import { markAsRead } from '../../../client/action/notifications';
 | 
			
		||||
import { roomToUnreadAtom } from '../../state/room/roomToUnread';
 | 
			
		||||
import { openInviteUser } from '../../../client/action/navigation';
 | 
			
		||||
| 
						 | 
				
			
			@ -65,6 +65,10 @@ import {
 | 
			
		|||
  getRoomNotificationModeIcon,
 | 
			
		||||
  useRoomsNotificationPreferencesContext,
 | 
			
		||||
} from '../../hooks/useRoomsNotificationPreferences';
 | 
			
		||||
import { JumpToTime } from './jump-to-time';
 | 
			
		||||
import { useRoomNavigate } from '../../hooks/useRoomNavigate';
 | 
			
		||||
import { useRoomCreators } from '../../hooks/useRoomCreators';
 | 
			
		||||
import { useRoomPermissions } from '../../hooks/useRoomPermissions';
 | 
			
		||||
 | 
			
		||||
type RoomMenuProps = {
 | 
			
		||||
  room: Room;
 | 
			
		||||
| 
						 | 
				
			
			@ -75,10 +79,13 @@ const RoomMenu = forwardRef<HTMLDivElement, RoomMenuProps>(({ room, requestClose
 | 
			
		|||
  const [hideActivity] = useSetting(settingsAtom, 'hideActivity');
 | 
			
		||||
  const unread = useRoomUnread(room.roomId, roomToUnreadAtom);
 | 
			
		||||
  const powerLevels = usePowerLevelsContext();
 | 
			
		||||
  const { getPowerLevel, canDoAction } = usePowerLevelsAPI(powerLevels);
 | 
			
		||||
  const canInvite = canDoAction('invite', getPowerLevel(mx.getUserId() ?? ''));
 | 
			
		||||
  const creators = useRoomCreators(room);
 | 
			
		||||
 | 
			
		||||
  const permissions = useRoomPermissions(creators, powerLevels);
 | 
			
		||||
  const canInvite = permissions.action('invite', mx.getSafeUserId());
 | 
			
		||||
  const notificationPreferences = useRoomsNotificationPreferencesContext();
 | 
			
		||||
  const notificationMode = getRoomNotificationMode(notificationPreferences, room.roomId);
 | 
			
		||||
  const { navigateRoom } = useRoomNavigate();
 | 
			
		||||
 | 
			
		||||
  const handleMarkAsRead = () => {
 | 
			
		||||
    markAsRead(mx, room.roomId, hideActivity);
 | 
			
		||||
| 
						 | 
				
			
			@ -175,6 +182,33 @@ const RoomMenu = forwardRef<HTMLDivElement, RoomMenuProps>(({ room, requestClose
 | 
			
		|||
            Room Settings
 | 
			
		||||
          </Text>
 | 
			
		||||
        </MenuItem>
 | 
			
		||||
        <UseStateProvider initial={false}>
 | 
			
		||||
          {(promptJump, setPromptJump) => (
 | 
			
		||||
            <>
 | 
			
		||||
              <MenuItem
 | 
			
		||||
                onClick={() => setPromptJump(true)}
 | 
			
		||||
                size="300"
 | 
			
		||||
                after={<Icon size="100" src={Icons.RecentClock} />}
 | 
			
		||||
                radii="300"
 | 
			
		||||
                aria-pressed={promptJump}
 | 
			
		||||
              >
 | 
			
		||||
                <Text style={{ flexGrow: 1 }} as="span" size="T300" truncate>
 | 
			
		||||
                  Jump to Time
 | 
			
		||||
                </Text>
 | 
			
		||||
              </MenuItem>
 | 
			
		||||
              {promptJump && (
 | 
			
		||||
                <JumpToTime
 | 
			
		||||
                  onSubmit={(eventId) => {
 | 
			
		||||
                    setPromptJump(false);
 | 
			
		||||
                    navigateRoom(room.roomId, eventId);
 | 
			
		||||
                    requestClose();
 | 
			
		||||
                  }}
 | 
			
		||||
                  onCancel={() => setPromptJump(false)}
 | 
			
		||||
                />
 | 
			
		||||
              )}
 | 
			
		||||
            </>
 | 
			
		||||
          )}
 | 
			
		||||
        </UseStateProvider>
 | 
			
		||||
      </Box>
 | 
			
		||||
      <Line variant="Surface" size="300" />
 | 
			
		||||
      <Box direction="Column" gap="100" style={{ padding: config.space.S100 }}>
 | 
			
		||||
| 
						 | 
				
			
			@ -230,7 +264,7 @@ export function RoomViewHeader() {
 | 
			
		|||
    ? mxcUrlToHttp(mx, avatarMxc, useAuthentication, 96, 96, 'crop') ?? undefined
 | 
			
		||||
    : undefined;
 | 
			
		||||
 | 
			
		||||
  const setPeopleDrawer = useSetSetting(settingsAtom, 'isPeopleDrawer');
 | 
			
		||||
  const [peopleDrawer, setPeopleDrawer] = useSetting(settingsAtom, 'isPeopleDrawer');
 | 
			
		||||
 | 
			
		||||
  const handleSearchClick = () => {
 | 
			
		||||
    const searchParams: _SearchPathSearchParams = {
 | 
			
		||||
| 
						 | 
				
			
			@ -404,7 +438,7 @@ export function RoomViewHeader() {
 | 
			
		|||
              offset={4}
 | 
			
		||||
              tooltip={
 | 
			
		||||
                <Tooltip>
 | 
			
		||||
                  <Text>Members</Text>
 | 
			
		||||
                  <Text>{peopleDrawer ? 'Hide Members' : 'Show Members'}</Text>
 | 
			
		||||
                </Tooltip>
 | 
			
		||||
              }
 | 
			
		||||
            >
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue