typescript-advanced-types

TypeScript Advanced Types

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "typescript-advanced-types" with this command: npx skills add beshkenadze/claude-skills-marketplace/beshkenadze-claude-skills-marketplace-typescript-advanced-types

TypeScript Advanced Types

Overview

Comprehensive guidance for mastering TypeScript's advanced type system including generics, conditional types, mapped types, template literal types, and utility types for building robust, type-safe applications.

When to Use This Skill

  • Building type-safe libraries or frameworks

  • Creating reusable generic components

  • Implementing complex type inference logic

  • Designing type-safe API clients

  • Building form validation systems

  • Creating strongly-typed configuration objects

  • Implementing type-safe state management

  • Migrating JavaScript codebases to TypeScript

Core Concepts

  1. Generics

Purpose: Create reusable, type-flexible components while maintaining type safety.

Basic Generic Function:

function identity<T>(value: T): T { return value; }

const num = identity<number>(42); // Type: number const str = identity<string>("hello"); // Type: string const auto = identity(true); // Type inferred: boolean

Generic Constraints:

interface HasLength { length: number; }

function logLength<T extends HasLength>(item: T): T { console.log(item.length); return item; }

logLength("hello"); // OK: string has length logLength([1, 2, 3]); // OK: array has length logLength({ length: 10 }); // OK: object has length // logLength(42); // Error: number has no length

Multiple Type Parameters:

function merge<T, U>(obj1: T, obj2: U): T & U { return { ...obj1, ...obj2 }; }

const merged = merge( { name: "John" }, { age: 30 } ); // Type: { name: string } & { age: number }

  1. Conditional Types

Purpose: Create types that depend on conditions, enabling sophisticated type logic.

Basic Conditional Type:

type IsString<T> = T extends string ? true : false;

type A = IsString<string>; // true type B = IsString<number>; // false

Extracting Return Types:

type ReturnType<T> = T extends (...args: any[]) => infer R ? R : never;

function getUser() { return { id: 1, name: "John" }; }

type User = ReturnType<typeof getUser>; // Type: { id: number; name: string; }

Distributive Conditional Types:

type ToArray<T> = T extends any ? T[] : never;

type StrOrNumArray = ToArray<string | number>; // Type: string[] | number[]

Nested Conditions:

type TypeName<T> = T extends string ? "string" : T extends number ? "number" : T extends boolean ? "boolean" : T extends undefined ? "undefined" : T extends Function ? "function" : "object";

type T1 = TypeName<string>; // "string" type T2 = TypeName<() => void>; // "function"

  1. Mapped Types

Purpose: Transform existing types by iterating over their properties.

Basic Mapped Type:

type Readonly<T> = { readonly [P in keyof T]: T[P]; };

interface User { id: number; name: string; }

type ReadonlyUser = Readonly<User>; // Type: { readonly id: number; readonly name: string; }

Optional Properties:

type Partial<T> = { [P in keyof T]?: T[P]; };

type PartialUser = Partial<User>; // Type: { id?: number; name?: string; }

Key Remapping:

