typescript 基于参数值的返回类型

4uqofj5v  于 2023-01-14  发布在  TypeScript
关注(0)|答案(1)|浏览(119)

我的目标是为不同的参数类型提供不同的类型
下面是我的代码

import React from "react";

type Response<T> = {
  data: T;
  description: string;
  date: string;
};
type Data = {
  title: string;
  name: string;
};
type DataWidthLabel = Data & { labels: string };
type Endpoint = "endpoint_1" | "endpoint_2";

type FnResponse<T> = 
  T extends "endpoint_1" ? Response<Data> :
  T extends "endpoint_2" ? Response<DataWidthLabel> : never;

type Props = {
  endpoint: Endpoint;
  children: (data: Response<Data> | Response<DataWidthLabel>) => void;
};

function queryFunction<T extends Endpoint>(endpoint: T): FnResponse<T> {
  if (endpoint === "endpoint_1") {
    return {
      data: {
        title: "title",
        name: "name"
      },
      description: "description",
      date: "date"
    } as FnResponse<T>;
  }

  return {
    data: {
      title: "title",
      name: "name",
      labels: "labels"
    },
    description: "description",
    date: "date"
  } as FnResponse<T>;
}

const DataComp: React.FC<Props> = ({ endpoint, children }) => {
  // queryFunction takes a type argument (<myType>) to type the data response
  const data = queryFunction(endpoint);

  return (
    <div>
      {children(data)}

      <div>
        <p>{data.description}</p>
        <p>{data.date}</p>
      </div>
    </div>
  );
};

const ParentComp1 = () => (
  <DataComp endpoint="endpoint_1">
    {(data) => {
      return data.data.name;
    }}
  </DataComp>
);

const ParentComp2 = () => (
  <DataComp endpoint="endpoint_2">
    {(data) => {
      return data.data.labels;
    }}
  </DataComp>
);

export default function App() {
  return (
    <div className="App">
      <ParentComp1 />
      -----
      <ParentComp2 />
    </div>
  );
}

我无法修复案例return data.data.labels;的错误

rjzwgtxy

rjzwgtxy1#

如果使Props成为泛型,则可以说data中的参数必须是endpoint字段的FnResponse

type Props<E extends Endpoint> = {
  endpoint: E;
  children: (data: FnResponse<E>) => void;
};

此外,您还可以将FnResponse更改为查找表:

type FnResponse<T extends string> = Response<({
    "endpoint_1": Data;
    "endpoint_2": DataWidthLabel;
} & { [key: string]: never })[T]>;

最后,组件本身需要是泛型的,这样我们就可以将泛型参数传递给Props

function DataComp<E extends Endpoint>({ endpoint, children }: Props<E>) {

// NO React.FC!
const DataComp = <E extends Endpoint>({ endpoint, children }: Props<E>) => {

Playground

相关问题