/**
 * Implementation of Drupal behavior.
 */
Drupal.behaviors.silver = function(context) {
  Drupal.silverToggle.attach(context);
};

Drupal.silverToggle = {
  /**
   * Stack of modifiers to retain between pages.
   */
  'modifiers': [],

  /**
   * Add a modifier to the stack, or remove it if it exists.
   */
  'toggleModifier': function(modifier) {
    for (var i in this.modifiers) {
      if (this.modifiers[i] == modifier) {
        delete this.modifiers[i];
        return true;
      }
    }
    this.modifiers.push(modifier);
    return true;
  },

  /**
   * Initialize and attach handlers.
   */
  'attach': function(context) {
    // If there are both main column and side column buttons,
    // only show the main column buttons if the user scrolls past
    // the ones to the side.
    if ($('div.column-main div.buttons').size()) {
      var offset = $('div.column-side div.buttons').height() + $('div.column-side div.buttons').offset().top;
      $(window).scroll(function () {
        if ($(this).scrollTop() > offset) {
          $('div.column-main div.buttons').show();
        }
        else {
          $('div.column-main div.buttons').hide();
        }
      });
    }

    var args = Drupal.silverToggle.parseHash(window.location.hash.substring(1));
    $('a.toggler:not(.silver-processed)', context).each(function() {
      var toggleable, params;

      // Parse the hash string.
      params = Drupal.silverToggle.parseHash($(this).attr('href').split('#')[1]);

      for (var key in params) {
        toggleable = $('#' + key);
        // Target toggleable exist.
        if (toggleable.size() > 0) {
          // Set init state.
          if (args[key] == 1) {
            toggleable.show();
            $(this).addClass('toggler-active');
            Drupal.silverToggle.toggleModifier(key + '=' + params[key]);
          }
          else {
            toggleable.hide();
            $(this).removeClass('toggler-active');
          }
          // Add click handler.
          $(this).click(function() {
            toggleable.toggle();
            $(this).toggleClass('toggler-active');
            Drupal.silverToggle.toggleModifier(key + '=' + params[key]);
            return false;
          });
        }
        // Target toggleable doesn't exist.
        else {
          $(this).addClass('toggler-disabled');
          $(this).click(function() { return false; });
        }
      }

      // Mark as processed.
      $(this).addClass('silver-processed');
      return false;
    });
    $('div.help-links a:not(.silver-processed)', context).each(function() {
      if ($(this).attr('href')) {
        // Rewrite this link's hash string when clicked.
        $(this).click(function() {
          var href = $(this).attr('href').split('#');
          var modifiers = Drupal.silverToggle.modifiers.join('&');
          if (modifiers) {
            href[1] = href[1] ? href[1] + '&' + modifiers : modifiers;
            $(this).attr('href', href.join('#'));
          }
        });
        // Mark as processed.
        $(this).addClass('silver-processed');
      }
    });
  },

  /**
   * Parse a hash string.
   */
  'parseHash': function(hash) {
    var parameters = {};
    var pairs = hash.split(/[&;]/);
    for(var i=0, len=pairs.length; i<len; ++i) {
      var keyValue = pairs[i].split('=');
      if (keyValue[0]) {
        var key = decodeURIComponent(keyValue[0]);
        var value = keyValue[1] || ''; //empty string if no value
        //decode individual values
        value = value.split(",");
        for(var j=0, jlen=value.length; j<jlen; j++) {
          value[j] = decodeURIComponent(value[j]);
        }
        //if there's only one value, do not return as array
        if (value.length == 1) {
          value = value[0];
        }
        parameters[key] = value;
      }
    }
    return parameters;
  }
};
