TypeScript 允许在JSX表达式和组件类型之间有不同的属性,

rt4zxlrg  于 8个月前  发布在  TypeScript
关注(0)|答案(2)|浏览(146)

建议

🔍 搜索词

  • jsx 功能组件属性
  • 属性类型
  • 参数类型
  • jsx 属性强制转换

✅ 可实现性检查清单

我的建议符合以下指导原则:

  • 这不会对现有的 TypeScript/JavaScript 代码造成破坏性的更改
  • 这不会改变现有 JavaScript 代码的运行时行为
  • 这可以在不根据表达式的类型发出不同的 JS 的情况下实现
  • 这不是运行时特性(例如库功能、带有 JavaScript 输出的非 ECMAScript 语法、新的 JS 语法糖等)
  • 这个特性将与 TypeScript's Design Goals 的其他部分一致。

⭐ 建议

基于 #14729 的概念
JSX 命名空间中引入一个名为 convertProperties 的新函数(可能会更改),该函数接受一个组件并返回表示其 JSX props 的对象。

  1. declare namespace JSX {
  2. // today's behavior
  3. function convertProperties<P>(component: (JSX.ElementClass & { new(): { props: P } }) | ((props: P) => JSX.Element)): P;
  4. // alternatively, if this were to supercede IntrinsicAttributes rather than just working alongside it
  5. function convertProperties<P>(component: (props: P) => JSX.Element): P & JSX.IntrinsicAttributes;
  6. function convertProperties<P, T extends { props: P }>(component: JSX.ElementClass & { new(): T }): P & JSX.IntrinsicAttributes & JSX.IntrinsicClassAttributes<T>;
  7. }

当类型检查器看到一个组件表达式(例如 <MyComponent prop={val} /> ),它将通过在提供的组件类型上执行虚拟调用 JSX.convertProperties 并使用返回类型来确定所需属性的类型。例如:

  1. type SignalProps<P> = { [K in keyof P]: () => P[K] };
  2. declare namespace JSX {
  3. function convertProperties<P>(component: ((props: SignalProps<P>) => JSX.Element)): P;
  4. }
  5. type MyProps = SignalProps<{
  6. title: string;
  7. count: number;
  8. }>;
  9. function MyComponent({ title, count }: MyProps) {
  10. return <div>{title()}: {count()}</div>;
  11. }
  12. function App() {
  13. // typeof MyComponent == (props: { title: () => string, count: () => number }) => JSX.Element
  14. // typeof JSX.convertProperties(MyComponent) == { title: string, count: number }
  15. return <MyComponent title="Hello, World!" count={3} />;
  16. }

📃 动机示例

Solid 框架与 React 非常相似,但与 React 在 useMemouseEffect 中的显式依赖项数组不同,Solid 需要调用函数来检索状态值,从而允许基于上次 memo/effect 回调运行时获取的先前状态跟踪隐式依赖关系(请参阅 https://www.solidjs.com/guides/reactivity#introducing-primitives)。这与 Knockout 的可观察对象类似。
然而,当涉及到组件属性时,Solid 使用代理解引用而不是显式函数调用。这导致了一个常见的陷阱,即像在 React 中那样解构 props 对象会阻止跟踪依赖关系(请参阅 https://www.solidjs.com/guides/faq#why-do-i-lose-reactivity-when-i-destructure-props)。另一种设计可能是让 jsx 工厂函数在捆绑到 props 对象之前为每个 prop Package 一个响应式getter。
通过将属性类型在组件创建和 JSX 使用之间分开,TypeScript 将能够在新的网络框架中支持这种设计。

💻 用例

请参阅上面的内容。
另一个类似的用例可能是:

  1. type DetailedProps<P> = {
  2. [K in keyof P]: {
  3. value: P[K];
  4. prevValue?: P[K];
  5. }
  6. }
  7. declare namespace JSX {
  8. function convertProperties<P>(component: ((props: DetailedProps<P>) => JSX.Element)): P;
  9. }

此功能还将允许执行 JSX.IntrinsicAttributes 的相反操作。例如:

  1. declare namespace JSX {
  2. function convertProperties<P>(component: ((props: P) => JSX.Element)): Omit<P, 'parent'>;
  3. }
  4. type PropsWithParent<T> = T & {
  5. parent?: JSX.Element;
  6. };
  7. function MyComponent({ parent }: PropsWithParent<{}>) {
  8. // ...
  9. }
  10. function App() {
  11. return <MyComponent />;
  12. }
j8yoct9x

j8yoct9x1#

CC @weswigham

kninwzqo

kninwzqo2#

看起来#14729可以满足这个用例以及其他用例(Playground)。

相关问题