The default Calendar/CalendarGroup controls do not provide positioning or dragdrop support natively, as the Container family's Overlay control and its subclasses do.
This example demonstrates how the Calendar control can be wrapped in a Container (a Dialog control in this case) which allows you to leverage Container family features to position the Calendar relative to a context element. It also show how you can setup click listeners on the document, to hide the dialog when the user clicks somewhere outside of the dialog.
In this example, we'll see how we can leverage functionality present in the Container family's Dialog widget, to provide a more advanced popup Calendar, than the basic popup features which Calendar supports inherently.
context
configuration property to position the Calendar relative to a calendar icon button.buttons
property to provide a "Reset" button to set the calendar page to the selected date, and a more prominent "Close" button to hide the popup Calendar.draggable
property to false
for this example, however if the application required it, we could set draggable
to true
to get a draggable popup CalendarThe code we use to construct the Dialog which will hold the Calendar is shown below:
1 | function resetHandler() { |
2 | // Reset the current calendar page to the select date, or |
3 | // to today if nothing is selected. |
4 | var selDates = calendar.getSelectedDates(); |
5 | var resetDate; |
6 | |
7 | if (selDates.length > 0) { |
8 | resetDate = selDates[0]; |
9 | } else { |
10 | resetDate = calendar.today; |
11 | } |
12 | |
13 | calendar.cfg.setProperty("pagedate", resetDate); |
14 | calendar.render(); |
15 | } |
16 | |
17 | function closeHandler() { |
18 | dialog.hide(); |
19 | } |
20 | |
21 | dialog = new YAHOO.widget.Dialog("container", { |
22 | visible:false, |
23 | context:["show", "tl", "bl"], |
24 | buttons:[ {text:"Reset", handler: resetHandler, isDefault:true}, {text:"Close", handler: closeHandler}], |
25 | draggable:false, |
26 | close:true |
27 | }); |
view plain | print | ? |
We don't set a width on the Dialog, so that it will shrink-wrap the Calendar, which does not have a specific width defined (the width of the calendar is controlled by the width of its contents).
The code to construct the Calendar is pretty straight forward, and is shown below:
1 | calendar = new YAHOO.widget.Calendar("cal", { |
2 | iframe:false, // Turn iframe off, since container has iframe support. |
3 | hide_blank_weeks:true // Enable, to demonstrate how we handle changing height, using changeContent |
4 | }); |
5 | calendar.render(); |
view plain | print | ? |
For this example, both the Dialog and the Calendar are constructed lazily. That is, we don't construct them until the "calendar" button is clicked on for the first time.
When creating a Calendar which is to be placed inside a Container control there are a few areas which require special attention.
Whenever the contents of the Calendar are changed we fire Dialog's changeContent
event so any of Dialog's rendered elements which need to be kept in sync are redrawn (such as the size of the shadow underlay for IE6/Safari2). We could optimize this, by checking for an actual change in dimensions before firing the changeContent
event, using Calendar's beforeRenderEvent
but the simpler approach is taken for the purposes of the example.
The Calendar has hide_blank_weeks
set to true
to illustrate the fact that the shadow is resized when the height of the Calendar changes.
1 | calendar.renderEvent.subscribe(function() { |
2 | // Tell Dialog it's contents have changed. |
3 | dialog.fireEvent("changeContent"); |
4 | }); |
view plain | print | ? |
Since we're using the yui-overlay-hidden
class to hide all tables in IE, while the container is hidden (see the discussion on
border-collapse:collapse, below), we also fire the changeContent
event for IE when showing the Dialog, to let it know that
the tables have been re-shown, which may impact the height of the Dialog:
1 | dialog.showEvent.subscribe(function() { |
2 | if (YAHOO.env.ua.ie) { |
3 | // Since we're hiding the table using yui-overlay-hidden, we |
4 | // want to let the dialog know that the content size has changed, |
5 | // when shown |
6 | dialog.fireEvent("changeContent"); |
7 | } |
8 | }); |
view plain | print | ? |
NOTE: Normally if you were to change the contents of the Dialog's header, body or footer elements (e.g. using dialog.setBody(...)
), changeContent
would be fired automatically, but in this case, we're changing the contents of the Calendar, and not the body element directly so we need to inform the Dialog that something inside the body changed.
Since dialog has a form element, we'll use it to clear Calendar's float:left and allow the Dialog body element to wrap the Calendar.
1 | /* Clear calendar's float for browsers which support :after */ |
2 | #container .bd form {clear:left} |
view plain | print | ? |
For IE6/7, we want to avoid activating hasLayout
(e.g. zoom:1;
) to clear the float for IE. If hasLayout
was activated, the Dialog would not shrink-wrap the Calendar correctly in IE6/7 and we'd need to define a specific width for the Dialog, hence we use an element to clear the left float.
Calendar's iframe
property is set to false
since the Dialog already provides iframe shim support and we want to avoid duplicating shims for performance reasons.
border-collapse:collapse
Issue
IE6 and IE7 have a known issue related to collapsed table borders remaining visible even though the table's containing element has its visibility
set to hidden
(See Container known issues).
Since the Sam skin Calendar uses border-collapse:collapse
and the Dialog is hidden using visibility:hidden
, we need to use the workaround mentioned in the known issues section above, to make sure Calendar's table borders get hidden when the Dialog is hidden.
We use the yui-overlay-hidden
class to define a rule which hides all tables, using display:none
, when the container is hidden:
1 | /* Prevent border-collapse:collapse from bleeding through in IE6, IE7 */ |
2 | #container_c.yui-overlay-hidden table { |
3 | *display:none; |
4 | } |
view plain | print | ? |
The example also shows how you can hide the dialog, when the user clicks on an area of the document outside of the dialog. A click
listener is set up on the document, so it receives notification whenever the user clicks on the document. Inside the listener, we try to determine if the element the user clicked on (the event target) is inside the dialog's containing element (dialog.element
) or the button which is used to display the dialog. It if is not, the dialog is hidden:
1 | // Hide Calendar if we click anywhere in the document other than the calendar |
2 | Event.on(document, "click", function(e) { |
3 | |
4 | var el = Event.getTarget(e); |
5 | var dialogEl = dialog.element; |
6 | |
7 | if (el != dialogEl && !Dom.isAncestor(dialogEl, el) && el != showBtn && !Dom.isAncestor(showBtn, el)) { |
8 | dialog.hide(); |
9 | } |
10 | }); |
view plain | print | ? |
As a side note, this example also shows how you can use the simpler version of the Calendar constructor, providing only the container id (available as of 2.4.0) and also how you can use Calendar's locale properties to create long date strings from a JavaScript Date object.
You can load the necessary JavaScript and CSS for this example from Yahoo's servers. Click here to load the YUI Dependency Configurator with all of this example's dependencies preconfigured.
Copyright © 2009 Yahoo! Inc. All rights reserved.
Privacy Policy - Terms of Service - Copyright Policy - Job Openings