What's the right way to enter and exit modal states with Ember router v2?
Asked Answered
M

2

7

I can't figure out the correct way to handle modal states/views with the new Ember router. More generally, how do you handle states that you can enter and exit without affecting the "main" state (the URL)?

For example, a "New Message" button that is always available regardless of the current leaf state. Clicking "New Message" should open the new message modal over the current view, without affecting the URL.

Currently, I'm using an approach like this:

Routes:

App.Router.map(function() {
   this.route('inbox');
   this.route('archive');
});

App.IndexRoute = Em.Route.extend({
  ...
  events: {
    newMessage: function() {
      this.render('new_message', { into: 'application', outlet: 'modalView' });
    },

    // Clicking 'Save' or 'Cancel' in the new message modal triggers this event to remove the view:
    hideModal: function() {
      // BAD - using private API
      this.router._lookupActiveView('application').disconnectOutlet('modalView');
    }
  }
});

App.InboxRoute = Em.Route.extend({
   ...
   renderTemplate: function(controller, model) {
     // BAD - need to specify the application template, instead of using default implementation
     this.render('inbox', { into: 'application' });
   }
});

App.ArchiveRoute = ... // basically the same as InboxRoute

application.handlebars:

<button {{action newMessage}}>New Message</button>
{{outlet}}
{{outlet modalView}}

I've obviously left out some code for brevity.

This approach 'works' but has the two problems identified above:

  1. I'm using a private API to remove the modal view in the hideModal event handler.
  2. I need to specify the application template in all of my subroutes, because if I don't, the default implementation of renderTemplate will attempt to render into the modal's template instead of into application if you open the modal, close it, and then navigate between the inbox and archive states (because the modal's template has become the lastRenderedTemplate for the IndexRoute).

Obviously, neither of these problems are dealbreakers but it would be nice to know if there is a better approach that I'm missing or if this is just a gap in the current router API.

Marshland answered 14/1, 2013 at 22:29 Comment(0)
L
4

We do kind of the same thing but without accessing the private API. I don't know if our solution is a best practice, but it works.

In the events of our RootRoute I have an event (same as your newMessage), where we create the view we need to render, and then append it.

events: {
    showNewSomething: function(){
        var newSomethingView = app.NewSomethingView.create({
            controller: this.controllerFor('newSomething')
        });
        newSomethingView.append();
    }
}

This appends the modal view into our app. On cancel or save in the newSomethingView we call this.remove() to destroy the view and removing it from the app again.

Again, this doesn't feel like a best practice, but it works. Feel free to comment on this if someone have a better solution.

Libertylibia answered 15/1, 2013 at 16:51 Comment(1)
Thanks - that makes sense to me and avoids some of the state issues in my solution.Marshland
C
0

Don't know if you are using the Bootstrap Modal script or which one, but if you are, this question has a proposed solution. Haven't figured out all the pieces myself yet, but is looking for a similar type of solution myself to be able to use Colorbox in an "Ember best practices"-compliant way.

Confiscable answered 8/4, 2013 at 18:19 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.