src/skills/web-utilities-date-fns/SKILL.md
date-fns patterns for TypeScript - formatting, parsing, manipulation, comparison, timezone handling, and internationalization
npx skillsauth add agents-inc/skills web-utilities-date-fnsInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Quick Guide: Use date-fns for modular, tree-shakeable date operations. Import only what you need. Use
parseISOfor ISO strings,formatwith Unicode tokens for display, and pure functions that return new Date objects. For timezones, use@date-fns/tzwithTZDate(v4+) ordate-fns-tzwithformatInTimeZone(v3.x). Never mutate dates.
<critical_requirements>
All code must follow project conventions in CLAUDE.md (kebab-case, named exports, import ordering,
import type, named constants)
(You MUST use parseISO() for ISO 8601 strings - NEVER use new Date(string) which has browser inconsistencies)
(You MUST import only needed functions - NEVER use import * as dateFns which defeats tree-shaking)
(You MUST use pure functions that return new dates - NEVER mutate dates with setDate() or similar)
(You MUST use named constants for format strings and durations - NO magic strings like 'yyyy-MM-dd' scattered in code)
</critical_requirements>
Auto-detection: date-fns, format, parseISO, addDays, subMonths, differenceInDays, formatDistance, isAfter, isBefore, eachDayOfInterval, date-fns-tz, @date-fns/tz, @date-fns/utc, TZDate, TZDateMini, UTCDate, UTCDateMini, tz(), transpose, tzName, tzScan, withTimeZone, locale
When to use:
When NOT to use:
Intl.DateTimeFormat (zero bundle cost)Date may sufficerrule.js alongsideDetailed Resources:
date-fns is a modular, functional date utility library. Each function is independent, enabling tree-shaking to include only what you use. All functions are pure - they return new Date objects rather than mutating inputs. This makes date operations predictable and testable.
Key principle: Import what you need, let bundlers remove the rest. A simple format import adds ~2KB, not the entire 80KB library.
// Tree-shakeable - only includes format and parseISO
import { format, parseISO } from "date-fns";
// Format constant at module level
const DATE_DISPLAY_FORMAT = "MMMM d, yyyy";
const date = parseISO("2026-01-15");
const display = format(date, DATE_DISPLAY_FORMAT); // "January 15, 2026"
v3+ is 100% TypeScript with built-in type definitions. No @types/date-fns needed.
date-fns uses Unicode Technical Standard #35 format tokens. These differ from Moment.js (yyyy not YYYY, dd not DD, EEEE not dddd).
import { format } from "date-fns";
// Define format constants at module level
const ISO_DATE_FORMAT = "yyyy-MM-dd";
const DISPLAY_DATE_FORMAT = "MMMM d, yyyy";
const DISPLAY_DATETIME_FORMAT = "MMMM d, yyyy 'at' h:mm a";
const date = new Date(2026, 0, 15, 14, 30, 0);
format(date, ISO_DATE_FORMAT); // "2026-01-15"
format(date, DISPLAY_DATE_FORMAT); // "January 15, 2026"
format(date, DISPLAY_DATETIME_FORMAT); // "January 15, 2026 at 2:30 PM"
format(date, "EEEE"); // "Thursday"
format(date, "h:mm a"); // "2:30 PM"
See reference.md for the full format token table.
Why good: named constants make format strings reusable and discoverable, Unicode tokens are standard across date libraries
Use P, PP, PPP, PPPP for locale-aware date formatting without specifying exact format.
import { format } from "date-fns";
import { enUS, de, ja, fr } from "date-fns/locale";
const date = new Date(2026, 0, 15);
// ✅ Good Example - Locale-aware formatting
// Short date
format(date, "P", { locale: enUS }); // "01/15/2026"
format(date, "P", { locale: de }); // "15.01.2026"
format(date, "P", { locale: ja }); // "2026/01/15"
// Medium date
format(date, "PP", { locale: enUS }); // "Jan 15, 2026"
format(date, "PP", { locale: de }); // "15. Jan. 2026"
// Long date
format(date, "PPP", { locale: enUS }); // "January 15th, 2026"
format(date, "PPP", { locale: fr }); // "15 janvier 2026"
// Full date
format(date, "PPPP", { locale: enUS }); // "Thursday, January 15th, 2026"
format(date, "PPPP", { locale: ja }); // "2026年1月15日木曜日"
// Time shortcuts
format(date, "p", { locale: enUS }); // "12:00 AM"
format(date, "pp", { locale: enUS }); // "12:00:00 AM"
// Combined date and time
format(date, "PPpp", { locale: enUS }); // "Jan 15, 2026, 12:00:00 AM"
Why good: P/PP/PPP adapt to locale conventions automatically, users see dates in familiar format for their region
// ❌ Bad Example - Hardcoded format ignores locale
format(date, "MM/dd/yyyy"); // "01/15/2026" - wrong for most of world!
Why bad: hardcoded MM/dd/yyyy format is US-specific, confusing for European and Asian users who expect different order
Use parseISO for ISO 8601 strings. Use parse for custom formats.
import { parseISO, parse, isValid } from "date-fns";
// ✅ Good - parseISO for ISO strings
const date = parseISO("2026-01-15T14:30:00Z");
// ✅ Good - parse for custom formats (3rd arg is reference date)
const CUSTOM_DATE_FORMAT = "dd/MM/yyyy";
const customDate = parse("15/01/2026", CUSTOM_DATE_FORMAT, new Date());
// Always validate parsed dates
if (!isValid(date)) {
/* handle invalid */
}
Why good: parseISO handles all ISO 8601 variants, isValid catches invalid dates
// ❌ Bad - Using Date constructor
const date1 = new Date("2026-01-15"); // Browser-inconsistent!
const date2 = new Date("01/15/2026"); // May fail in non-US browsers
Why bad: new Date(string) parsing varies by browser and locale
See examples/core.md for strict round-trip parsing that catches invalid dates like Feb 30.
Use pure functions for date manipulation. All return new Date objects.
import {
addDays,
addMonths,
addYears,
subDays,
subMonths,
subYears,
addHours,
addMinutes,
} from "date-fns";
// Duration constants
const WEEK_IN_DAYS = 7;
const BILLING_CYCLE_MONTHS = 1;
const TRIAL_PERIOD_DAYS = 14;
const date = new Date(2026, 0, 15);
// ✅ Good Example - Pure functions return new dates
const nextWeek = addDays(date, WEEK_IN_DAYS); // Jan 22, 2026
const nextMonth = addMonths(date, BILLING_CYCLE_MONTHS); // Feb 15, 2026
const nextYear = addYears(date, 1); // Jan 15, 2027
const lastWeek = subDays(date, WEEK_IN_DAYS); // Jan 8, 2026
const trialEnd = addDays(date, TRIAL_PERIOD_DAYS); // Jan 29, 2026
// Chain operations
const twoWeeksFromNextMonth = addDays(addMonths(date, 1), WEEK_IN_DAYS * 2);
// Original date is unchanged
console.log(date); // Still Jan 15, 2026
Why good: pure functions are predictable and testable, original date unchanged, constants document business logic
// ❌ Bad Example - Mutating dates
const date = new Date(2026, 0, 15);
date.setDate(date.getDate() + 7); // Mutates original!
// Original date is now changed - causes bugs in shared references
Why bad: mutation creates side effects, especially problematic when date is passed as prop or stored in state
Use boundary functions for consistent start/end of periods.
import {
startOfDay,
endOfDay,
startOfWeek,
endOfWeek,
startOfMonth,
endOfMonth,
startOfQuarter,
endOfQuarter,
startOfYear,
endOfYear,
} from "date-fns";
const date = new Date(2026, 0, 15, 14, 30, 0);
// ✅ Good Example - Boundary functions for consistent ranges
const dayStart = startOfDay(date); // Jan 15, 2026 00:00:00
const dayEnd = endOfDay(date); // Jan 15, 2026 23:59:59.999
// Week boundaries (weekStartsOn: 1 = Monday)
const weekStart = startOfWeek(date, { weekStartsOn: 1 }); // Jan 13, 2026
const weekEnd = endOfWeek(date, { weekStartsOn: 1 }); // Jan 19, 2026
const monthStart = startOfMonth(date); // Jan 1, 2026
const monthEnd = endOfMonth(date); // Jan 31, 2026 23:59:59.999
Why good: boundary functions handle edge cases (month lengths, leap years), consistent for database queries and filtering
See examples/core.md for range generation utilities (getMonthRange, preset ranges).
Use comparison functions instead of manual timestamp comparisons.
import {
isAfter,
isBefore,
isEqual,
isSameDay,
isSameMonth,
isSameYear,
isWithinInterval,
isToday,
isPast,
isFuture,
isWeekend,
} from "date-fns";
const date1 = new Date(2026, 0, 15);
const date2 = new Date(2026, 0, 20);
// ✅ Good Example - Semantic comparison functions
const isLater = isAfter(date2, date1); // true
const isEarlier = isBefore(date1, date2); // true
const areSame = isEqual(date1, date1); // true
// Same period checks (ignore time)
const sameDay = isSameDay(date1, date1); // true
const sameMonth = isSameMonth(date1, date2); // true (both January)
const sameYear = isSameYear(date1, date2); // true (both 2026)
// Range check
const isInRange = isWithinInterval(new Date(2026, 0, 17), {
start: date1,
end: date2,
}); // true
// Convenience checks
const todayCheck = isToday(new Date()); // true
const pastCheck = isPast(date1); // depends on current date
const futureCheck = isFuture(date2); // depends on current date
const weekendCheck = isWeekend(date1); // false (Thursday)
Why good: semantic function names make code readable, handles edge cases like time zone differences
// ❌ Bad Example - Manual timestamp comparison
const isLater = date2.getTime() > date1.getTime();
const sameDay =
date1.getFullYear() === date2.getFullYear() &&
date1.getMonth() === date2.getMonth() &&
date1.getDate() === date2.getDate();
Why bad: verbose, error-prone, doesn't handle edge cases, hard to read intent
</patterns>@date-fns/tz for timezone handling with TZDate class and tz() helperdate-fns-tz package with formatInTimeZone, toZonedTime, fromZonedTimedate-fns/constantsv4 Bundle Size:
TZDate: 1.2 KB | TZDateMini: 916 B (use Mini for internal calculations)UTCDate: 504 B | UTCDateMini: 239 B (use Mini for internal calculations)v4 Timezone Functions (@date-fns/tz):
tz(): Creates timezone context for the in optiontzName(): Returns human-readable timezone name (short, long, generic formats)tzScan(): Detects DST transitions within a date rangetzOffset(): Returns UTC offset in minutes.withTimeZone(): TZDate method for timezone conversionv4 Core Functions (from date-fns):
transpose(): Converts dates between timezones (replaces toZonedTime/fromZonedTime)<red_flags>
new Date(string) for parsing - Browser-inconsistent, use parseISO or parsedate.setDate() causes side effects, use pure functions like addDaysimport * from date-fns - Defeats tree-shaking, import individual functionsformat(date, 'yyyy-MM-dd') scattered in code, use named constantsisValid check - Parsed dates can be invalid, always validate after parsingYYYY/DD (Moment) vs yyyy/dd (date-fns)MM/dd/yyyy is US-only, use P/PP/PPP with localeisWithinInterval with inverted start/end - Throws error if start > endSee reference.md for the complete red flags and gotchas list.
</red_flags>
<critical_reminders>
All code must follow project conventions in CLAUDE.md
(You MUST use parseISO() for ISO 8601 strings - NEVER use new Date(string) which has browser inconsistencies)
(You MUST import only needed functions - NEVER use import * as dateFns which defeats tree-shaking)
(You MUST use pure functions that return new dates - NEVER mutate dates with setDate() or similar)
(You MUST use named constants for format strings and durations - NO magic strings like 'yyyy-MM-dd' scattered in code)
Failure to follow these rules will cause browser inconsistencies, bundle bloat, and mutation bugs.
</critical_reminders>
development
Material Design component library for Vue 3
development
VitePress 1.x — Vue-powered static site generator for documentation sites, built on Vite
tools
Docusaurus 3.x documentation framework — site configuration, docs/blog plugins, sidebars, versioning, MDX, swizzling, and deployment
development
TanStack Form patterns - useForm, form.Field, validators, arrays, linked fields, createFormHook, type safety