npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

navstack

v0.4.3

Published

Manages multiple screens with mobile-friendly transitions

Downloads

122

Readme

Navstack

Manage a stack of multiple views. Perfect for tabs, navigation stacks, and similar. Inspired by iOS's UINavigationController.

  • Framework-agnostic: made to play fair with Backbone, Ractive, React.js and more. Can even work with plain old jQuery.

  • Router-friendly: made to be able to work with pushstate routers (like page.js), giving you back/forward button support.

  • Mobile-like transitions: buttery-smooth transitions are available out of the box, modeled after iOS7.

Status

Installation

Navstack is a JS + CSS bundle.

Or get it via Bower or NPM:

$ npm install --save navstack
$ bower install --save navstack

Then use it:

<script src="navstack.js"></script>
<link rel="stylesheet" href="navstack.css">

Getting started

Create your stack by instanciating Navstack. Pass a jQuery object or a DOM node to el.

stage = new Navstack({
  el: $('#stage')
});

Adding panes

Use .push() to create your panes. It takes 2 arguments:

  • name (string): the ID of the pane. This will the unique identifier that will identify your pane.
  • initializer (function): a function to return the pane's contents.
// Navigate to new pages using push.
stage.push('/home', function() {
  return $("<div class='full-screen'>This is the home screen</div>");
});

// The first parameter is an ID for the pane to be pushed.
// This will animate the stage to slide into the new view.
stage.push('/task/1', function() {
  return $("<div class='full-screen'>Task #1 details: ...</div>");
});

Libraries support: The initializer can return jQuery elements, Backbone views, Ractive instances, or React.js components.

stage.push('task:1', function() {
  /* Backbone: */
  return new Backbone.View({ ... });

  /* Ractive: */
  return new Ractive({ template: '...' });

  /* React.js: */
  var MyComponent = React.createClass({ ... });
  return new MyComponent({ name: "John" });
});

Returning back: Calling .push() again with a pane that is already part of the stack will make the stage will animate backwards (slide left) to that old pane. If the pane is recent (ie, last 5 panes used or so), the pane's DOM element is previously hidden and will be made visible. If it's an old pane, it will be recreated based on the initializer first passed onto .push().

stage.push('/home', function() { ... });

stage.push('/task/1', function() { ... });

// this will slide left into the first screen.
stage.push('/home', function() { ... });

Groups & modal dialogs

Group panes together by passing the { group: 'groupname' } option to push().

This allows you to create logical sections of your app UI. Panes of the same group will slide left-and-right by default, while panes of a different group will pop up like modal dialogs.

In this example below, the settings pages will pop up in a modal:

stage = new Navstack({ el: ... });

// Use no group names for the main parts of your app.
stage.push('home', function () { ... });
stage.push('messages', function () { ... });
stage.push('message/user1', function () { ... });

// this will pop up the `config` settings dialog with a modal popup animation.
// the next one, `account` settings, will animate by sliding to the right,
// since it's in the same group as the previous pane.
stage.push('config',  {group: 'settings'}, function () { ... });
stage.push('account', {group: 'settings'}, function () { ... });

// by going back to `home`, there will be a modal exit animation, since you're 
// transitioning from one group to another.
stage.push('home', function () { ... });

Prefixes: You can also prefix names with groupname! (eg, settings!account) -- this accomplishes the same thing.

// these two are equivalent
stage.push('config',  {group: 'settings'}, function () { ... });
stage.push('settings!config', function () { ... });

Sleeping and waking

When a view is about to be hidden, a navstack:sleep event is called. When a view is about to be shown, a navstack:wake event is called. These are triggered as jQuery, Backbone, Ractive or React events, depending on what your pane is.

var $box = $("<div>hello</div>");
$box.on('navstack:sleep', function () { ... });
$box.on('navstack:wake', function () { ... });

stage.push('home', function () {
  return $box;
});

Use with routers

To take full advantage of Navstack, it's recommended to use it with a router to manage browser history states (read: makes the browser "Back" button work). Here's an example usage of Navstack with page.js:

var stack = new Navstack();

page('/home', function (ctx) {
  stack.push(ctx.canonicalPath, function () {
    return $("<div>...</div>");
  });
});

page('/book/:id', function (ctx) {
  stack.push(ctx.canonicalPath, function () {
    return $("<div>...</div>");
  });
});

