| name | typeclass-design |
| description | Implement typeclasses with curried signatures and dual APIs for both data-first and data-last usage Use when this capability is needed. |
| metadata | {"author":"kriegcloud"} |
Typeclass Design Skill
Use this skill when implementing typeclasses that provide reusable abstractions across multiple types.
Pattern: Curried Typeclass Functions
All typeclass functions must be fully curried to enable partial application:
import { Duration } from "effect"
declare interface Durable<A> {
readonly getDuration: (self: A) => Duration.Duration
}
export const isMoreThan =
<A>(D: Durable<A>) =>
(minimum: Duration.Input) =>
(self: A): boolean => {
const current = D.getDuration(self)
return Duration.greaterThanOrEqualTo(current, minimum)
}
Pattern: Dual APIs
Provide both data-first (uncurried) and data-last (curried) variants using Function.dual:
import { Duration } from "effect"
import * as Function from "effect/Function"
declare interface Durable<A> {
readonly getDuration: (self: A) => Duration.Duration
}
export const isMoreThan = <A>(D: Durable<A>) =>
Function.dual<
(minimum: Duration.Input) => (self: A) => boolean,
(self: A, minimum: Duration.Input) => boolean
>(
2,
(self: A, minimum: Duration.Input): boolean => {
const current = D.getDuration(self)
return Duration.greaterThanOrEqualTo(current, minimum)
}
)
Usage Patterns
The dual API enables both styles:
import { pipe } from "effect/Function"
import * as Duration from "effect/Duration"
import * as Function from "effect/Function"
declare interface Durable<A> {
readonly getDuration: (self: A) => Duration.Duration
}
declare const isMoreThan: <A>(D: Durable<A>) => {
(minimum: Duration.Input): (self: A) => boolean
(self: A, minimum: Duration.Input): boolean
}
declare interface Appointment {
duration: Duration.Duration
}
declare const Appointment: {
Durable: Durable<Appointment>
}
declare const appointment: Appointment
declare const appointments: Appointment[]
const hasMinimum = isMoreThan(Appointment.Durable)(
appointment,
Duration.hours(1)
)
const hasMinimum2 = pipe(
appointment,
isMoreThan(Appointment.Durable)(Duration.hours(1))
)
const longAppointments = appointments.filter(
isMoreThan(Appointment.Durable)(Duration.hours(1))
)
Complete Typeclass Example
import { Duration, Order } from "effect"
import * as Function from "effect/Function"
export interface Durable<A> {
readonly getDuration: (self: A) => Duration.Duration
readonly setDuration: (self: A, duration: Duration.Input) => A
}
export const make = <A>(
getDuration: (self: A) => Duration.Duration,
setDuration: (self: A, duration: Duration.Input) => A
): Durable<A> => ({
getDuration,
setDuration
})
export const isMoreThan = <A>(D: Durable<A>) =>
Function.dual<
(minimum: Duration.Input) => (self: A) => boolean,
(self: A, minimum: Duration.Input) => boolean
>(
2,
(self: A, minimum: Duration.Input): boolean =>
Duration.greaterThanOrEqualTo(
D.getDuration(self),
Duration.decode(minimum)
)
)
export const OrderByDuration = <A>(D: Durable<A>): Order.Order<A> =>
Order.mapInput(Duration.Order, (self: A) => D.getDuration(self))
When to Use
- Creating reusable abstractions (Schedulable, Durable, Priceable)
- Implementing operations that work across multiple types
- Providing composable, pipe-friendly APIs
- Enabling partial application for filtering/mapping
Key Principles
- Curry everything - Enable partial application
- Dual APIs always - Support both usage styles
- Typeclass first - First parameter is always the typeclass instance
- Type lambda for HKT - Use TypeLambda pattern when needed
Converted and distributed by TomeVault — claim your Tome and manage your conversions.