Skip to content

Commit

Permalink
Yet another PR adding TypeScript declarations (#530)
Browse files Browse the repository at this point in the history
* Adds TypeScript declarations for moment-timezone

* Remove unnecessary `moment.` qualification in typings

* Move exported types to top-level so that they can all be imported as named imports

- Zone class must exist at `tz.Zone` because that's where it's exposed at runtime, but the `Zone` interface is still re-exported as a named export

* tsc invocation always uses locally installed tsc

Co-authored-by: Elena Sharovar <[email protected]>
  • Loading branch information
cspotcode and ellenaua authored May 16, 2020
1 parent d50bf27 commit ed529ea
Show file tree
Hide file tree
Showing 7 changed files with 461 additions and 5 deletions.
14 changes: 10 additions & 4 deletions Gruntfile.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,10 @@ module.exports = function(grunt) {

clean: {
data: ['temp']
},

exec: {
'typing-tests': './node_modules/.bin/tsc --project ./typing-tests'
}
});

Expand All @@ -54,11 +58,13 @@ module.exports = function(grunt) {
grunt.loadNpmTasks('grunt-contrib-jshint');
grunt.loadNpmTasks('grunt-contrib-uglify');
grunt.loadNpmTasks('grunt-contrib-clean');
grunt.loadNpmTasks('grunt-exec');

grunt.registerTask('release', ['jshint', 'data', 'nodeunit', 'typing-tests', 'build', 'uglify']);

/** creates a release */
grunt.registerTask('release', ['jshint', 'data', 'nodeunit', 'build', 'uglify']);
grunt.registerTask('releaseNoData', ['jshint', 'nodeunit', 'typing-tests', 'build', 'uglify']);

grunt.registerTask('releaseNoData', ['jshint', 'nodeunit', 'build', 'uglify']);
grunt.registerTask('typing-tests', ['exec:typing-tests']);

grunt.registerTask('default', ['jshint', 'nodeunit']);
grunt.registerTask('default', ['jshint', 'nodeunit', 'typing-tests']);
};
127 changes: 127 additions & 0 deletions index.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,127 @@
import * as moment from 'moment';

// require('moment-timezone') === require('moment')
export = moment;

declare module 'moment' {
interface Moment {
/** Set the timezone and update the offset */
tz(zone: string): Moment;
/** Return the timezone name or undefined if a zone has not yet been set */
tz(): string | undefined;

/** Get the zone abbreviation. This is what moment.js uses when formatting the z token. */
zoneAbbr(): string;

/** Get the zone long form name. This is what moment.js uses when formatting the zz token. */
zoneName(): string;
}

// Match normal moment constructor but with an extra timezone argument
// Here's a copy-paste of the normal moment constructor's signature, from https://github.com/moment/moment/blob/develop/moment.d.ts#L1-L2
// declare function moment(inp?: moment.MomentInput, format?: moment.MomentFormatSpecification, strict?: boolean): moment.Moment;
// declare function moment(inp?: moment.MomentInput, format?: moment.MomentFormatSpecification, language?: string, strict?: boolean): moment.Moment;

// Should be sorted from tightest to loosest. TypeScript picks the first signature that matches, going top to bottom.

/** create a moment with a time zone */
function tz(inp: MomentInput, format: MomentFormatSpecification, language: string, strict: boolean, zone: string): Moment;
/** create a moment with a time zone */
function tz(inp: MomentInput, format: MomentFormatSpecification, language: string, zone: string): Moment;
/** create a moment with a time zone */
function tz(inp: MomentInput, format: MomentFormatSpecification, strict: boolean, zone: string): Moment;
/** create a moment with a time zone */
function tz(inp: MomentInput, format: MomentFormatSpecification, zone: string): Moment;
/** create a moment with a time zone */
function tz(inp: MomentInput, zone: string): Moment;
/** create a moment with a time zone */
function tz(zone?: string): Moment;

namespace tz {
/** Version of moment-timezone */
const version: string;

/**
* Change the default timezone of newly created Moment instances.
* By default new instances are created in the local timezone.
*/
function setDefault(zone: string): typeof moment;

/** Reset the default timezone to local. */
function setDefault(): typeof moment;

/**
* Retrieve or guess the user's timezone. Uses the browser's Internationalization API if available.
* Otherwise, guesses by sampling offsets from different points in time and comparing them to available zone data.
*/
function guess(): string;

interface Zone extends UnpackedZone {}
class Zone {
/** Get the abbreviation for a given timestamp from a Zone. */
abbr(timestamp: number): string;

/** Get the offset for a given timestamp from a Zone. */
offset(timestamp: number): number;

/** Parse an offset for a timestamp constructed from Date.UTC in that zone. */
parse(timestamp: number): number;
}

/** Return a timezone by name or null if timezone by that name is not loaded. */
function zone(name: string): Zone | null;

/** Add zone data for a timezone. */
function add(packedZone: string): void;
/** Add zone data for multiple timezones. */
function add(packedZones: Array<string>): void;

/** Link two zone names to the same data */
function link(packedLink: string): void;
/** Add multiple links at once */
function link(packedLinks: Array<string>): void;

/** load a bundle of zone data and links */
function load(bundle: PackedZoneBundle): void;

/** get a list of all available time zone names */
function names(): Array<string>;

/** Convert a packed string to an unpacked zone data object */
function unpack(packedZone: string): UnpackedZone;
/** Convert a base 60 string to a base 10 number. */
function unpackBase60(base60String: string): number;
}

type Zone = tz.Zone;

/** Parsed / unpacked zone data. */
interface UnpackedZone {
/** The uniquely identifying name of the time zone. */
name: string;

/** zone abbreviations */
abbrs: Array<string>;

/** (measured in milliseconds) */
untils: Array<number | null>;

/** (measured in minutes) */
offsets: Array<number>;
}

/** Bundle of zone data and links for multiple timezones */
interface PackedZoneBundle {
version: string;
zones: Array<string>;
links: Array<string>;
}

/** Bundle of zone data and links for multiple timezones */
interface UnpackedZoneBundle {
version: string;
zones: Array<UnpackedZone>;
links: Array<string>;
}

}
41 changes: 41 additions & 0 deletions moment-timezone-utils.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import * as moment from 'moment';