document.body.appendChild(stack.el);

Or with Backbone.Router:

var stack = new Navstack();

App.Router = Backbone.Router.extend({
  routes: {
    '': 'home',
    'book/:id': 'showBook'
  },

  home: function () {
    stack.push('home', function () {
      return new HomeView(...);
    });
  },

  showBook: function (id) {
    stack.push('book:' + id, function () {
      var book = new Book(id: id);
      return new BookView(book);
    });
  }
});

$(function () {
  $(stack.el).appendTo('body');
  Backbone.history.start();
});

Navstack

new Navstack(options)

Instanciates a new Navstack stage that manages multiple panes.

stage = new Navstack({
  el: '#stack'
});

You may pass any of these options below. All of them are optional.

  • el a selector, a jQuery object, or a DOM element.
  • transition a string of the transition name to use.
  • groupTransition a string of the transition to use in between groups.

You'll then use push to add panes into the stage.

stage.push('home', function () {
  return $("<div>Hello</div>");
});

Attributes

panes

Index of panes that have been registered with this Navstack. Object with pane names as keys and Pane instances as values.

stage.push('home', function () { ... });

stage.panes['home']
stage.panes['home'].name   //=> 'home'
stage.panes['home'].el     //=> DOMElement
stage.panes['home'].view

active

A reference to the active pane. This is a Navstack.Pane instance.

stage.push('home', function() { ... });

// later:
stage.active.name   //=> 'home'
stage.active.el     //=> DOMElement
stage.active.view

It is a pointer to the active pane in the panes object.

stage.push('home', function() { ... });

// later:
stage.active === stage.panes['home']

stack

Ordered array of pane names of what are the panes present in the stack. When doing push(), you are adding an item to the stack.

stage.push('home', function() { ... });
stage.stack
=> ['home']

stage.push('timeline', function() { ... });
stage.stack
=> ['home', 'timeline']

transition

The transition name to be used. Defaults to "slide". This can either be a String (a transition name), a Function, or false (no animations).

stage = new Navstack({
  transition: 'slide',
  groupTransition: 'modal'
});

// the second push here will use the slide animation.
stage.push('home', function() { ... });
stage.push('mentions', function() { ... });

// this will use the modal transition, as its in a different group.
stage.push('auth!login', function() { ... });

groupTransition

Pane transition to use in between groups. Defaults to "modal". See transition for more details.

el

The DOM element of the stack. You may specify this while creating a Navstack instance. When no el is given, it will default to creating a new <div> element.

stage = new Navstack({
  el: document.getElementById('#box')
});

You may also pass a jQuery object here for convenience.

stage = new Navstack({
  el: $('#box')
});

You can access this later in the Navstack instance:

$(stage.el).show()

Methods

push

.push(name, [options], [fn])

Registers a pane with the given name.

The function will specify the initializer that will return the view to be pushed. It can return a DOM node, a jQuery object, a Backbone view, Ractive instance, or a React component.

stage.push('home', function() {
  return $("<div>...</div>");
});

You can specify a pane's group by prefixing the name with the group name and a bang.

stage.push('modal!form', function() {
  return $("<div>...</div>");
});

You can specify options.

stage.push('home', { group: 'root' }, function() {
  return $("<div>...</div>");
});

Available options are (all are optional):

  • group (String) the group name that the pane should belong to.
  • transition (String) the name of the transition to use. See Navstack.transitions.

init

Constructor. You may override this function when subclassing via Navstack.extend to run some code when subclassed stack is instanciated.

var MyStack = Navstack.extend({
  init: function() {
    // initialize here
  }
});

goNow

.goNow(name, [options])

Performs the actual moving, as delegated to by .go(), which is then delegated from .push().

For external API, Use .push() instead.

remove

Destroys the Navstack instance, removes the DOM element associated with it.

stage = new Navstack({ el: '#stack' });
stage.remove();

This is also aliased as .teardown(), following Ractive's naming conventions.

ready

ready(fn)

Runs a function fn when transitions have elapsed. If no transitions are happening, run the function immediately.

nav = new Navstack();
nav.push('home', function () { ... });
nav.push('messages', function () { ... });

nav.ready(function () {
  // gets executed only after transitions are done
});

Events

A stack may emit events, which you can listen to via [on()].

stage = new Navstack();

