Compare commits
12 Commits
next
...
7aef7ae796
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7aef7ae796 | ||
|
|
1bfbeba934 | ||
|
|
fc4295894f | ||
|
|
ab0dce107d | ||
|
|
9436602443 | ||
|
|
66de31706c | ||
|
|
8925fe6336 | ||
|
|
fe8ca23d37 | ||
|
|
d48f20a59d | ||
|
|
b7b4abcbe2 | ||
|
|
129ab04ea7 | ||
|
|
870fe479c3 |
24
bun.lock
24
bun.lock
@@ -46,9 +46,21 @@
|
||||
"vite": "^6.2.6",
|
||||
},
|
||||
},
|
||||
"packages/extension-form": {
|
||||
"name": "@reffuse/extension-form",
|
||||
"version": "0.1.0",
|
||||
"devDependencies": {
|
||||
"reffuse": "workspace:*",
|
||||
},
|
||||
"peerDependencies": {
|
||||
"effect": "^3.13.0",
|
||||
"react": "^19.0.0",
|
||||
"reffuse": "^0.1.6",
|
||||
},
|
||||
},
|
||||
"packages/extension-lazyref": {
|
||||
"name": "@reffuse/extension-lazyref",
|
||||
"version": "0.1.1",
|
||||
"version": "0.1.2",
|
||||
"devDependencies": {
|
||||
"reffuse": "workspace:*",
|
||||
},
|
||||
@@ -57,12 +69,12 @@
|
||||
"@types/react": "^19.0.0",
|
||||
"effect": "^3.13.0",
|
||||
"react": "^19.0.0",
|
||||
"reffuse": "^0.1.4",
|
||||
"reffuse": "^0.1.6",
|
||||
},
|
||||
},
|
||||
"packages/extension-query": {
|
||||
"name": "@reffuse/extension-query",
|
||||
"version": "0.1.2",
|
||||
"version": "0.1.3",
|
||||
"devDependencies": {
|
||||
"reffuse": "workspace:*",
|
||||
},
|
||||
@@ -73,12 +85,12 @@
|
||||
"@types/react": "^19.0.0",
|
||||
"effect": "^3.13.0",
|
||||
"react": "^19.0.0",
|
||||
"reffuse": "^0.1.4",
|
||||
"reffuse": "^0.1.6",
|
||||
},
|
||||
},
|
||||
"packages/reffuse": {
|
||||
"name": "reffuse",
|
||||
"version": "0.1.5",
|
||||
"version": "0.1.6",
|
||||
"peerDependencies": {
|
||||
"@types/react": "^19.0.0",
|
||||
"effect": "^3.13.0",
|
||||
@@ -363,6 +375,8 @@
|
||||
|
||||
"@reffuse/example": ["@reffuse/example@workspace:packages/example"],
|
||||
|
||||
"@reffuse/extension-form": ["@reffuse/extension-form@workspace:packages/extension-form"],
|
||||
|
||||
"@reffuse/extension-lazyref": ["@reffuse/extension-lazyref@workspace:packages/extension-lazyref"],
|
||||
|
||||
"@reffuse/extension-query": ["@reffuse/extension-query@workspace:packages/extension-query"],
|
||||
|
||||
9
packages/extension-form/README.md
Normal file
9
packages/extension-form/README.md
Normal file
@@ -0,0 +1,9 @@
|
||||
# LazyRef extension for Reffuse
|
||||
|
||||
Extension to integrate `@typed/lazy-ref` with Reffuse.
|
||||
|
||||
## Peer dependencies
|
||||
- `@typed/lazy-ref`
|
||||
- `reffuse` 0.1.3+
|
||||
- `effect` 3.13+
|
||||
- `react` & `@types/react` 19+
|
||||
40
packages/extension-form/package.json
Normal file
40
packages/extension-form/package.json
Normal file
@@ -0,0 +1,40 @@
|
||||
{
|
||||
"name": "@reffuse/extension-form",
|
||||
"version": "0.1.0",
|
||||
"type": "module",
|
||||
"files": [
|
||||
"./README.md",
|
||||
"./dist"
|
||||
],
|
||||
"license": "MIT",
|
||||
"repository": {
|
||||
"url": "git+https://github.com/Thiladev/reffuse.git"
|
||||
},
|
||||
"types": "./dist/index.d.ts",
|
||||
"exports": {
|
||||
".": {
|
||||
"types": "./dist/index.d.ts",
|
||||
"default": "./dist/index.js"
|
||||
},
|
||||
"./*": {
|
||||
"types": "./dist/*.d.ts",
|
||||
"default": "./dist/*.js"
|
||||
}
|
||||
},
|
||||
"scripts": {
|
||||
"build": "tsc",
|
||||
"lint:tsc": "tsc --noEmit",
|
||||
"pack": "npm pack",
|
||||
"clean:cache": "rm -f tsconfig.tsbuildinfo",
|
||||
"clean:dist": "rm -rf dist",
|
||||
"clean:node": "rm -rf node_modules"
|
||||
},
|
||||
"devDependencies": {
|
||||
"reffuse": "workspace:*"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"effect": "^3.13.0",
|
||||
"react": "^19.0.0",
|
||||
"reffuse": "^0.1.7"
|
||||
}
|
||||
}
|
||||
9
packages/extension-form/src/FormExtension.ts
Normal file
9
packages/extension-form/src/FormExtension.ts
Normal file
@@ -0,0 +1,9 @@
|
||||
import { ReffuseExtension, type ReffuseNamespace } from "reffuse"
|
||||
|
||||
|
||||
export const FormExtension = ReffuseExtension.make(() => ({
|
||||
useForm<A, E, R>(
|
||||
this: ReffuseNamespace.ReffuseNamespace<R>,
|
||||
) {
|
||||
},
|
||||
}))
|
||||
1
packages/extension-form/src/index.ts
Normal file
1
packages/extension-form/src/index.ts
Normal file
@@ -0,0 +1 @@
|
||||
export * from "./FormExtension.js"
|
||||
6
packages/extension-form/src/internal/Form.ts
Normal file
6
packages/extension-form/src/internal/Form.ts
Normal file
@@ -0,0 +1,6 @@
|
||||
import { Schema } from "effect"
|
||||
|
||||
|
||||
export interface Form<A, I, R> {
|
||||
readonly schema: Schema.Schema<A, I, R>
|
||||
}
|
||||
44
packages/extension-form/src/internal/FormField.ts
Normal file
44
packages/extension-form/src/internal/FormField.ts
Normal file
@@ -0,0 +1,44 @@
|
||||
import type { Schema } from "effect"
|
||||
import type * as FormTree from "./FormTree.ts"
|
||||
|
||||
|
||||
export interface FormField<S extends Schema.Schema.Any> {
|
||||
readonly schema: S
|
||||
}
|
||||
|
||||
export interface GenericFormField<S extends Schema.Schema.Any> extends FormField<S> {
|
||||
readonly _tag: "GenericFormField"
|
||||
readonly value: S["Type"]
|
||||
}
|
||||
|
||||
export interface TupleFormField<S extends Schema.Tuple<readonly Schema.Schema.Any[]>> extends FormField<S> {
|
||||
readonly _tag: "TupleFormField"
|
||||
readonly elements: { readonly [K in keyof S["elements"]]: FormTree.FormTree<
|
||||
S["elements"][K] extends Schema.Schema.Any
|
||||
? S["elements"][K]
|
||||
: never
|
||||
> }
|
||||
}
|
||||
|
||||
export interface Tuple2FormField<S extends Schema.Tuple2<Schema.Schema.Any, Schema.Schema.Any>> extends FormField<S> {
|
||||
readonly _tag: "Tuple2FormField"
|
||||
readonly elements: { readonly [K in keyof S["elements"]]: FormTree.FormTree<
|
||||
S["elements"][K] extends Schema.Schema.Any
|
||||
? S["elements"][K]
|
||||
: never
|
||||
> }
|
||||
}
|
||||
|
||||
export interface ArrayFormField<S extends Schema.Array$<Schema.Schema.AnyNoContext>> extends FormField<S> {
|
||||
readonly _tag: "ArrayFormField"
|
||||
readonly elements: readonly FormTree.FormTree<S["value"]>[]
|
||||
}
|
||||
|
||||
export interface StructFormField<S extends Schema.Struct<{
|
||||
readonly [x: string]: Schema.Schema.AnyNoContext
|
||||
readonly [x: number]: Schema.Schema.AnyNoContext
|
||||
readonly [x: symbol]: Schema.Schema.AnyNoContext
|
||||
}>> extends FormField<S> {
|
||||
readonly _tag: "StructFormField"
|
||||
readonly fields: { readonly [K in keyof S["fields"]]: FormTree.FormTree<S["fields"][K]> }
|
||||
}
|
||||
23
packages/extension-form/src/internal/FormTree.ts
Normal file
23
packages/extension-form/src/internal/FormTree.ts
Normal file
@@ -0,0 +1,23 @@
|
||||
import { Schema } from "effect"
|
||||
import type * as FormField from "./FormField.js"
|
||||
|
||||
|
||||
export type FormTree<S extends Schema.Schema.Any> = (
|
||||
S extends Schema.Tuple<any> ? FormField.TupleFormField<S> :
|
||||
S extends Schema.Tuple2<any, any> ? FormField.Tuple2FormField<S> :
|
||||
S extends Schema.Array$<any> ? FormField.ArrayFormField<S> :
|
||||
S extends Schema.Struct<any> ? FormField.StructFormField<S> :
|
||||
FormField.GenericFormField<S>
|
||||
)
|
||||
|
||||
|
||||
const User = Schema.Struct({
|
||||
name: Schema.String,
|
||||
roles: Schema.Tuple(Schema.Literal("Admin"), Schema.Literal("Moderator"), Schema.Literal("User")),
|
||||
values: Schema.Array(Schema.String),
|
||||
})
|
||||
|
||||
type TestFormTree = FormTree<typeof User>
|
||||
declare const testFormTree: TestFormTree
|
||||
|
||||
testFormTree.fields.roles.elements
|
||||
37
packages/extension-form/src/internal/guards.ts
Normal file
37
packages/extension-form/src/internal/guards.ts
Normal file
@@ -0,0 +1,37 @@
|
||||
import { Array, Predicate, Record, Schema, Tuple } from "effect"
|
||||
|
||||
|
||||
export const isTupleSchema = (u: unknown): u is Schema.Tuple<any> => (
|
||||
Schema.isSchema(u) &&
|
||||
Predicate.hasProperty(u, "elements") && Array.isArray(u.elements) &&
|
||||
Predicate.hasProperty(u, "rest") && Array.isArray(u.rest)
|
||||
)
|
||||
|
||||
export const isArraySchema = (u: unknown): u is Schema.Array$<any> => (
|
||||
Schema.isSchema(u) &&
|
||||
Predicate.hasProperty(u, "elements") && Array.isArray(u.elements) && Array.isEmptyArray(u.elements) &&
|
||||
Predicate.hasProperty(u, "rest") && Array.isArray(u.rest) && Tuple.isTupleOf(u.rest, 1) &&
|
||||
Predicate.hasProperty(u, "value")
|
||||
)
|
||||
|
||||
export const isStructSchema = (u: unknown): u is Schema.Struct<any> => (
|
||||
Schema.isSchema(u) &&
|
||||
Predicate.hasProperty(u, "fields") && Predicate.isObject(u.fields) &&
|
||||
Predicate.hasProperty(u, "records") && Array.isArray(u.records) && Array.isEmptyArray(u.records)
|
||||
)
|
||||
|
||||
export const isRecordSchema = (u: unknown): u is Schema.Record$<any, any> => (
|
||||
Schema.isSchema(u) &&
|
||||
Predicate.hasProperty(u, "fields") && Predicate.isObject(u.fields) && Record.isEmptyRecord(u.fields) &&
|
||||
Predicate.hasProperty(u, "records") && Array.isArray(u.records) &&
|
||||
Predicate.hasProperty(u, "key") &&
|
||||
Predicate.hasProperty(u, "value")
|
||||
)
|
||||
|
||||
|
||||
const myTuple = Schema.Tuple(Schema.String)
|
||||
const myArray = Schema.Array(Schema.String)
|
||||
const myStruct = Schema.Struct({})
|
||||
const myRecord = Schema.Record({ key: Schema.String, value: Schema.String })
|
||||
|
||||
console.log(isArraySchema(myTuple))
|
||||
1
packages/extension-form/src/internal/index.ts
Normal file
1
packages/extension-form/src/internal/index.ts
Normal file
@@ -0,0 +1 @@
|
||||
export * as Form from "./Form.js"
|
||||
33
packages/extension-form/tsconfig.json
Normal file
33
packages/extension-form/tsconfig.json
Normal file
@@ -0,0 +1,33 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
// Enable latest features
|
||||
"lib": ["ESNext", "DOM"],
|
||||
"target": "ESNext",
|
||||
"module": "NodeNext",
|
||||
"moduleDetection": "force",
|
||||
"jsx": "react-jsx",
|
||||
// "allowJs": true,
|
||||
|
||||
// Bundler mode
|
||||
"moduleResolution": "NodeNext",
|
||||
// "allowImportingTsExtensions": true,
|
||||
"verbatimModuleSyntax": true,
|
||||
// "noEmit": true,
|
||||
|
||||
// Best practices
|
||||
"strict": true,
|
||||
"skipLibCheck": true,
|
||||
"noFallthroughCasesInSwitch": true,
|
||||
|
||||
// Some stricter flags (disabled by default)
|
||||
"noUnusedLocals": false,
|
||||
"noUnusedParameters": false,
|
||||
"noPropertyAccessFromIndexSignature": false,
|
||||
|
||||
// Build
|
||||
"outDir": "./dist",
|
||||
"declaration": true
|
||||
},
|
||||
|
||||
"include": ["./src"]
|
||||
}
|
||||
Reference in New Issue
Block a user