GitXplorerGitXplorer
q

ko-router

public
7 stars
1 forks
1 issues

Commits

List of commits on branch master.
Unverified
bff1d18524ecbeadee27a9aab59dd5ab947b399b

fix #4

qqwelias committed 8 years ago
Unverified
9479f263ac630b263fc4559d518b6c5f51ed6b25

clarify polyfill.io

qqwelias committed 8 years ago
Unverified
d29dfaeb6250178c5c07bae1b96df925421b5d3c

fix typo

qqwelias committed 8 years ago
Unverified
0080e684ee2af18ce52eb5d963a4f21f1d7f36d2

start option endWithComma

qqwelias committed 8 years ago
Unverified
9d558ac24b9102da9647fa99b14ce7240ba312b6

add nav example

qqwelias committed 8 years ago
Unverified
f0c57e0e2cb98e9e5bb54a08d8689188bd29e43d

Update README.md

qqwelias committed 8 years ago

README

The README file for this repository.

ko-router

KnockoutJS HTML5 History router.

Supports template lazy-load, (a)synchronous pre-, post-hooks and is just under 250 lines.

Requirements

  • knockout v3.4+
  • HTML5 History
  • URL
  • Promise
  • fetch
  • es6-shim
  • polyfill.io recommended ( URL, Promise, fetch, es6-shim )

Usage

Upon load creates window.router.

Examples

index.js

// on ready
window.ko.applyBindings( app );

// optional loader animation
if ( window.someLoader ) {
    window.router.loader.start = window.someLoader.start;
    window.router.loader.done = window.someLoader.done;
};

window.router.root.guard = function ( route ) {
    // if location == '/home/foo', then route == [ 'home', 'foo' ]
    // only root guard will receive whole route
    // navigate to /home if location == '/'
    if ( !route.length ) return Promise.reject( window.router.navigate( '/home' ) ); // reject current navigation
};

return window.router.start();

index.html

<body>
    <!-- some page content -->

    <!-- /home -->
    <section data-bind="page: {
        route: 'home',
        template: 'tmpl-page-home',
        guard: function( id ) {
            // id === 'home'
            if ( !app.me._id ) return Promise.reject( router.navigate( '/login' ) ); // reject current navigation
        },
        src: '/pages/home.html',
        title: 'home'
    }">
    </section>

    <!-- /login -->
    <section data-bind="page: {
        route: 'login',
        template: 'tmpl-page-login',
        guard: function( id ) {
            // id === 'login'
            if ( app.me._id ) return Promise.reject( router.navigate( '/home' ) ); // reject current navigation
        },
        src: '/pages/login.html',
        title: 'login'
    }">
    </section>

    <!-- some page content -->

</body>

page/home.html

<script type="text/html"
    id="tmpl-page-home">

    <!-- some page content -->

    <!-- link to /home/foo -->
    <a data-bind="nav: 'foo'">foo</a>

    <!-- /home/foo -->
    <div data-bind="page: {
        route: '*',
        template: 'tmpl-page-content',
        guard: function ( id ) {

            // guard called on any location change that involved the page
            // so resolve if already was opened
            if ( id === this.current ) return Promise.resolve();

            // reject if misnavigation
            if ( !app[ id ] ) return Promise.reject();

            // reload foo table from server
            return app[ id ].table.reload(); // returns Promise
        },
        src: '/pages/content.html'
    }">
    </div>

    <!-- some page content -->

</script>

window.router

.route

ko.observableArray containing pathname splitted by "/".

Updated after successful navigation.

.navigate( href )

href is either a relative or absolute local path.

Returns a Promise.

.resolvePath( ...paths )

Will resolve paths in relation to location.origin

.start()

Will enable router and return a Promise.

.loader.start()

Dummy function, called when navigation starts.

Assign it something that will show some loader.

.loader.done()

Dummy function, called when navigation ends.

Assign it something that will hide some loader.

.on404()

Called if path not found. Override it.

.root

A root Page, created upon load.

window.router.Page( data )

Page class.

.element = data.element

Element page bound to, none for root.

.route = data.route

Route to match, none for root.

Values:

  • '/' - will match empty route, usefull for setting a default page at some level.
  • string - actual string to match.
  • '*' - match anything none empty.

.title = data.title

Title to set to the document, defaults to parent's title.

.template = data.template

Template value to pass to Knockout's template binding.

.guard = data.guard

Function to be called on an attempt to open the page.

Function accepts it's piece of route as first argument and is boud to a page, should return a Promise. Root page is the exception as it accepts whole array of routes. Useful for any kind of (a)synchronous checks/prehooks.

If you want to return a window.router.navigate( href ) wrap it into Promise.reject so that previous navigation stops.

.onclose = data.onclose

Function to be called on an attempt to close the page.

Bound to a page.

If you want to return a window.router.navigate( href ) wrap it into Promise.reject so that previous navigation stops.

.current = ''

Current page route.

.src = data.src

Template source to load if not found.

.context = data.context

Knockout's bindingContext of the parent's element.

.to = {}

References to child pages. Filled automatically.

.path()

Return a path to the page.

.parent()

Return parent.

.next( route )

Return child page matched to a route.

.check()

Internal function.

Return a Promise.all result of the page's guards.

.ensureTemplate()

Internal function.

Literally the function name.

.open( current )

Internal function.

Apply template binding to the page's elemtent.

Expands Knockout's bindingContext of the element with _page as a reference to this page.

.close()

Internal function.

Yeah.

.closeChildren()

Internal function.

Yep.

.closeSiblings()

Internal function.

Hmm.

Bindings

page

Creates Page instance, passing value to the constructor, and bounds it to a parent's to upon init.

Or updates existing one.

nav

Pass either relative or absolute local path to navigate to upon element click event.

Would not call navigate and will pass an event, so a new tab is opened, if clicked with either MMB or Ctrl + LMB, to preserve native browser behavior.

Warnings

Internal functions should not be called manually, but hey!