// require('moment-timezone') === require('moment')
export = moment;

declare module 'moment' {
namespace tz {
/** Converts zone data in the unpacked format to the packed format. */
function pack(unpackedObject: UnpackedZone): string;

/** Convert a base 10 number to a base 60 string. */
function packBase60(input: number, precision?: number): string;

/** Create links out of two zones that share data.
* @returns A new ZoneBundle with duplicate zone data replaced by links
*/
function createLinks(unlinked: UnpackedZoneBundle): PackedZoneBundle;

/**
* Filter out data for years outside a certain range.
* @return a new, filtered UnPackedZone object
*/
function filterYears(unpackedZone: UnpackedZone, startYear: number, endYear: number): UnpackedZone;
/**
* Filter out data for years outside a certain range.
* @return a new, filtered UnPackedZone object
*/
function filterYears(unpackedZone: UnpackedZone, startAndEndYear: number): UnpackedZone;

/**
* Combines packing, link creation, and subsetting of years into one simple interface.
* Pass in an unpacked bundle, start year, and end year and get a filtered, linked, packed bundle back.
*/
function filterLinkPack(unpackedBundle: UnpackedZoneBundle, startYear: number, endYear: number): PackedZoneBundle;
/**
* Combines packing, link creation, and subsetting of years into one simple interface.
* Pass in an unpacked bundle, start year, and end year and get a filtered, linked, packed bundle back.
*/
function filterLinkPack(unpackedBundle: UnpackedZoneBundle, startAndEndYear: number): PackedZoneBundle;
}
}
6 changes: 6 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 4 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
"tz"
],
"main": "./index.js",
"typings": "./index.d.ts",
"engines": {
"node": "*"
},
Expand All @@ -34,7 +35,9 @@
"grunt-contrib-clean": "^2.0.0",
"grunt-contrib-jshint": "^2.1.0",
"grunt-contrib-nodeunit": "^2.0.0",
"grunt-contrib-uglify": "^4.0.1"
"grunt-contrib-uglify": "^4.0.1",
"grunt-exec": "^3.0.0",
"typescript": "^3.5.1"
},
"jspm": {
"main": "builds/moment-timezone-with-data",
Expand Down
Loading

0 comments on commit ed529ea

Please sign in to comment.