#ViewRenderer.ts - an MVC approach with TypeScript
"I was inspired by the many good javascript frameworks out there so I made my own".
Not. This is something I did to learn about what it takes to build simple client side functionality MVC style with TypeScript, without a framework.
In my approach I used the three MVC components this way:
- The model ** A TypeScript class or a plain javascript model.
- The view ** Essentially a function that returns html.
- The controller ** The controller is invoked on view render. It hooks upp bindings. Can be user actions, timers, router events, pubsub publishs, clicks or whatever.
The components are tied together with a ViewRenderer class (25 lines of code). It takes the mentioned components as parameters and has a render function which renders the view on a given jQuery element.
So the the ViewRenderer is the only code that is my own here. Other than that I use jQuery and the minimalistic Html-builder Dom-O in my sample to get a really nice DOM-experience.
Dom-O by jed is the thing that got me inspired to write this, and what really makes this interesting since it's possible to write dynamic views in good looking and pure ts/js with it. Yep, no external templating whatsoever.
To get structure in the application, the components can for example be spread across different files (asp net mvc style) or bound together in TypeScript modules. In the sample I use the latter approach.
Check out the samples previews:
- Simple views sample
- Multiple views sample A bit more code. I built it as a lottery game simulator (using swedish lottery "Triss" winning scheme). One master view and one instantiatable "game" view. It's using local pubsub with amplify to communicate between page parts. And it's using websockets library Xsockets to store results and broadcast them to other game players.
In the simple sample I've got two modules, "masterModule" and "personModule".
The masterModule looks like this and shows how the the Module and the View are bound together with a viewrenderer:
module masterModule {
class MasterModel {
subViewName: string;
}
var masterModel = new MasterModel();
masterModel.subViewName = "personView";
// here's how the view syntax look like with "Dom-o", build html with functions
var masterView = (model: MasterModel) =>
DIV(
H1("Sample MVC"),
P("Rendered at time : " + new Date()),
DIV({ id: model.subViewName })
);
export var masterViewRenderer = new ViewRenderer(masterView, masterModel);
}
Next step is to render the view to an element. I do that in my application module:
// setup masterview to use body as container
masterModule.masterViewRenderer.render($("body"));
Note that the class instantiations take place inside the modules, and in the main module (app.ts) I just tell my existing views where they are going to be rendered. That is of course a matter of preference.
The personModule has a view and a model and also adds a controller (controllerbindings), to handle some user interaction:
var personControllerBindings = (model: PersonModel, viewRenderer:ViewRenderer) => {
$("#button1").on("click", () => {
model.firstName = $("#firstname").val();
model.secondName = $("#secondname").val();
viewRenderer.render();
});
};
export var personViewRenderer = new ViewRenderer(personView, personModel, personControllerBindings);
The fullName is a function within the module:
class PersonModel {
firstName: string;
secondName: string;
fullName() {
return this.firstName + " " + this.secondName;
}
}
And the personView renders the input boxes as well as the full name to the view:
P(
STRONG("Hello : " +
model.fullName()))
);
##A helper to build the input control Since we have two similar input "controls" we like to put their code inside a function. Here's how:
var inputControl = (label: string, id: string, value:()=> string) =>
DIV(LABEL(label + ": "),
INPUT({ id: id, value: value() }),
BR());
Which means our view can be written as:
var personView = (model: PersonModel) =>
DIV(
inputControl("First name", "firstname", () =>model.firstName),
inputControl("Second name", "secondname", () =>model.secondName),