4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

LeveragesAdvent Calendar 2018

Day 4

typescriptの型拡張型でライブラリのメソッド入力に制限を行う

Last updated at Posted at 2018-12-03

今回はよく使う便利な型拡張のパターンを紹介します。

型拡張のやり方

今回題材として、react-navigation(今回諸事情で1系なのはご了承ください。)のnavigateメソッドをあつかいます。

このメソッドの第一引数はstring型なのですが、実際は登録したRoute以外の文字列を引数に入力すると、何も起こらず望んだ挙動になりません。

そこで、入力できる引数を自分たちが定義した文字列に限るようにするというのが今回のテーマです。こういった入力制限を型拡張でできることを覚えておくととても便利です。

以下はambientファイルと呼ばれるtypescriptの型定義ファイルです。これはtypescriptのincludeパスに入っているとコンパイル前に自動でこの型定義を使用してコンパイルしてくれるc言語でいうヘッダファイルのようなものです。module宣言はその後に続く文字列で指定したパスをimportしているソースファイルに対して、型定義の解析をしてからimportを行なってくれるため、既存のライブラリに存在する型定義に拡張を施すことができます。

関数のオーバーロードでは、さらに、入力引数に対する制約条件をいれることができるため、これによって、

react-navigation.d.ts
export * from 'react-navigation';
import { Routes } from '~/components/route';

declare module 'react-navigation' {
  export type RouteName = Routes;

  export namespace NavigationActions {
    function navigate<T>(options: T): NavigationNavigateAction;
  }

  export type TSRouteConfigMap = Partial<{ [routeName in RouteName]: NavigationRouteConfig }>;

  export function TabNavigator(routeConfigMap: TSRouteConfigMap, drawConfig?: TabNavigatorConfig): NavigationContainer;

  export function DrawerNavigator(
    routeConfigMap: TSRouteConfigMap,
    drawerConfig?: DrawerNavigatorConfig
  ): NavigationContainer;

  export function SwitchNavigator(
    routeConfigMap: TSRouteConfigMap,
    switchConfig?: SwitchNavigatorConfig
  ): NavigationContainer;

  export function StackNavigator(
    routeConfigMap: TSRouteConfigMap,
    stackConfig?: StackNavigatorConfig
  ): NavigationContainer;

  export interface NavigationScreenProp<S, P = NavigationParams> {
    navigate<T>(options: { routeName: T; params?: NavigationParams; action?: NavigationAction; key?: string }): boolean;
    navigate<T>(routeNameOrOptions: T, params?: NavigationParams, action?: NavigationAction): boolean;
  }
}

また components/routes.tsの中身は例えば、以下のように定義されています。

routes.ts
// .../部分抜粋
export type Routes =
  'Entrance'
  | 'Entrance/SignUp'
  | 'Entrance/Login'
  | 'Entrance/SignUp/MailSignUp'
  | 'Entrance/SignUp/Register'
  | 'Entrance/Register'

呼び出し先で何が起きるか

以上がうちのチームで導入指定しているreact-navigationの型定義になり複雑そうに見えますが、やってることはRouteの登録時に行うrouteConfigMapの制約条件として、RouteNameという新たな型を導入しそこで定義されている文字列以外にkeyにすることができない制限と、navigateメソッドを使用する可能性がある型のnavigateメソッドにGeneric型を適用しているだけです。なぜnavigateメソッドの引数をRouteNameのオーバーロードにしないかというと、オーバーロードにすると、navigateはすでにstringを許容するメソッドであるため、補完をすることができても入力制限まではできませんが、navigateという型パラメータ付きで、実行することでなく文字列補完だけではなく、入力制限まで行うことができるからです。

以下はGeneric補完で、ジェネリックメソッドに型パラメータRouteを指定したことによりstring型のオーバーロードの影響を受けず警告が出ている図
スクリーンショット 2018-11-25 18.51.40.png
以下は、オーバーロード引数stringが有効になってしまっているため警告が出ていない図です。
スクリーンショット 2018-11-25 18.53.51.png

createNavigatorの引数に対してのRoutesNameの制限に関しては、以下のように、独自定義型なので、一度変数にいれて、検証することで、解決していますがこちらも、Genericにして解決しても構いません。

スクリーンショット 2018-11-25 19.03.48.png

以上のようにすでに、string型で入力可能な状態になっているライブラリ内のメソッドであってもGeneric型をうまく使いこなすことによって、このテクニックは様々な場面で大活躍するので、覚えておいて損はないでしょう。

4
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?