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 🙏

© 2025 – Pkg Stats / Ryan Hefner

ts-modal

v1.0.3

Published

Loosely based off of Angular Bootstrap Modal

Downloads

5

Readme

ts-modal [tsModalService] (An Angular Service for directives)

Loosely based off of Angular Bootstrap Modal

Reasoning

Why build a new modal service when the bootstrap one works fine?

Modularity

Problem: Angular-Bootstrap's modal is messy. Developers are required to use a controller/template system, which means if you want to modularize your component to be used in a modal (using a directive), you'd have to pass data through the modal resolve, include it in the controller $scope, then pass each variable into attributes on the directive element, and lastly, you'll need to handle them all again in the directive. Also, if you want to store your controller/template elsewhere and pass references in, the two aren't inherently paired (unlike directives).

Solution: This component uses directives instead. You explicitly declare the directive name, and the component is compiled inside the modal:

tsModalService.open({
  directive: 'directive-name'
});

Animation

Problem: No control over animation

Solution: Use optional dependency VelocityJS for animations. Allows for more control.

Install

There are currently only 2 ways to use this module:

  • NPM: npm install ts-modal --save
  • Clone repo: Use the files in the dist folder

Using the module as a dependency

This module was purposefully built to be a dependency in an Angular application, so the ng-module needs to be passed in as an argument.

Also, the package needs to be imported, so some sort of module loader is required (webpack, browserfy, systemJS, etc).

import the module

import modal from 'ts-modal';

initialize and pass in your ng-module (app)

let app = angular.module('app', []);
modal(app);

now the service (tsModalService) is available to be injected

app.controller('ctrl', function(tsModalService) {
  tsModalService.open({
      directive: 'directiveName',
      resolve: {
          data: Promise.resolve('data'),
          coolStuff: 'I am Legend'
      }
  });
});

The modal service generates the following html behind the scenes and passes in the resolved data as unique identifiers:

<div class="ts-modal">
    <directive-name data="data['id']" cool-stuff="data['id']"></directive-name>
</div>

Example of handling the modal inside declared directive:

app.directive('directiveName', function() {
  return {
    restrict: 'E',                     // "Element" is required
    scope: {
      data: '=',                       // data is passed in with two-way binding,
      coolStuff: '='                   // all items in the 'resolve' must be declared
    },
    controller: function($scope) {
      console.log($scope.data, $scope.coolStuff);  // you're data is available
    }
  };
});

Include the SCSS file in your styles:

@import '../../node_modules/dist/_modal.scss';

Methods

.open()

Open a new modal. You can have an unlimited amount of modals open at the same time, but keep in mind that .submit() and .cancel() methods only apply to the most recent or 'active' modal.

Returns a promise that is resolved when .submit() is is called, and rejected when .cancel() is called.

tsModalService.open({
    directive: 'directiveName',
    resolve: {
        data: Promise.resolve('data'),
        coolStuff: 'I am Legend'
    },
    size: 'medium', // (small|medium|large)
    display: 'notification', // (component|notification)
    closeBackdrop: true,
    closeEscape: true,
    animate: true,
    animateDuration: 400, // velocityJS required
});

options

  • directive (string) REQUIRED -- The name of the angular directive you wish to pass in.
  • resolve (object) -- Data to pass to the directive.
    • You can pass an unlimited amount of items into this object. You can also pass in Promises which will wait till they resolve before building the modal and passing it to the directive.
    • Data is passed to the directive by adding each item to the scope. You must declare them in the scope: scope: { data: '=' }.
  • size (string) DEFAULT: 'medium' -- The width of the modal. ( small | medium | large )
  • display (string) DEFAULT: 'notification' -- The type of modal to display ( notification | component ).
    • notification: Used for things like confirmations and warnings.
    • component: Used for larger components that do heavy lifting (eg pipeline component).
  • closeBackdrop (boolean) DEFAULT: true -- Whether clicking the backdrop closes the modal or not.
  • closeEscape (boolean) DEFAULT: true -- Whether pressing escape on the keyboard closes the modal or not.
  • animate (boolean) DEFAULT: true -- Whether to use VelocityJS for animation (Always use animation - it looks ugly with out it).
  • animateDuration (integer) DEFAULT: 400 -- The duration of the animations

.submit() & .cancel()

Pass data back through the Promise.resolve() and Promise.reject() callbacks.

app.directive('directiveName', function() {
  return {
    restrict: 'E',     // "Element" is required
    scope: {
      data: '=',
      coolStuff: '='
    },
    controller: function($scope) {
      console.log($scope.data, $scope.coolStuff);  // you're data is available
      
      // submit
      $scope.submit = function() {
        tsModalService.submit({data: 'data'});
      };
      
      // cancel
      $scope.cancel = function() {
        tsModalService.cancel({data: 'data'});
      };
    }
  };
});

// modal promise resolves or rejects
tsModalService.open({...}).then(
    function submitCb(data) {}, // submit
    function cancelCb(data) {} // cancel
);

Modal Helpers

It became necessary, due to rendering issues with other components, to create modal helpers that you can use to get helpful hints about the modal status and condition. Although only one helper is available now, more can be added as needs arise.

modalReady Boolean

An attribute added to the scope that tells you when the modal is built and the animation is complete.

app.directive('directiveName', function() {
  return {
    restrict: 'E',
    scope: {
      modalReady: '='      // attribute ready to be added into scope
    },
    controller: function($scope) {
      $scope.$watch('modalReady', function(newVal) {
        if (newVal) {
          console.log('modal is built and animation is complete');
        }
      });
    }
  }
});

Dev Setup

  • clone repo
  • install global dependencies npm install webpack webpack-dev-server
  • install local dependencies (in repo root) npm install

Run Dev Environment

  • npm start
  • go to localhost:8080 in your browser