@jgamaraalv/ts-dev-kit 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (117) hide show
  1. package/.claude-plugin/marketplace.json +24 -0
  2. package/.claude-plugin/plugin.json +24 -0
  3. package/CHANGELOG.md +24 -0
  4. package/LICENSE +21 -0
  5. package/README.md +128 -0
  6. package/agents/accessibility-pro.md +139 -0
  7. package/agents/api-builder.md +110 -0
  8. package/agents/code-reviewer.md +190 -0
  9. package/agents/database-expert.md +138 -0
  10. package/agents/debugger.md +241 -0
  11. package/agents/docker-expert.md +51 -0
  12. package/agents/multi-agent-coordinator.md +378 -0
  13. package/agents/nextjs-expert.md +136 -0
  14. package/agents/performance-engineer.md +138 -0
  15. package/agents/playwright-expert.md +126 -0
  16. package/agents/react-specialist.md +97 -0
  17. package/agents/security-scanner.md +105 -0
  18. package/agents/test-generator.md +221 -0
  19. package/agents/typescript-pro.md +253 -0
  20. package/agents/ux-optimizer.md +93 -0
  21. package/docs/rules/orchestration.md.template +126 -0
  22. package/package.json +28 -0
  23. package/skills/bullmq/SKILL.md +225 -0
  24. package/skills/bullmq/references/flows-and-schedulers.md +186 -0
  25. package/skills/bullmq/references/job-types-and-options.md +163 -0
  26. package/skills/bullmq/references/patterns.md +273 -0
  27. package/skills/bullmq/references/production.md +308 -0
  28. package/skills/composition-patterns/SKILL.md +58 -0
  29. package/skills/composition-patterns/references/architecture-avoid-boolean-props.md +87 -0
  30. package/skills/composition-patterns/references/architecture-compound-components.md +107 -0
  31. package/skills/composition-patterns/references/patterns-children-over-render-props.md +77 -0
  32. package/skills/composition-patterns/references/patterns-explicit-variants.md +87 -0
  33. package/skills/composition-patterns/references/react19-no-forwardref.md +37 -0
  34. package/skills/composition-patterns/references/state-context-interface.md +194 -0
  35. package/skills/composition-patterns/references/state-decouple-implementation.md +96 -0
  36. package/skills/composition-patterns/references/state-lift-state.md +126 -0
  37. package/skills/conventional-commits/SKILL.md +148 -0
  38. package/skills/docker/SKILL.md +55 -0
  39. package/skills/docker/references/compose-configs.md +95 -0
  40. package/skills/docker/references/monorepo-dockerfile.md +111 -0
  41. package/skills/drizzle-pg/SKILL.md +202 -0
  42. package/skills/drizzle-pg/references/advanced.md +299 -0
  43. package/skills/drizzle-pg/references/migrations.md +214 -0
  44. package/skills/drizzle-pg/references/queries.md +321 -0
  45. package/skills/drizzle-pg/references/relations.md +272 -0
  46. package/skills/drizzle-pg/references/schema-pg.md +256 -0
  47. package/skills/drizzle-pg/references/sql-operator.md +215 -0
  48. package/skills/fastify-best-practices/SKILL.md +143 -0
  49. package/skills/fastify-best-practices/references/hooks-and-lifecycle.md +122 -0
  50. package/skills/fastify-best-practices/references/plugins-and-encapsulation.md +137 -0
  51. package/skills/fastify-best-practices/references/request-reply-errors.md +189 -0
  52. package/skills/fastify-best-practices/references/routes-and-handlers.md +134 -0
  53. package/skills/fastify-best-practices/references/server-and-options.md +127 -0
  54. package/skills/fastify-best-practices/references/typescript-and-logging.md +223 -0
  55. package/skills/fastify-best-practices/references/validation-and-serialization.md +190 -0
  56. package/skills/ioredis/SKILL.md +51 -0
  57. package/skills/ioredis/references/advanced-patterns.md +312 -0
  58. package/skills/ioredis/references/cluster-sentinel.md +280 -0
  59. package/skills/ioredis/references/connection-options.md +187 -0
  60. package/skills/ioredis/references/core-api.md +179 -0
  61. package/skills/nextjs-best-practices/SKILL.md +194 -0
  62. package/skills/nextjs-best-practices/references/async-patterns.md +84 -0
  63. package/skills/nextjs-best-practices/references/bundling.md +192 -0
  64. package/skills/nextjs-best-practices/references/data-patterns.md +310 -0
  65. package/skills/nextjs-best-practices/references/debug-tricks.md +127 -0
  66. package/skills/nextjs-best-practices/references/directives.md +74 -0
  67. package/skills/nextjs-best-practices/references/error-handling.md +237 -0
  68. package/skills/nextjs-best-practices/references/file-conventions.md +152 -0
  69. package/skills/nextjs-best-practices/references/font.md +175 -0
  70. package/skills/nextjs-best-practices/references/functions.md +116 -0
  71. package/skills/nextjs-best-practices/references/hydration-error.md +86 -0
  72. package/skills/nextjs-best-practices/references/image.md +184 -0
  73. package/skills/nextjs-best-practices/references/metadata.md +305 -0
  74. package/skills/nextjs-best-practices/references/parallel-routes.md +299 -0
  75. package/skills/nextjs-best-practices/references/route-handlers.md +154 -0
  76. package/skills/nextjs-best-practices/references/rsc-boundaries.md +168 -0
  77. package/skills/nextjs-best-practices/references/runtime-selection.md +40 -0
  78. package/skills/nextjs-best-practices/references/scripts.md +148 -0
  79. package/skills/nextjs-best-practices/references/self-hosting.md +210 -0
  80. package/skills/nextjs-best-practices/references/suspense-boundaries.md +67 -0
  81. package/skills/owasp-security-review/SKILL.md +98 -0
  82. package/skills/owasp-security-review/references/a01-broken-access-control.md +78 -0
  83. package/skills/owasp-security-review/references/a02-security-misconfiguration.md +81 -0
  84. package/skills/owasp-security-review/references/a03-supply-chain-failures.md +65 -0
  85. package/skills/owasp-security-review/references/a04-cryptographic-failures.md +82 -0
  86. package/skills/owasp-security-review/references/a05-injection.md +106 -0
  87. package/skills/owasp-security-review/references/a06-insecure-design.md +76 -0
  88. package/skills/owasp-security-review/references/a07-authentication-failures.md +83 -0
  89. package/skills/owasp-security-review/references/a08-integrity-failures.md +72 -0
  90. package/skills/owasp-security-review/references/a09-logging-alerting-failures.md +76 -0
  91. package/skills/owasp-security-review/references/a10-exceptional-conditions.md +131 -0
  92. package/skills/postgresql/SKILL.md +50 -0
  93. package/skills/postgresql/references/ddl-schema.md +300 -0
  94. package/skills/postgresql/references/indexes.md +257 -0
  95. package/skills/postgresql/references/jsonb.md +261 -0
  96. package/skills/postgresql/references/performance.md +291 -0
  97. package/skills/postgresql/references/psql-cli.md +153 -0
  98. package/skills/postgresql/references/queries.md +287 -0
  99. package/skills/postgresql/references/transactions.md +280 -0
  100. package/skills/react-best-practices/SKILL.md +110 -0
  101. package/skills/react-best-practices/references/advanced-patterns.md +91 -0
  102. package/skills/react-best-practices/references/async-patterns.md +233 -0
  103. package/skills/react-best-practices/references/bundle-optimization.md +201 -0
  104. package/skills/react-best-practices/references/client-patterns.md +178 -0
  105. package/skills/react-best-practices/references/js-performance.md +210 -0
  106. package/skills/react-best-practices/references/rendering-performance.md +209 -0
  107. package/skills/react-best-practices/references/rerender-optimization.md +316 -0
  108. package/skills/react-best-practices/references/server-performance.md +274 -0
  109. package/skills/service-worker/SKILL.md +195 -0
  110. package/skills/service-worker/references/api-reference.md +114 -0
  111. package/skills/service-worker/references/caching-strategies.md +202 -0
  112. package/skills/service-worker/references/push-and-sync.md +261 -0
  113. package/skills/typescript-conventions/SKILL.md +51 -0
  114. package/skills/ui-ux-guidelines/SKILL.md +105 -0
  115. package/skills/ui-ux-guidelines/references/accessibility-and-interaction.md +74 -0
  116. package/skills/ui-ux-guidelines/references/forms-content-checklist.md +126 -0
  117. package/skills/ui-ux-guidelines/references/layout-typography-animation.md +95 -0
