mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-07 12:15:51 +00:00
cdf001b203
bug 278423, r=stuart. npotb
617 lines
19 KiB
JavaScript
617 lines
19 KiB
JavaScript
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (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.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is OEone Calendar Code, released October 31st, 2001.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* OEone Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s): Garth Smedley <garths@oeone.com>
|
|
* Mike Potter <mikep@oeone.com>
|
|
* Chris Charabaruk <coldacid@meldstar.com>
|
|
* Colin Phillips <colinp@oeone.com>
|
|
* ArentJan Banck <ajbanck@planet.nl>
|
|
* Curtis Jewell <csjewell@mail.freeshell.org>
|
|
* Eric Belhaire <eric.belhaire@ief.u-psud.fr>
|
|
* Mark Swaffer <swaff@fudo.org>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
/*-----------------------------------------------------------------
|
|
* U N I F I N D E R
|
|
*
|
|
* This is a hacked in interface to the unifinder. We will need to
|
|
* improve this to make it usable in general.
|
|
*/
|
|
|
|
const ToDoUnifinderTreeName = "unifinder-todo-tree";
|
|
|
|
var gTaskArray = new Array();
|
|
|
|
/**
|
|
* Observer for the calendar event data source. This keeps the unifinder
|
|
* display up to date when the calendar event data is changed
|
|
*/
|
|
|
|
var unifinderToDoDataSourceObserver =
|
|
{
|
|
onStartBatch: function () { },
|
|
onEndBatch: function () {
|
|
toDoUnifinderRefresh();
|
|
},
|
|
|
|
onLoad: function () {
|
|
toDoUnifinderRefresh();
|
|
},
|
|
|
|
onAddItem: function (aItem) {
|
|
if (aItem instanceof Components.interfaces.calITodo)
|
|
toDoUnifinderRefresh();
|
|
},
|
|
|
|
onModifyItem: function (aNewItem, aOldItem) {
|
|
if (aNewItem instanceof Components.interfaces.calITodo)
|
|
toDoUnifinderRefresh();
|
|
},
|
|
|
|
onDeleteItem: function (aItem) {
|
|
if (aItem instanceof Components.interfaces.calITodo)
|
|
toDoUnifinderRefresh();
|
|
},
|
|
|
|
onAlarm: function (aAlarmItem) { },
|
|
|
|
onError: function (aErrNo, aMessage) { }
|
|
};
|
|
|
|
/**
|
|
* Called when the calendar is loaded
|
|
*/
|
|
|
|
function prepareCalendarToDoUnifinder()
|
|
{
|
|
var calendar = getCalendar();
|
|
calendar.addObserver(unifinderToDoDataSourceObserver, calendar.ITEM_FILTER_TYPE_TODO);
|
|
toDoUnifinderRefresh();
|
|
}
|
|
|
|
/**
|
|
* Called when the calendar is unloaded
|
|
*/
|
|
|
|
function finishCalendarToDoUnifinder()
|
|
{
|
|
var calendar = createCalendar();
|
|
calendar.removeObserver(unifinderToDoDataSourceObserver);
|
|
}
|
|
|
|
/**
|
|
* Helper function to display todo datetimes in the unifinder
|
|
*/
|
|
|
|
function formatUnifinderToDoDateTime(aDateTime)
|
|
{
|
|
// datetime is from todo object, it is not a javascript date
|
|
if (aDateTime && aDateTime.isValid)
|
|
return gCalendarWindow.dateFormater.formatDateTime(aDateTime.jsDate, true);
|
|
return "";
|
|
}
|
|
|
|
/**
|
|
* Called by event observers to update the display
|
|
*/
|
|
|
|
function toDoUnifinderRefresh()
|
|
{
|
|
var hideCompleted = document.getElementById("hide-completed-checkbox").checked;
|
|
|
|
var savedThis = this;
|
|
var refreshListener = {
|
|
mTaskArray: new Array(),
|
|
|
|
onOperationComplete: function (aCalendar, aStatus, aOperationType, aId, aDateTime) {
|
|
setTimeout(function () { refreshToDoTree (refreshListener.mTaskArray); }, 0);
|
|
},
|
|
|
|
onGetResult: function (aCalendar, aStatus, aItemType, aDetail, aCount, aItems) {
|
|
for (var i = 0; i < aCount; i++) {
|
|
refreshListener.mTaskArray.push(aItems[i]);
|
|
}
|
|
}
|
|
};
|
|
|
|
var calendar = getCalendar();
|
|
var filter = 0;
|
|
if (hideCompleted)
|
|
filter |= calendar.ITEM_FILTER_COMPLETED_NO;
|
|
else
|
|
filter |= calendar.ITEM_FILTER_COMPLETED_ALL;
|
|
|
|
filter |= calendar.ITEM_FILTER_TYPE_TODO;
|
|
|
|
calendar.getItems (filter, 0, null, null, refreshListener);
|
|
}
|
|
|
|
function getToDoFromEvent( event )
|
|
{
|
|
var tree = document.getElementById( ToDoUnifinderTreeName );
|
|
|
|
var row = tree.treeBoxObject.getRowAt( event.clientX, event.clientY );
|
|
|
|
if( row != -1 && row < tree.view.rowCount )
|
|
{
|
|
return( tree.taskView.getCalendarTaskAtRow( row ) );
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function getSelectedToDo()
|
|
{
|
|
var tree = document.getElementById( ToDoUnifinderTreeName );
|
|
// .toDo object sometimes isn't available?
|
|
return( tree.taskView.getCalendarTaskAtRow(tree.currentIndex) );
|
|
|
|
}
|
|
|
|
/**
|
|
* This is attached to the onclik attribute of the to do list shown in the unifinder
|
|
*/
|
|
|
|
function modifyToDoCommand( event )
|
|
{
|
|
// we only care about button 0 (left click) events
|
|
if (event.button != 0) return;
|
|
|
|
//open the edit todo dialog box
|
|
var ThisToDo = getToDoFromEvent( event );
|
|
|
|
if( ThisToDo )
|
|
editToDo( ThisToDo );
|
|
else
|
|
newToDoCommand();
|
|
}
|
|
|
|
/**
|
|
* Set the context menu on mousedown to change it before it is opened
|
|
*/
|
|
|
|
function unifinderMouseDownToDo( event )
|
|
{
|
|
var tree = document.getElementById( ToDoUnifinderTreeName );
|
|
var treechildren = tree.getElementsByTagName( "treechildren" )[0];
|
|
|
|
var ThisToDo = getToDoFromEvent( event );
|
|
if( ThisToDo )
|
|
{
|
|
// TODO HACK notifiers should be rewritten to integrate events and todos
|
|
document.getElementById( "delete_todo_command" ).removeAttribute( "disabled" );
|
|
document.getElementById( "print_command" ).setAttribute( "disabled", "true" );
|
|
} else
|
|
{
|
|
tree.view.selection.clearSelection();
|
|
|
|
// TODO HACK notifiers should be rewritten to integrate events and todos
|
|
document.getElementById( "delete_todo_command" ).setAttribute( "disabled", "true" );
|
|
// printing tasks not supported
|
|
document.getElementById( "print_command" ).setAttribute( "disabled", "true" );
|
|
}
|
|
}
|
|
|
|
function checkboxClick(thisTodo, completed)
|
|
{
|
|
var newTodo = thisTodo.clone().QueryInterface(Components.interfaces.calITodo);
|
|
if(completed) {
|
|
newTodo.completedDate.jsDate = new Date();
|
|
newTodo.percentComplete = 100;
|
|
} else {
|
|
newTodo.completedDate.reset();
|
|
if (newTodo.percentComplete == 100)
|
|
newTodo.percentComplete = 0;
|
|
}
|
|
|
|
var calendar = getCalendar();
|
|
calendar.modifyItem (newTodo, null);
|
|
}
|
|
|
|
|
|
/*
|
|
This function return the progress state of a ToDo task :
|
|
completed, overdue, duetoday, inprogress, future
|
|
*/
|
|
function ToDoProgressAtom( aTodo )
|
|
{
|
|
var now = new Date();
|
|
var completed = (aTodo.percentComplete == 100);
|
|
|
|
if (completed)
|
|
return "completed";
|
|
|
|
if (aTodo.dueDate.isValid) {
|
|
if (aTodo.dueDate.jsDate.getTime() < now.getTime())
|
|
return "overdue";
|
|
else if (aTodo.dueDate.year == now.getFullYear() &&
|
|
aTodo.dueDate.month == now.getMonth() &&
|
|
aTodo.dueDate.day == now.getDate())
|
|
return "duetoday";
|
|
}
|
|
|
|
if (aTodo.entryDate.isValid &&
|
|
aTodo.entryDate.jsDate.getTime() < now.getTime())
|
|
return "inprogress";
|
|
|
|
return "future";
|
|
}
|
|
|
|
var toDoTreeView =
|
|
{
|
|
rowCount : gTaskArray.length,
|
|
selectedColumn : null,
|
|
sortDirection : null,
|
|
sortStartedTime : new Date().getTime(), // updated just before sort
|
|
|
|
isContainer : function(){return false;},
|
|
// By getCellProperties, the properties defined with
|
|
// treechildren:-moz-tree-cell-text in CSS are used.
|
|
// It is use here to color a particular row with a color
|
|
// given by the progress state of the ToDo task.
|
|
getCellProperties : function( row,column, props )
|
|
{
|
|
calendarToDo = gTaskArray[row];
|
|
var aserv=Components.classes["@mozilla.org/atom-service;1"].getService(Components.interfaces.nsIAtomService);
|
|
|
|
// Moz1.8 trees require column.id, moz1.7 and earlier trees use column.
|
|
if( (typeof(column)=="object" ? column.id : column) == "unifinder-todo-tree-col-priority" )
|
|
{
|
|
if(calendarToDo.priority > 0 && calendarToDo.priority < 5)
|
|
props.AppendElement(aserv.getAtom("highpriority"));
|
|
if(calendarToDo.priority > 5 && calendarToDo.priority < 10)
|
|
props.AppendElement(aserv.getAtom("lowpriority"));
|
|
}
|
|
props.AppendElement(aserv.getAtom(ToDoProgressAtom(calendarToDo)));
|
|
},
|
|
getColumnProperties : function(){return false;},
|
|
// By getRowProperties, the properties defined with
|
|
// treechildren:-moz-tree-row in CSS are used.
|
|
// It is used here to color the background of a selected
|
|
// ToDo task with a color
|
|
// given by the progress state of the ToDo task.
|
|
getRowProperties : function( row,props ){
|
|
calendarToDo = gTaskArray[row];
|
|
|
|
var aserv=Components.classes["@mozilla.org/atom-service;1"].getService(Components.interfaces.nsIAtomService);
|
|
props.AppendElement(aserv.getAtom(ToDoProgressAtom( calendarToDo )));
|
|
},
|
|
isSorted : function(){return false;},
|
|
isEditable : function(){return true;},
|
|
isSeparator : function(){return false;},
|
|
// Return the empty string in order
|
|
// to use moz-tree-image pseudoelement :
|
|
// it is mandatory to return "" and not false :-(
|
|
getImageSrc : function(){return("");},
|
|
cycleCell : function(row,col)
|
|
{
|
|
calendarToDo = gTaskArray[row];
|
|
if( !calendarToDo ) return;
|
|
|
|
// Moz1.8 trees require column.id, moz1.7 and earlier trees use column.
|
|
if( (typeof(col)=="object" ? col.id : col) == "unifinder-todo-tree-col-completed") {
|
|
if (calendarToDo.completedDate.isValid)
|
|
checkboxClick( calendarToDo, false ) ;
|
|
else
|
|
checkboxClick( calendarToDo, true ) ;
|
|
}
|
|
},
|
|
cycleHeader : function(col, element) // element parameter used in Moz1.7-
|
|
{ // not in Moz1.8+
|
|
var sortActive;
|
|
var treeCols;
|
|
|
|
// Moz1.8 trees require column.id, moz1.7 and earlier trees use column.
|
|
this.selectedColumn = (typeof(col)=="object" ? col.id : col);
|
|
if (!element) element = col.element; // in Moz1.8+, get element from col
|
|
sortActive = element.getAttribute("sortActive");
|
|
this.sortDirection = element.getAttribute("sortDirection");
|
|
|
|
if (sortActive != "true")
|
|
{
|
|
treeCols = document.getElementsByTagName("treecol");
|
|
for (var i = 0; i < treeCols.length; i++)
|
|
{
|
|
treeCols[i].removeAttribute("sortActive");
|
|
treeCols[i].removeAttribute("sortDirection");
|
|
}
|
|
sortActive = true;
|
|
this.sortDirection = "ascending";
|
|
}
|
|
else
|
|
{
|
|
sortActive = true;
|
|
if (this.sortDirection == "" || this.sortDirection == "descending")
|
|
{
|
|
this.sortDirection = "ascending";
|
|
}
|
|
else
|
|
{
|
|
this.sortDirection = "descending";
|
|
}
|
|
}
|
|
element.setAttribute("sortActive", sortActive);
|
|
element.setAttribute("sortDirection", this.sortDirection);
|
|
this.sortStartedTime = new Date().getTime(); // for null dates during sort
|
|
gTaskArray.sort( compareTasks );
|
|
|
|
document.getElementById( ToDoUnifinderTreeName ).view = this;
|
|
},
|
|
setTree : function( tree ){this.tree = tree;},
|
|
getCellText : function(row,column)
|
|
{
|
|
calendarToDo = gTaskArray[row];
|
|
if( !calendarToDo )
|
|
return false;
|
|
|
|
// Moz1.8 trees require column.id, moz1.7 and earlier trees use column.
|
|
switch( typeof(column)=="object" ? column.id : column )
|
|
{
|
|
case "unifinder-todo-tree-col-completed":
|
|
return( "" );
|
|
|
|
case "unifinder-todo-tree-col-priority":
|
|
return( "" );
|
|
|
|
case "unifinder-todo-tree-col-title":
|
|
// return title, or "Untitled" if empty/null
|
|
return calendarToDo.title || gCalendarBundle.getString( "eventUntitled" );
|
|
case "unifinder-todo-tree-col-startdate":
|
|
return( formatUnifinderToDoDateTime( calendarToDo.start ) );
|
|
case "unifinder-todo-tree-col-duedate":
|
|
return( formatUnifinderToDoDateTime( calendarToDo.due ) );
|
|
case "unifinder-todo-tree-col-completeddate":
|
|
return( formatUnifinderToDoDateTime( calendarToDo.completed ) );
|
|
case "unifinder-todo-tree-col-percentcomplete":
|
|
return( calendarToDo.percent+"%" );
|
|
case "unifinder-todo-tree-col-categories":
|
|
return( calendarToDo.categories );
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
function compareTasks( taskA, taskB )
|
|
{
|
|
var modifier = 1;
|
|
if (toDoTreeView.sortDirection == "descending")
|
|
{
|
|
modifier = -1;
|
|
}
|
|
|
|
switch(toDoTreeView.selectedColumn)
|
|
{
|
|
case "unifinder-todo-tree-col-priority": // 0-9
|
|
return compareNumber(taskA.priority, taskB.priority) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-title":
|
|
return compareString(taskA.title, taskB.title) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-startdate":
|
|
return compareDate(taskA.start, taskB.start) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-duedate":
|
|
return compareDate(taskA.due, taskB.due) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-completed": // checkbox if date exists
|
|
case "unifinder-todo-tree-col-completeddate":
|
|
return compareDate(taskA.completed, taskB.completed) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-percentcomplete":
|
|
return compareNumber(taskA.percent, taskB.percent) * modifier;
|
|
|
|
case "unifinder-todo-tree-col-categories":
|
|
return compareString(taskA.categories, taskB.categories) * modifier;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
function compareString(a, b) {
|
|
a = nullToEmpty(a);
|
|
b = nullToEmpty(b);
|
|
return ((a < b) ? -1 :
|
|
(a > b) ? 1 : 0);
|
|
}
|
|
|
|
function nullToEmpty(value) {
|
|
return value == null? "" : value;
|
|
}
|
|
|
|
function compareNumber(a, b) {
|
|
// Number converts a date to msecs since 1970, and converts a null to 0.
|
|
a = Number(a);
|
|
b = Number(b);
|
|
return ((a < b) ? -1 : // avoid underflow problems of subtraction
|
|
(a > b) ? 1 : 0);
|
|
}
|
|
function compareDate(a, b) {
|
|
a = dateToMilliseconds(a);
|
|
b = dateToMilliseconds(b);
|
|
return ((a < b) ? -1 : // avoid underflow problems of subtraction
|
|
(a > b) ? 1 : 0);
|
|
}
|
|
function dateToMilliseconds(oeICalDateTime) {
|
|
// Treat unset time as "time when sort started", so incomplete tasks
|
|
// stay current. "Time when sort started" is computed once per sort
|
|
// (just before sort) so sort is stable.
|
|
if (oeICalDateTime && oeICalDateTime.isSet)
|
|
return oeICalDateTime.getTime();
|
|
else
|
|
return treeView.sortStartedTime;
|
|
}
|
|
|
|
|
|
function calendarTaskView( taskArray )
|
|
{
|
|
this.taskArray = taskArray;
|
|
}
|
|
|
|
calendarTaskView.prototype.getCalendarTaskAtRow = function( i )
|
|
{
|
|
return( gTaskArray[ i ] );
|
|
}
|
|
|
|
calendarTaskView.prototype.getRowOfCalendarTask = function( Task )
|
|
{
|
|
for( var i = 0; i < this.gTaskArray.length; i++ )
|
|
{
|
|
if( this.gTaskArray[i].id == Event.id )
|
|
return( i );
|
|
}
|
|
return( "null" );
|
|
}
|
|
|
|
/**
|
|
* Redraw the categories unifinder tree
|
|
*/
|
|
|
|
function refreshToDoTree( taskArray )
|
|
{
|
|
if( taskArray === false )
|
|
taskArray = getTaskTable();
|
|
|
|
gTaskArray = taskArray;
|
|
|
|
toDoTreeView.rowCount = gTaskArray.length;
|
|
|
|
var ArrayOfTreeCols = document.getElementById( ToDoUnifinderTreeName ).getElementsByTagName( "treecol" );
|
|
|
|
for( var i = 0; i < ArrayOfTreeCols.length; i++ )
|
|
{
|
|
if( ArrayOfTreeCols[i].getAttribute( "sortActive" ) == "true" )
|
|
{
|
|
toDoTreeView.selectedColumn = ArrayOfTreeCols[i].getAttribute( "id" );
|
|
toDoTreeView.sortDirection = ArrayOfTreeCols[i].getAttribute("sortDirection");
|
|
toDoTreeView.sortStartedTime = new Date().getTime(); //for null/0 dates
|
|
gTaskArray.sort(compareTasks);
|
|
break;
|
|
}
|
|
}
|
|
|
|
document.getElementById( ToDoUnifinderTreeName ).view = toDoTreeView;
|
|
|
|
document.getElementById( ToDoUnifinderTreeName ).taskView = new calendarTaskView( gTaskArray );
|
|
}
|
|
|
|
|
|
function getTaskTable( )
|
|
{
|
|
return gTaskArray;
|
|
}
|
|
|
|
|
|
function contextChangeProgress( event, Progress )
|
|
{
|
|
var tree = document.getElementById( ToDoUnifinderTreeName );
|
|
|
|
if (tree.view.selection.count > 0)
|
|
{
|
|
var todoItem = tree.taskView.getCalendarTaskAtRow( tree.currentIndex );
|
|
if(todoItem)
|
|
{
|
|
var newItem = todoItem.clone().QueryInterface(Components.interfaces.calITodo);
|
|
newItem.percentComplete = Progress;
|
|
var calendar = getCalendar();
|
|
calendar.modifyItem (newItem, null);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function contextChangePriority( event, Priority )
|
|
{
|
|
var tree = document.getElementById( ToDoUnifinderTreeName );
|
|
|
|
if (tree.view.selection.count > 0)
|
|
{
|
|
var todoItem = tree.taskView.getCalendarTaskAtRow( tree.currentIndex );
|
|
if(todoItem)
|
|
{
|
|
var newItem = todoItem.clone().QueryInterface(Components.interfaces.calITodo);
|
|
newItem.priority = Priority;
|
|
var calendar = getCalendar();
|
|
calendar.modifyItem (newItem, null);
|
|
}
|
|
}
|
|
}
|
|
|
|
function changeToolTipTextForToDo( event )
|
|
{
|
|
var toDoItem = getToDoFromEvent( event );
|
|
|
|
var toolTip = document.getElementById( "taskTooltip" );
|
|
|
|
while( toolTip.hasChildNodes() )
|
|
{
|
|
toolTip.removeChild( toolTip.firstChild );
|
|
}
|
|
var holderBox = getPreviewForTask( toDoItem );
|
|
if( holderBox )
|
|
toolTip.appendChild( holderBox );
|
|
}
|
|
|
|
function changeContextMenuForToDo( event )
|
|
{
|
|
if (event.target.id != "taskitem-context-menu")
|
|
return;
|
|
|
|
var toDoItem = getToDoFromEvent( event );
|
|
if( toDoItem )
|
|
{
|
|
document.getElementById("is_editable").removeAttribute("disabled");
|
|
var liveList = document.getElementById( "taskitem-context-menu" ).getElementsByAttribute( "checked", "true" );
|
|
// Delete in reverse order. Moz1.8+ getElementsByAttribute list is
|
|
// 'live', so when attribute is deleted the indexes of later elements
|
|
// change, but Moz1.7- is not. Reversed order works with both.
|
|
for (var i = liveList.length - 1; i >= 0; i-- )
|
|
{
|
|
liveList.item(i).removeAttribute( "checked" );
|
|
}
|
|
|
|
if( document.getElementById( "percent-"+toDoItem.percent+"-menuitem" ) )
|
|
{
|
|
document.getElementById( "percent-"+toDoItem.percent+"-menuitem" ).setAttribute( "checked", "true" );
|
|
}
|
|
|
|
if( document.getElementById( "priority-"+toDoItem.priority+"-menuitem" ) )
|
|
{
|
|
document.getElementById( "priority-"+toDoItem.priority+"-menuitem" ).setAttribute( "checked", "true" );
|
|
}
|
|
} else {
|
|
document.getElementById("is_editable").setAttribute("disabled", "true");
|
|
}
|
|
}
|
|
|