define([
"./ViewBase",
"dijit/_TemplatedMixin",
"./_VerticalScrollBarBase",
"dojo/text!./templates/SimpleColumnView.html",
"dojo/_base/declare",
"dojo/_base/event",
"dojo/_base/lang",
"dojo/_base/array",
"dojo/_base/sniff",
"dojo/_base/fx",
"dojo/_base/html",
"dojo/on",
"dojo/dom",
"dojo/dom-class",
"dojo/dom-style",
"dojo/dom-geometry",
"dojo/dom-construct",
"dojo/mouse",
"dojo/query",
"dojox/html/metrics"],
function(
ViewBase,
_TemplatedMixin,
_VerticalScrollBarBase,
template,
declare,
event,
lang,
arr,
has,
fx,
html,
on,
dom,
domClass,
domStyle,
domGeometry,
domConstruct,
mouse,
query,
metrics){
/*=====
var __ColumnClickEventArgs = {
// summary:
// A column click event.
// index: Integer
// The column index.
// date: Date
// The date displayed by the column.
// triggerEvent: Event
// The origin event.
};
=====*/
return declare("dojox.calendar.SimpleColumnView", [ViewBase, _TemplatedMixin], {
// summary:
// The simple column view is displaying a day per column. Each cell of a column is a time slot.
baseClass: "dojoxCalendarSimpleColumnView",
templateString: template,
// viewKind: String
// Type of the view. Used by the calendar widget to determine how to configure the view.
// This view kind is "columns".
viewKind: "columns",
// scroll container is the focusable item to enable scrolling using up and down arrows
_setTabIndexAttr: "domNode",
// renderData: Object
// The render data is the object that contains all the properties needed to render the component.
renderData: null,
// startDate: Date
// The start date of the time interval displayed.
// If not set at initialization time, will be set to current day.
startDate: null,
// columnCount: Integer
// The number of column to display (from the startDate).
columnCount: 7,
// minHours: Integer
// The minimum hour to be displayed. It must be in the [0,23] interval and must be lower than the maxHours.
minHours: 8,
// maxHours: Integer
// The maximum hour to be displayed. It must be in the [1,24] interval and must be greater than the minHours.
maxHours: 18,
// hourSize: Integer
// The desired size in pixels of an hour on the screen.
// Note that the effective size may be different as the time slot size must be an integer.
hourSize: 100,
// timeSlotDuration: Integer
// Duration of the time slot in minutes. Must be a divisor of 60.
timeSlotDuration: 15,
// rowHeaderGridSlotDuration: Integer
// Duration of the time slot in minutes in the row header. Must be a divisor of 60 and a multiple/divisor of timeSlotDuration.
rowHeaderGridSlotDuration: 60,
// rowHeaderLabelSlotDuration: Integer
// Duration of the time slot in minutes in the row header labels. Must be a divisor of 60 and a multiple/divisor of timeSlotDuration.
rowHeaderLabelSlotDuration: 60,
// rowHeaderLabelOffset: Integer
// Offset of the row label from the top of the row header cell in pixels.
rowHeaderLabelOffset: 2,
// rowHeaderFirstLabelOffset: Integer
// Offset of the first row label from the top of the first row header cell in pixels.
rowHeaderFirstLabelOffset: 2,
// verticalRenderer: Class
// The class use to create vertical renderers.
verticalRenderer: null,
// percentOverlap: Integer
// The percentage of the renderer width used to superimpose one item renderer on another
// when two events are overlapping.
percentOverlap: 70,
// horizontalGap: Integer
// The number of pixels between two item renderers that are overlapping each other if the percentOverlap property is 0.
horizontalGap: 4,
_columnHeaderHandlers: null,
constructor: function(){
this.invalidatingProperties = ["columnCount", "startDate", "minHours", "maxHours", "hourSize", "verticalRenderer",
"rowHeaderTimePattern", "columnHeaderDatePattern", "timeSlotDuration", "rowHeaderGridSlotDuration", "rowHeaderLabelSlotDuration",
"rowHeaderLabelOffset", "rowHeaderFirstLabelOffset","percentOverlap", "horizontalGap", "scrollBarRTLPosition","itemToRendererKindFunc",
"layoutPriorityFunction", "formatItemTimeFunc", "textDir", "items"];
this._columnHeaderHandlers = [];
},
destroy: function(preserveDom){
this._cleanupColumnHeader();
if(this.scrollBar){
this.scrollBar.destroy(preserveDom);
}
this.inherited(arguments);
},
_scrollBar_onScroll: function(value){
this._setScrollPosition(value);
},
buildRendering: function(){
this.inherited(arguments);
if(this.vScrollBar){
this.scrollBar = new _VerticalScrollBarBase(
{content: this.vScrollBarContent},
this.vScrollBar);
this.scrollBar.on("scroll", lang.hitch(this, this._scrollBar_onScroll));
this._viewHandles.push(
on(this.scrollContainer, mouse.wheel,
dojo.hitch(this, this._mouseWheelScrollHander)));
}
},
postscript: function(){
this.inherited(arguments);
this._initialized = true;
if(!this.invalidRendering){
this.refreshRendering();
}
},
_setVerticalRendererAttr: function(value){
this._destroyRenderersByKind("vertical");
this._set("verticalRenderer", value);
},
_createRenderData: function(){
var renderData = {};
renderData.minHours = this.get("minHours");
renderData.maxHours = this.get("maxHours");
renderData.hourSize = this.get("hourSize");
renderData.hourCount = renderData.maxHours - renderData.minHours;
renderData.slotDuration = this.get("timeSlotDuration"); // must be consistent with previous statement
renderData.rowHeaderGridSlotDuration = this.get("rowHeaderGridSlotDuration");
renderData.slotSize = Math.ceil(renderData.hourSize / (60 / renderData.slotDuration));
renderData.hourSize = renderData.slotSize * (60 / renderData.slotDuration);
renderData.sheetHeight = renderData.hourSize * renderData.hourCount;
renderData.scrollbarWidth = metrics.getScrollbar().w + 1;
renderData.dateLocaleModule = this.dateLocaleModule;
renderData.dateClassObj = this.dateClassObj;
renderData.dateModule = this.dateModule; // arithmetics on Dates
renderData.dates = [];
renderData.columnCount = this.get("columnCount");
var d = this.get("startDate");
if (d == null){
d = new renderData.dateClassObj();
}
d = this.floorToDay(d, false, renderData);
this.startDate = d;
for(var col = 0; col < renderData.columnCount ; col++){
renderData.dates.push(d);
d = renderData.dateModule.add(d, "day", 1);
d = this.floorToDay(d, false, renderData);
}
renderData.startTime = new renderData.dateClassObj(renderData.dates[0]);
renderData.startTime.setHours(renderData.minHours);
renderData.endTime = new renderData.dateClassObj(renderData.dates[renderData.columnCount-1]);
renderData.endTime.setHours(renderData.maxHours);
if(this.displayedItemsInvalidated && !this._isEditing){
// while editing in no live layout we must not to recompute items (duplicate renderers)
this._computeVisibleItems(renderData);
}else if (this.renderData){
renderData.items = this.renderData.items;
}
return renderData;
},
_validateProperties: function() {
this.inherited(arguments);
var v = this.minHours;
if(v < 0 || v>23 || isNaN(v)){
this.minHours = 0;
}
v = this.maxHours;
if (v < 1 || v>24 || isNaN(v)){
this.minHours = 24;
}
if(this.minHours > this.maxHours){
var t = this.maxHours;
this.maxHours = this.minHours;
this.minHours = t;
}
if (this.maxHours - this.minHours < 1){
this.minHours = 0;
this.maxHours = 24;
}
if (this.columnCount<1 || isNaN(this.columnCount)){
this.columnCount = 1;
}
v = this.percentOverlap;
if(v < 0 ||v > 100 || isNaN(v)){
this.percentOverlap = 70;
}
if(this.hourSize<5 || isNaN(this.hourSize)){
this.hourSize = 10;
}
v = this.timeSlotDuration;
if (v < 1 || v > 60 || isNaN(v)) {
this.timeSlotDuration = 15;
}
},
_setStartDateAttr: function(value){
this.displayedItemsInvalidated = true;
this._set("startDate", value);
},
_setColumnCountAttr: function(value){
this.displayedItemsInvalidated = true;
this._set("columnCount", value);
},
__fixEvt:function(e){
// tags:
// private
e.sheet = "primary";
e.source = this;
return e;
},
//////////////////////////////////////////
//
// Formatting functions
//
//////////////////////////////////////////
_formatRowHeaderLabel: function(/*Date*/d){
// summary:
// Computes the row header label for the specified time of day.
// By default a formatter is used, optionally the rowHeaderTimePattern
property can be used to set a custom time pattern to the formatter.
// d: Date
// The date to format
// tags:
// protected
return this.renderData.dateLocaleModule.format(d, {
selector: "time",
timePattern: this.rowHeaderTimePattern
});
},
_formatColumnHeaderLabel: function(/*Date*/d){
// summary:
// Computes the column header label for the specified date.
// By default a formatter is used, optionally the columnHeaderDatePattern
property can be used to set a custom date pattern to the formatter.
// d: Date
// The date to format
// tags:
// protected
return this.renderData.dateLocaleModule.format(d, {
selector: "date",
datePattern: this.columnHeaderDatePattern,
formatLength: "medium"
});
},
//////////////////////////////////////////
//
// Time of day management
//
//////////////////////////////////////////
// startTimeOfDay: Object
// The scroll position of the view. The value is an object made of "hours" and "minutes" properties.
startTimeOfDay: null,
// scrollBarRTLPosition: String
// Position of the scroll bar in right-to-left display.
// Valid values are "left" and "right", default value is "left".
scrollBarRTLPosition: "left",
_getStartTimeOfDay: function(){
// summary:
// Returns the visible first time of day.
// tags:
// protected
// returns: Object
var v = (this.get("maxHours") - this.get("minHours")) *
this._getScrollPosition() / this.renderData.sheetHeight;
return {
hours: this.renderData.minHours + Math.floor(v),
minutes: (v - Math.floor(v)) * 60
};
},
_getEndTimeOfDay: function(){
// summary:
// Returns the visible last time of day.
// tags:
// protected
// returns: Integer[]
var v = (this.get("maxHours") - this.get("minHours")) *
(this._getScrollPosition() + this.scrollContainer.offsetHeight) / this.renderData.sheetHeight;
return {
hours: this.renderData.minHours + Math.floor(v),
minutes: (v - Math.floor(v)) * 60
};
},
// startTimeOfDay: Object
// First time (hour/minute) of day displayed, if reachable.
// An object containing "hours" and "minutes" properties.
startTimeOfDay: 0,
_setStartTimeOfDayAttr: function(value){
if(this.renderData){
this._setStartTimeOfDay(value.hours, value.minutes, value.duration, value.easing);
}else{
this._startTimeOfDayInvalidated = true;
}
this._set("startTimeOfDay", value);
},
_getStartTimeOfDayAttr: function(){
if(this.renderData){
return this._getStartTimeOfDay();
}else{
return this._get("startTimeOfDay");
}
},
_setStartTimeOfDay: function(hour, minutes, maxDuration, easing){
// summary:
// Scrolls the view to show the specified first time of day.
// hour: Integer
// The hour of the start time of day.
// minutes: Integer
// The minutes part of the start time of day.
// maxDuration: Integer
// The max duration of the scroll animation.
// tags:
// protected
var rd = this.renderData;
hour = hour || rd.minHours;
minutes = minutes || 0;
maxDuration = maxDuration || 0;
if (minutes < 0){
minutes = 0;
}else if (minutes > 59){
minutes = 59;
}
if (hour < 0){
hour = 0;
}else if (hour > 24){
hour = 24;
}
var timeInMinutes = hour * 60 + minutes;
var minH = rd.minHours*60;
var maxH = rd.maxHours*60;
if (timeInMinutes < minH){
timeInMinutes = minH;
}else if(timeInMinutes > maxH){
timeInMinutes = maxH;
}
var pos = (timeInMinutes - minH) * rd.sheetHeight / (maxH - minH);
pos = Math.min(rd.sheetHeight - this.scrollContainer.offsetHeight, pos);
this._scrollToPosition(pos, maxDuration, easing);
},
_scrollToPosition: function(position, maxDuration, easing){
// summary:
// Scrolls the view to show the specified first time of day.
// position: Integer
// The position in pixels.
// maxDuration: Integer
// The max duration of the scroll animation.
// tags:
// protected
if (maxDuration) {
if(this._scrollAnimation){
this._scrollAnimation.stop();
}
var scrollPos = this._getScrollPosition();
var duration = Math.abs(((position - scrollPos) * maxDuration) / this.renderData.sheetHeight);
this._scrollAnimation = new fx.Animation({
curve: [scrollPos, position],
duration: duration,
easing: easing,
onAnimate: lang.hitch(this, function(position) {
this._setScrollImpl(position);
})
});
this._scrollAnimation.play();
}else{
this._setScrollImpl(position);
}
},
_setScrollImpl: function(v){
this._setScrollPosition(v);
if(this.scrollBar){
this.scrollBar.set("value", v);
}
},
ensureVisibility: function(start, end, visibilityTarget, margin, duration){
// summary:
// Scrolls the view if the [start, end] time range is not visible or only partially visible.
// start: Date
// Start time of the range of interest.
// end: Date
// End time of the range of interest.
// margin: Integer
// Margin in minutes around the time range.
// visibilityTarget: String
// The end(s) of the time range to make visible.
// Valid values are: "start", "end", "both".
// duration: Number
// Optional, the maximum duration of the scroll animation.
margin = margin == undefined ? this.renderData.slotDuration : margin;
if(this.scrollable && this.autoScroll){
var s = start.getHours() * 60 + start.getMinutes() - margin;
var e = end.getHours() * 60 + end.getMinutes() + margin;
var vs = this._getStartTimeOfDay();
var ve = this._getEndTimeOfDay();
var viewStart = vs.hours * 60 + vs.minutes;
var viewEnd = ve.hours * 60 + ve.minutes;
var visible = false;
var target = null;
switch(visibilityTarget){
case "start":
visible = s >= viewStart && s <= viewEnd;
target = s ;
break;
case "end":
visible = e >= viewStart && e <= viewEnd;
target = e - (viewEnd - viewStart);
break;
case "both":
visible = s >= viewStart && e <= viewEnd;
target = s;
break;
}
if(!visible){
this._setStartTimeOfDay(Math.floor(target/60), target%60, duration);
}
}
},
scrollView: function(dir){
// summary:
// Scrolls the view to the specified direction of one time slot duration.
// dir: Integer
// Direction of the scroll. Valid values are -1 and 1.
//
var t = this._getStartTimeOfDay();
t = t.hours*60 + t.minutes + (dir * this.timeSlotDuration);
this._setStartTimeOfDay(Math.floor(t/60), t%60);
},
_mouseWheelScrollHander: function(e){
// summary:
// Mouse wheel handler.
// tags:
// protected
this.scrollView(e.wheelDelta > 0 ? -1 : 1);
},
//////////////////////////////////////////
//
// HTML structure management
//
//////////////////////////////////////////
refreshRendering: function(){
if(!this._initialized){
return;
}
this._validateProperties();
var oldRd = this.renderData;
var rd = this._createRenderData();
this.renderData = rd;
this._createRendering(rd, oldRd);
this._layoutRenderers(rd);
},
_createRendering: function(/*Object*/renderData, /*Object*/oldRenderData){
// tags:
// private
domStyle.set(this.sheetContainer, "height", renderData.sheetHeight + "px");
// padding for the scroll bar.
this._configureScrollBar(renderData);
this._buildColumnHeader(renderData, oldRenderData);
this._buildRowHeader(renderData, oldRenderData);
this._buildGrid(renderData, oldRenderData);
this._buildItemContainer(renderData, oldRenderData);
this._commitProperties(renderData);
},
_commitProperties: function(renderData){
if(this._startTimeOfDayInvalidated){
this._startTimeOfDayInvalidated = false;
var v = this.startTimeOfDay;
if(v != null){
this._setStartTimeOfDay(v.hours, v.minutes == undefined ? 0 : v.minutes); // initial position, no animation
}
}
},
_configureScrollBar: function(renderData){
// summary:
// Sets the scroll bar size and position.
// renderData: Object
// The render data.
// tags:
// protected
if(has("ie") && this.scrollBar){
domStyle.set(this.scrollBar.domNode, "width", (renderData.scrollbarWidth + 1) + "px");
}
var atRight = this.isLeftToRight() ? true : this.scrollBarRTLPosition == "right";
var rPos = atRight ? "right" : "left";
var lPos = atRight? "left" : "right";
if(this.scrollBar){
this.scrollBar.set("maximum", renderData.sheetHeight);
domStyle.set(this.scrollBar.domNode, rPos, 0);
domStyle.set(this.scrollBar.domNode, atRight? "left" : "right", "auto");
}
domStyle.set(this.scrollContainer, rPos, renderData.scrollbarWidth + "px");
domStyle.set(this.scrollContainer, lPos, "0");
domStyle.set(this.header, rPos, renderData.scrollbarWidth + "px");
domStyle.set(this.header, lPos, "0");
if(this.buttonContainer && this.owner != null && this.owner.currentView == this){
domStyle.set(this.buttonContainer, rPos, renderData.scrollbarWidth + "px");
domStyle.set(this.buttonContainer, lPos, "0");
}
},
_columnHeaderClick: function(e){
// tags:
// private
event.stop(e);
var index = query("td", this.columnHeaderTable).indexOf(e.currentTarget);
this._onColumnHeaderClick({
index: index,
date: this.renderData.dates[index],
triggerEvent: e
});
},
_buildColumnHeader: function(renderData, oldRenderData){
// summary:
// Creates incrementally the HTML structure of the column header and configures its content.
//
// renderData:
// The render data to display.
//
// oldRenderData:
// The previously render data displayed, if any.
// tags:
// private
var table = this.columnHeaderTable;
if (!table){
return;
}
var count = renderData.columnCount - (oldRenderData ? oldRenderData.columnCount : 0);
if(has("ie") == 8){
// workaround Internet Explorer 8 bug.
// if on the table, width: 100% and table-layout: fixed are set
// and columns are removed, width of remaining columns is not
// recomputed: must rebuild all.
if(this._colTableSave == null){
this._colTableSave = lang.clone(table);
}else if(count < 0){
this._cleanupColumnHeader();
this.columnHeader.removeChild(table);
domConstruct.destroy(table);
table = lang.clone(this._colTableSave);
this.columnHeaderTable = table;
this.columnHeader.appendChild(table);
count = renderData.columnCount;
}
} // else incremental dom add/remove for real browsers.
var tbodies = query("tbody", table);
var trs = query("tr", table);
var tbody, tr, td;
if (tbodies.length == 1){
tbody = tbodies[0];
}else{
tbody = html.create("tbody", null, table);
}
if (trs.length == 1){
tr = trs[0];
}else{
tr = domConstruct.create("tr", null, tbody);
}
// Build HTML structure (incremental)
if(count > 0){ // creation
for(var i=0; i < count; i++){
td = domConstruct.create("td", null, tr);
var h = [];
h.push(on(td, "click", lang.hitch(this, this._columnHeaderClick)));
if(has("touch")){
h.push(on(td, "touchstart", function(e){
event.stop(e);
domClass.add(e.currentTarget, "Active");
}));
h.push(on(td, "touchend", function(e){
event.stop(e);
domClass.remove(e.currentTarget, "Active");
}));
}else{
h.push(on(td, "mousedown", function(e){
event.stop(e);
domClass.add(e.currentTarget, "Active");
}));
h.push(on(td, "mouseup", function(e){
event.stop(e);
domClass.remove(e.currentTarget, "Active");
}));
h.push(on(td, "mouseover", function(e){
event.stop(e);
domClass.add(e.currentTarget, "Hover");
}));
h.push(on(td, "mouseout", function(e){
event.stop(e);
domClass.remove(e.currentTarget, "Hover");
}));
}
this._columnHeaderHandlers.push(h);
}
}else{ // deletion
count = -count;
for(var i=0; i < count; i++){
td = tr.lastChild;
tr.removeChild(td);
domConstruct.destroy(td);
var list = this._columnHeaderHandlers.pop();
while(list.length>0){
list.pop().remove();
}
}
}
// fill & configure
query("td", table).forEach(function(td, i){
td.className = "";
if(i == 0){
domClass.add(td, "first-child");
}else if(i == this.renderData.columnCount-1){
domClass.add(td, "last-child");
}
var d = renderData.dates[i];
this._setText(td, this._formatColumnHeaderLabel(d));
this.styleColumnHeaderCell(td, d, renderData);
}, this);
if(this.yearColumnHeaderContent){
var d = renderData.dates[0];
this._setText(this.yearColumnHeaderContent, renderData.dateLocaleModule.format(d,
{selector: "date", datePattern:"yyyy"}));
}
},
_cleanupColumnHeader: function(){
while(this._columnHeaderHandlers.length > 0){
var list = this._columnHeaderHandlers.pop();
while(list.length > 0){
list.pop().remove();
}
}
},
styleColumnHeaderCell: function(node, date, renderData){
// summary:
// Styles the CSS classes to the node that displays a column header cell.
// By default this method is setting:
// - "dojoxCalendarToday" class name if the date displayed is the current date,
// - "dojoxCalendarWeekend" if the date represents a weekend,
// - the CSS class corresponding of the displayed day of week ("Sun", "Mon" and so on).
// node: Node
// The DOM node that displays the column in the grid.
// date: Date
// The date displayed by this column
// renderData: Object
// The render data.
// tags:
// protected
domClass.add(node, this._cssDays[date.getDay()]);
if(this.isToday(date)){
domClass.add(node, "dojoxCalendarToday");
} else if(this.isWeekEnd(date)){
domClass.add(node, "dojoxCalendarWeekend");
}
},
_addMinutesClasses: function(node, minutes){
switch(minutes){
case 0:
domClass.add(node, "hour");
break;
case 30:
domClass.add(node, "halfhour");
break;
case 15:
case 45:
domClass.add(node, "quarterhour");
break;
}
},
_buildRowHeader: function(renderData, oldRenderData){
// summary:
// Creates incrementally the HTML structure of the row header and configures its content.
//
// renderData:
// The render data to display.
//
// oldRenderData:
// The previously render data displayed, if any.
// tags:
// private
var rowHeaderTable = this.rowHeaderTable;
if (!rowHeaderTable){
return;
}
if(this._rowHeaderLabelContainer == null){
this._rowHeaderLabelContainer = domConstruct.create("div", {"class": "dojoxCalendarRowHeaderLabelContainer"}, this.rowHeader);
}
domStyle.set(rowHeaderTable, "height", renderData.sheetHeight + "px");
var tbodies = query("tbody", rowHeaderTable);
var tbody, tr, td;
if (tbodies.length == 1){
tbody = tbodies[0];
}else{
tbody = domConstruct.create("tbody", null, rowHeaderTable);
}
var nbRows = Math.floor(60 / renderData.rowHeaderGridSlotDuration) * renderData.hourCount;
var count = nbRows -
(oldRenderData ? Math.floor(60 / oldRenderData.rowHeaderGridSlotDuration) * oldRenderData.hourCount : 0);
// Build HTML structure
if(count>0){ // creation
for(var i=0; i < count; i++){
tr = domConstruct.create("tr", null, tbody);
td = domConstruct.create("td", null, tr);
}
}else{
count = -count;
// deletion of existing nodes
for(var i=0; i < count; i++){
tbody.removeChild(tbody.lastChild);
}
}
// fill labels
var rd = this.renderData;
var size = Math.ceil(renderData.hourSize / (60 / renderData.rowHeaderGridSlotDuration));
var d = new Date(2000, 0, 1, 0, 0, 0);
query("tr", rowHeaderTable).forEach(function(tr, i){
var td = query("td", tr)[0];
td.className = "";
domStyle.set(tr, "height", (has("ie") == 7)?size-2*(60 / renderData.rowHeaderGridSlotDuration):size + "px");
this.styleRowHeaderCell(td, d.getHours(), d.getMinutes(), rd);
var m = (i * this.renderData.rowHeaderGridSlotDuration) % 60;
this._addMinutesClasses(td, m);
}, this);
var lc = this._rowHeaderLabelContainer;
count = (Math.floor(60 / this.rowHeaderLabelSlotDuration) * renderData.hourCount) - lc.childNodes.length;
var span;
if(count>0){ // creation
for(var i=0; i < count; i++){
span = domConstruct.create("span", null, lc);
domClass.add(span, "dojoxCalendarRowHeaderLabel");
}
}else{
count = -count;
// deletion of existing nodes
for(var i=0; i < count; i++){
lc.removeChild(lc.lastChild);
}
}
size = Math.ceil(renderData.hourSize / (60 / this.rowHeaderLabelSlotDuration));
query(">span", lc).forEach(function(span, i){
d.setHours(0);
d.setMinutes(renderData.minHours * 60 + (i*this.rowHeaderLabelSlotDuration));
this._configureRowHeaderLabel(span, d, i, size*i, rd);
}, this);
},
_configureRowHeaderLabel: function(node, d, index, pos, renderData){
// summary:
// Configures the label of a row header cell.
// node: DOMNode
// The DOM node that is the parent of the label.
// d:Date
// A date object that contains the hours and minutes displayed by this row header cell.
// index: Integer
// The index of this row header cell
// pos: Integer
// The computed position of the row header cell
// renderData: Object
// The render data.
this._setText(node, this._formatRowHeaderLabel(d));
domStyle.set(node, "top", (pos + (index==0?this.rowHeaderFirstLabelOffset:this.rowHeaderLabelOffset))+"px");
var m = (index * this.rowHeaderLabelSlotDuration) % 60;
domClass.remove(node, ["hour", "halfhour", "quarterhour"]);
this._addMinutesClasses(node, m);
},
styleRowHeaderCell: function(node, h, m, renderData){
// summary:
// Styles the CSS classes to the node that displays a row header cell.
// By default this method is doing nothing.
// node: Node
// The DOM node that displays the column in the grid.
// h: Integer
// The time of day displayed by this row header cell.
// renderData: Object
// The render data.
// tags:
// protected
},
_buildGrid: function (renderData, oldRenderData){
// summary:
// Creates incrementally the HTML structure of the grid and configures its content.
//
// renderData:
// The render data to display.
//
// oldRenderData:
// The previously render data displayed, if any.
// tags:
// private
var table = this.gridTable;
if (!table){
return;
}
domStyle.set(table, "height", renderData.sheetHeight + "px");
var nbRows = Math.floor(60 / renderData.slotDuration) * renderData.hourCount;
var rowDiff = nbRows -
(oldRenderData ? Math.floor(60 / oldRenderData.slotDuration) * oldRenderData.hourCount : 0);
var addRows = rowDiff > 0;
var colDiff = renderData.columnCount - (oldRenderData ? oldRenderData.columnCount : 0);
if(has("ie") == 8){
// workaround Internet Explorer 8 bug.
// if on the table, width: 100% and table-layout: fixed are set
// and columns are removed, width of remaining columns is not
// recomputed: must rebuild all.
if(this._gridTableSave == null){
this._gridTableSave = lang.clone(table);
}else if(colDiff < 0){
this.grid.removeChild(table);
domConstruct.destroy(table);
table = lang.clone(this._gridTableSave);
this.gridTable = table;
this.grid.appendChild(table);
colDiff = renderData.columnCount;
rowDiff = nbRows;
addRows = true;
}
}
var tbodies = query("tbody", table);
var tbody;
if (tbodies.length == 1){
tbody = tbodies[0];
}else{
tbody = domConstruct.create("tbody", null, table);
}
// Build time slots (lines) HTML structure (incremental)
if(addRows){ // creation
for(var i=0; i0;
colDiff = addCols ? colDiff : -colDiff;
query("tr", table).forEach(function(tr, i){
if(addCols){ // creation
var len = i >= rowIndex ? renderData.columnCount : colDiff;
for(var i=0; i0){ // creation
for(var i=0; i < count; i++){
td = domConstruct.create("td", null, tr);
domConstruct.create("div", {"className": "dojoxCalendarContainerColumn"}, td);
}
}else{ // deletion
count = -count;
for(var i=0; i < count; i++){
tr.removeChild(tr.lastChild);
}
}
query("td>div", table).forEach(function(div, i){
domStyle.set(div, {
"height": renderData.sheetHeight + "px"
});
bgCols.push(div);
}, this);
renderData.cells = bgCols;
},
///////////////////////////////////////////////////////////////
//
// Layout
//
///////////////////////////////////////////////////////////////
_overlapLayoutPass2: function(lanes){
// summary:
// Second pass of the overlap layout (optional). Compute the extent of each layout item.
// lanes:
// The array of lanes.
// tags:
// private
var i,j,lane, layoutItem;
// last lane, no extent possible
lane = lanes[lanes.length-1];
for(j = 0; j < lane.length; j++){
lane[j].extent = 1;
}
for(i=0; i 0){
this._layoutVerticalItems(renderData, index, start, end, verticalItems);
}
},
_layoutVerticalItems: function(/*Object*/renderData, /*Integer*/index, /*Date*/startTime, /*Date*/endTime, /*Object[]*/items){
// tags:
// private
if(this.verticalRenderer == null){
return;
}
var cell = renderData.cells[index];
var layoutItems = [];
// step 1 compute projected position and size
for(var i = 0; i < items.length; i++){
var item = items[i];
var overlap = this.computeRangeOverlap(renderData, item.startTime, item.endTime, startTime, endTime);
var top = this.computeProjectionOnDate(renderData, startTime, overlap[0], renderData.sheetHeight);
var bottom = this.computeProjectionOnDate(renderData, startTime, overlap[1], renderData.sheetHeight);
if (bottom > top){
var litem = lang.mixin({
start: top,
end: bottom,
range: overlap,
item: item
}, item);
layoutItems.push(litem);
}
}
// step 2: compute overlapping layout
var numLanes = this.computeOverlapping(layoutItems, this._overlapLayoutPass2).numLanes;
var hOverlap = this.percentOverlap / 100;
// step 3: create renderers and apply layout
for(i=0; i r.w){
x = r.w-1;
}
if (y < 0){
y = 0;
}else if(y > r.h){
y = r.h-1;
}
var col = Math.floor(x / (domGeometry.getMarginBox(this.itemContainer).w / this.renderData.columnCount));
var t = this.getTimeOfDay(y, this.renderData);
var date = null;
if(col < this.renderData.dates.length){
date = this.newDate(this.renderData.dates[col]);
date = this.floorToDay(date, true);
date.setHours(t.hours);
date.setMinutes(t.minutes);
}
return date;
},
///////////////////////////////////////////////////////////////
//
// Events
//
///////////////////////////////////////////////////////////////
_onGridMouseUp: function(e){
// tags:
// private
this.inherited(arguments);
if (this._gridMouseDown) {
this._gridMouseDown = false;
this._onGridClick({
date: this.getTime(e),
triggerEvent: e
});
}
},
_onGridTouchStart: function(e){
// tags:
// private
this.inherited(arguments);
var g = this._gridProps;
g.moved= false;
g.start= e.touches[0].screenY;
g.scrollTop= this._getScrollPosition();
},
_onGridTouchMove: function(e){
// tags:
// private
this.inherited(arguments);
if (e.touches.length > 1 && !this._isEditing){
event.stop(e);
return;
}
if(this._gridProps && !this._isEditing){
var touch = {x: e.touches[0].screenX, y: e.touches[0].screenY};
var p = this._edProps;
if (!p || p &&
(Math.abs(touch.x - p.start.x) > 25 ||
Math.abs(touch.y - p.start.y) > 25)) {
this._gridProps.moved = true;
var d = e.touches[0].screenY - this._gridProps.start;
var value = this._gridProps.scrollTop - d;
var max = this.itemContainer.offsetHeight - this.scrollContainer.offsetHeight;
if (value < 0){
this._gridProps.start = e.touches[0].screenY;
this._setScrollImpl(0);
this._gridProps.scrollTop = 0;
}else if(value > max){
this._gridProps.start = e.touches[0].screenY;
this._setScrollImpl(max);
this._gridProps.scrollTop = max;
}else{
this._setScrollImpl(value);
}
}
}
},
_onGridTouchEnd: function(e){
// tags:
// private
//event.stop(e);
this.inherited(arguments);
var g = this._gridProps;
if(g){
if(!this._isEditing){
if(!g.moved){
// touched on grid and on touch start editing was ongoing.
if(!g.fromItem && !g.editingOnStart){
this.selectFromEvent(e, null, null, true);
}
if(!g.fromItem){
if(this._pendingDoubleTap && this._pendingDoubleTap.grid){
this._onGridDoubleClick({
date: this.getTime(this._gridProps.event),
triggerEvent: this._gridProps.event
});
clearTimeout(this._pendingDoubleTap.timer);
delete this._pendingDoubleTap;
}else{
this._onGridClick({
date: this.getTime(this._gridProps.event),
triggerEvent: this._gridProps.event
});
this._pendingDoubleTap = {
grid: true,
timer: setTimeout(lang.hitch(this, function(){
delete this._pendingDoubleTap;
}), this.doubleTapDelay)
};
}
}
}
}
this._gridProps = null;
}
},
_onColumnHeaderClick: function(e){
// tags:
// private
this._dispatchCalendarEvt(e, "onColumnHeaderClick");
},
onColumnHeaderClick: function(e){
// summary:
// Event dispatched when a column header cell is dispatched.
// e: __ColumnClickEventArgs
// The event has the following properties
// tags:
// callback
},
getTimeOfDay: function (pos, rd) {
// summary:
// Return the time of day associated to the specified position.
// pos: Integer
// The position in pixels.
// rd: Object
// The render data.
var minH = rd.minHours*60;
var maxH = rd.maxHours*60;
var minutes = minH + (pos * (maxH - minH) / rd.sheetHeight);
return {
hours: Math.floor(minutes / 60),
minutes: Math.floor(minutes % 60)
};
},
///////////////////////////////////////////////////////////////
//
// View limits
//
///////////////////////////////////////////////////////////////
_isItemInView: function(item){
// subclassed to add some tests
var res = this.inherited(arguments);
if(res){
// test if time range is overlapping [maxHours, next day min hours]
var rd = this.renderData;
var len = rd.dateModule.difference(item.startTime, item.endTime, "millisecond");
var vLen = (24 - rd.maxHours + rd.minHours) * 3600000; // 60 * 60 * 1000, number of milliseconds in 1 minute
if(len > vLen){ // longer events are always visible
return true;
}
var sMin = item.startTime.getHours()*60 + item.startTime.getMinutes();
var eMin = item.endTime.getHours()*60 + item.endTime.getMinutes();
var sV = rd.minHours * 60;
var eV = rd.maxHours * 60;
if(sMin > 0 && sMin < sV || sMin > eV && sMin <= 1440){
return false;
}
if(eMin > 0 && eMin < sV || eMin > eV && eMin <= 1440){
return false;
}
}
return res;
},
_ensureItemInView: function(item){
var fixed;
var startTime = item.startTime;
var endTime = item.endTime;
// test if time range is overlapping [maxHours, next day min hours]
var rd = this.renderData;
var cal = rd.dateModule;
var len = Math.abs(cal.difference(item.startTime, item.endTime, "millisecond"));
var vLen = (24 - rd.maxHours + rd.minHours) * 3600000;
if(len > vLen){ // longer events are always visible
return false;
}
var sMin = startTime.getHours()*60 + startTime.getMinutes();
var eMin = endTime.getHours()*60 + endTime.getMinutes();
var sV = rd.minHours * 60;
var eV = rd.maxHours * 60;
if(sMin > 0 && sMin < sV){
this.floorToDay(item.startTime, true, rd);
item.startTime.setHours(rd.minHours);
item.endTime = cal.add(item.startTime, "millisecond", len);
fixed = true;
}else if(sMin > eV && sMin <= 1440){
// go on next visible time
this.floorToDay(item.startTime, true, rd);
item.startTime = cal.add(item.startTime, "day", 1);
// if we are going out of the view, the super() will fix it
item.startTime.setHours(rd.minHours);
item.endTime = cal.add(item.startTime, "millisecond", len);
fixed = true;
}
if(eMin > 0 && eMin < sV){
// go on previous day
this.floorToDay(item.endTime, true, rd);
item.endTime = cal.add(item.endTime, "day", -1);
item.endTime.setHours(rd.maxHours);
item.startTime = cal.add(item.endTime, "millisecond", -len);
fixed = true;
}else if(eMin > eV && eMin <= 1440){
this.floorToDay(item.endTime, true, rd);
item.endTime.setHours(rd.maxHours);
item.startTime = cal.add(item.endTime, "millisecond", -len);
fixed = true;
}
fixed = fixed || this.inherited(arguments);
return fixed;
},
_onScrollTimer_tick: function(){
// tags:
// private
this._scrollToPosition(this._getScrollPosition() + this._scrollProps.scrollStep);
},
////////////////////////////////////////////
//
// Editing
//
///////////////////////////////////////////
snapUnit: "minute",
snapSteps: 15,
minDurationUnit: "minute",
minDurationSteps: 15,
liveLayout: false,
stayInView: true,
allowStartEndSwap: true,
allowResizeLessThan24H: true
});
});