A humble library for building SPAs
Create a new directory for your project, and then create a npm package.
npm init
Install loki
npm i @skapoor8/loki
To create a new loki project
npx loki new myApp
This will create a loki-js project with the following files:
Your app will look like this:
import Loki from "@skapoor8/loki"; class App extends Loki.Component { static selector = "loki-app"; render() { return ` <h1>Loki App</h1> <p>Welcome to your Loki app!</p> `; } } export default App;
Now you can serve your app:
npx loki build npx loki serve
Your app should be live on localhost:3012
Creating a component is as simple as adding a new file. In the render method, use Embedded JavaScript (EJS) to embed variables, use if branches and for loops. Choose a component selector using the static selector property on your component.
// src/example-component.js import Loki from "@skapoor8/loki"; export class MyList extends Loki.Component { // define a component selector here static selector = "my-list"; /* * A components template goes here. Attach event handlers as shown below * for the click event. The templating language is EJS. */ render() { this.state = { list: [1, 2, 3], }; return /* html */ ` <div > <h1>My List</h1> <ol> <% for (item of list) { %> <li><%= item %></li> <% } %> </ol> </div> `; } }
Now you can register your new component in your app, and use its selector in your app component's template.
// src/app.js import Loki from "@skapoor8/loki"; import { ExampleComponent } from "./example-component"; class App extends Loki.Component { static selector = "loki-app"; static components = [ExampleComponent]; render() { return ` <my-list></my-list> `; } } export default App;
Any changes in your components should be reflected immediately in the browser while in development. This behavior is controlled by the "mode" property in your loki.json file, which is set to "local" by default.
Listen to events in the DOM with event handlers. Pass data through dataset attributes.
... export class MyList extends Loki.Component { static selector = "my-list"; render() { this.state = { list: [1, 2, 3], }; return /* html */ ` <div (click)="handleClick" data-list="<%= list %>"> <h1>My List</h1> <ol> <% for (item of list) { %> <li><%= item %></li> <% } %> </ol> </div> `; } handleClick(e) { console.log('click:', e.target.dataset.list); } }
Lifecycle methods onBeforeInit, onInit, onBeforeDestroy, onDestroy, and onUpdateState are provided to initialize component data, and register side-effects. For example, onBeforeInit is where initial state of a component should be set.
... export class MyList extends Loki.Component { static selector = "my-list"; render() { return /* html */ ` <div (click)="handleClick" data-list="<%= list %>"> <h1>My List</h1> <ol> <% for (item of list) { %> <li><%= item %></li> <% } %> </ol> </div> `; } onBeforeInit() { this.state = { list: [1, 2, 3], }; } handleClick(e) { console.log('click:', e.target.dataset.list); } }
Use the setState method to re-render your component with its updated state
... export class MyList extends Loki.Component { static selector = "my-list"; render() { return /* html */ ` <div > <h1>My List</h1> <ol> <% for (item of list) { %> <li><%= item %></li> <% } %> </ol> <button (click)="handleAdd">Add</button> </div> `; } onBeforeInit() { this.state = { list: [1, 2, 3], }; } handleAdd() { this.setState({ list: [...this.state.list, his.state.list.length + 1] }); } }
Use the components static property to register child components which will be used in your component's template. The state property can be used on all Loki components to initialize component states.
export class MyList extends Loki.Component { static selector = "my-list"; static components = [MyListItem]; render() { return /* html */ ` <div > <h1>My List</h1> <% for (item of list) { %> <my-list-item state="<%= item %>"></my-list-item> <% } %> <button (click)="handleAdd">Add</button> </div> `; } onBeforeInit() { this.state = { list: [1, 2, 3], }; } handleAdd() { this.setState({ list: [...this.state.list, his.state.list.length + 1], }); } } class MyListItem extends Loki.Component { static selector = "my-list-item"; render() { return /* html */ ` <span><%= value %></span> `; } }
You can create custom events in your components, and dispatch them with the emit method.
export class MyList extends Loki.Component { static selector = "my-list"; static components = [MyListItem]; static events = ["add"]; render() { return /* html */ ` <div > <h1>My List</h1> <% for (item of list) { %> <my-list-item state="<%= item %>"></my-list-item> <% } %> <button (click)="handleAdd">Add</button> </div> `; } onBeforeInit() { this.state = { list: [1, 2, 3], }; } handleAdd() { const toAdd = this.state.list.length + 1; this.setState({ list: [...this.state.list, toAdd], }); this.emit("add", { added: toAdd }); } } class MyListItem extends Loki.Component { static selector = "my-list-item"; render() { return /* html */ ` <span><%= value %></span> `; } }
Use the style static method to add styles to your component's template. All css will automatically be prefixed with your component's selector to scope the css properly.
export class MyList extends Loki.Component { static selector = "my-list"; static components = [MyListItem]; static events = ["add"]; static style() { return /* css */ ` .container { width: 100%; display: flex; flex-direction: column; align-items: stretch; } h1 { border-bottom: thin solid gray; } `; } render() { return /* html */ ` <div class="container"> <h1>My List</h1> <% for (item of list) { %> <my-list-item state="<%= item %>"></my-list-item> <% } %> <button (click)="handleAdd">Add</button> </div> `; } onBeforeInit() { this.state = { list: [1, 2, 3], }; } handleAdd() { const toAdd = this.state.list.length + 1; this.setState({ list: [...this.state.list, toAdd], }); this.emit("add", { added: toAdd }); } } class MyListItem extends Loki.Component { static selector = "my-list-item"; render() { return /* html */ ` <span><%= value %></span> `; } }
Loki implements a simple DI system, whereby services that inherit from Loki.Service will be injected at the highest level in the component subtree from where they are first registered via the services static property on components.
Services implement lifecycle methods onLoad and onUnload to handle any data loading and cleanup. Services also provide a static services property to allow them to inject other services, or data stores.
For example, say we wish to put all our code the interacts with the api layer in a service. We can do so by creating a MyApiService class in our project. We may have a function loadList on this class to fetch our list from our api, instead of initiliazing it with meaningless values as we do now. We can now inject this service, and use it as follows:
export class MyList extends Loki.Component { static selector = "my-list"; static components = [MyListItem]; static events = ["add"]; static services = { api: MyApiService }; static style() { return /* css */ ` ... `; } render() { return /* html */ ` <div class="container"> <h1>My List</h1> <% if (isLoading ) { %> <span>Loading...</span> <% } else { %> <% for (item of list) { %> <my-list-item state="<%= item %>"></my-list-item> <% } %> <button (click)="handleAdd">Add</button> <% } %> </div> `; } onBeforeInit() { this.state = { list: [], isLoading: true, }; } // api call goes here onInit() { const { api } = this.services; api.loadList().then((data) => this.setState({ list: data, isLoading: false, }) ); } handleAdd() { const toAdd = this.state.list.length + 1; this.setState({ list: [...this.state.list, toAdd], }); this.emit("add", { added: toAdd }); } } class MyListItem extends Loki.Component { static selector = "my-list-item"; render() { return /* html */ ` <span><%= value %></span> `; } }
Using a 3rd party library is the recommended way of handling reactivity in a Loki App. However, a simple data store is provided as well. Create a data store as follows:
import Loki from "loki"; export class UIStore extends Loki.Store { /* * Initialize the state of your store here */ init() { this.payloads = { lists: [], isLoading: true, }; } }
To use a store, inject it in a component or a service. To publish values to a store property, use the pub api.
myStore.pub("isLoading", false);
To react to updates in store properties, use the sub api.
const sub = myStore.sub( "isLoading", (propVal) => { // do something }, skipFirst ); // you can unsubscribe when the subscription should stop running // to prevent memory leaks sub.unsubscribe();
skipFirst is a boolean value, which is true if omitted. This means, be default, store subscriptions do not run with an initial value, akin to RxJS subjects. Passing a true value instead of skip first will make the subscription behave like an RxJS Behavior Subject, i.e. the subscription will fire with the property's initial value.
I believe a great way to understand how something works is to attempt to build a primitive version of it. This is exactly what loki-js is. I love frameworks like Angular and React, and loki-js is an attempt to re-create some of their magic.
While building loki-js, I got to interact first-hand with the complexities of maintaining and propagating component states, and the challenges of reactive programming, and gained an appreciation for the functionality provided by many beloved frameworks.
View my iOS reminders app clone built with Loki HERE.
Read more about it on its own project page HERE.