I'm pretty new to typescript and I'm writing a small prototyping framework for WebGl. I'm currently refactoring my project and have run into some problems as how to organize my project, as both (modules and namespaces) approaches seem to have serious drawbacks.
This post is not about HOW to use these patterns, but how to overcome the problems each of these brings.
Coming from C# this seemed like the most natural way to go. Every class/module gets it's appropriate namespace and I supply the "outFile" parameter in the tsconfig.json so everything gets concatenated into one large file. After compilation I have my root namespace as a global object. Dependencies are not built into the project, so you manually have to supply the needed *.js files in you html (not good)
Example file
namespace Cross.Eye {     export class SpriteFont {            //code omitted     }     } Example usage (You have to make sure the Cross namespace is loaded into the global namespace before by suppling the js file in the html)
namespace Examples {     export class _01_BasicQuad {         context: Cross.Eye.Context;         shader: Cross.Eye.ShaderProgram;          //code omitted     } } For most projects we recommend using external modules and using namespace for quick demos and porting old JavaScript code.
from https://basarat.gitbooks.io/typescript/content/docs/project/namespaces.html
Typescript supports ES6 Modules, they are new and shiny and everybody seems to agree they are the way to go. The idea seems to be that each file is a module and by suppling the files in import statements you can define your dependencies very explicitly which makes it easy for bundling tools to efficently pack your code. I mostly have one class per file which doesn't seem to work to well with dhte module pattern.
This is my file structure after the refactor:

Also I'm having an index.ts file in each folder so I can import all of its classes by import * as FolderModule from "./folder" 
export * from "./AggregateLoader"; export * from "./ImageLoader"; export * from "./TiledLoader"; export * from "./XhrLoaders"; export * from "./XmlSpriteFontLoader"; Example file - I think the problem becomes clearly visible here..
import {SpriteFont} from "./SpriteFont"; import {ISpriteTextGlyph, ISpriteChar} from "./Interfaces"; import {Event,EventArgs} from "../../Core"; import {Attribute, AttributeConfiguration} from "../Attributes"; import {DataType} from "../GlEnums"; import {VertexStore} from "../VertexStore"; import {IRectangle} from "../Geometry"; import {vec3} from "gl-matrix";  export class SpriteText {     // code omitted } Example usage. As you can see I no longer have to walk through the namespaces, because I can import the classes directly.
import {     Context,     Shader,     ShaderProgram,     Attribute,     AttributeConfiguration,     VertexStore,     ShaderType,     VertexBuffer,     PrimitiveType } from "../cross/src/Eye";  import {     Assets,     TextLoader } from "../cross/src/Load";  export class _01_BasicQuad {     context: Context;     shader: ShaderProgram;      // code omitted. } IMO both approaches seem to flawed. Namespaces seem to be terribly outdated, impractical for large projects and incompatible with common tools while using modules is quite inconvenient and breaks some of the features I was adapting typescript for in the first place.
In a perfect world I would write my framework using the namespace pattern and export it as a module which then can be imported and bundled with its dependencies. However this doesn't seem to be possible without some ugly hacks.
So here's my question: How have you dealt with these problems? How can I minimize the drawbacks each approach implies?
After gaining a bit more experience with typescript and javascript developement in general, I have to point out that modules are probably the way to go for 90% of all use cases.
The last 10% are hopefully legacy projects which use global namespaces, that you want to spice up with a little typescript (which works great by the way).
Much of my critic for modules can be (and has been) resolved by better IDE support. Visual Studio Code has since added automatic module resolution which works great.
A module is a way which is used to organize the code in separate files and can execute in their local scope, not in the global scope. A namespace is a way which is used for logical grouping of functionalities with local scoping.
Namespaces are a TypeScript-specific way to organize code. Namespaces are simply named JavaScript objects in the global namespace. This makes namespaces a very simple construct to use. Unlike modules, they can span multiple files, and can be concatenated using outFile .
TypeScript is an extension of the JavaScript language that uses JavaScript's runtime with a compile-time type checker. In TypeScript, you can use namespaces to organize your code. Previously known as internal modules, namespaces in TypeScript are based on an early draft of the ECMAScript modules.
tl;dr: Do not choose the past. Choose the future: Modules.
In early drafts of the ES6 modules specification, there was an inline modules notion, which then has been eliminated in September 2013. However, this notion was already implemented by the TypeScript team, in 2012, with the first beta versions of the language: it was internal modules. Then, the final specification for ES6 modules has been released in July 2014 without inline modules. A year later, in July 2015, with the version 1.5 of TypeScript, internal modules has been renamed to namespaces in order to avoid confusion with the standard.
Namespaces are a legacy feature. It won't be a part of the language ECMAScript. And the TypeScript team will continue to follow the standard. There has been no improvement regarding TS namespaces since the release of the ECMAScript modules standard in July 2014.
Cons [of ES6 modules]
- Very tedious if every class is a different file, you will have to type the same import statements over and over.
- Renaming files will break your code (bad).
- Refactoring class names won't propagate through to your imports (very bad - might depend on your IDE though, I'm using vs-code)
We can hope some improvements on these issues with future IDEs. The first one is already resolved by WebStorm.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With