diff --git a/examples/basicApp/src/app/Dialog.ts b/examples/basicApp/src/app/Dialog.ts index 7fb2c4b..47b70a5 100644 --- a/examples/basicApp/src/app/Dialog.ts +++ b/examples/basicApp/src/app/Dialog.ts @@ -1,29 +1,31 @@ -import * as declare from 'dojo/_base/declare'; -import * as Dialog from 'dijit/Dialog'; +import * as DijitDialog from 'dijit/Dialog'; +import * as messages from 'dojo/i18n!./nls/main'; +import declare from './declareDecorator'; -/* We are using the ES6 module format here, as it is more forward compatible */ +/** + * `dojo/_base/declare` can infer typings, but since we are generating our declare + * constructor with a decorator (see below), we need to define our widget interface + * explicitly. + */ +interface Dialog extends DijitDialog {} -/* While type inference works with declare, if you want to refer to the type at - * some other point, you cannot easily extract the type. Therefore we will do - * essentially what the dojo.declare typings will do and create type reference. - * This does not emit anything, but it allows others using this module in - * TypeScript to have a clear reference to the type we are creating here */ -export type DialogType = dijit.Dialog & { - title: string; - content: string; -}; +/** + * By using a decorator to create constructors with `dojo/_base/declare`, we can + * write our widgets using TypeScript classes. The one caveat is that we are then + * required to use merged declarations to define our interface, so the widget + * class has to be declared as the default export in a separate statement. + */ +@declare(DijitDialog) +class Dialog { + title: string = messages.dialogTitle; + content: string = messages.dialogContent; +} -/* dojo.declare can infer the typings here, but as explained above, we want to - * specfically assert the type, so we are providing it as a generic argument - * for decalare - * - * In addition, we are using `export default` here, which is part of the ES6 - * module format. If other consumers of the module are compatible with default - * exports, like other modules written in TypeScript, then this is fine, but if - * someone consumes this module, that insead of this being the return value of - * the module, it will be located under Dialog.default. +/* + * We are using `export default` here, which is part of the ES6 module format. + * If other consumers of the module are compatible with default exports, like + * other modules written in TypeScript, then this is fine, but if someone + * consumes this module, that instead of this being the return value of the module, + * it will be located under Dialog.default. */ -export default declare(Dialog, { - title: 'Hello World', - content: 'Loaded successfully!' -}); +export default Dialog; diff --git a/examples/basicApp/src/app/declareDecorator.ts b/examples/basicApp/src/app/declareDecorator.ts new file mode 100644 index 0000000..f1ab6cf --- /dev/null +++ b/examples/basicApp/src/app/declareDecorator.ts @@ -0,0 +1,12 @@ +import * as declare from 'dojo/_base/declare'; + +/** + * A decorator that converts a TypeScript class into a declare constructor. + * This allows declare constructors to be defined as classes, which nicely + * hides away the `declare([], {})` boilerplate. + */ +export default function (... mixins: Object[]): ClassDecorator { + return function (target: Function) { + return declare(mixins, target.prototype); + }; +} diff --git a/examples/basicApp/src/app/main.ts b/examples/basicApp/src/app/main.ts index b7db7d1..59cbd43 100644 --- a/examples/basicApp/src/app/main.ts +++ b/examples/basicApp/src/app/main.ts @@ -1,19 +1,11 @@ -import Dialog, { DialogType } from './Dialog'; +import Dialog from './Dialog'; import 'dojo/domReady!'; -/* Using just "import 'mid'"" above loads the modules for and causes the - * appropriate side affects, but doesn't create any reference in the module - * - * Also we are using the destrcuring syntax to also import a reference to the - * DialogType type above. This will not actually emit anything because there - * is no runtime emit for the type - */ - -/* So others can consume our application module better under TypeScript, lets +/* So others can consume our application module better under TypeScript, let's * specify an interface for our application and export it */ export interface App { - dialog: DialogType; + dialog: Dialog; } /* Now we are creating our application object */ diff --git a/examples/basicApp/src/app/nls/main.ts b/examples/basicApp/src/app/nls/main.ts new file mode 100644 index 0000000..a6b3316 --- /dev/null +++ b/examples/basicApp/src/app/nls/main.ts @@ -0,0 +1,4 @@ +export const root = { + dialogTitle: 'Hello World', + dialogContent: 'Loaded successfully!' +}; diff --git a/examples/basicApp/tsconfig.json b/examples/basicApp/tsconfig.json index 9c486a4..4d32549 100644 --- a/examples/basicApp/tsconfig.json +++ b/examples/basicApp/tsconfig.json @@ -5,12 +5,14 @@ "outDir": "_build", "removeComments": false, "sourceMap": true, - "target": "ES5" + "target": "ES5", + "experimentalDecorators": true }, + "include": [ + "./src/**/*.ts" + ], "files": [ - "./src/app/main.ts", - "./src/app/Dialog.ts", "../../dojo/1.11/modules.d.ts", "../../dijit/1.11/modules.d.ts" ] -} \ No newline at end of file +}