@@ -0,0 +1,87 @@
1
+ ## Avoid Boolean Prop Proliferation
2
+
3
+ Don't add boolean props like `isThread`, `isEditing`, `isDMThread` to customize
4
+ component behavior. Each boolean doubles possible states and creates
5
+ unmaintainable conditional logic. Use composition instead.
6
+
7
+ **Incorrect (boolean props create exponential complexity):**
8
+
9
+ ```tsx
10
+ function Composer({
11
+ onSubmit,
12
+ isThread,
13
+ channelId,
14
+ isDMThread,
15
+ dmId,
16
+ isEditing,
17
+ isForwarding,
18
+ }: Props) {
19
+ return (
20
+ <form>
21
+ <Header />
22
+ <Input />
23
+ {isDMThread ? (
24
+ <AlsoSendToDMField id={dmId} />
25
+ ) : isThread ? (
26
+ <AlsoSendToChannelField id={channelId} />
27
+ ) : null}
28
+ {isEditing ? <EditActions /> : isForwarding ? <ForwardActions /> : <DefaultActions />}
29
+ <Footer onSubmit={onSubmit} />
30
+ </form>
31
+ );
32
+ }
33
+ ```
34
+
35
+ **Correct (composition eliminates conditionals):**
36
+
37
+ ```tsx
38
+ // Channel composer
39
+ function ChannelComposer() {
40
+ return (
41
+ <Composer.Frame>
42
+ <Composer.Header />
43
+ <Composer.Input />
44
+ <Composer.Footer>
45
+ <Composer.Attachments />
46
+ <Composer.Formatting />
47
+ <Composer.Emojis />
48
+ <Composer.Submit />
49
+ </Composer.Footer>
50
+ </Composer.Frame>
51
+ );
52
+ }
53
+
54
+ // Thread composer - adds "also send to channel" field
55
+ function ThreadComposer({ channelId }: { channelId: string }) {
56
+ return (
57
+ <Composer.Frame>
58
+ <Composer.Header />
59
+ <Composer.Input />
60
+ <AlsoSendToChannelField id={channelId} />
61
+ <Composer.Footer>
62
+ <Composer.Formatting />
63
+ <Composer.Emojis />
64
+ <Composer.Submit />
65
+ </Composer.Footer>
66
+ </Composer.Frame>
67
+ );
68
+ }
69
+
70
+ // Edit composer - different footer actions
71
+ function EditComposer() {
72
+ return (
73
+ <Composer.Frame>
74
+ <Composer.Input />
75
+ <Composer.Footer>
76
+ <Composer.Formatting />
77
+ <Composer.Emojis />
78
+ <Composer.CancelEdit />
79
+ <Composer.SaveEdit />
80
+ </Composer.Footer>
81
+ </Composer.Frame>
82
+ );
83
+ }
84
+ ```
85
+
86
+ Each variant is explicit about what it renders. We can share internals without
87
+ sharing a single monolithic parent.
@@ -0,0 +1,107 @@
1
+ ## Use Compound Components
2
+
3
+ ### Table of Contents
4
+
5
+ - [Incorrect: Monolithic Component with Render Props](#incorrect-monolithic-component-with-render-props)
6
+ - [Correct: Compound Components with Shared Context](#correct-compound-components-with-shared-context)
7
+ - [Usage](#usage)
8
+
9
+ Structure complex components as compound components with a shared context. Each
10
+ subcomponent accesses shared state via context, not props. Consumers compose the
11
+ pieces they need.
12
+
13
+ **Incorrect (monolithic component with render props):**
14
+
15
+ ```tsx
16
+ function Composer({
17
+ renderHeader,
18
+ renderFooter,
19
+ renderActions,
20
+ showAttachments,
21
+ showFormatting,
22
+ showEmojis,
23
+ }: Props) {
24
+ return (
25
+ <form>
26
+ {renderHeader?.()}
27
+ <Input />
28
+ {showAttachments && <Attachments />}
29
+ {renderFooter ? (
30
+ renderFooter()
31
+ ) : (
32
+ <Footer>
33
+ {showFormatting && <Formatting />}
34
+ {showEmojis && <Emojis />}
35
+ {renderActions?.()}
36
+ </Footer>
37
+ )}
38
+ </form>
39
+ );
40
+ }
41
+ ```
42
+
43
+ **Correct (compound components with shared context):**
44
+
45
+ ```tsx
46
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
47
+
48
+ function ComposerProvider({ children, state, actions, meta }: ProviderProps) {
49
+ return <ComposerContext value={{ state, actions, meta }}>{children}</ComposerContext>;
50
+ }
51
+
52
+ function ComposerFrame({ children }: { children: React.ReactNode }) {
53
+ return <form>{children}</form>;
54
+ }
55
+
56
+ function ComposerInput() {
57
+ const {
58
+ state,
59
+ actions: { update },
60
+ meta: { inputRef },
61
+ } = use(ComposerContext);
62
+ return (
63
+ <TextInput
64
+ ref={inputRef}
65
+ value={state.input}
66
+ onChangeText={(text) => update((s) => ({ ...s, input: text }))}
67
+ />
68
+ );
69
+ }
70
+
71
+ function ComposerSubmit() {
72
+ const {
73
+ actions: { submit },
74
+ } = use(ComposerContext);
75
+ return <Button onPress={submit}>Send</Button>;
76
+ }
77
+
78
+ // Export as compound component
79
+ const Composer = {
80
+ Provider: ComposerProvider,
81
+ Frame: ComposerFrame,
82
+ Input: ComposerInput,
83
+ Submit: ComposerSubmit,
84
+ Header: ComposerHeader,
85
+ Footer: ComposerFooter,
86
+ Attachments: ComposerAttachments,
87
+ Formatting: ComposerFormatting,
88
+ Emojis: ComposerEmojis,
89
+ };
90
+ ```
91
+
92
+ **Usage:**
93
+
94
+ ```tsx
95
+ <Composer.Provider state={state} actions={actions} meta={meta}>
96
+ <Composer.Frame>
97
+ <Composer.Header />
98
+ <Composer.Input />
99
+ <Composer.Footer>
100
+ <Composer.Formatting />
101
+ <Composer.Submit />
102
+ </Composer.Footer>
103
+ </Composer.Frame>
104
+ </Composer.Provider>
105
+ ```
106
+
107
+ Consumers explicitly compose exactly what they need. No hidden conditionals. And the state, actions and meta are dependency-injected by a parent provider, allowing multiple usages of the same component structure.
@@ -0,0 +1,77 @@
1
+ ## Prefer Children Over Render Props
2
+
3
+ Use `children` for composition instead of `renderX` props. Children are more
4
+ readable, compose naturally, and don't require understanding callback
5
+ signatures.
6
+
7
+ **Incorrect (render props):**
8
+
9
+ ```tsx
10
+ function Composer({
11
+ renderHeader,
12
+ renderFooter,
13
+ renderActions,
14
+ }: {
15
+ renderHeader?: () => React.ReactNode;
16
+ renderFooter?: () => React.ReactNode;
17
+ renderActions?: () => React.ReactNode;
18
+ }) {
19
+ return (
20
+ <form>
21
+ {renderHeader?.()}
22
+ <Input />
23
+ {renderFooter ? renderFooter() : <DefaultFooter />}
24
+ {renderActions?.()}
25
+ </form>
26
+ );
27
+ }
28
+
29
+ // Usage is awkward and inflexible
30
+ return (
31
+ <Composer
32
+ renderHeader={() => <CustomHeader />}
33
+ renderFooter={() => (
34
+ <>
35
+ <Formatting />
36
+ <Emojis />
37
+ </>
38
+ )}
39
+ renderActions={() => <SubmitButton />}
40
+ />
41
+ );
42
+ ```
43
+
44
+ **Correct (compound components with children):**
45
+
46
+ ```tsx
47
+ function ComposerFrame({ children }: { children: React.ReactNode }) {
48
+ return <form>{children}</form>;
49
+ }
50
+
51
+ function ComposerFooter({ children }: { children: React.ReactNode }) {
52
+ return <footer className="flex">{children}</footer>;
53
+ }
54
+
55
+ // Usage is flexible
56
+ return (
57
+ <Composer.Frame>
58
+ <CustomHeader />
59
+ <Composer.Input />
60
+ <Composer.Footer>
61
+ <Composer.Formatting />
62
+ <Composer.Emojis />
63
+ <SubmitButton />
64
+ </Composer.Footer>
65
+ </Composer.Frame>
66
+ );
67
+ ```
68
+
69
+ **When render props are appropriate:**
70
+
71
+ ```tsx
72
+ // Render props work well when you need to pass data back
73
+ <List data={items} renderItem={({ item, index }) => <Item item={item} index={index} />} />
74
+ ```
75
+
76
+ Use render props when the parent needs to provide data or state to the child.
77
+ Use children when composing static structure.
@@ -0,0 +1,87 @@
1
+ ## Create Explicit Component Variants
2
+
3
+ Instead of one component with many boolean props, create explicit variant
4
+ components. Each variant composes the pieces it needs. The code documents
5
+ itself.
6
+
7
+ **Incorrect (one component, many modes):**
8
+
9
+ ```tsx
10
+ // What does this component actually render?
11
+ <Composer isThread isEditing={false} channelId="abc" showAttachments showFormatting={false} />
12
+ ```
13
+
14
+ **Correct (explicit variants):**
15
+
16
+ ```tsx
17
+ // Immediately clear what this renders
18
+ <ThreadComposer channelId="abc" />
19
+
20
+ // Or
21
+ <EditMessageComposer messageId="xyz" />
22
+
23
+ // Or
24
+ <ForwardMessageComposer messageId="123" />
25
+ ```
26
+
27
+ Each implementation is unique, explicit and self-contained. Yet they can each
28
+ use shared parts.
29
+
30
+ **Implementation:**
31
+
32
+ ```tsx
33
+ function ThreadComposer({ channelId }: { channelId: string }) {
34
+ return (
35
+ <ThreadProvider channelId={channelId}>
36
+ <Composer.Frame>
37
+ <Composer.Input />
38
+ <AlsoSendToChannelField channelId={channelId} />
39
+ <Composer.Footer>
40
+ <Composer.Formatting />
41
+ <Composer.Emojis />
42
+ <Composer.Submit />
43
+ </Composer.Footer>
44
+ </Composer.Frame>
45
+ </ThreadProvider>
46
+ );
47
+ }
48
+
49
+ function EditMessageComposer({ messageId }: { messageId: string }) {
50
+ return (
51
+ <EditMessageProvider messageId={messageId}>
52
+ <Composer.Frame>
53
+ <Composer.Input />
54
+ <Composer.Footer>
55
+ <Composer.Formatting />
56
+ <Composer.Emojis />
57
+ <Composer.CancelEdit />
58
+ <Composer.SaveEdit />
59
+ </Composer.Footer>
60
+ </Composer.Frame>
61
+ </EditMessageProvider>
62
+ );
63
+ }
64
+
65
+ function ForwardMessageComposer({ messageId }: { messageId: string }) {
66
+ return (
67
+ <ForwardMessageProvider messageId={messageId}>
68
+ <Composer.Frame>
69
+ <Composer.Input placeholder="Add a message, if you'd like." />
70
+ <Composer.Footer>
71
+ <Composer.Formatting />
72
+ <Composer.Emojis />
73
+ <Composer.Mentions />
74
+ </Composer.Footer>
75
+ </Composer.Frame>
76
+ </ForwardMessageProvider>
77
+ );
78
+ }
79
+ ```
80
+
81
+ Each variant is explicit about:
82
+
83
+ - What provider/state it uses
84
+ - What UI elements it includes
85
+ - What actions are available
86
+
87
+ No boolean prop combinations to reason about. No impossible states.
@@ -0,0 +1,37 @@
1
+ ## React 19 API Changes
2
+
3
+ > **⚠️ React 19+ only.** Skip this if you're on React 18 or earlier.
4
+
5
+ In React 19, `ref` is now a regular prop (no `forwardRef` wrapper needed), and `use()` replaces `useContext()`.
6
+
7
+ **Incorrect (forwardRef in React 19):**
8
+
9
+ ```tsx
10
+ const ComposerInput = forwardRef<HTMLInputElement, Props>((props, ref) => {
11
+ return <input ref={ref} {...props} />;
12
+ });
13
+ ```
14
+
15
+ **Correct (ref as a regular prop):**
16
+
17
+ ```tsx
18
+ function ComposerInput({ ref, ...props }: Props & { ref?: React.Ref<HTMLInputElement> }) {
19
+ return <input ref={ref} {...props} />;
20
+ }
21
+ ```
22
+
23
+ **Incorrect (useContext in React 19):**
24
+
25
+ ```tsx
26
+ const value = useContext(MyContext);
27
+ ```
28
+
29
+ **Correct (use instead of useContext):**
30
+
31
+ ```tsx
32
+ import { use } from "react";
33
+
34
+ const value = use(MyContext);
35
+ ```
36
+
37
+ `use()` can also be called conditionally, unlike `useContext()`.
@@ -0,0 +1,194 @@
1
+ ## Define Generic Context Interfaces for Dependency Injection
2
+
3
+ ### Table of Contents
4
+
5
+ - [Core Principle](#core-principle)
6
+ - [Incorrect: UI Coupled to Specific State](#incorrect-ui-coupled-to-specific-state)
7
+ - [Correct: Generic Interface](#correct-generic-interface)
8
+ - [UI Components Consume the Interface](#ui-components-consume-the-interface)
9
+ - [Different Providers Implement the Same Interface](#different-providers-implement-the-same-interface)
10
+ - [Same Composed UI Works with Both](#same-composed-ui-works-with-both)
11
+ - [Custom UI Outside the Component](#custom-ui-outside-the-component)
12
+
13
+ Define a **generic interface** for your component context with three parts:
14
+ `state`, `actions`, and `meta`. This interface is a contract that any provider
15
+ can implement—enabling the same UI components to work with completely different
16
+ state implementations.
17
+
18
+ **Core principle:** Lift state, compose internals, make state
19
+ dependency-injectable.
20
+
21
+ **Incorrect (UI coupled to specific state implementation):**
22
+
23
+ ```tsx
24
+ function ComposerInput() {
25
+ // Tightly coupled to a specific hook
26
+ const { input, setInput } = useChannelComposerState();
27
+ return <TextInput value={input} onChangeText={setInput} />;
28
+ }
29
+ ```
30
+
31
+ **Correct (generic interface enables dependency injection):**
32
+
33
+ ```tsx
34
+ // Define a GENERIC interface that any provider can implement
35
+ interface ComposerState {
36
+ input: string;
37
+ attachments: Attachment[];
38
+ isSubmitting: boolean;
39
+ }
40
+
41
+ interface ComposerActions {
42
+ update: (updater: (state: ComposerState) => ComposerState) => void;
43
+ submit: () => void;
44
+ }
45
+
46
+ interface ComposerMeta {
47
+ inputRef: React.RefObject<TextInput>;
48
+ }
49
+
50
+ interface ComposerContextValue {
51
+ state: ComposerState;
52
+ actions: ComposerActions;
53
+ meta: ComposerMeta;
54
+ }
55
+
56
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
57
+ ```
58
+
59
+ **UI components consume the interface, not the implementation:**
60
+
61
+ ```tsx
62
+ function ComposerInput() {
63
+ const {
64
+ state,
65
+ actions: { update },
66
+ meta,
67
+ } = use(ComposerContext);
68
+
69
+ // This component works with ANY provider that implements the interface
70
+ return (
71
+ <TextInput
72
+ ref={meta.inputRef}
73
+ value={state.input}
74
+ onChangeText={(text) => update((s) => ({ ...s, input: text }))}
75
+ />
76
+ );
77
+ }
78
+ ```
79
+
80
+ **Different providers implement the same interface:**
81
+
82
+ ```tsx
83
+ // Provider A: Local state for ephemeral forms
84
+ function ForwardMessageProvider({ children }: { children: React.ReactNode }) {
85
+ const [state, setState] = useState(initialState);
86
+ const inputRef = useRef(null);
87
+ const submit = useForwardMessage();
88
+
89
+ return (
90
+ <ComposerContext
91
+ value={{
92
+ state,
93
+ actions: { update: setState, submit },
94
+ meta: { inputRef },
95
+ }}
96
+ >
97
+ {children}
98
+ </ComposerContext>
99
+ );
100
+ }
101
+
102
+ // Provider B: Global synced state for channels
103
+ function ChannelProvider({ channelId, children }: Props) {
104
+ const { state, update, submit } = useGlobalChannel(channelId);
105
+ const inputRef = useRef(null);
106
+
107
+ return (
108
+ <ComposerContext
109
+ value={{
110
+ state,
111
+ actions: { update, submit },
112
+ meta: { inputRef },
113
+ }}
114
+ >
115
+ {children}
116
+ </ComposerContext>
117
+ );
118
+ }
119
+ ```
120
+
121
+ **The same composed UI works with both:**
122
+
123
+ ```tsx
124
+ // Works with ForwardMessageProvider (local state)
125
+ <ForwardMessageProvider>
126
+ <Composer.Frame>
127
+ <Composer.Input />
128
+ <Composer.Submit />
129
+ </Composer.Frame>
130
+ </ForwardMessageProvider>
131
+
132
+ // Works with ChannelProvider (global synced state)
133
+ <ChannelProvider channelId="abc">
134
+ <Composer.Frame>
135
+ <Composer.Input />
136
+ <Composer.Submit />
137
+ </Composer.Frame>
138
+ </ChannelProvider>
139
+ ```
140
+
141
+ **Custom UI outside the component can access state and actions:**
142
+
143
+ The provider boundary is what matters—not the visual nesting. Components that
144
+ need shared state don't have to be inside the `Composer.Frame`. They just need
145
+ to be within the provider.
146
+
147
+ ```tsx
148
+ function ForwardMessageDialog() {
149
+ return (
150
+ <ForwardMessageProvider>
151
+ <Dialog>
152
+ {/* The composer UI */}
153
+ <Composer.Frame>
154
+ <Composer.Input placeholder="Add a message, if you'd like." />
155
+ <Composer.Footer>
156
+ <Composer.Formatting />
157
+ <Composer.Emojis />
158
+ </Composer.Footer>
159
+ </Composer.Frame>
160
+
161
+ {/* Custom UI OUTSIDE the composer, but INSIDE the provider */}
162
+ <MessagePreview />
163
+
164
+ {/* Actions at the bottom of the dialog */}
165
+ <DialogActions>
166
+ <CancelButton />
167
+ <ForwardButton />
168
+ </DialogActions>
169
+ </Dialog>
170
+ </ForwardMessageProvider>
171
+ );
172
+ }
173
+
174
+ // This button lives OUTSIDE Composer.Frame but can still submit based on its context!
175
+ function ForwardButton() {
176
+ const {
177
+ actions: { submit },
178
+ } = use(ComposerContext);
179
+ return <Button onPress={submit}>Forward</Button>;
180
+ }
181
+
182
+ // This preview lives OUTSIDE Composer.Frame but can read composer's state!
183
+ function MessagePreview() {
184
+ const { state } = use(ComposerContext);
185
+ return <Preview message={state.input} attachments={state.attachments} />;
186
+ }
187
+ ```
188
+
189
+ The `ForwardButton` and `MessagePreview` are not visually inside the composer
190
+ box, but they can still access its state and actions. This is the power of
191
+ lifting state into providers.
192
+
193
+ The UI is reusable bits you compose together. The state is dependency-injected
194
+ by the provider. Swap the provider, keep the UI.
@@ -0,0 +1,96 @@
1
+ ## Decouple State Management from UI
2
+
3
+ The provider component should be the only place that knows how state is managed.
4
+ UI components consume the context interface—they don't know if state comes from
5
+ useState, Zustand, or a server sync.
6
+
7
+ **Incorrect (UI coupled to state implementation):**
8
+
9
+ ```tsx
10
+ function ChannelComposer({ channelId }: { channelId: string }) {
11
+ // UI component knows about global state implementation
12
+ const state = useGlobalChannelState(channelId);
13
+ const { submit, updateInput } = useChannelSync(channelId);
14
+
15
+ return (
16
+ <Composer.Frame>
17
+ <Composer.Input value={state.input} onChange={(text) => sync.updateInput(text)} />
18
+ <Composer.Submit onPress={() => sync.submit()} />
19
+ </Composer.Frame>
20
+ );
21
+ }
22
+ ```
23
+
24
+ **Correct (state management isolated in provider):**
25
+
26
+ ```tsx
27
+ // Provider handles all state management details
28
+ function ChannelProvider({
29
+ channelId,
30
+ children,
31
+ }: {
32
+ channelId: string;
33
+ children: React.ReactNode;
34
+ }) {
35
+ const { state, update, submit } = useGlobalChannel(channelId);
36
+ const inputRef = useRef(null);
37
+
38
+ return (
39
+ <Composer.Provider state={state} actions={{ update, submit }} meta={{ inputRef }}>
40
+ {children}
41
+ </Composer.Provider>
42
+ );
43
+ }
44
+
45
+ // UI component only knows about the context interface
46
+ function ChannelComposer() {
47
+ return (
48
+ <Composer.Frame>
49
+ <Composer.Header />
50
+ <Composer.Input />
51
+ <Composer.Footer>
52
+ <Composer.Submit />
53
+ </Composer.Footer>
54
+ </Composer.Frame>
55
+ );
56
+ }
57
+
58
+ // Usage
59
+ function Channel({ channelId }: { channelId: string }) {
60
+ return (
61
+ <ChannelProvider channelId={channelId}>
62
+ <ChannelComposer />
63
+ </ChannelProvider>
64
+ );
65
+ }
66
+ ```
67
+
68
+ **Different providers, same UI:**
69
+
70
+ ```tsx
71
+ // Local state for ephemeral forms
72
+ function ForwardMessageProvider({ children }) {
73
+ const [state, setState] = useState(initialState);
74
+ const forwardMessage = useForwardMessage();
75
+
76
+ return (
77
+ <Composer.Provider state={state} actions={{ update: setState, submit: forwardMessage }}>
78
+ {children}
79
+ </Composer.Provider>
80
+ );
81
+ }
82
+
83
+ // Global synced state for channels
84
+ function ChannelProvider({ channelId, children }) {
85
+ const { state, update, submit } = useGlobalChannel(channelId);
86
+
87
+ return (
88
+ <Composer.Provider state={state} actions={{ update, submit }}>
89
+ {children}
90
+ </Composer.Provider>
91
+ );
92
+ }
93
+ ```
94
+
95
+ The same `Composer.Input` component works with both providers because it only
96
+ depends on the context interface, not the implementation.