stage.on('push', function (e) {
  e.direction  // 'forward' or 'backward'
  e.current    // current pane
  e.previous   // previous pane
});

// to listen for a specific pane:
stage.on('push:NameHere', function (e) {
  ...
});

Available events are:

  • push -- called after a push() succeeds
  • push:NAME -- called after a pane with the name NAME is pushed
  • purge -- called when a pane is purged from being obsolete
  • purge:NAME -- called when pane NAME is purged
  • remove -- called when removing the stack

on

.on(event, function)

Binds an event handler.

stage.on('remove', function() {
  // do things
});

off

.off(event, callback)

Removes an event handler.

stage.off('remove', myfunction);

one

.one(event, callback)

Works like .on, except it unbinds itself right after.

Navstack.Pane

Panes are accessible via navstack.panes['name'] or navstack.active.

stage = new Navstack();
pane = stage.active;

pane.name
pane.initializer  // function
pane.el
pane.view

You'll find these properties:

  • name (String) the identifier for this pane as passed onto push().
  • parent a reference to the Navstack instance.
  • el DOM element.
  • view the view instance created by the initializer passed onto push().
  • adaptor a wrapped version of view (internal).

Static members

These are static members you can access from the global Navstack object.

Navstack.extend

extend(prototype)

Subclasses Navstack to create your new Navstack class. This allows you to create 'presets' of the options to be passed onto the constructor.

var Mystack = Navstack.extend({
  transition: 'slide'
});

// doing this is equivalent to passing `transition: 'slide'` to the
// options object.
var stack = new Mystack({ el: '#stack' });

Navstack.transitions

The global transitions registry. It's an Object where transition functions are stored.

Available transitions are:

  • default show new panes immediately, no animation
  • slide slides the new panes horizontally like iOS7
  • modal slides the new panes vertically

Whenever a transition is used on a Navstack (eg, with new Navstack({ transition: 'slide' })), it is first looked up in the stack's own registry (stage.transitions). If it's not found there, it's then looked up in the global transitions registry, Navstack.transitions.

You can define your own transitions via:

Navstack.transitions.foo = function (direction, current, previous) {

  // this function should return an object with 3 keys: `before`,
  // `run`, and `after`. Each of them are asynchronous functions
  // that will perform different phases of the transition.
  //
  // you can use the arguments:
  //
  //   direction - this is either "first", "forward", or "backward".
  //   previous  - the previous pane. This an instance of [Pane].
  //   current   - the pane to transition to.

  return {
    before: function (next) {
      // things to perform in preparation of a transition,
      // such as hide the current pane.
      // invoke next() after it's done.

      if (current) $(current.el).hide();
      next();
    },

    run: function (next) {
      // run the actual transition.
      // invoke next() after it's done.

      if (current)  $(current.el).show();
      if (previous) $(previous.el).hide();
      next();
    },

    after: function (next) {
      // things to perform after running the transition.
      // invoke next() after it's done.
      next();
    }
  }
};

Navstack.jQuery

Pointer to the instance of jQuery to optionally use. Set this if you would like Navstack to utilize [jQuery.queue].

Navstack.jQuery = jQuery;

Cheat sheet

// Basic usage
stage = new Navstack();

// All options are optional
stage = new Navstack({
  el: 'body', /* selector, or jQuery object */

  adapt: ['backbone'],
  adaptors: { backbone: ... },

  transition: 'slide' | 'modal',
  transitions: { slide: ... }, /* custom transitions */
});

stage.push('pane_id', function () {
  return $("<div>...</div>");
  return new Ractive(...);
  return new ReactComponent(...);
});

// Return to old pane
stage.push('pane_id');

// The main element
stage.el;           //=> <div>

// Access the active pane
stage.active;
stage.active.el;    //=> <div>
stage.active.view;  //=> whatever's returned in the initializer
stage.active.name;  //=> "home"

// Access the stack
stage.stack;
stage.stack['pane_id'].view;
stage.stack.length;

// Global repositories
Navstack.adaptors = {...};
Navstack.transitions = {...};

Thanks

Navstack © 2014+, Rico Sta. Cruz. Released under the MIT License. Authored and maintained by Rico Sta. Cruz with help from contributors.

ricostacruz.com  ·  GitHub @rstacruz  ·  Twitter @rstacruz