/**
 * Initialize a client-side view from the server. The main idea here is to
 * give server-side views a way to integrate with client-side views.
 *
 * The idea is that a client-side view will have an accompanying
 * thin server-side component. The server-side component renders a placeholder
 * element in the document, and then it will invoke this  behavior to initialize
 * the view into the placeholder.
 *
 * Because server-side views may be nested, we need to add complexity to
 * handle nesting properly.
 *
 * Assuming a server-side view design that looks like hierarchical views,
 * we have to handle structures like
 *
 * <server:component>
 *   <client:component id="1">
 *     <server:component>
 *       <client:component id="2">
 *       </client:component>
 *     </server:component>
 *   </client:component>
 * </server:component>
 *
 * This leads to a problem: Client component 1 needs to initialize the behavior
 * with its children, which includes client component 2. So client component
 * 2 must be rendered first. When client component 2 is rendered, it will also
 * initialize a copy of this behavior. If behaviors are run in the order they
 * are initialized, the child component will run before the parent, and its
 * placeholder won't exist yet.
 *
 * To solve this problem, placeholder behaviors are initialized with the token
 * of a containing view that must be rendered first (if any) and a token
 * representing it for its own children to depend on. This means the server code
 * is free to call initBehavior in any order.
 *
 * In Phabricator, AphrontJavelinView demonstrates how to handle this correctly.
 *
 * config: {
 *   id: Node id to replace.
 *   view: class of view, without the 'JX.' prefix.
 *   params: view parameters
 *   children: messy and loud, cute when drunk
 *   trigger_id: id of containing view that must be rendered first
 * }
 *
 * @provides javelin-behavior-view-placeholder
 * @requires javelin-behavior
 *           javelin-dom
 *           javelin-view-renderer
 *           javelin-install
 */



JX.behavior('view-placeholder', function(config, statics) {
  JX.ViewPlaceholder.register(config.trigger_id, config.id, function() {
    var replace = JX.$(config.id);

    var children = config.children;
    if (typeof children === "string") {
      children = JX.$H(children);
    }

    var view = new JX[config.view](config.params, children);
    var rendered = JX.ViewRenderer.render(view);

    JX.DOM.replace(replace, rendered);
  });
});

JX.install('ViewPlaceholder', {
  statics: {
    register: function(wait_on_token, token, cb) {
      var ready_q = [];
      var waiting;

      if (!wait_on_token || wait_on_token in JX.ViewPlaceholder.ready) {
        ready_q.push({token: token, cb: cb});
      } else {
        waiting = JX.ViewPlaceholder.waiting;
        waiting[wait_on_token] = waiting[wait_on_token] || [];
        waiting[wait_on_token].push({token: token, cb: cb});
      }

      while(ready_q.length) {
        var ready = ready_q.shift();

        waiting = JX.ViewPlaceholder.waiting[ready.token];
        if (waiting) {
          for (var ii = 0; ii < waiting.length; ii++) {
            ready_q.push(waiting[ii]);
          }
          delete JX.ViewPlaceholder.waiting[ready.token];
        }
        ready.cb();

        JX.ViewPlaceholder.ready[token] = true;
      }

    },
    ready: {},
    waiting: {}
  }
});