Compare commits

...

111 Commits

Author SHA1 Message Date
Anselm
8b88ae25f1 Fix Highlight component 2025-03-25 11:55:30 +00:00
Anselm
f21089bd59 Fix darkmode diff colours 2025-03-25 11:43:54 +00:00
Anselm
4760b84828 Reintroduce Jazz colors 2025-03-25 11:41:02 +00:00
Anselm
e021068051 Typing in docs working 2025-03-24 12:55:21 +00:00
Anselm
16311977e9 First attempt to make twoslash work 2025-03-21 17:40:10 +00:00
Anselm
ab2970413e Upgrade shiki and use built-in transformers for diffs 2025-03-21 17:00:35 +00:00
Anselm
34ff647221 More details in deep loading 2025-03-21 12:41:44 +00:00
Anselm
ee01e2f059 Divide upgrade guide into breaking / new features 2025-03-21 10:58:35 +00:00
Anselm
525f24ffc1 Remove incorrect stuff 2025-03-21 10:50:27 +00:00
Anselm
82bb7733ff Clean up manual subscription & hooks 2025-03-21 10:48:32 +00:00
Anselm
e307ecc524 Include current version of history & time travel 2025-03-20 13:41:26 +00:00
Anselm
f0f43a8b85 "loading depth" -> "resolve query" 2025-03-20 13:34:30 +00:00
Anselm
77d2909999 Container-like -> collections 2025-03-20 13:25:52 +00:00
Trisha Lim
8c951fb08b move AI tools and Inspector in its own section 2025-03-20 13:21:36 +00:00
Trisha Lim
d5f43c5225 reduce spacing between docs nav items 2025-03-20 13:21:36 +00:00
Trisha Lim
4fe3d25adc remove "coming soon" legend on docs nav 2025-03-20 13:21:36 +00:00
Trisha Lim
bd84d87e94 install lucide-react on jazz-inspector 2025-03-20 13:21:36 +00:00
Trisha Lim
5e9b33fded remove covalue inspector form from dom if not shown 2025-03-20 13:21:35 +00:00
Trisha Lim
e2cbe62887 set input length 2025-03-20 13:21:35 +00:00
Trisha Lim
39734cb19f missing key 2025-03-20 13:21:35 +00:00
Trisha Lim
a1b1bab466 move ui components to separate dir 2025-03-20 13:21:35 +00:00
Trisha Lim
9383a0dd47 account selector UI improvement 2025-03-20 13:21:35 +00:00
Trisha Lim
1e7ca2d008 add changeset 2025-03-20 13:21:34 +00:00
Trisha Lim
65d1e94efc "add account" form improvement 2025-03-20 13:21:34 +00:00
Trisha Lim
fcd8e21bfb UI fixes 2025-03-20 13:21:34 +00:00
Trisha Lim
e0d171cf79 spacing and typography improvements 2025-03-20 13:21:34 +00:00
Trisha Lim
aa45457f44 show array contents in a co.json 2025-03-20 13:21:34 +00:00
James Vickery
7189a1ea17 fix lint 2025-03-20 13:21:33 +00:00
James Vickery
0ddf1c871e fix rss urls and set canonical url 2025-03-20 13:21:33 +00:00
James Vickery
abc74dd567 rss 2025-03-20 13:21:33 +00:00
Trisha Lim
69689289a1 update meta tags to align with new copy 2025-03-20 13:21:33 +00:00
Anselm Eickhoff
2e6ea42d21 bring back zero-config magic 2025-03-20 13:21:32 +00:00
Anselm Eickhoff
f1b8f4b5c0 more down-to-earth hero section 2025-03-20 13:21:32 +00:00
Trisha Lim
89f39406ba update homepage copy 2025-03-20 13:21:32 +00:00
Giordano Ricci
8cc79473db fix bar width on small screen to avoid horizontal scroll 2025-03-20 13:21:32 +00:00
Giordano Ricci
690d9e2587 fill missing data on the left 2025-03-20 13:21:31 +00:00
Giordano Ricci
04d1c18378 Fix status latency query, adjust probes 2025-03-20 13:21:31 +00:00
Trisha Lim
ac8081ff01 create-jazz-app: replace project name param with directory name (#1612) 2025-03-20 13:21:31 +00:00
Anselm Eickhoff
7339782684 Comment grammar fix 2025-03-20 13:21:31 +00:00
Guido D'Orsi
b8d64ba00a fix: align the profile migration in the passkey starter with the last best pratice 2025-03-20 13:21:31 +00:00
pax-k
71b2dc21a9 chore: changeset 2025-03-20 13:21:30 +00:00
pax-k
3f822c3cac fix(examples): removed when="singedUp" from examples apps' Jazz providers 2025-03-20 13:21:30 +00:00
Trisha Lim
789cdab504 set inspector widget height 2025-03-20 13:21:30 +00:00
Trisha Lim
22c3b2b92b fix button exports 2025-03-20 13:21:30 +00:00
Trisha Lim
cd14f30129 clean up 2025-03-20 13:21:29 +00:00
Trisha Lim
2423b3ff46 fit more data onscreen 2025-03-20 13:21:29 +00:00
Trisha Lim
91ada6fd2d set default cursor 2025-03-20 13:21:29 +00:00
Trisha Lim
2052c98094 use Input component for all inputs 2025-03-20 13:21:29 +00:00
Trisha Lim
daa8ddf890 use Button element for all buttons 2025-03-20 13:21:29 +00:00
Trisha Lim
760c90ccc0 remove unused packages and css 2025-03-20 13:21:28 +00:00
Trisha Lim
5003067f47 fix missing types when running dev 2025-03-20 13:21:28 +00:00
Trisha Lim
b36ef59aea fix account export to inspector app 2025-03-20 13:21:28 +00:00
Trisha Lim
2fe509774b reuse jazz-inspector components in inspector app 2025-03-20 13:21:28 +00:00
Trisha Lim
7e254cf9f5 reuse jazz-inspector components in inspector app 2025-03-20 13:21:27 +00:00
Benjamin S. Leveritt
dd25b5f926 Update filestreams.mdx 2025-03-20 13:21:27 +00:00
Benjamin S. Leveritt
1bb51398aa Adds CreateImg config, annotates other methods 2025-03-20 13:21:27 +00:00
Benjamin S. Leveritt
5deec8ddfd Adds React Native docs 2025-03-20 13:21:27 +00:00
Benjamin S. Leveritt
45ec2fabe3 Add Creating Images section to vanilla 2025-03-20 13:21:26 +00:00
Benjamin S. Leveritt
c7f9d9d263 Adds comment about knowness 2025-03-20 13:21:26 +00:00
Benjamin S. Leveritt
3285b85a90 Reword to synced 2025-03-20 13:21:26 +00:00
Benjamin S. Leveritt
64a88b7f54 Tweak presentation of FileStream 2025-03-20 13:21:26 +00:00
Benjamin S. Leveritt
dfc4b21fc8 Switches sections 2025-03-20 13:21:26 +00:00
Benjamin S. Leveritt
fdc86bcb9a Add link to Writing to FileStreams 2025-03-20 13:21:25 +00:00
Benjamin S. Leveritt
dca7edbf47 Add links to Vanilla docs 2025-03-20 13:21:25 +00:00
Benjamin S. Leveritt
56d8d87195 Add React docs 2025-03-20 13:21:25 +00:00
Benjamin S. Leveritt
b7731c8fca Adds revokeObjectURL references and best practices 2025-03-20 13:21:25 +00:00
Benjamin S. Leveritt
3e8afdafdc Fix note section 2025-03-20 13:21:24 +00:00
Benjamin S. Leveritt
b39dc35fae Adds example for fallback behaviour 2025-03-20 13:21:24 +00:00
Benjamin S. Leveritt
31d8831faa Adds imageDef docs 2025-03-20 13:21:24 +00:00
Benjamin S. Leveritt
673e194034 Removes mentions of images, pointing to ImageDef 2025-03-20 13:21:24 +00:00
Benjamin S. Leveritt
b579b32c69 Adds tests for imageDef 2025-03-20 13:21:23 +00:00
Benjamin S. Leveritt
c0e53dc8b8 Adds CodeGroups 2025-03-20 13:21:23 +00:00
Benjamin S. Leveritt
99ad5008eb Add FileStream progress test 2025-03-20 13:21:23 +00:00
Benjamin S. Leveritt
2c86de4176 Adds FileStream doc draft 2025-03-20 13:21:23 +00:00
Trisha Lim
ba3a0d4032 hide jazz-inspector if not dev 2025-03-20 13:21:23 +00:00
Trisha Lim
8f8d02e43e make dev generate app.js instead of jazz-inspector.js 2025-03-20 13:21:22 +00:00
Trisha Lim
2118c46efe add basic json viewer 2025-03-20 13:21:22 +00:00
Trisha Lim
0073ea4950 add covalue search to nav 2025-03-20 13:21:22 +00:00
Trisha Lim
744fcf53b2 bare minimum dark styles 2025-03-20 13:21:22 +00:00
Trisha Lim
37103fdff9 add inspector to chat app 2025-03-20 13:21:21 +00:00
Trisha Lim
7345bc6ca4 replace indigo with blue 2025-03-20 13:21:21 +00:00
Trisha Lim
042b407f2d remove javascript hover styles 2025-03-20 13:21:21 +00:00
Trisha Lim
ddcf91a014 match colors to jazz brand 2025-03-20 13:21:21 +00:00
Trisha Lim
366e1b1cc1 add button label 2025-03-20 13:21:21 +00:00
Trisha Lim
d8edad8041 rewrite styles to tailwind 2025-03-20 13:21:20 +00:00
Trisha Lim
20d823d6fd install twind 2025-03-20 13:21:20 +00:00
Trisha Lim
efb08caeae clean up 2025-03-20 13:21:20 +00:00
Trisha Lim
1a0e37eaed TOC fixes 2025-03-20 13:21:20 +00:00
Trisha Lim
86bc03ac4a fix coming soon TOC 2025-03-20 13:21:19 +00:00
Trisha Lim
748c7160ba set TOC items in context 2025-03-20 13:21:19 +00:00
Trisha Lim
c77c617aaa move docs layout 2025-03-20 13:21:19 +00:00
github-actions[bot]
9fb4961215 Version Packages 2025-03-20 13:21:19 +00:00
Guido D'Orsi
513ebd8fb9 fix: downgrade the WasmCrypto initialization error logging to a warning 2025-03-20 13:21:18 +00:00
github-actions[bot]
6b22fe9a93 Version Packages 2025-03-20 13:21:18 +00:00
Guido D'Orsi
3076a90a51 fix: use PureJSCrypto as fallback when WasmCrypto fails to initialize 2025-03-20 13:21:18 +00:00
Guido D'Orsi
7e6316b930 chore: move the cloudflare test to use WASM 2025-03-20 13:21:17 +00:00
Trisha Lim
f7cdfd5767 add changeset 2025-03-20 13:21:16 +00:00
Benjamin S. Leveritt
ca6bb1c006 Change to markdown + format
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 13:15:32 +00:00
Benjamin S. Leveritt
ba9f7eb487 Fix ‘for example’
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 12:18:12 +00:00
Anselm
3523ebf2bf Missing green line 2025-03-20 11:47:59 +00:00
Anselm
9131894f40 Upgrade guide improvments 2025-03-20 11:39:36 +00:00
Benjamin S. Leveritt
618be5d016 Fix date
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 09:45:29 +00:00
Benjamin S. Leveritt
8a18f30ba5 Update CoMap.Record resolve example
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 09:43:09 +00:00
Benjamin S. Leveritt
3128dda432 Remove auto-loading section
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 09:38:03 +00:00
Benjamin S. Leveritt
517662635e Add Vanilla examples
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-20 09:31:55 +00:00
Benjamin S. Leveritt
c119a613a7 Tweak copy
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-19 10:33:06 +00:00
Benjamin S. Leveritt
ddd4bc63c3 Fix permissions in lists comment
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-19 09:51:53 +00:00
Benjamin S. Leveritt
e802cabd0f Move Framework detail to just under the fold
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-19 09:49:27 +00:00
Benjamin S. Leveritt
dfabf2d295 Update Subs and Loading docs
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-18 10:20:58 +00:00
Benjamin S. Leveritt
834607f8bc Update subs doc
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-18 10:20:53 +00:00
Benjamin S. Leveritt
bda61d8440 Fix version and date
Signed-off-by: Benjamin S. Leveritt <benjamin@leveritt.co.uk>
2025-03-18 10:20:37 +00:00
218 changed files with 4384 additions and 3542 deletions

View File

@@ -0,0 +1,18 @@
---
"jazz-tailwind-demo-auth-starter": patch
"file-share-svelte": patch
"jazz-password-manager": patch
"version-history": patch
"passkey-svelte": patch
"chat-rn-clerk": patch
"jazz-example-music-player": patch
"passphrase": patch
"multiauth": patch
"reactions": patch
"passkey": patch
"clerk": patch
"jazz-example-pets": patch
"jazz-example-todo": patch
---
Removed when="singedUp" from examples apps' Jazz providers. This is a really niche use-case option and can lead to broken-feeling experiences when anonymous users try to load something.

View File

@@ -1,5 +0,0 @@
---
"jazz-inspector": patch
---
add z-index of 1 to inspector

View File

@@ -0,0 +1,5 @@
---
"create-jazz-app": patch
---
add directory param to create-jazz-app

View File

@@ -0,0 +1,6 @@
---
"jazz-inspector": patch
"jazz-inspector-app": patch
---
UI and JSON display improvements

View File

@@ -1,5 +0,0 @@
---
"jazz-tools": patch
---
Throw when assigning invalid values to ref fields

View File

@@ -1,5 +0,0 @@
---
"jazz-tools": patch
---
Use RegisteredAccount types for `by` props

View File

@@ -1,5 +0,0 @@
---
"jazz-example-chat": patch
---
In the chat app example, show a "Message not readable" placeholder, if messages from a chat list are not readable by the user.

View File

@@ -1,5 +0,0 @@
---
"jazz-tools": patch
---
Fixes coList.splice to handle insertions at start of list

View File

@@ -1,5 +1,26 @@
# chat-rn-clerk
## 1.0.86
### Patch Changes
- jazz-react-native@0.11.5
- jazz-react-native-auth-clerk@0.11.5
- jazz-tools@0.11.5
- jazz-react-native-media-images@0.11.5
## 1.0.85
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react-native@0.11.4
- jazz-react-native-auth-clerk@0.11.4
- jazz-react-native-media-images@0.11.4
## 1.0.84
### Patch Changes

View File

@@ -1,7 +1,7 @@
{
"name": "chat-rn-clerk",
"main": "index.js",
"version": "1.0.84",
"version": "1.0.86",
"scripts": {
"build": "expo export -p ios",
"start": "expo start",

View File

@@ -12,7 +12,6 @@ export function JazzAndAuth({ children }: PropsWithChildren) {
storage="sqlite"
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp", // This makes the app work in local mode when the user is not authenticated
}}
>
{children}

View File

@@ -1,5 +1,22 @@
# chat-rn
## 1.0.82
### Patch Changes
- jazz-react-native@0.11.5
- jazz-tools@0.11.5
## 1.0.81
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react-native@0.11.4
## 1.0.80
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "chat-rn",
"version": "1.0.80",
"version": "1.0.82",
"main": "index.js",
"scripts": {
"build": "expo export -p ios",

View File

@@ -1,5 +1,24 @@
# chat-vue
## 0.0.67
### Patch Changes
- jazz-browser@0.11.5
- jazz-tools@0.11.5
- jazz-vue@0.11.5
## 0.0.66
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser@0.11.4
- jazz-vue@0.11.4
## 0.0.65
### Patch Changes

View File

@@ -11,12 +11,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example chat-vue --project-name chat-vue
npx create-jazz-app@latest chat-vue-app --example chat-vue
```
Go to the new project directory.
```bash
cd chat-vue
cd chat-vue-app
```
Run the dev server.

View File

@@ -1,6 +1,6 @@
{
"name": "chat-vue",
"version": "0.0.65",
"version": "0.0.67",
"private": true,
"type": "module",
"scripts": {

View File

@@ -1,5 +1,25 @@
# jazz-example-chat
## 0.0.164
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
- jazz-browser-media-images@0.11.5
## 0.0.163
### Patch Changes
- 2074e45: In the chat app example, show a "Message not readable" placeholder, if messages from a chat list are not readable by the user.
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser-media-images@0.11.4
- jazz-react@0.11.4
## 0.0.162
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example chat --project-name chat
npx create-jazz-app@latest chat-app --example chat
```
Go to the new project directory.
```bash
cd chat
cd chat-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-example-chat",
"private": true,
"version": "0.0.162",
"version": "0.0.164",
"type": "module",
"scripts": {
"dev": "vite",
@@ -16,6 +16,7 @@
"clsx": "^2.0.0",
"hash-slash": "workspace:*",
"jazz-browser-media-images": "workspace:*",
"jazz-inspector": "workspace:*",
"jazz-react": "workspace:*",
"jazz-tools": "workspace:*",
"lucide-react": "^0.274.0",

View File

@@ -1,6 +1,7 @@
import { apiKey } from "@/apiKey.ts";
import { getRandomUsername, inIframe, onChatLoad } from "@/util.ts";
import { useIframeHashRouter } from "hash-slash";
import { JazzInspector } from "jazz-inspector";
import { JazzProvider, useAccount } from "jazz-react";
import { Group, ID } from "jazz-tools";
import { StrictMode } from "react";
@@ -61,6 +62,7 @@ createRoot(document.getElementById("root")!).render(
defaultProfileName={defaultProfileName}
>
<App />
<JazzInspector />
</JazzProvider>
</StrictMode>
</ThemeProvider>,

View File

@@ -1,5 +1,24 @@
# minimal-auth-clerk
## 0.0.63
### Patch Changes
- jazz-react@0.11.5
- jazz-react-auth-clerk@0.11.5
- jazz-tools@0.11.5
## 0.0.62
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
- jazz-react-auth-clerk@0.11.4
## 0.0.61
### Patch Changes

View File

@@ -15,12 +15,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example clerk --project-name clerk
npx create-jazz-app@latest clerk-app --example clerk
```
Go to the new project directory.
```bash
cd clerk
cd clerk-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "clerk",
"private": true,
"version": "0.0.61",
"version": "0.0.63",
"type": "module",
"scripts": {
"dev": "vite",
@@ -13,7 +13,7 @@
"dependencies": {
"@clerk/clerk-react": "^5.4.1",
"jazz-react": "workspace:*",
"jazz-react-auth-clerk": "workspace:0.11.3",
"jazz-react-auth-clerk": "workspace:0.11.5",
"jazz-tools": "workspace:*",
"react": "^18.3.1",
"react-dom": "^18.3.1"

View File

@@ -21,7 +21,6 @@ function JazzProvider({ children }: { children: React.ReactNode }) {
clerk={clerk}
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp", // This makes the app work in local mode when the user is not authenticated
}}
>
{children}

View File

@@ -1,5 +1,22 @@
# file-share-svelte
## 0.0.47
### Patch Changes
- jazz-svelte@0.11.5
- jazz-tools@0.11.5
## 0.0.46
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-svelte@0.11.4
## 0.0.45
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "file-share-svelte",
"version": "0.0.45",
"version": "0.0.47",
"private": true,
"type": "module",
"scripts": {

View File

@@ -27,7 +27,6 @@
AccountSchema={FileShareAccount}
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
>
<PasskeyAuthBasicUI appName="File Share">

View File

@@ -1,5 +1,22 @@
# jazz-tailwind-demo-auth-starter
## 0.0.3
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.2
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.52
### Patch Changes

View File

@@ -1,7 +1,7 @@
{
"name": "filestream",
"private": true,
"version": "0.0.1",
"version": "0.0.3",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -1,5 +1,24 @@
# form
## 0.1.5
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
- jazz-browser-media-images@0.11.5
## 0.1.4
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser-media-images@0.11.4
- jazz-react@0.11.4
## 0.1.3
### Patch Changes

View File

@@ -28,12 +28,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example form --project-name form
npx create-jazz-app@latest form-app --example form
```
Go to the new project directory.
```bash
cd form
cd form-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "form",
"private": true,
"version": "0.1.3",
"version": "0.1.5",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -1,5 +1,24 @@
# image-upload
## 0.0.61
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
- jazz-browser-media-images@0.11.5
## 0.0.60
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser-media-images@0.11.4
- jazz-react@0.11.4
## 0.0.59
### Patch Changes

View File

@@ -15,12 +15,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example image-upload --project-name image-upload
npx create-jazz-app@latest image-upload-app --example image-upload
```
Go to the new project directory.
```bash
cd image-upload
cd image-upload-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "image-upload",
"private": true,
"version": "0.0.59",
"version": "0.0.61",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -1,5 +1,21 @@
# jazz-example-inspector
## 0.0.114
### Patch Changes
- Updated dependencies [60f5b3f]
- cojson@0.11.5
- cojson-transport-ws@0.11.5
## 0.0.113
### Patch Changes
- Updated dependencies [7f036c1]
- cojson@0.11.4
- cojson-transport-ws@0.11.4
## 0.0.112
### Patch Changes

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-inspector-app",
"private": true,
"version": "0.0.112",
"version": "0.0.114",
"type": "module",
"scripts": {
"dev": "vite",
@@ -11,27 +11,19 @@
"preview": "vite preview"
},
"dependencies": {
"@radix-ui/react-checkbox": "^1.0.4",
"@radix-ui/react-slot": "^1.1.1",
"@radix-ui/react-toast": "^1.1.4",
"class-variance-authority": "^0.7.0",
"jazz-inspector": "workspace:*",
"clsx": "^2.0.0",
"cojson": "workspace:0.11.3",
"cojson-transport-ws": "workspace:0.11.3",
"cojson": "workspace:0.11.5",
"cojson-transport-ws": "workspace:0.11.5",
"hash-slash": "workspace:0.2.2",
"lucide-react": "^0.274.0",
"qrcode": "^1.5.3",
"react": "^18.3.1",
"react-dom": "^18.3.1",
"react-router": "^6.16.0",
"react-router-dom": "^6.16.0",
"react-use": "^17.4.0",
"tailwind-merge": "^1.14.0",
"tailwindcss-animate": "^1.0.7",
"uniqolor": "^1.1.0"
"react-use": "^17.4.0"
},
"devDependencies": {
"@types/qrcode": "^1.5.1",
"@types/react": "^18.3.12",
"@types/react-dom": "^18.3.1",
"@vitejs/plugin-react-swc": "^3.3.2",

View File

@@ -1,92 +1,3 @@
@tailwind base;
@tailwind components;
@tailwind utilities;
@layer base {
:root {
--background: 0 0% 100%;
--foreground: 20 14.3% 4.1%;
--card: 0 0% 100%;
--card-foreground: 20 14.3% 4.1%;
--popover: 0 0% 100%;
--popover-foreground: 20 14.3% 4.1%;
--primary: 24 9.8% 10%;
--primary-foreground: 60 9.1% 97.8%;
--secondary: 60 4.8% 95.9%;
--secondary-foreground: 24 9.8% 10%;
--muted: 60 4.8% 95.9%;
--muted-foreground: 25 5.3% 44.7%;
--accent: 60 4.8% 95.9%;
--accent-foreground: 24 9.8% 10%;
--destructive: 0 84.2% 60.2%;
--destructive-foreground: 60 9.1% 97.8%;
--border: 20 5.9% 90%;
--input: 20 5.9% 90%;
--ring: 20 14.3% 4.1%;
--radius: 0.5rem;
}
.dark {
--background: 20 14.3% 4.1%;
--foreground: 60 9.1% 97.8%;
--card: 20 14.3% 4.1%;
--card-foreground: 60 9.1% 97.8%;
--popover: 20 14.3% 4.1%;
--popover-foreground: 60 9.1% 97.8%;
--primary: 60 9.1% 97.8%;
--primary-foreground: 24 9.8% 10%;
--secondary: 12 6.5% 15.1%;
--secondary-foreground: 60 9.1% 97.8%;
--muted: 12 6.5% 15.1%;
--muted-foreground: 24 5.4% 63.9%;
--accent: 12 6.5% 15.1%;
--accent-foreground: 60 9.1% 97.8%;
--destructive: 0 62.8% 30.6%;
--destructive-foreground: 60 9.1% 97.8%;
--border: 12 6.5% 15.1%;
--input: 12 6.5% 15.1%;
--ring: 24 5.7% 82.9%;
}
}
@layer base {
* {
@apply border-border;
}
body {
@apply bg-background text-foreground;
margin: 0;
padding: 0;
}
}
.animate-in {
animation: slideIn 0.3s ease-out;
}
@keyframes slideIn {
from {
transform: translateZ(400px) translateY(30px) scale(1.05);
opacity: 0.4;
}
to {
transform: translateZ(0) scale(1);
opacity: 1;
}
}

View File

@@ -1,18 +0,0 @@
export function LinkIcon() {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
strokeWidth={1.5}
stroke="currentColor"
className="w-3 h-3"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M13.19 8.688a4.5 4.5 0 0 1 1.242 7.244l-4.5 4.5a4.5 4.5 0 0 1-6.364-6.364l1.757-1.757m13.35-.622 1.757-1.757a4.5 4.5 0 0 0-6.364-6.364l-4.5 4.5a4.5 4.5 0 0 0 1.242 7.244"
/>
</svg>
);
}

View File

@@ -1,39 +0,0 @@
import React from "react";
import { PageInfo } from "./types";
interface BreadcrumbsProps {
path: PageInfo[];
onBreadcrumbClick: (index: number) => void;
}
export const Breadcrumbs: React.FC<BreadcrumbsProps> = ({
path,
onBreadcrumbClick,
}) => {
return (
<div className="z-20 relative bg-indigo-400/10 backdrop-blur-sm rounded-lg inline-flex px-2 py-1 whitespace-pre transition-all items-center space-x-1 min-h-10">
<button
onClick={() => onBreadcrumbClick(-1)}
className="flex items-center justify-center p-1 rounded-sm hover:bg-indigo-500/10 transition-colors"
aria-label="Go to home"
>
<img src="jazz-logo.png" alt="Jazz Logo" className="size-5" />
</button>
{path.map((page, index) => {
return (
<span key={index} className="inline-block first:pl-1 last:pr-1">
{index === 0 ? null : (
<span className="text-indigo-500/30">{" / "}</span>
)}
<button
onClick={() => onBreadcrumbClick(index)}
className="text-indigo-700 hover:underline"
>
{index === 0 ? page.name || "Root" : page.name}
</button>
</span>
);
})}
</div>
);
};

View File

@@ -1,344 +0,0 @@
import {
CoID,
LocalNode,
RawBinaryCoStream,
RawCoStream,
RawCoValue,
} from "cojson";
import { base64URLtoBytes } from "cojson";
import { BinaryStreamItem, BinaryStreamStart, CoStreamItem } from "cojson";
import { JsonObject, JsonValue } from "cojson";
import { ArrowDownToLine } from "lucide-react";
import { useEffect, useState } from "react";
import { PageInfo } from "./types";
import { AccountOrGroupPreview } from "./value-renderer";
// typeguard for BinaryStreamStart
function isBinaryStreamStart(item: unknown): item is BinaryStreamStart {
return (
typeof item === "object" &&
item !== null &&
"type" in item &&
item.type === "start"
);
}
function detectCoStreamType(value: RawCoStream | RawBinaryCoStream) {
const firstKey = Object.keys(value.items)[0];
if (!firstKey)
return {
type: "unknown",
};
const items = value.items[firstKey as never]?.map((v) => v.value);
if (!items)
return {
type: "unknown",
};
const firstItem = items[0];
if (!firstItem)
return {
type: "unknown",
};
// This is a binary stream
if (isBinaryStreamStart(firstItem)) {
return {
type: "binary",
items: items as BinaryStreamItem[],
};
} else {
return {
type: "coStream",
};
}
}
async function getBlobFromCoStream({
items,
onlyFirstChunk = false,
}: {
items: BinaryStreamItem[];
onlyFirstChunk?: boolean;
}) {
if (onlyFirstChunk && items.length > 1) {
items = items.slice(0, 2);
}
const chunks: Uint8Array[] = [];
const binary_U_prefixLength = 8;
let lastProgressUpdate = Date.now();
for (const item of items.slice(1)) {
if (item.type === "end") {
break;
}
if (item.type !== "chunk") {
console.error("Invalid binary stream chunk", item);
return undefined;
}
const chunk = base64URLtoBytes(item.chunk.slice(binary_U_prefixLength));
// totalLength += chunk.length;
chunks.push(chunk);
if (Date.now() - lastProgressUpdate > 100) {
lastProgressUpdate = Date.now();
}
}
const defaultMime = "mimeType" in items[0] ? items[0].mimeType : null;
const blob = new Blob(chunks, defaultMime ? { type: defaultMime } : {});
const mimeType =
defaultMime === "" ? await detectPDFMimeType(blob) : defaultMime;
return {
blob,
mimeType: mimeType as string,
unfinishedChunks: items.length > 1,
totalSize:
"totalSizeBytes" in items[0]
? (items[0].totalSizeBytes as number)
: undefined,
};
}
const detectPDFMimeType = async (blob: Blob): Promise<string> => {
const arrayBuffer = await blob.slice(0, 4).arrayBuffer();
const uint8Array = new Uint8Array(arrayBuffer);
const header = uint8Array.reduce(
(acc, byte) => acc + String.fromCharCode(byte),
"",
);
if (header === "%PDF") {
return "application/pdf";
}
return "application/octet-stream";
};
const BinaryDownloadButton = ({
pdfBlob,
fileName = "document",
label,
mimeType,
}: {
pdfBlob: Blob;
mimeType?: string;
fileName?: string;
label: string;
}) => {
const downloadFile = () => {
const url = URL.createObjectURL(
new Blob([pdfBlob], mimeType ? { type: mimeType } : {}),
);
const link = document.createElement("a");
link.href = url;
link.download =
mimeType === "application/pdf" ? `${fileName}.pdf` : fileName;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
URL.revokeObjectURL(url);
};
return (
<button
className="flex items-center gap-2 px-2 py-1 text-gray-900 border border-gray-900/10 bg-clip-border shadow-sm transition-colors rounded bg-gray-50 text-sm"
onClick={downloadFile}
>
<ArrowDownToLine size={16} />
{label}
{/* Download {mimeType === "application/pdf" ? "PDF" : "File"} */}
</button>
);
};
const LabelContentPair = ({
label,
content,
}: {
label: string;
content: React.ReactNode;
}) => {
return (
<div className="flex flex-col gap-1.5 ">
<span className="uppercase text-xs font-medium text-gray-600 tracking-wide">
{label}
</span>
<span>{content}</span>
</div>
);
};
function RenderCoBinaryStream({
value,
items,
}: {
items: BinaryStreamItem[];
value: RawBinaryCoStream;
}) {
const [file, setFile] = useState<
| {
blob: Blob;
mimeType: string;
unfinishedChunks: boolean;
totalSize: number | undefined;
}
| undefined
| null
>(null);
const [isLoading, setIsLoading] = useState(true);
useEffect(() => {
// load only the first chunk to get the mime type and size
getBlobFromCoStream({
items,
onlyFirstChunk: true,
})
.then((v) => {
if (v) {
setFile(v);
if (v.mimeType.includes("image")) {
// If it's an image, load the full blob
getBlobFromCoStream({
items,
}).then((s) => {
if (s) setFile(s);
});
}
}
})
.finally(() => setIsLoading(false));
}, [items]);
if (!isLoading && !file) return <div>No blob</div>;
if (isLoading) return <div>Loading...</div>;
if (!file) return <div>No blob</div>;
const { blob, mimeType } = file;
const sizeInKB = (file.totalSize || 0) / 1024;
return (
<div className="space-y-8 mt-4">
<div className="grid grid-cols-3 gap-2 max-w-3xl">
<LabelContentPair
label="Mime Type"
content={
<span className="font-mono bg-gray-100 rounded px-2 py-1 text-sm">
{mimeType || "No mime type"}
</span>
}
/>
<LabelContentPair
label="Size"
content={<span>{sizeInKB.toFixed(2)} KB</span>}
/>
<LabelContentPair
label="Download"
content={
<BinaryDownloadButton
fileName={value.id.toString()}
pdfBlob={blob}
mimeType={mimeType}
label={
mimeType === "application/pdf"
? "Download PDF"
: "Download File"
}
/>
}
/>
</div>
{mimeType === "image/png" || mimeType === "image/jpeg" ? (
<LabelContentPair
label="Preview"
content={
<div className="bg-gray-50 p-3 rounded-sm">
<RenderBlobImage blob={blob} />
</div>
}
/>
) : null}
</div>
);
}
function RenderCoStream({
value,
node,
}: {
value: RawCoStream;
node: LocalNode;
}) {
const streamPerUser = Object.keys(value.items);
const userCoIds = streamPerUser.map((stream) => stream.split("_session")[0]);
return (
<div className="grid grid-cols-3 gap-2">
{userCoIds.map((id, idx) => (
<div
className="bg-gray-100 p-3 rounded-lg transition-colors overflow-hidden bg-white border hover:bg-gray-100/5 cursor-pointer shadow-sm"
key={id}
>
<AccountOrGroupPreview coId={id as CoID<RawCoValue>} node={node} />
{/* @ts-expect-error - TODO: fix types */}
{value.items[streamPerUser[idx]]?.map(
(item: CoStreamItem<JsonValue>) => (
<div>
{new Date(item.madeAt).toLocaleString()}{" "}
{JSON.stringify(item.value)}
</div>
),
)}
</div>
))}
</div>
);
}
export function CoStreamView({
value,
node,
}: {
data: JsonObject;
onNavigate: (pages: PageInfo[]) => void;
node: LocalNode;
value: RawCoStream;
}) {
// if (!value) return <div>No value</div>;
const streamType = detectCoStreamType(value);
if (streamType.type === "binary") {
if (streamType.items === undefined) {
return <div>No binary stream</div>;
}
return (
<RenderCoBinaryStream
value={value as RawBinaryCoStream}
items={streamType.items}
/>
);
}
if (streamType.type === "coStream") {
return <RenderCoStream value={value} node={node} />;
}
if (streamType.type === "unknown") return <div>Unknown stream type</div>;
return <div>Unknown stream type</div>;
}
function RenderBlobImage({ blob }: { blob: Blob }) {
const urlCreator = window.URL || window.webkitURL;
return <img src={urlCreator.createObjectURL(blob)} />;
}

View File

@@ -1,65 +0,0 @@
import clsx from "clsx";
import { CoID, LocalNode, RawCoValue } from "cojson";
import { JsonObject } from "cojson";
import { ResolveIcon } from "./type-icon";
import { PageInfo, isCoId } from "./types";
import { CoMapPreview, ValueRenderer } from "./value-renderer";
export function GridView({
data,
onNavigate,
node,
}: {
data: JsonObject;
onNavigate: (pages: PageInfo[]) => void;
node: LocalNode;
}) {
const entries = Object.entries(data);
return (
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 p-2">
{entries.map(([key, child], childIndex) => (
<div
key={childIndex}
className={clsx(
"bg-gray-100 p-3 rounded-lg transition-colors overflow-hidden",
isCoId(child)
? "bg-white border hover:bg-gray-100/5 cursor-pointer shadow-sm"
: "bg-gray-50",
)}
onClick={() =>
isCoId(child) &&
onNavigate([{ coId: child as CoID<RawCoValue>, name: key }])
}
>
<h3 className="truncate">
{isCoId(child) ? (
<span className="font-medium flex justify-between">
{key}
<div className="px-2 py-1 text-xs bg-gray-100 rounded">
<ResolveIcon coId={child as CoID<RawCoValue>} node={node} />
</div>
</span>
) : (
<span>{key}</span>
)}
</h3>
<div className="mt-2 text-sm">
{isCoId(child) ? (
<CoMapPreview coId={child as CoID<RawCoValue>} node={node} />
) : (
<ValueRenderer
json={child}
onCoIDClick={(coId) => {
onNavigate([{ coId, name: key }]);
}}
compact
/>
)}
</div>
</div>
))}
</div>
);
}

View File

@@ -1,7 +1,6 @@
import { LocalNode } from "cojson";
import { Breadcrumbs } from "./breadcrumbs";
import { PageStack } from "./page-stack";
import { PageInfo } from "./types";
import { Breadcrumbs, PageStack } from "jazz-inspector";
import type { PageInfo } from "jazz-inspector";
import { usePagePath } from "./use-page-path";
export default function CoJsonViewer({

View File

@@ -9,10 +9,15 @@ import {
} from "cojson";
import { createWebSocketPeer } from "cojson-transport-ws";
import { WasmCrypto } from "cojson/crypto/WasmCrypto";
import { Trash2 } from "lucide-react";
import {
Breadcrumbs,
Button,
Icon,
Input,
PageStack,
Select,
} from "jazz-inspector";
import React, { useState, useEffect } from "react";
import { Breadcrumbs } from "./breadcrumbs";
import { PageStack } from "./page-stack";
import { usePagePath } from "./use-page-path";
import { resolveCoValue, useResolvedCoValue } from "./use-resolve-covalue";
@@ -121,15 +126,23 @@ export default function CoJsonViewerApp() {
}
return (
<div className="w-full h-screen bg-gray-100 p-4 overflow-hidden flex flex-col">
<div className="flex items-center mb-4 gap-4">
<div
className={clsx(
"h-screen overflow-hidden flex flex-col",
" text-stone-700 bg-white",
"dark:text-stone-300 dark:bg-stone-950",
)}
>
<header className="flex items-center gap-4 p-3">
<Breadcrumbs path={path} onBreadcrumbClick={goToIndex} />
<div className="flex-1">
<form onSubmit={handleCoValueIdSubmit}>
{path.length !== 0 && (
<input
className="border p-2 rounded-lg min-w-[21rem] font-mono"
<Input
className="min-w-[21rem] font-mono"
placeholder="co_z1234567890abcdef123456789"
label="CoValue ID"
hideLabel
value={coValueId}
onChange={(e) =>
setCoValueId(e.target.value as CoID<RawCoValue>)
@@ -145,7 +158,7 @@ export default function CoJsonViewerApp() {
deleteCurrentAccount={deleteCurrentAccount}
localNode={localNode}
/>
</div>
</header>
<PageStack
path={path}
@@ -153,49 +166,39 @@ export default function CoJsonViewerApp() {
goBack={goBack}
addPages={addPages}
>
{!currentAccount ? (
<AddAccountForm addAccount={addAccount} />
) : (
{!currentAccount && <AddAccountForm addAccount={addAccount} />}
{currentAccount && path.length <= 0 && (
<form
onSubmit={handleCoValueIdSubmit}
aria-hidden={path.length !== 0}
className={clsx(
"flex flex-col justify-center items-center gap-2 h-full w-full mb-20 ",
"transition-all duration-150",
path.length > 0
? "opacity-0 -translate-y-2 scale-95"
: "opacity-100",
)}
className="flex flex-col relative -top-6 justify-center gap-2 h-full w-full max-w-sm mx-auto"
>
<fieldset className="flex flex-col gap-2 text-sm">
<h2 className="text-3xl font-medium text-gray-950 text-center mb-4">
Jazz CoValue Inspector
</h2>
<input
className="border p-4 rounded-lg min-w-[21rem] font-mono"
placeholder="co_z1234567890abcdef123456789"
value={coValueId}
onChange={(e) =>
setCoValueId(e.target.value as CoID<RawCoValue>)
}
/>
<button
type="submit"
className="bg-indigo-500 hover:bg-indigo-500/80 text-white px-4 py-2 rounded-md"
>
Inspect
</button>
<hr />
<button
type="button"
className="border inline-block px-2 py-1.5 text-black rounded"
onClick={() => {
setPage(currentAccount.id);
}}
>
Inspect My Account
</button>
</fieldset>
<h1 className="text-lg text-center font-medium mb-4 text-stone-900 dark:text-white">
Jazz CoValue Inspector
</h1>
<Input
label="CoValue ID"
className="font-mono"
hideLabel
placeholder="co_z1234567890abcdef123456789"
value={coValueId}
onChange={(e) => setCoValueId(e.target.value as CoID<RawCoValue>)}
/>
<Button type="submit" variant="primary">
Inspect CoValue
</Button>
<p className="text-center">or</p>
<Button
variant="secondary"
onClick={() => {
setPage(currentAccount.id);
}}
>
Inspect my account
</Button>
</form>
)}
</PageStack>
@@ -217,8 +220,10 @@ function AccountSwitcher({
localNode: LocalNode | null;
}) {
return (
<div className="relative flex items-center gap-1">
<select
<div className="relative flex items-stretch gap-1">
<Select
label="Account to inspect"
className="label:sr-only max-w-96"
value={currentAccount?.id || "add-account"}
onChange={(e) => {
if (e.target.value === "add-account") {
@@ -228,7 +233,6 @@ function AccountSwitcher({
setCurrentAccount(account || null);
}
}}
className="p-2 px-4 bg-gray-100/50 border border-indigo-500/10 backdrop-blur-sm rounded-md text-indigo-700 appearance-none"
>
{accounts.map((account) => (
<option key={account.id} value={account.id}>
@@ -240,15 +244,16 @@ function AccountSwitcher({
</option>
))}
<option value="add-account">Add account</option>
</select>
</Select>
{currentAccount && (
<button
<Button
variant="secondary"
onClick={deleteCurrentAccount}
className="p-3 rounded hover:bg-gray-200 transition-colors"
title="Delete Account"
className="rounded-md p-2 ml-1"
aria-label="Remove account"
>
<Trash2 size={16} className="text-gray-500" />
</button>
<Icon name="delete" className="text-gray-500" />
</Button>
)}
</div>
);
@@ -272,30 +277,34 @@ function AddAccountForm({
return (
<form
onSubmit={handleSubmit}
className="flex flex-col gap-2 max-w-md mx-auto h-full justify-center"
className="flex flex-col gap-3 max-w-md mx-auto h-full justify-center"
>
<h2 className="text-2xl font-medium text-gray-900 mb-3">
Add an Account to Inspect
<h2 className="text-2xl font-medium text-gray-900 dark:text-white">
Add an account to inspect
</h2>
<input
className="border py-2 px-3 rounded-md"
placeholder="Account ID"
<p className="leading-relaxed mb-5">
Use the{" "}
<code className="whitespace-nowrap text-stone-900 dark:text-white font-semibold">
jazz-logged-in-secret
</code>{" "}
local storage key from within your Jazz app for your account
credentials.
</p>
<Input
label="Account ID"
value={id}
placeholder="co_z1234567890abcdef123456789"
onChange={(e) => setId(e.target.value)}
/>
<input
<Input
label="Account secret"
type="password"
className="border py-2 px-3 rounded-md"
placeholder="Account Secret"
value={secret}
onChange={(e) => setSecret(e.target.value)}
/>
<button
type="submit"
className="bg-indigo-500 text-white px-4 py-2 rounded-md"
>
Add Account
</button>
<Button className="mt-3" type="submit">
Add account
</Button>
</form>
);
}

View File

@@ -1,53 +0,0 @@
import { CoID, LocalNode, RawCoValue } from "cojson";
import { Page } from "./page"; // Assuming you have a Page component
// Define the structure of a page in the path
interface PageInfo {
coId: CoID<RawCoValue>;
name?: string;
}
// Props for the PageStack component
interface PageStackProps {
path: PageInfo[];
node?: LocalNode | null;
goBack: () => void;
addPages: (pages: PageInfo[]) => void;
children?: React.ReactNode;
}
export function PageStack({
path,
node,
goBack,
addPages,
children,
}: PageStackProps) {
return (
<div className="relative mt-4 h-[calc(100vh-6rem)]">
{children && <div className="absolute inset-0 pb-20">{children}</div>}
{node &&
path.map((page, index) => (
<Page
key={`${page.coId}-${index}`}
coId={page.coId}
node={node}
name={page.name || page.coId}
onHeaderClick={goBack}
onNavigate={addPages}
isTopLevel={index === path.length - 1}
style={{
transform: `translateZ(${(index - path.length + 1) * 200}px) scale(${
1 - (path.length - index - 1) * 0.05
}) translateY(${-(index - path.length + 1) * -4}%)`,
opacity: 1 - (path.length - index - 1) * 0.05,
zIndex: index,
transitionProperty: "transform, opacity",
transitionDuration: "0.3s",
transitionTimingFunction: "ease-out",
}}
/>
))}
</div>
);
}

View File

@@ -1,138 +0,0 @@
import clsx from "clsx";
import { CoID, LocalNode, RawCoStream, RawCoValue } from "cojson";
import { useEffect, useState } from "react";
import { CoStreamView } from "./co-stream-view";
import { GridView } from "./grid-view";
import { TableView } from "./table-viewer";
import { TypeIcon } from "./type-icon";
import { PageInfo } from "./types";
import { useResolvedCoValue } from "./use-resolve-covalue";
import { AccountOrGroupPreview } from "./value-renderer";
type PageProps = {
coId: CoID<RawCoValue>;
node: LocalNode;
name: string;
onNavigate: (newPages: PageInfo[]) => void;
onHeaderClick?: () => void;
isTopLevel?: boolean;
style: React.CSSProperties;
};
export function Page({
coId,
node,
name,
onNavigate,
onHeaderClick,
style,
isTopLevel,
}: PageProps) {
const { value, snapshot, type, extendedType } = useResolvedCoValue(
coId,
node,
);
const [viewMode, setViewMode] = useState<"grid" | "table">("grid");
const supportsTableView = type === "colist" || extendedType === "record";
// Automatically switch to table view if the page is a CoMap record
useEffect(() => {
if (supportsTableView) {
setViewMode("table");
}
}, [supportsTableView]);
if (snapshot === "unavailable") {
return <div style={style}>Data unavailable</div>;
}
if (!snapshot) {
return <div style={style}></div>;
}
return (
<div
style={style}
className={clsx(
"absolute inset-0 border border-gray-900/5 bg-clip-padding bg-white rounded-xl shadow-lg p-6 animate-in",
)}
>
{!isTopLevel && (
<div
className="absolute inset-x-0 top-0 h-10"
aria-label="Back"
onClick={() => {
onHeaderClick?.();
}}
aria-hidden="true"
></div>
)}
<div className="flex justify-between items-center mb-4">
<div className="flex flex-col gap-2">
<h2 className="text-2xl font-bold flex items-start flex-col gap-1">
<span>
{name}
{typeof snapshot === "object" && "name" in snapshot ? (
<span className="text-gray-600 font-medium">
{" "}
{
(
snapshot as {
name: string;
}
).name
}
</span>
) : null}
</span>
</h2>
<div className="flex items-center gap-2">
<span className="text-xs text-gray-700 font-medium py-0.5 px-1 -ml-0.5 rounded bg-gray-700/5 inline-block font-mono">
{type && <TypeIcon type={type} extendedType={extendedType} />}
</span>
<span className="text-xs text-gray-700 font-medium py-0.5 px-1 -ml-0.5 rounded bg-gray-700/5 inline-block font-mono">
{coId}
</span>
</div>
</div>
{/* {supportsTableView && (
<button
onClick={toggleViewMode}
className="px-4 py-2 bg-blue-500 text-white rounded hover:bg-blue-600 transition-colors"
>
{viewMode === "grid" ? "Table View" : "Grid View"}
</button>
)} */}
</div>
<div className="overflow-auto max-h-[calc(100%-4rem)]">
{type === "costream" ? (
<CoStreamView
data={snapshot}
onNavigate={onNavigate}
node={node}
value={value as RawCoStream}
/>
) : viewMode === "grid" ? (
<GridView data={snapshot} onNavigate={onNavigate} node={node} />
) : (
<TableView data={snapshot} node={node} onNavigate={onNavigate} />
)}
{/* --- */}
{extendedType !== "account" && extendedType !== "group" && (
<div className="text-xs text-gray-500 mt-4">
Owned by{" "}
<AccountOrGroupPreview
coId={value.group.id}
node={node}
showId
onClick={() => {
onNavigate([{ coId: value.group.id, name: "owner" }]);
}}
/>
</div>
)}
</div>
</div>
);
}

View File

@@ -1,133 +0,0 @@
import { CoID, LocalNode, RawCoValue } from "cojson";
import { JsonObject } from "cojson";
import { useMemo, useState } from "react";
import { LinkIcon } from "../link-icon";
import { PageInfo } from "./types";
import { useResolvedCoValues } from "./use-resolve-covalue";
import { ValueRenderer } from "./value-renderer";
export function TableView({
data,
node,
onNavigate,
}: {
data: JsonObject;
node: LocalNode;
onNavigate: (pages: PageInfo[]) => void;
}) {
const [visibleRowsCount, setVisibleRowsCount] = useState(10);
const [coIdArray, visibleRows] = useMemo(() => {
const coIdArray = Array.isArray(data)
? data
: Object.values(data).every(
(k) => typeof k === "string" && k.startsWith("co_"),
)
? Object.values(data).map((k) => k as CoID<RawCoValue>)
: [];
const visibleRows = coIdArray.slice(0, visibleRowsCount);
return [coIdArray, visibleRows];
}, [data, visibleRowsCount]);
const resolvedRows = useResolvedCoValues(visibleRows, node);
const hasMore = visibleRowsCount < coIdArray.length;
if (!coIdArray.length) {
return <div>No data to display</div>;
}
if (resolvedRows.length === 0) {
return <div>Loading...</div>;
}
const keys = Array.from(
new Set(resolvedRows.flatMap((item) => Object.keys(item.snapshot || {}))),
);
const loadMore = () => {
setVisibleRowsCount((prevVisibleRows) => prevVisibleRows + 10);
};
return (
<div>
<table className="min-w-full divide-y divide-gray-200">
<thead className="sticky top-0 border-b">
<tr>
{["", ...keys].map((key) => (
<th
key={key}
className="px-4 py-3 bg-gray-50 text-left text-xs font-medium text-gray-500 rounded"
>
{key}
</th>
))}
</tr>
</thead>
<tbody className="bg-white divide-y divide-gray-200">
{resolvedRows.slice(0, visibleRowsCount).map((item, index) => (
<tr key={index}>
<td className="px-1 py-0">
<button
onClick={() =>
onNavigate([
{
coId: item.value!.id,
name: index.toString(),
},
])
}
className="px-4 py-4 whitespace-nowrap text-sm text-gray-500 hover:text-blue-500 hover:bg-gray-100 rounded"
>
<LinkIcon />
</button>
</td>
{keys.map((key) => (
<td
key={key}
className="px-4 py-4 whitespace-nowrap text-sm text-gray-500"
>
<ValueRenderer
json={(item.snapshot as JsonObject)[key]}
onCoIDClick={(coId) => {
async function handleClick() {
onNavigate([
{
coId: item.value!.id,
name: index.toString(),
},
{
coId: coId,
name: key,
},
]);
}
handleClick();
}}
/>
</td>
))}
</tr>
))}
</tbody>
</table>
<div className="py-4 text-gray-500 flex items-center justify-between gap-2">
<span>
Showing {Math.min(visibleRowsCount, coIdArray.length)} of{" "}
{coIdArray.length}
</span>
{hasMore && (
<div className="text-center">
<button
onClick={loadMore}
className="px-4 py-2 bg-blue-500 text-white rounded hover:bg-blue-600"
>
Load More
</button>
</div>
)}
</div>
</div>
);
}

View File

@@ -1,47 +0,0 @@
import { CoID, LocalNode, RawCoValue } from "cojson";
import {
CoJsonType,
ExtendedCoJsonType,
useResolvedCoValue,
} from "./use-resolve-covalue";
export const TypeIcon = ({
type,
extendedType,
}: {
type: CoJsonType;
extendedType?: ExtendedCoJsonType;
}) => {
const iconMap: Record<ExtendedCoJsonType | CoJsonType, string> = {
record: "{} Record",
image: "🖼️ Image",
comap: "{} CoMap",
costream: "≋ CoStream",
colist: "☰ CoList",
account: "👤 Account",
group: "👥 Group",
};
const iconKey = extendedType || type;
const icon = iconMap[iconKey as keyof typeof iconMap];
return icon ? <span className="font-mono">{icon}</span> : null;
};
export const ResolveIcon = ({
coId,
node,
}: {
coId: CoID<RawCoValue>;
node: LocalNode;
}) => {
const { type, extendedType, snapshot } = useResolvedCoValue(coId, node);
if (snapshot === "unavailable" && !type) {
return <div className="text-gray-600 font-medium">Unavailable</div>;
}
if (!type) return <div className="whitespace-pre w-14 font-mono"> </div>;
return <TypeIcon type={type} extendedType={extendedType} />;
};

View File

@@ -1,9 +0,0 @@
import { CoID, RawCoValue } from "cojson";
export type PageInfo = {
coId: CoID<RawCoValue>;
name?: string;
};
export const isCoId = (coId: unknown): coId is CoID<RawCoValue> =>
typeof coId === "string" && coId.startsWith("co_");

View File

@@ -1,6 +1,6 @@
import { CoID, RawCoValue } from "cojson";
import { PageInfo } from "jazz-inspector";
import { useCallback, useEffect, useState } from "react";
import { PageInfo } from "./types";
export function usePagePath(defaultPath?: PageInfo[]) {
const [path, setPath] = useState<PageInfo[]>(() => {

View File

@@ -1,260 +0,0 @@
import clsx from "clsx";
import { CoID, JsonValue, LocalNode, RawCoValue } from "cojson";
import React, { useEffect, useState } from "react";
import { LinkIcon } from "../link-icon";
import {
isBrowserImage,
resolveCoValue,
useResolvedCoValue,
} from "./use-resolve-covalue";
// Is there a chance we can pass the actual CoValue here?
export function ValueRenderer({
json,
compact,
onCoIDClick,
}: {
json: JsonValue | undefined;
compact?: boolean;
onCoIDClick?: (childNode: CoID<RawCoValue>) => void;
}) {
const [isExpanded, setIsExpanded] = useState(false);
if (typeof json === "undefined" || json === undefined) {
return <span className="text-gray-400">undefined</span>;
}
if (json === null) {
return <span className="text-gray-400">null</span>;
}
if (typeof json === "string" && json.startsWith("co_")) {
return (
<span
className={clsx(
"inline-flex gap-1 items-center",
onCoIDClick && "text-blue-500 cursor-pointer hover:underline",
)}
onClick={() => {
onCoIDClick?.(json as CoID<RawCoValue>);
}}
>
{json}
{onCoIDClick && <LinkIcon />}
</span>
);
}
if (typeof json === "string") {
return (
<span className="text-green-900 font-mono">
{/* <span className="select-none opacity-70">{'"'}</span> */}
{json}
{/* <span className="select-none opacity-70">{'"'}</span> */}
</span>
);
}
if (typeof json === "number") {
return <span className="text-purple-500">{json}</span>;
}
if (typeof json === "boolean") {
return (
<span
className={clsx(
json
? "text-green-700 bg-green-700/5"
: "text-amber-700 bg-amber-500/5",
"font-mono",
"inline-block px-1 py-0.5 rounded",
)}
>
{json.toString()}
</span>
);
}
if (Array.isArray(json)) {
return (
<span title={JSON.stringify(json)}>
Array <span className="text-gray-500">({json.length})</span>
</span>
);
}
if (typeof json === "object") {
return (
<span
title={JSON.stringify(json, null, 2)}
className="inline-block max-w-64"
>
{compact ? (
<span>
Object{" "}
<span className="text-gray-500">({Object.keys(json).length})</span>
<pre className="mt-1 text-sm whitespace-pre-wrap">
{isExpanded
? JSON.stringify(json, null, 2)
: JSON.stringify(json, null, 2)
.split("\n")
.slice(0, 3)
.join("\n") + (Object.keys(json).length > 2 ? "\n..." : "")}
</pre>
<button
onClick={() => setIsExpanded(!isExpanded)}
className="text-xs text-gray-500 hover:text-gray-700"
>
{isExpanded ? "Show less" : "Show more"}
</button>
</span>
) : (
<pre className="whitespace-pre-wrap">
{JSON.stringify(json, null, 2)}
</pre>
)}
</span>
);
}
return <span>{String(json)}</span>;
}
export const CoMapPreview = ({
coId,
node,
limit = 6,
}: {
coId: CoID<RawCoValue>;
node: LocalNode;
limit?: number;
}) => {
const { value, snapshot, type, extendedType } = useResolvedCoValue(
coId,
node,
);
if (!snapshot) {
return (
<div className="rounded bg-gray-100 animate-pulse whitespace-pre w-24">
{" "}
</div>
);
}
if (snapshot === "unavailable" && !value) {
return <div className="text-gray-500">Unavailable</div>;
}
if (extendedType === "image" && isBrowserImage(snapshot)) {
return (
<div>
<img
src={snapshot.placeholderDataURL}
className="size-8 border-2 border-white drop-shadow-md my-2"
/>
<span className="text-gray-500 text-sm">
{snapshot.originalSize[0]} x {snapshot.originalSize[1]}
</span>
{/* <CoMapPreview coId={value[]} node={node} /> */}
{/* <ProgressiveImg image={value}>
{({ src }) => <img src={src} className={clsx("w-full")} />}
</ProgressiveImg> */}
</div>
);
}
if (extendedType === "record") {
return (
<div>
Record{" "}
<span className="text-gray-500">({Object.keys(snapshot).length})</span>
</div>
);
}
if (type === "colist") {
return (
<div>
List{" "}
<span className="text-gray-500">
({(snapshot as unknown as []).length})
</span>
</div>
);
}
return (
<div className="text-sm flex flex-col gap-2 items-start">
<div className="grid grid-cols-[auto_1fr] gap-2">
{Object.entries(snapshot)
.slice(0, limit)
.map(([key, value]) => (
<React.Fragment key={key}>
<span className="font-medium">{key}: </span>
<span>
<ValueRenderer json={value} />
</span>
</React.Fragment>
))}
</div>
{Object.entries(snapshot).length > limit && (
<div className="text-left text-xs text-gray-500 mt-2">
{Object.entries(snapshot).length - limit} more
</div>
)}
</div>
);
};
export function AccountOrGroupPreview({
coId,
node,
showId = false,
onClick,
}: {
coId: CoID<RawCoValue>;
node: LocalNode;
showId?: boolean;
onClick?: (name?: string) => void;
}) {
const { snapshot, extendedType } = useResolvedCoValue(coId, node);
const [name, setName] = useState<string | null>(null);
useEffect(() => {
if (extendedType === "account") {
resolveCoValue(
(snapshot as unknown as { profile: CoID<RawCoValue> }).profile,
node,
).then(({ snapshot }) => {
if (
typeof snapshot === "object" &&
"name" in snapshot &&
typeof snapshot.name === "string"
) {
setName(snapshot.name);
}
});
}
}, [snapshot, node, extendedType]);
if (!snapshot) return <span>Loading...</span>;
if (extendedType !== "account" && extendedType !== "group") {
return <span>CoID is not an account or group</span>;
}
const displayName = extendedType === "account" ? name || "Account" : "Group";
const displayText = showId ? `${displayName} (${coId})` : displayName;
const props = onClick
? {
onClick: () => onClick(displayName),
className: "text-blue-500 cursor-pointer hover:underline",
}
: {
className: "text-gray-500",
};
return <span {...props}>{displayText}</span>;
}

View File

@@ -1,5 +1,30 @@
import type { Config } from "tailwindcss";
import animate from "tailwindcss-animate";
import plugin from "tailwindcss/plugin";
const stonePalette = {
50: "oklch(0.988281 0.002 75)",
100: "oklch(0.980563 0.002 75)",
200: "oklch(0.917969 0.002 75)",
300: "oklch(0.853516 0.002 75)",
400: "oklch(0.789063 0.002 75)",
500: "oklch(0.726563 0.002 75)",
600: "oklch(0.613281 0.002 75)",
700: "oklch(0.523438 0.002 75)",
800: "oklch(0.412109 0.002 75)",
900: "oklch(0.302734 0.002 75)",
925: "oklch(0.220000 0.002 75)",
950: "oklch(0.193359 0.002 75)",
};
const stonePaletteWithAlpha = { ...stonePalette };
Object.keys(stonePalette).forEach((key) => {
// @ts-ignore
stonePaletteWithAlpha[key] = stonePaletteWithAlpha[key].replace(
")",
"/ <alpha-value>)",
);
});
const config: Config = {
content: [
@@ -18,62 +43,26 @@ const config: Config = {
},
extend: {
colors: {
border: "hsl(var(--border))",
input: "hsl(var(--input))",
ring: "hsl(var(--ring))",
background: "hsl(var(--background))",
foreground: "hsl(var(--foreground))",
primary: {
DEFAULT: "hsl(var(--primary))",
foreground: "hsl(var(--primary-foreground))",
stone: stonePaletteWithAlpha,
gray: stonePaletteWithAlpha,
blue: {
50: "#f5f7ff",
100: "#ebf0fe",
200: "#d6e0fd",
300: "#b3c7fc",
400: "#8aa6f9",
500: "#5870F1",
600: "#3651E7",
700: "#3313F7",
800: "#2A12BE",
900: "#12046A",
950: "#1e1b4b",
DEFAULT: "#3313F7",
},
secondary: {
DEFAULT: "hsl(var(--secondary))",
foreground: "hsl(var(--secondary-foreground))",
},
destructive: {
DEFAULT: "hsl(var(--destructive))",
foreground: "hsl(var(--destructive-foreground))",
},
muted: {
DEFAULT: "hsl(var(--muted))",
foreground: "hsl(var(--muted-foreground))",
},
accent: {
DEFAULT: "hsl(var(--accent))",
foreground: "hsl(var(--accent-foreground))",
},
popover: {
DEFAULT: "hsl(var(--popover))",
foreground: "hsl(var(--popover-foreground))",
},
card: {
DEFAULT: "hsl(var(--card))",
foreground: "hsl(var(--card-foreground))",
},
},
borderRadius: {
lg: "var(--radius)",
md: "calc(var(--radius) - 2px)",
sm: "calc(var(--radius) - 4px)",
},
keyframes: {
"accordion-down": {
from: { height: "0" },
to: { height: "var(--radix-accordion-content-height)" },
},
"accordion-up": {
from: { height: "var(--radix-accordion-content-height)" },
to: { height: "0" },
},
},
animation: {
"accordion-down": "accordion-down 0.2s ease-out",
"accordion-up": "accordion-up 0.2s ease-out",
},
},
},
plugins: [animate],
plugins: [plugin(({ addVariant }) => addVariant("label", "& :is(label)"))],
};
export default config;

View File

@@ -1,5 +1,24 @@
# multiauth
## 0.0.4
### Patch Changes
- jazz-react@0.11.5
- jazz-react-auth-clerk@0.11.5
- jazz-tools@0.11.5
## 0.0.3
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
- jazz-react-auth-clerk@0.11.4
## 0.0.2
### Patch Changes

View File

@@ -13,11 +13,11 @@ To run this example, you may either:
1. Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example counter --project-name counter
npx create-jazz-app@latest counter-app --example counter
```
2. Navigate to the new project and start the development server.
```bash
cd counter
cd counter-app
npm run dev
```

View File

@@ -1,7 +1,7 @@
{
"name": "multiauth",
"private": true,
"version": "0.0.2",
"version": "0.0.4",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -19,7 +19,6 @@ createRoot(document.getElementById("root")!).render(
<OmniAuth
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp", // This makes the app work in local mode when the user is not authenticated
}}
>
<App />

View File

@@ -1,5 +1,26 @@
# jazz-example-musicplayer
## 0.0.85
### Patch Changes
- jazz-inspector@0.11.5
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.84
### Patch Changes
- Updated dependencies [d133d47]
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- Updated dependencies [e902405]
- jazz-inspector@0.11.4
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.83
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example music-player --project-name music-player
npx create-jazz-app@latest music-player-app --example music-player
```
Go to the new project directory.
```bash
cd music-player
cd music-player-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-example-music-player",
"private": true,
"version": "0.0.83",
"version": "0.0.85",
"type": "module",
"scripts": {
"dev": "vite",
@@ -22,8 +22,8 @@
"class-variance-authority": "^0.7.0",
"clsx": "^2.0.0",
"jazz-inspector": "workspace:*",
"jazz-react": "workspace:0.11.3",
"jazz-tools": "workspace:0.11.3",
"jazz-react": "workspace:0.11.5",
"jazz-tools": "workspace:0.11.5",
"lucide-react": "^0.274.0",
"react": "^18.3.1",
"react-dom": "^18.3.1",

View File

@@ -72,7 +72,6 @@ ReactDOM.createRoot(document.getElementById("root")!).render(
<JazzProvider
sync={{
peer,
when: "signedUp", // This makes the app work in local mode when the user is anonymous
}}
storage="indexedDB"
AccountSchema={MusicaAccount}

View File

@@ -1,5 +1,22 @@
# organization
## 0.0.57
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.56
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.55
### Patch Changes

View File

@@ -16,12 +16,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example organization --project-name organization
npx create-jazz-app@latest organization-app --example organization
```
Go to the new project directory.
```bash
cd organization
cd organization-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "organization",
"private": true,
"version": "0.0.55",
"version": "0.0.57",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -1,5 +1,17 @@
# passkey-svelte
## 0.0.51
### Patch Changes
- jazz-svelte@0.11.5
## 0.0.50
### Patch Changes
- jazz-svelte@0.11.4
## 0.0.49
### Patch Changes

View File

@@ -21,12 +21,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example passkey-svelte --project-name passkey-svelte
npx create-jazz-app@latest passkey-svelte-app --example passkey-svelte
```
Go to the new project directory.
```bash
cd passkey-svelte
cd passkey-svelte-app
```
Run the dev server.

View File

@@ -1,6 +1,6 @@
{
"name": "passkey-svelte",
"version": "0.0.49",
"version": "0.0.51",
"type": "module",
"private": true,
"scripts": {

View File

@@ -8,7 +8,6 @@
<JazzProvider
sync={{
peer: `wss://cloud.jazz.tools/?key={apiKey}`,
when: "signedUp",
}}
>
<PasskeyAuthBasicUI appName="minimal-svelte-auth-passkey">

View File

@@ -1,5 +1,22 @@
# minimal-auth-passkey
## 0.0.62
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.61
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.60
### Patch Changes

View File

@@ -15,12 +15,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example passkey --project-name passkey
npx create-jazz-app@latest passkey-app --example passkey
```
Go to the new project directory.
```bash
cd passkey
cd passkey-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "passkey",
"private": true,
"version": "0.0.60",
"version": "0.0.62",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -10,7 +10,6 @@ function JazzAndAuth({ children }: { children: React.ReactNode }) {
<JazzProvider
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
>
<PasskeyAuthBasicUI appName="Jazz Minimal Auth Passkey Example">

View File

@@ -1,5 +1,22 @@
# passphrase
## 0.0.59
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.58
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.57
### Patch Changes

View File

@@ -12,12 +12,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example passphrase --project-name passphrase
npx create-jazz-app@latest passphrase-app --example passphrase
```
Go to the new project directory.
```bash
cd passphrase
cd passphrase-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "passphrase",
"private": true,
"version": "0.0.57",
"version": "0.0.59",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -142,7 +142,6 @@ function JazzAndAuth({ children }: { children: React.ReactNode }) {
<JazzProvider
sync={{
peer: "wss://cloud.jazz.tools/?key=minimal-auth-passphrase-example@garden.co",
when: "signedUp",
}}
>
<PassphraseAuthBasicUI

View File

@@ -1,5 +1,22 @@
# jazz-password-manager
## 0.0.83
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.82
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.81
### Patch Changes

View File

@@ -15,12 +15,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example password-manager --project-name password-manager
npx create-jazz-app@latest password-manager-app --example password-manager
```
Go to the new project directory.
```bash
cd password-manager
cd password-manager-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-password-manager",
"private": true,
"version": "0.0.81",
"version": "0.0.83",
"type": "module",
"scripts": {
"dev": "vite",
@@ -12,8 +12,8 @@
"clean-install": "rm -rf node_modules pnpm-lock.yaml && pnpm install"
},
"dependencies": {
"jazz-react": "workspace:0.11.3",
"jazz-tools": "workspace:0.11.3",
"jazz-react": "workspace:0.11.5",
"jazz-tools": "workspace:0.11.5",
"react": "^18.3.1",
"react-dom": "^18.3.1",
"react-hook-form": "^7.41.5",

View File

@@ -12,7 +12,6 @@ function JazzAndAuth({ children }: { children: React.ReactNode }) {
AccountSchema={PasswordManagerAccount}
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
>
<PasskeyAuthBasicUI appName="Jazz Password Manager">

View File

@@ -1,5 +1,24 @@
# jazz-example-pets
## 0.0.181
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
- jazz-browser-media-images@0.11.5
## 0.0.180
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser-media-images@0.11.4
- jazz-react@0.11.4
## 0.0.179
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example pets --project-name pets
npx create-jazz-app@latest pets-app --example pets
```
Go to the new project directory.
```bash
cd pets
cd pets-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-example-pets",
"private": true,
"version": "0.0.179",
"version": "0.0.181",
"type": "module",
"scripts": {
"dev": "vite",
@@ -19,9 +19,9 @@
"@radix-ui/react-toast": "^1.1.4",
"class-variance-authority": "^0.7.0",
"clsx": "^2.0.0",
"jazz-browser-media-images": "workspace:0.11.3",
"jazz-react": "workspace:0.11.3",
"jazz-tools": "workspace:0.11.3",
"jazz-browser-media-images": "workspace:0.11.5",
"jazz-react": "workspace:0.11.5",
"jazz-tools": "workspace:0.11.5",
"lucide-react": "^0.274.0",
"qrcode": "^1.5.3",
"react": "^18.3.1",
@@ -41,7 +41,7 @@
"@vitejs/plugin-react-swc": "^3.3.2",
"autoprefixer": "^10.4.20",
"is-ci": "^3.0.1",
"jazz-run": "workspace:0.11.3",
"jazz-run": "workspace:0.11.5",
"postcss": "^8.4.27",
"tailwindcss": "^3.4.17",
"typescript": "~5.6.2",

View File

@@ -52,7 +52,6 @@ ReactDOM.createRoot(document.getElementById("root")!).render(
<JazzProvider
sync={{
peer,
when: "signedUp",
}}
AccountSchema={PetAccount}
>

View File

@@ -1,5 +1,24 @@
# reactions
## 0.0.61
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
- jazz-browser-media-images@0.11.5
## 0.0.60
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser-media-images@0.11.4
- jazz-react@0.11.4
## 0.0.59
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example reactions --project-name reactions
npx create-jazz-app@latest reactions-app --example reactions
```
Go to the new project directory.
```bash
cd reactions
cd reactions-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "reactions",
"private": true,
"version": "0.0.59",
"version": "0.0.61",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -10,7 +10,6 @@ createRoot(document.getElementById("root")!).render(
<JazzProvider
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
>
<PasskeyAuthBasicUI appName="Jazz Reactions Example">

View File

@@ -1,5 +1,24 @@
# todo-vue
## 0.0.65
### Patch Changes
- jazz-browser@0.11.5
- jazz-tools@0.11.5
- jazz-vue@0.11.5
## 0.0.64
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-browser@0.11.4
- jazz-vue@0.11.4
## 0.0.63
### Patch Changes

View File

@@ -11,12 +11,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example todo-vue --project-name todo-vue
npx create-jazz-app@latest todo-vue-app --example todo-vue
```
Go to the new project directory.
```bash
cd todo-vue
cd todo-vue-app
```
Run the dev server.

View File

@@ -1,6 +1,6 @@
{
"name": "todo-vue",
"version": "0.0.63",
"version": "0.0.65",
"private": true,
"type": "module",
"scripts": {

View File

@@ -1,5 +1,22 @@
# jazz-example-todo
## 0.0.180
### Patch Changes
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.179
### Patch Changes
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.178
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example todo --project-name todo
npx create-jazz-app@latest todo-app --example todo
```
Go to the new project directory.
```bash
cd todo
cd todo-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "jazz-example-todo",
"private": true,
"version": "0.0.178",
"version": "0.0.180",
"type": "module",
"scripts": {
"dev": "vite",
@@ -16,8 +16,8 @@
"@radix-ui/react-toast": "^1.1.4",
"class-variance-authority": "^0.7.0",
"clsx": "^2.0.0",
"jazz-react": "workspace:0.11.3",
"jazz-tools": "workspace:0.11.3",
"jazz-react": "workspace:0.11.5",
"jazz-tools": "workspace:0.11.5",
"lucide-react": "^0.274.0",
"qrcode": "^1.5.3",
"react": "^18.3.1",

View File

@@ -42,7 +42,6 @@ function JazzAndAuth({ children }: { children: React.ReactNode }) {
<JazzProvider
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
AccountSchema={TodoAccount}
>

View File

@@ -1,5 +1,26 @@
# version-history
## 0.0.58
### Patch Changes
- jazz-inspector@0.11.5
- jazz-react@0.11.5
- jazz-tools@0.11.5
## 0.0.57
### Patch Changes
- Updated dependencies [d133d47]
- Updated dependencies [57a3dbe]
- Updated dependencies [a717754]
- Updated dependencies [a91f343]
- Updated dependencies [e902405]
- jazz-inspector@0.11.4
- jazz-tools@0.11.4
- jazz-react@0.11.4
## 0.0.56
### Patch Changes

View File

@@ -13,12 +13,12 @@ You can either
Create a new Jazz project, and use this example as a template.
```bash
npx create-jazz-app@latest --example version-history --project-name version-history
npx create-jazz-app@latest version-history-app --example version-history
```
Go to the new project directory.
```bash
cd version-history
cd version-history-app
```
Run the dev server.

View File

@@ -1,7 +1,7 @@
{
"name": "version-history",
"private": true,
"version": "0.0.56",
"version": "0.0.58",
"type": "module",
"scripts": {
"dev": "vite",

View File

@@ -11,7 +11,6 @@ createRoot(document.getElementById("root")!).render(
<JazzProvider
sync={{
peer: `wss://cloud.jazz.tools/?key=${apiKey}`,
when: "signedUp",
}}
>
<DemoAuthBasicUI appName="Jazz Version History Example">

View File

@@ -0,0 +1,37 @@
import { metaTags } from "@/app/layout";
import { posts } from "@/lib/posts";
import { Feed } from "feed";
import { NextResponse } from "next/server";
export async function GET() {
const feed = new Feed({
title: "Garden Computing Blog",
description: "News from Garden Computing",
id: metaTags.url,
link: `${metaTags.url}/news`,
language: "en",
image: `${metaTags.url}/social-image.png`,
favicon: `${metaTags.url}/favicon.ico`,
copyright: `${new Date().getFullYear()} Garden Computing, Inc.`,
});
posts.forEach((post) => {
feed.addItem({
title: post.meta.title,
description: post.meta.subtitle,
id: post.meta.slug,
link: `${metaTags.url}/news/${post.meta.slug}`,
date: new Date(post.meta.date),
author: [{ name: post.meta.author.name }],
guid: post.meta.slug,
image: `${metaTags.url}${post.meta.coverImage}`,
});
});
return new NextResponse(feed.rss2(), {
headers: {
"Content-Type": "application/xml",
"Cache-Control": "public, s-maxage=3600, stale-while-revalidate=86400",
},
});
}

View File

@@ -41,7 +41,7 @@ const commitMono = localFont({
display: "swap",
});
const metaTags = {
export const metaTags = {
title: "garden computing",
description:
"Computers are magic. So why do we put up with so much complexity? We believe just a few new ideas can make all the difference.",
@@ -70,6 +70,16 @@ export const metadata: Metadata = {
},
],
},
alternates: {
canonical: metaTags.url,
types: {
"application/rss+xml": `${
process.env.VERCEL_URL
? `https://${process.env.VERCEL_URL}`
: "http://localhost:3000"
}/api/rss`,
},
},
};
export default function RootLayout({

Some files were not shown because too many files have changed in this diff Show More