// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Copyright 2006 Google Inc. All Rights Reserved.
/**
* @fileoverview Zippy widget implementation.
*
* @see ../demos/zippy.html
*/
goog.provide('goog.ui.Zippy');
goog.provide('goog.ui.ZippyEvent');
goog.require('goog.dom');
goog.require('goog.dom.classes');
goog.require('goog.events');
goog.require('goog.events.Event');
goog.require('goog.events.EventTarget');
goog.require('goog.events.EventType');
goog.require('goog.events.KeyCodes');
/**
* Zippy widget. Expandable/collapsible container, clicking the header toggles
* the visibility of the content.
*
* @extends {goog.events.EventTarget}
* @param {Element|string} header Header element, either element reference or
* string id.
* @param {Element|string} opt_content Content element (if any), either element
* reference or string id. If skipped, the caller
* should handle the TOGGLE event in its own way.
* @param {boolean} opt_expanded Initial expanded/visibility state. Defaults to
* false.
* @constructor
*/
goog.ui.Zippy = function(header, opt_content, opt_expanded) {
goog.events.EventTarget.call(this);
/**
* Header element.
* @type {Element}
* @private
*/
this.elHeader_ = goog.dom.getElement(header);
/**
* Content element.
* @type {?Element}
* @private
*/
this.elContent_ = opt_content ? goog.dom.getElement(opt_content) : null;
/**
* Expanded state.
* @type {boolean}
* @private
*/
this.expanded_ = opt_expanded == true;
// Listen for click and keydown events on header
this.elHeader_.tabIndex = 0;
goog.events.listen(this.elHeader_, goog.events.EventType.CLICK,
this.onHeaderClick_, false, this);
goog.events.listen(this.elHeader_, goog.events.EventType.KEYDOWN,
this.onHeaderKeyDown_, false, this);
// initialize based on expanded state
this.setExpanded(this.expanded_);
};
goog.inherits(goog.ui.Zippy, goog.events.EventTarget);
/**
* Constants for event names
*
* @type {Object}
*/
goog.ui.Zippy.Events = {
TOGGLE: 'toggle'
};
/**
* Destroys widget and removes all event listeners.
*/
goog.ui.Zippy.prototype.disposeInternal = function() {
goog.events.removeAll(this.elHeader_);
goog.ui.Zippy.superClass_.disposeInternal.call(this);
};
/**
* Expands content pane.
*/
goog.ui.Zippy.prototype.expand = function() {
this.setExpanded(true);
};
/**
* Collapses content pane.
*/
goog.ui.Zippy.prototype.collapse = function() {
this.setExpanded(false);
};
/**
* Toggles expanded state.
*/
goog.ui.Zippy.prototype.toggle = function() {
this.setExpanded(!this.expanded_);
};
/**
* Sets expanded state.
*
* @param {boolean} expanded Expanded/visibility state.
*/
goog.ui.Zippy.prototype.setExpanded = function(expanded) {
if (this.elContent_) {
// Hide the element, if one is provided.
this.elContent_.style.display = expanded ? '' : 'none';
}
// Update header image, if any.
this.updateHeaderClassName_(expanded);
this.expanded_ = expanded;
// Fire toggle event
this.dispatchEvent(new goog.ui.ZippyEvent(goog.ui.Zippy.Events.TOGGLE,
this, this.expanded_));
};
/**
* @return {boolean} Whether the zippy is expanded.
*/
goog.ui.Zippy.prototype.isExpanded = function() {
return this.expanded_;
};
/**
* Updates the header element's className
*
* @param {boolean} expanded Expanded/visibility state.
* @private
*/
goog.ui.Zippy.prototype.updateHeaderClassName_ = function(expanded) {
if (expanded) {
goog.dom.classes.remove(this.elHeader_, 'goog-zippy-collapsed');
goog.dom.classes.add(this.elHeader_, 'goog-zippy-expanded');
} else {
goog.dom.classes.remove(this.elHeader_, 'goog-zippy-expanded');
goog.dom.classes.add(this.elHeader_, 'goog-zippy-collapsed');
}
};
/**
* KeyDown event handler for header element. Enter and space toggles expanded
* state.
*
* @param {goog.events.BrowserEvent} event KeyDown event.
* @private
*/
goog.ui.Zippy.prototype.onHeaderKeyDown_ = function(event) {
if (event.keyCode == goog.events.KeyCodes.ENTER ||
event.keyCode == goog.events.KeyCodes.SPACE) {
this.toggle();
// Prevent enter key from submiting form.
event.preventDefault();
event.stopPropagation();
}
};
/**
* Click event handler for header element.
*
* @param {goog.events.BrowserEvent} event Click event.
* @private
*/
goog.ui.Zippy.prototype.onHeaderClick_ = function(event) {
this.toggle();
};
/**
* Object representing a zippy toggle event.
*
* @param {string} type Event type.
* @param {goog.ui.Zippy} target Zippy widget initiating event.
* @param {boolean} expanded Expanded state.
* @extends {goog.events.Event}
* @constructor
*/
goog.ui.ZippyEvent = function(type, target, expanded) {
goog.events.Event.call(this, type, target);
/**
* The expanded state.
* @type {boolean}
*/
this.expanded = expanded;
};
goog.inherits(goog.ui.ZippyEvent, goog.events.Event);