type Getters<T> = { [K in keyof T as get${Capitalize&#x3C;string &#x26; K>}]: () => T[K] };

interface Person { name: string; age: number; }

type PersonGetters = Getters<Person>; // Type: { getName: () => string; getAge: () => number; }

Filtering Properties:

type PickByType<T, U> = { [K in keyof T as T[K] extends U ? K : never]: T[K] };

interface Mixed { id: number; name: string; age: number; active: boolean; }

type OnlyNumbers = PickByType<Mixed, number>; // Type: { id: number; age: number; }

  1. Template Literal Types

Purpose: Create string-based types with pattern matching and transformation.

Basic Template Literal:

type EventName = "click" | "focus" | "blur"; type EventHandler = on${Capitalize&#x3C;EventName>}; // Type: "onClick" | "onFocus" | "onBlur"

String Manipulation:

type UppercaseGreeting = Uppercase<"hello">; // "HELLO" type LowercaseGreeting = Lowercase<"HELLO">; // "hello" type CapitalizedName = Capitalize<"john">; // "John" type UncapitalizedName = Uncapitalize<"John">; // "john"

Path Building:

type Path<T> = T extends object ? { [K in keyof T]: K extends string ? ${K} | ${K}.${Path&#x3C;T[K]>} : never }[keyof T] : never;

interface Config { server: { host: string; port: number; }; database: { url: string; }; }

type ConfigPath = Path<Config>; // Type: "server" | "database" | "server.host" | "server.port" | "database.url"

  1. Utility Types

Built-in Utility Types:

// Partial<T> - Make all properties optional type PartialUser = Partial<User>;

// Required<T> - Make all properties required type RequiredUser = Required<PartialUser>;

// Readonly<T> - Make all properties readonly type ReadonlyUser = Readonly<User>;

// Pick<T, K> - Select specific properties type UserName = Pick<User, "name" | "email">;

// Omit<T, K> - Remove specific properties type UserWithoutPassword = Omit<User, "password">;

// Exclude<T, U> - Exclude types from union type T1 = Exclude<"a" | "b" | "c", "a">; // "b" | "c"

// Extract<T, U> - Extract types from union type T2 = Extract<"a" | "b" | "c", "a" | "b">; // "a" | "b"

// NonNullable<T> - Exclude null and undefined type T3 = NonNullable<string | null | undefined>; // string

// Record<K, T> - Create object type with keys K and values T type PageInfo = Record<"home" | "about", { title: string }>;

Advanced Patterns

Pattern 1: Type-Safe Event Emitter

type EventMap = { "user:created": { id: string; name: string }; "user:updated": { id: string }; "user:deleted": { id: string }; };

class TypedEventEmitter<T extends Record<string, any>> { private listeners: { [K in keyof T]?: Array<(data: T[K]) => void>; } = {};

on<K extends keyof T>(event: K, callback: (data: T[K]) => void): void { if (!this.listeners[event]) { this.listeners[event] = []; } this.listeners[event]!.push(callback); }

emit<K extends keyof T>(event: K, data: T[K]): void { const callbacks = this.listeners[event]; if (callbacks) { callbacks.forEach(callback => callback(data)); } } }

const emitter = new TypedEventEmitter<EventMap>();

emitter.on("user:created", (data) => { console.log(data.id, data.name); // Type-safe! });

emitter.emit("user:created", { id: "1", name: "John" }); // emitter.emit("user:created", { id: "1" }); // Error: missing 'name'

Pattern 2: Deep Readonly/Partial

type DeepReadonly<T> = { readonly [P in keyof T]: T[P] extends object ? T[P] extends Function ? T[P] : DeepReadonly<T[P]> : T[P]; };

type DeepPartial<T> = { [P in keyof T]?: T[P] extends object ? T[P] extends Array<infer U> ? Array<DeepPartial<U>> : DeepPartial<T[P]> : T[P]; };

interface Config { server: { host: string; port: number; ssl: { enabled: boolean; cert: string; }; }; }

type ReadonlyConfig = DeepReadonly<Config>; // All nested properties are readonly

type PartialConfig = DeepPartial<Config>; // All nested properties are optional

Pattern 3: Discriminated Unions

type Success<T> = { status: "success"; data: T; };

type Error = { status: "error"; error: string; };

type Loading = { status: "loading"; };

type AsyncState<T> = Success<T> | Error | Loading;

function handleState<T>(state: AsyncState<T>): void { switch (state.status) { case "success": console.log(state.data); // Type: T break; case "error": console.log(state.error); // Type: string break; case "loading": console.log("Loading..."); break; } }

Type Inference Techniques

  1. Infer Keyword

// Extract array element type type ElementType<T> = T extends (infer U)[] ? U : never;

type NumArray = number[]; type Num = ElementType<NumArray>; // number

// Extract promise type type PromiseType<T> = T extends Promise<infer U> ? U : never;

type AsyncNum = PromiseType<Promise<number>>; // number

// Extract function parameters type Parameters<T> = T extends (...args: infer P) => any ? P : never;

function foo(a: string, b: number) {} type FooParams = Parameters<typeof foo>; // [string, number]

  1. Type Guards

function isString(value: unknown): value is string { return typeof value === "string"; }

function isArrayOf<T>( value: unknown, guard: (item: unknown) => item is T ): value is T[] { return Array.isArray(value) && value.every(guard); }

const data: unknown = ["a", "b", "c"];

if (isArrayOf(data, isString)) { data.forEach(s => s.toUpperCase()); // Type: string[] }

  1. Assertion Functions

function assertIsString(value: unknown): asserts value is string { if (typeof value !== "string") { throw new Error("Not a string"); } }

function processValue(value: unknown) { assertIsString(value); // value is now typed as string console.log(value.toUpperCase()); }

Quick Reference

Utility Purpose

Partial<T>

All properties optional

Required<T>

All properties required

Readonly<T>

All properties readonly

Pick<T, K>

Select properties

Omit<T, K>

Remove properties

Exclude<T, U>

Remove from union

Extract<T, U>

Keep from union

NonNullable<T>

Remove null/undefined

Record<K, T>

Object with keys K, values T

ReturnType<T>

Function return type

Parameters<T>

Function parameter types

Best Practices

Do

  • Use unknown over any

  • enforce type checking

  • Prefer interface for object shapes - better error messages

  • Use type for unions and complex types

  • Leverage type inference - let TypeScript infer when possible

  • Create helper types - build reusable utilities

  • Use const assertions - preserve literal types

  • Use type guards instead of assertions

  • Enable strict mode in tsconfig

Don't

  • Over-use any

  • defeats TypeScript's purpose

  • Ignore strict null checks - leads to runtime errors

  • Create overly complex types - slows compilation

  • Forget readonly modifiers - allows unintended mutations

  • Use type assertions when guards work - loses type safety

Common Pitfalls

Pitfall Solution

Circular type references Break cycle with explicit types

Deeply nested conditionals Simplify or use helper types

Missing discriminant Add type or kind field

Forgetting infer scope Use inside extends clause

Union distribution unexpected Wrap in tuple [T]

Resources

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

swiftui-developer

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

python-uv

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

dev-workflow

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

codex-code-review

No summary provided by upstream source.

Repository SourceNeeds Review