by null @uid11
TypeScript は構造的型システムを持っていますが、場合によっては(公称型システムのように)あらかじめ定義されたユニークなオブジェクトだけを受け入れ、required なフィールドを持つオブジェクトは受け入れないようにしたいこともあるでしょう。
引数にオブジェクトを受け取り、引数に含まれる全てのオブジェクトの文字列と数値のキー、およびこれらのキーの値のプロパティーを保持しながら、引数のオブジェクトとそれに含まれるネストしたオブジェクト全てをユニークにする型を実装します。
元の型と実装した型のアウトプットの型は相互に代入可能でなければなりませんが、同一の型であってはいけません。
例えば、
import { Equal } from '@type-challenges/utils';
type Foo = { foo: 2; bar: { 0: 1 }; baz: { 0: 1 } };
type UniqFoo = DeepObjectToUniq<Foo>;
declare let foo: Foo;
declare let uniqFoo: UniqFoo;
uniqFoo = foo; // ok
foo = uniqFoo; // ok
type T0 = Equal<UniqFoo, Foo>; // false
type T1 = UniqFoo['foo']; // 2
type T2 = Equal<UniqFoo['bar'], UniqFoo['baz']>; // false
type T3 = UniqFoo['bar'][0]; // 1
type T4 = Equal<keyof Foo & string, keyof UniqFoo & string>; // true