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

backbone-model-factory

v1.3.0

Published

Provides a factory for generating model constructors that will never duplicate instances of a model with the same unique identifier. Useful for sharing model instances across views.

Downloads

3,335

Readme

Backbone.ModelFactory

Provides a function for generating model constructors which will never produce multiple instances with the same unique identifier.

When you create a new instance of a model created with Backbone.ModelFactory and give it an id it will never create duplicate instances of the same model with a given id.

var user1 = new User({id: 1});
var user2 = new User({id: 1});
var user3 = new User({id: 2});

console.log(user1 === user2); // true
console.log(user3 === user1); // false

Benefits

When architecting loosely-coupled web applications, it is generally considered a good practice for modules which control separate pieces of functionality to not rely on each other. ModelFactory helps with that!

Using ModelFactory models, you can eliminate the need to pass model instances between unrelated views.

Additionally, collections using ModelFactory models that fetch data will always be populated by existing model instances if they already exist in the cache. This prevents creating multiple model instances which represent the same resource.

ModelFactory makes sharing models between collections, views, routers, etc. almost completely hands-off.

Dependencies

Backbone.ModelFactory 1.2.0+ depends on the following libraries:

  • Underscore: 1.2.0+
  • Backbone: 0.9.9+

Earlier versions of ModelFactory will work with Backbone 0.9.0-1.0.0 and do not depend on Underscore directly.

Inclusion

Backbone.ModelFactory supports three methods of inclusion.

  1. Node:
var Backbone = require('backbone-model-factory');
  1. AMD/RequireJS:
require(['backbone-model-factory'], function (Backbone) {
  // Do stuff...
});
  1. Browser Globals:
<script src="path/to/backbone.js"></script>
<script src="path/to/backbone-model-factory.js"></script>

Usage

Rather than extending Backbone.Model, constructors are created by a call to Backbone.ModelFactory. Instead of this:

var User = Backbone.Model.extend({
  defaults: {
    firstName: 'John',
    lastName: 'Doe'
  }
});

...do this:

var User = Backbone.ModelFactory({
  defaults: {
    firstName: 'John',
    lastName: 'Doe',
    isAdmin: false
  }
});

ModelFactory also supports inheritance, so model constructors can extend each other by providing a model constructor (whether generated by ModelFactory or not) as the first argument:

var Admin = Backbone.ModelFactory(User, {
  defaults: _.defaults({
    isAdmin: true
  }, User.prototype.defaults)
});

Consequences of Extending Models

Models created with ModelFactory will not share their unique-enforcement behavior with models which they extend or which extend them. For example, using the User and Admin models above giving each the same id would not result in the same object:

var user = new User({id: 1});
var admin = new Admin({id: 1});

console.log(user === admin); // false

Subclassing to achieve different defaults is usually a bad idea. The User/Admin example models are not intended to be an example of good practices. :)

Potential Gotchas

Relationships and Recursion

If you are using any sort of nested relationships (models or collections within models) and operating on those relationships in a recursive manner, it can cause an infinite loop if an identical model instance exists in its own relationship graph.

For example, let's say a User has a collection of Posts and each Post has a User. Having an infinite path of object references (User -> Posts -> Post -> User -> Posts -> ad infinitum) is not harmful, but if you do any automated/recursive serialization of these relationships (into JSON, etc), you might encounter an infinite loop if the serialization is unchecked.

This entirely depends on how you're using Backbone and what you've built upon it.

Caching and Wiping

Since model instances are cached, the potential exists for unneeded objects to hang around in memory.

As of ModelFactory 1.2.0 you can wipe model instances once you are done with them. Both the model instances themselves and the constructor function have a wipe method which can be used:

var User = Backbone.ModelFactory();

var jake = new User({id: 1});
var joe1 = new User({id: 2});
var joe2 = new User({id: 2});
var jane = new User({id: 3});
var jen = new User({id: 4});
var josh = new User({id: 5});
var james = new User({id: 6});

// Wipe the instance of User with id: 2 from cache. 
// "jake" will still exist in memory.
jake.wipe();

// Wipe a single cached instance from the model. 
// "joe1" and "joe2" will still exist in memory.
User.wipe(joe);

// Wipe multiple cached instances in an array or a collection. 
// As before, "jane", "jen", "josh", and "james" exist in 
// memory still.
User.wipe([jane, jen]);
User.wipe(new Backbone.Collection([josh, james]));

console.log(_.keys(User._cache).length); // 0

Note: This will only remove them from the internal cache - any references in your views, collections, or elsewhere will not be deleted! To manage your references to models, hook into the various wipe events...

Cleaning up Your References with Wipe Events

Inspired by an issue raised by GitHub user @niksy, starting in ModelFactory 1.3.0, developers can clean up their own model references easily using wipe events. These are triggered with the following logic:

  • Calling model.wipe() will trigger "wipe" on that instance. The instance is passed as an argument to any callback(s).
    • Calling model.wipe() will also trigger the model/constructor-level events (see below).
  • Calling MyModel.wipe() will trigger individual "wipe" events on every instance being wiped and...
    • If no models are wiped, no events are fired.
    • If wiping all instances (one or more), trigger "wipe:all".
    • If wiping fewer than all, trigger "wipe:some".
    • In both cases, callbacks receive the following arguments:
      • Constructor: The model constructor on which the event was triggered. In this case, MyModel.
      • models: An array of models that were removed from the cache.
      • remainder: An array of models remaining in the cache (empty in the case of "wipe:all").

In all cases, {silent: true} will be respected.

var User = Backbone.ModelFactory();

var jake = new User({id: 1});
var jane = new User({id: 2});
var jen = new User({id: 3});
var josh = new User({id: 4});

// Triggers "wipe" on jake.
// Triggers "wipe" and "wipe:some" on User with arguments:
//   - [User, [jake], [jane, jen, josh]]
jake.wipe();

// Triggers individual "wipe" events on jane, jen, and josh. 
// Triggers "wipe" and  "wipe:all" on User with arguments:
//   - [User, [jane, jen, josh], []];
User.wipe([jane, jen, josh]);

Tests

Inclusion

There are 3 files in test/inclusion and they account for the 3 supported methods of including this module. To execute these tests, simply open the HTML files in a browser or install the npm dependencies and run node test/inclusion/node-module.js.

Unit

Mocha tests exist in test/test.js.

To run the tests, make sure development dependencies are installed and use npm test.

To test against different versions of Backbone (or Underscore), install the version desired (e.g., npm install backbone@~1.0.0).

Inspriation

This is inspired by SoundCloud's approach detailed in Building the Next SoundCloud under "Sharing Models between Views."