<title>Coming Next</title>\r
\r
<style type="text/css">\r
-/* -----------------------------------------------------------------------\r
-here you can customize background color, font color, font size etc...\r
---------------------------------------------------------------------------- */\r
-.background { color:#ffffff; background-color:#000000; font:normal 12pt; } /* Defines the background of the widget. If you want to use a background image, set useBackgroundImage = true below */\r
- /* for the default themes, black, gray, and light blue, codes are #292029, #e7dfe7, #009aef */\r
-.weekDay { } /* Defines the appearance of all week day texts */\r
-.date { } /* Defines the appearance of all date texts */\r
-.today { color:#ff0000; } /* Defines the appearance of "Today" text */\r
-.time { } /* Defines the appearance of all time texts */\r
-.now { color:#ff00ff; } /* Defines the appearance of "Now" text */\r
-.description { } /* Defines the appearance of all event descriptions */\r
-.icon { width:15px; height:15px; } /* Defines size and appearance of icons */\r
+/* The following classes can be modified by widget settings */\r
+.background { color:#ffffff; background-color:#000000 }\r
+.backgroundFullscreen { color:#ffffff; background-color:#000000 }\r
+.weekDay { }\r
+.date { }\r
+.today { color:#ff0000 }\r
+.tomorrow { color:#0000ff }\r
+.time { }\r
+.now { color:#ff00ff }\r
+.description { }\r
+.icon { width:15px; height:15px }\r
+.overdue { color:#ffff00 }\r
+.calendar1 { background-color:#0757cf }\r
+.calendar2 { background-color:#579f37 }\r
+.calendar3 { background-color:#ff9f07 }\r
+.calendar4 { background-color:#af8fef }\r
+.calendar5 { background-color:#57afbf }\r
+.calendar6 { background-color:#9fdf57 }\r
</style>\r
\r
+<script type="text/javascript" src="localizedTextStrings.js" charset="utf-8" />\r
+\r
<script>\r
+// valid types for the config object are 'Int', 'Bool', 'String', 'Enum', 'UID', 'Array'\r
+var config = {\r
+ monthRange: { Type: 'Int', Default: 2, Value: 2,},\r
+ includeTodos: { Type: 'Bool', Default: true, Value: true,},\r
+ useBackgroundImage: { Type: 'Bool', Default: true, Value: true,},\r
+ backgroundImageLocation: { Type: 'Enum', Default: 'internal', Value: 'internal', ValidValues: ['internal', 'external']},\r
+ showCombinedDateTime: { Type: 'Bool', Default: false, Value: false,},\r
+ showLocation: { Type: 'Bool', Default: true, Value: true,},\r
+ showTodayAsText: { Type: 'Bool', Default: true, Value: true,},\r
+ todayText: { Type: 'String', Default: getLocalizedText('settings.default.todayText'), Value: getLocalizedText('settings.default.todayText'),},\r
+ tomorrowText: { Type: 'String', Default: getLocalizedText('settings.default.tomorrowText'), Value: getLocalizedText('settings.default.tomorrowText'),},\r
+ showNowAsText: { Type: 'Bool', Default: true, Value: true,},\r
+ nowText: { Type: 'String', Default: getLocalizedText('settings.default.nowText'), Value: getLocalizedText('settings.default.nowText'),},\r
+ markOverdueTodos: { Type: 'Bool', Default: true, Value: true,},\r
+ overdueText: {Type: 'String', Default: getLocalizedText('settings.default.overdueText'), Value: getLocalizedText('settings.default.overdueText'),},\r
+ dateSeparator: { Type: 'String', Default: getLocalizedText('settings.default.dateSeparator'), Value: getLocalizedText('settings.default.dateSeparator'),},\r
+ dateFormat: { Type: 'Enum', Default: 'auto', Value: 'auto', ValidValues: ['auto', 'DDMM', 'MMDD'],},\r
+ weekDayLength: { Type: 'Int', Default: 2, Value: 2,},\r
+ updateDataInterval: { Type: 'Int', Default: 5, Value: 5,},\r
+ calendarApp: { Type: 'UID', Default: 0x10005901, Value: 0x10005901,},\r
+ eventsPerWidget: { Type: 'Int', Default: 4, Value: 4,},\r
+ showNothingText: { Type: 'Bool', Default: true, Value: true,},\r
+ nothingText: { Type: 'String', Default: getLocalizedText('settings.default.nothingText'), Value: getLocalizedText('settings.default.nothingText'),},\r
+ enableDaylightSaving: { Type: 'Bool', Default: true, Value: true,},\r
+ daylightSavingOffset: { Type: 'Int', Default: 1, Value: 1,},\r
+ hideWidgetOnCalendarOpen: { Type: 'Bool', Default: false, Value: false,},\r
+ showCalendarIndicator: { Type: 'Bool', Default: true, Value: true,},\r
+ excludedCalendars: { Type: 'Array', Default: [], Value: [],},\r
+ enableLogging: { Type: 'Bool', Default: false, Value: false,},\r
+ cssStyle_background: { Type: 'String', Default: 'color:#ffffff; background-color:#000000', Value: 'color:#ffffff; background-color:#000000',},\r
+ cssStyle_backgroundFullscreen: { Type: 'String', Default: 'color:#ffffff; background-color:#000000', Value: 'color:#ffffff; background-color:#000000',},\r
+ cssStyle_weekDay: { Type: 'String', Default: '', Value: '',},\r
+ cssStyle_date: { Type: 'String', Default: '', Value: '',},\r
+ cssStyle_today: { Type: 'String', Default: 'color:#ff0000', Value: 'color:#ff0000',},\r
+ cssStyle_tomorrow: { Type: 'String', Default: 'color:#0000ff', Value: 'color:#0000ff',},\r
+ cssStyle_time: { Type: 'String', Default: '', Value: '',},\r
+ cssStyle_now: { Type: 'String', Default: 'color:#ff00ff', Value: 'color:#ff00ff',},\r
+ cssStyle_description: { Type: 'String', Default: '', Value: '',},\r
+ cssStyle_icon: { Type: 'String', Default: 'width:15px; height:15px', Value: 'width:15px; height:15px',},\r
+ cssStyle_overdue: { Type: 'String', Default: 'color:#ffff00', Value: 'color:#ffff00',},\r
+ cssStyle_calendar1: { Type: 'String', Default: 'background-color:#0757cf', Value: 'background-color:#0757cf',},\r
+ cssStyle_calendar2: { Type: 'String', Default: 'background-color:#579f37', Value: 'background-color:#579f37',},\r
+ cssStyle_calendar3: { Type: 'String', Default: 'background-color:#ff9f07', Value: 'background-color:#ff9f07',},\r
+ cssStyle_calendar4: { Type: 'String', Default: 'background-color:#af8fef', Value: 'background-color:#af8fef',},\r
+ cssStyle_calendar5: { Type: 'String', Default: 'background-color:#57afbf', Value: 'background-color:#57afbf',},\r
+ cssStyle_calendar6: { Type: 'String', Default: 'background-color:#9fdf57', Value: 'background-color:#9fdf57',},\r
+}\r
+\r
\r
-//---------------------------------------------------------------\r
-// The following section contains settings you may want to tweak\r
-//---------------------------------------------------------------\r
-var monthRange = 2; // number of months to include in the event list\r
-var includeTodos = true; // disable to remove ToDos from event list\r
-var useBackgroundImage = true; // use background_portrait.png and background_landscape.png to fake transparency. Set to "false" to use a solid background color\r
-var showCombinedDateTime = false;// only show the time for events happening today, otherwise just show the date\r
-var showLocation = true; // show the location for meeting events\r
-var showTodayAsText = true; // if enabled, the current date will be shown as "Today" instead of "31.12"\r
-var todayText = 'Today'; // text to display for "Today"\r
-var showNowAsText = true; // if enabled, the appointment time will be shown as "Now" instead of "12:00"\r
-var nowText = 'Now'; // text to display for "Now"\r
-var dateSeparator = '.'; // separator for dates. e.g. "31.12" or "31/12"\r
-var dateFormat = 'auto' // how dates will be displayed. 'auto' will autodetect your phone's date format setting. 'MMDD' will write month first, 'DDMM' will write day first\r
-var weekDayLength = 2; // defines how many characters of the weekday will be shown. E.g. 2 will cut "Friday" to "Fr"\r
-var updateDataInterval = 5; // how many minutes to wait before updating the displayed data. The higher the number, the less battery is used\r
-var calendarApp = 0x10005901; // UID of the calendar app to run when clicking the widget. 0x10005901 = buildin calendar, 0x20004ec1 = Epocware Handy Calendar\r
-var eventsPerWidget = 4; // number of events to show per widget. Default is 4\r
-var showNothingText = true; // if set to "true", show a text if no events are in the list\r
-var nothingText = 'No further events within ' + monthRange + ' months'; // text to show when no events are in the list\r
\r
//-------------------------------------------------------\r
// Nothing of interest from here on...\r
//-------------------------------------------------------\r
var panelNum = 0; // use 1 for second panel\r
+var version = "1.32";\r
+var versionURL = "http://comingnext.sourceforge.net/version.xml";\r
var calendarService = null;\r
var cacheEntriesHtml = [];\r
var months_translated = [];\r
var orientation = '';\r
var now = new Date();\r
+var mode = 0; // 0 = homescreen, 1 = fullscreen, 2 = settings, 3 = about, 4 = check for update\r
+var reqV = null; \r
+var settingsCalEntryId = null;\r
+var settingsCache = null;\r
+var notificationRequests = new Array();\r
+var calendarList = [];\r
+var calendarColors = [];\r
+var updateTimer = null;\r
+var screenRotationTimer = null;\r
+var lastUpdateTime = now; // last time we updated the display\r
+var lastReloadTime = null; // last time we fetched calendar data\r
+var reloadInterval = 6 * 60 * 60 * 1000; // = 6 hours; time interval for reloading calendar data\r
+var errorOccured = false;\r
+var entryLists = null; // stores all fetched calendar entries until data is refreshed\r
\r
-window.onload = init;\r
-window.onresize = updateScreen;\r
-window.onshow = updateScreen;\r
+// vars for daylight saving time\r
+var summertime = false; // true, if current date is in summer, false if in winter\r
+var daylightSavingDates = new Object(); // caches calculated DST winter and summer time shift dates\r
+\r
+// this is a list of data fields a calendar event can have\r
+var entryFields = [\r
+ "id",\r
+ "Type",\r
+ "CalendarName",\r
+ "Summary",\r
+ "Location",\r
+ "Status",\r
+ "StartTime",\r
+ "EndTime",\r
+ "InstanceStartTime",\r
+ "InstanceEndTime"\r
+];\r
+\r
+function isLeapYear( year ) {\r
+ if (( year % 4 == 0 && year % 100 != 0 ) || year % 400 == 0 )\r
+ return true;\r
+ else\r
+ return false;\r
+}\r
+\r
+function calcLeapYear(year, days)\r
+{\r
+ if (isLeapYear(year))\r
+ return ++days;\r
+ else\r
+ return days;\r
+}\r
+ \r
+function subToSunday(myDate, year, days, prevMonthDays)\r
+{\r
+ for (i = myDate.getDay(); i > 0 ;i--)\r
+ days--;\r
+ days -= prevMonthDays;\r
+ days = isLeapYear(year) ? --days : days;\r
+ return days;\r
+}\r
+\r
+function isSummertime(curDate)\r
+{\r
+ var summer = false;\r
+ var winter = false;\r
+\r
+ // if we already calculated DST summer and winter time dates for this year, use cached values\r
+ var dst = daylightSavingDates[curDate.getFullYear()];\r
+ if (!dst) {\r
+ var thisYearS = new Date(curDate.getFullYear(), 3, 0, 0, 0, 0 );\r
+ var thisYearW = new Date(curDate.getFullYear(), 10, 0, 0, 0, 0 );\r
+ var nextYearS = new Date(curDate.getFullYear() + 1, 3, 0, 0, 0, 0 );\r
+ var nextYearW = new Date(curDate.getFullYear() + 1, 10, 0, 0, 0, 0 );\r
+ \r
+ thisYearSDays = nextYearSDays = 90;\r
+ thisYearWDays = nextYearWDays = 304;\r
+ \r
+ thisYearSDays = calcLeapYear(curDate.getFullYear(), thisYearSDays);\r
+ thisYearWDays = calcLeapYear(curDate.getFullYear(), thisYearWDays);\r
+ nextYearSDays = calcLeapYear(curDate.getFullYear() + 1, nextYearSDays);\r
+ nextYearWDays = calcLeapYear(curDate.getFullYear() + 1, nextYearWDays);\r
+ \r
+ thisYearSDays = subToSunday(thisYearS, curDate.getFullYear(), thisYearSDays, 59);\r
+ thisYearWDays = subToSunday(thisYearW, curDate.getFullYear(), thisYearWDays, 273);\r
+ nextYearSDays = subToSunday(nextYearS, curDate.getFullYear() + 1, nextYearSDays, 59);\r
+ nextYearWDays = subToSunday(nextYearW, curDate.getFullYear() + 1, nextYearWDays, 273);\r
+ \r
+ dst = {\r
+ Summer: new Date (curDate.getFullYear(), 03-1, thisYearSDays, 2, 0, 0),\r
+ Winter: new Date (curDate.getFullYear(), 10-1, thisYearWDays, 2, 0, 0),\r
+ }\r
+ daylightSavingDates[curDate.getFullYear()] = dst;\r
+ }\r
+\r
+ if (dst.Summer < curDate)\r
+ summer = true;\r
+ if (dst.Winter < curDate)\r
+ winter = true;\r
+ if (summer && !winter)\r
+ return true;\r
+ else\r
+ return false;\r
+}\r
\r
function error(message)\r
{\r
console.info('Error: ' + message);\r
document.getElementById("calendarList").innerHTML = 'Error: ' + message;\r
+ document.getElementById("fullscreenCalendarList").innerHTML = 'Error: ' + message;\r
+ errorOccured = true;\r
+ document.onclick = null;\r
+}\r
+\r
+function areDatesEqual(date1, date2)\r
+{\r
+ return (date1.getFullYear() == date2.getFullYear() && \r
+ date1.getMonth() == date2.getMonth() && \r
+ date1.getDate() == date2.getDate());\r
+}\r
+\r
+function isTomorrow(date)\r
+{\r
+ // tommorow = now + 1 day\r
+ // ToDo: some days can be shorter as 24 hours(daylight saving change day)\r
+ return areDatesEqual(date, new Date (now.getTime() + 24*60*60*1000));\r
}\r
\r
function isToday(date)\r
{\r
- if (date.getDate() == now.getDate() && date.getMonth() == now.getMonth())\r
- return true;\r
- return false;\r
+ return areDatesEqual(date, now);\r
}\r
\r
function collectLocales()\r
{\r
- var tmpyear = ((panelNum == 0) ? 2000 : 2001);\r
+ var tmpyear = 2000 + panelNum;\r
var month = 0;\r
\r
if (months_translated.length > 0)\r
month = tmp;\r
}\r
\r
- console.info(entry.StartTime + ' -> ' + month + ' ' + counter);\r
+ log(entry.StartTime + ' -> ' + month + ' ' + counter);\r
ids[counter] = entry.id;\r
months_translated[month] = counter + 1;\r
counter++;\r
error(e + ', line ' + e.line);\r
return;\r
}\r
- console.info(ids);\r
+ log(ids);\r
try {\r
var criteria = new Object();\r
criteria.Type = "CalendarEntry";\r
{\r
var criteria = new Object();\r
criteria.Type = "CalendarEntry";\r
+ criteria.Filter = new Object();\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ criteria.Filter.CalendarName = calendarList[i];\r
+ try {\r
+ var notificationRequest = calendarService.IDataSource.RequestNotification(criteria, callback);\r
+ if (notificationRequest.ErrorCode)\r
+ error('requestNotification failed with error code ' + notificationRequest.ErrorCode);\r
+ notificationRequests.push(notificationRequest);\r
+ } catch (e) {\r
+ error("requestNotification: " + e + ', line ' + e.line);\r
+ }\r
+ }\r
\r
+ var criteria2 = new Object();\r
+ criteria2.Type = "CalendarEntry";\r
+ criteria2.Filter = new Object();\r
+ criteria2.Filter.LocalIdList = new Array();\r
+ criteria2.Filter.LocalIdList[0] = settingsCalEntryId;\r
try {\r
- var result = calendarService.IDataSource.RequestNotification(criteria, callback);\r
- if (result.ErrorCode)\r
- error('loading Calendar items list');\r
+ var notificationRequest = calendarService.IDataSource.RequestNotification(criteria2, settingsCallback);\r
+ if (notificationRequest.ErrorCode)\r
+ error('requestNotification failed with error code ' + notificationRequest.ErrorCode);\r
+ notificationRequests.push(notificationRequest);\r
} catch (e) {\r
error("requestNotification: " + e + ', line ' + e.line);\r
}\r
}\r
\r
+function cancelNotification()\r
+{\r
+ for(var i=0; i < notificationRequests.length; i++) {\r
+ try {\r
+ var result = calendarService.IDataSource.Cancel(notificationRequests[i]);\r
+ if (result.ErrorCode)\r
+ error('cancelNotification failed with error code ' + result.ErrorCode);\r
+ } catch (e) {\r
+ error("cancelNotification: " + e + ', line ' + e.line);\r
+ }\r
+ }\r
+}\r
+\r
function callback(transId, eventCode, result)\r
{\r
+ log("callback(): panelNum: " + panelNum + " transId: " + transId + " eventCode: " + eventCode + " result.ErrorCode: " + result.ErrorCode);\r
+ lastReloadTime = null; // force calendar data reload on next update\r
updateData();\r
}\r
\r
+function settingsCallback(transId, eventCode, result)\r
+{\r
+ log("settingsCallback(): panelNum: " + panelNum + " transId: " + transId + " eventCode: " + eventCode + " result.ErrorCode: " + result.ErrorCode);\r
+ loadSettings();\r
+}\r
+\r
function parseDate(dateString)\r
{\r
/*\r
if (dateArr.length == 6 && dateArr[5].toLowerCase() == 'am' && hours == 12)\r
hours = 0;\r
\r
- console.info('year=' + year + ' month=' + month + ' day=' + day + ' hours=' + hours + ' minutes=' + minutes+ ' seconds=' + seconds);\r
+ var result = new Date(year, month - 1, day, hours, minutes, seconds);\r
+ \r
+ // take care of daylight saving time\r
+ if (config['enableDaylightSaving'].Value) {\r
+ \r
+ // determine if date is in summer or winter time\r
+ var dateSummerTime = isSummertime(result);\r
\r
- return new Date(year, month - 1, day, hours, minutes, seconds);\r
+ // work around bug in Nokias calendar api resulting in dates within a different DST to be off by 1 hour\r
+ if (summertime && !dateSummerTime) {\r
+ result = new Date(result.getTime() - 1000 * 60 * 60 * config['daylightSavingOffset'].Value); // -1 hour\r
+ log('parseDate(): fixing time -1h: ' + result);\r
+ }\r
+ else if (!summertime && dateSummerTime) {\r
+ result = new Date(result.getTime() + 1000 * 60 * 60 * config['daylightSavingOffset'].Value); // +1 hour\r
+ log('parseDate(): fixing time +1h: ' + result);\r
+ }\r
+ }\r
+\r
+ return result;\r
}\r
\r
// returns a short date as string ("31.12" or "12.31") based on the format string which should look like "Wednesday, 26 August, 2009 12:00:00 am"\r
while (day.length < 2) { day = '0' + day; }\r
while (month.length < 2) { month = '0' + month; }\r
\r
- if (showTodayAsText && isToday(date))\r
- return '<span class="today">' + todayText + '</span>';\r
+ if (config['showTodayAsText'].Value && isToday(date))\r
+ return '<span class="today">' + config['todayText'].Value + '</span>';\r
+ if (config['showTodayAsText'].Value && isTomorrow(date))\r
+ return '<span class="tomorrow">' + config['tomorrowText'].Value + '</span>';\r
\r
var dateArr = format.replace(/,/g,'').replace(/\./g,':').replace(/ /g,' ').split(' ');\r
if (dateArr.length != 5 && dateArr.length != 6) {\r
// we don't know how to format this\r
- if (dateFormat == 'auto' || dateFormat == 'DDMM')\r
- return day + dateSeparator + month;\r
+ if (config['dateFormat'].Value == 'auto' || config['dateFormat'].Value == 'DDMM')\r
+ return day + config['dateSeparator'].Value + month;\r
else\r
- return month + dateSeparator + day;\r
+ return month + config['dateSeparator'].Value + day;\r
}\r
\r
var dayFirst = true;\r
- if (dateFormat == 'MMDD')\r
+ if (config['dateFormat'].Value == 'MMDD')\r
dayFirst = false;\r
- else if (dateFormat == 'DDMM')\r
+ else if (config['dateFormat'].Value == 'DDMM')\r
dayFirst = true;\r
else {\r
- // dateFormat == 'auto', try to detect system setting\r
+ // config['dateFormat'].Value == 'auto', try to detect system setting\r
// parse date\r
var day_ = dateArr[1];\r
var month_ = dateArr[2];\r
}\r
\r
if (dayFirst)\r
- return day + dateSeparator + month;\r
+ return day + config['dateSeparator'].Value + month;\r
else\r
- return month + dateSeparator + day;\r
+ return month + config['dateSeparator'].Value + day;\r
}\r
\r
function formatTime(date)\r
var time = date.toLocaleTimeString().replace(/[\.:]00/, ''); // remove seconds from string\r
if (time.replace(/\./, ':').split(':')[0].length < 2)\r
time = '0' + time;\r
- if (showNowAsText && date.getTime() == now.getTime())\r
- time = '<span class="now">' + nowText + '</span>';\r
+ if (config['showNowAsText'].Value && date.getTime() == now.getTime())\r
+ time = '<span class="now">' + config['nowText'].Value + '</span>';\r
return time;\r
}\r
\r
function updateData()\r
{\r
- try {\r
- // meetings have time\r
- // note: anniveraries have a start time of 12:00am. So since we want to include them, we have to query the whole day and check if events have passed later\r
- now = new Date();\r
- var meetingListFiltering = {\r
- Type:'CalendarEntry',\r
- Filter:{\r
- StartRange: (new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0, 0, 0)),\r
- EndRange: (new Date(now.getFullYear(), now.getMonth() + monthRange, now.getDate(), 0, 0, 0))\r
+ log('updateData()');\r
+ if (errorOccured) {\r
+ return;\r
+ }\r
+\r
+ // check if we got additional or less calendars since our last update\r
+ var newCalendarList = listCalendars();\r
+ if (newCalendarList.length != calendarList.length) {\r
+ calendarList = newCalendarList;\r
+ updateCalendarColors();\r
+ cancelNotification();\r
+ requestNotification();\r
+ lastReloadTime = null; // force calendar data reload on this update\r
+ }\r
+\r
+ now = new Date();\r
+ \r
+ // only reload calendar data every 6 hours, visual updates occure more often\r
+ if (!lastReloadTime || now.getTime() - lastReloadTime.getTime() > reloadInterval) {\r
+ log('updateData(): reloading calendar data');\r
+ try {\r
+ // meetings have time\r
+ // note: anniveraries have a start time of 12:00am. So since we want to include them, we have to query the whole day and check if events have passed later\r
+ summertime = isSummertime(now); // cache summer time info for today\r
+ var meetingList = [];\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ // ignore excluded calendars\r
+ if (config['excludedCalendars'].Value.indexOf(calendarList[i]) != -1)\r
+ continue;\r
+ var meetingListFiltering = {\r
+ Type:'CalendarEntry',\r
+ Filter:{\r
+ CalendarName: calendarList[i],\r
+ StartRange: (new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0, 0, 0)),\r
+ EndRange: (new Date(now.getFullYear(), now.getMonth() + config['monthRange'].Value, now.getDate(), 0, 0, 0))\r
+ }\r
+ }\r
+ var meetingResult = calendarService.IDataSource.GetList(meetingListFiltering);\r
+ if (meetingResult.ErrorCode != 0)\r
+ throw("Error fetching calendar data: " + meetingResult.ErrorCode + ': ' + meetingResult.ErrorMessage);\r
+ var list = meetingResult.ReturnValue;\r
+ meetingList = meetingList.concat(listToArray(list, calendarList[i]));\r
}\r
- }\r
- var meetingResult = calendarService.IDataSource.GetList(meetingListFiltering);\r
- var meetingList = meetingResult.ReturnValue;\r
-\r
- // todos don't, they start on 00:00 hrs., but should be visible anyway\r
- // this will generate a list of passed todos. We have to check if they have been marked as "done" yet\r
- if (includeTodos) {\r
- var todayTodoListFiltering = {\r
- Type:'CalendarEntry',\r
- Filter:{\r
- Type: 'ToDo',\r
- StartRange: (new Date(now.getFullYear() - 1, now.getMonth(), now.getDate(), 0, 0, 0)),\r
- EndRange: (new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0, 0, 1))\r
+ log("updateData(): meetingList.sort()");\r
+ meetingList.sort(sortCalendarEntries);\r
+\r
+ // todos don't, they start on 00:00 hrs., but should be visible anyway\r
+ // this will generate a list of passed todos. We have to check if they have been marked as "done" yet\r
+ if (config['includeTodos'].Value) {\r
+ var todayTodoList = [];\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ // ignore excluded calendars\r
+ if (config['excludedCalendars'].Value.indexOf(calendarList[i]) != -1)\r
+ continue;\r
+ var todayTodoListFiltering = {\r
+ Type:'CalendarEntry',\r
+ Filter:{\r
+ CalendarName: calendarList[i],\r
+ Type: 'ToDo',\r
+ StartRange: (new Date(now.getFullYear() - 1, now.getMonth(), now.getDate(), 0, 0, 0)),\r
+ EndRange: (new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0, 0, 1))\r
+ }\r
+ }\r
+ var todayTodoResult = calendarService.IDataSource.GetList(todayTodoListFiltering);\r
+ var list = todayTodoResult.ReturnValue;\r
+ todayTodoList = todayTodoList.concat(listToArray(list, calendarList[i]));\r
}\r
+ log("updateData(): todayTodoList.sort()");\r
+ todayTodoList.sort(sortCalendarEntries);\r
+ entryLists = [todayTodoList, meetingList];\r
+ } else {\r
+ entryLists = [meetingList];\r
}\r
- var todayTodoResult = calendarService.IDataSource.GetList(todayTodoListFiltering);\r
- var todayTodoList = todayTodoResult.ReturnValue;\r
- var entryLists = [todayTodoList, meetingList];\r
- } else {\r
- var entryLists = [meetingList];\r
+ lastReloadTime = new Date();\r
+ } catch(e) {\r
+ error('loading Calendar items list:' + e + ', line ' + e.line);\r
+ return;\r
}\r
- } catch(e) {\r
- error('loading Calendar items list:' + e + ', line ' + e.line);\r
- return;\r
}\r
\r
try {\r
var counter = 0;\r
var entryDate = '';\r
var dateArr = [];\r
- var entriesHtml = '<table>';\r
+ var fontsize = 'normal';\r
+ if (mode == 0) {\r
+ if (config['eventsPerWidget'].Value == 3) {\r
+ fontsize = '17pt';\r
+ changeCssClass('.icon', 'width:20px; height:20px');\r
+ }\r
+ else if (config['eventsPerWidget'].Value == 5) {\r
+ fontsize = '10pt';\r
+ changeCssClass('.icon', 'width:10px; height:10px');\r
+ }\r
+ else if (config['eventsPerWidget'].Value == 6) {\r
+ fontsize = '8pt';\r
+ changeCssClass('.icon', 'width:8px; height:8px');\r
+ }\r
+ }\r
+ else\r
+ changeCssClass('.icon', config['cssStyle_icon'].Value);\r
+ var entriesHtml = '<table style="font-size:' + fontsize + ';">';\r
var eventIds = [];\r
- var max = ((panelNum == 0) ? eventsPerWidget : 2 * eventsPerWidget);\r
+ var max;\r
+ if (mode == 0)\r
+ max = (panelNum + 1) * config['eventsPerWidget'].Value;\r
+ else\r
+ max = 30; // we can display a lot more events in fullscreen mode\r
\r
+ if (config['enableLogging'].Value) {\r
+ var listinfo = "";\r
+ for (var i=0; i < entryLists.length; i++) {\r
+ listinfo = listinfo + " " + entryLists[i].length;\r
+ var entrieslist = "";\r
+ for (var j=0; j < entryLists[i].length; j++) {\r
+ entrieslist += entryLists[i][j].Summary + ", ";\r
+ }\r
+ log("updateData(): entrieslist: " + entrieslist);\r
+ }\r
+ log("updateData(): inner loop, " + entryLists.length + " lists, [" + listinfo + "] entries");\r
+ }\r
+ \r
// the first outer loop iteration is for passed ToDos, the second loop is for all upcomming events (may also include ToDos)\r
for (var i=0; counter < max && i < entryLists.length; i++) {\r
- while (counter < max && (entry = entryLists[i].getNext()) != undefined) {\r
+ for (var j=0; (counter < max) && (j < entryLists[i].length); j++) {\r
+ entry = entryLists[i][j];\r
counter++;\r
\r
// output event info for debugging\r
- console.info(\r
- 'event: Id=' + entry.id + \r
- ',Type=' + entry.Type + \r
- ',Summary=' + entry.Summary + \r
- ',Location=' + entry.Location + \r
- ',Status=' + entry.Status + \r
- ',StartTime=' + entry.StartTime +\r
- ',EndTime=' + entry.EndTime +\r
- ',InstanceStartTime=' + entry.InstanceStartTime +\r
- ',InstanceEndTime=' + entry.InstanceEndTime\r
- );\r
+ var entryInfo = "event: ";\r
+ for(var k=0; k < entryFields.length; ++k) {\r
+ if (entry[entryFields[k]] != undefined) {\r
+ entryInfo += entryFields[k] + "=" + entry[entryFields[k]] + ",";\r
+ }\r
+ }\r
+ log(entryInfo);\r
\r
// we don't want ToDos when includeTodos == false or when they are completed\r
- if (entry.Type == 'ToDo' && (entry.Status == "TodoCompleted" || !includeTodos)) {\r
- console.info('skipping ' + entry.id );\r
+ if (entry.Type == 'ToDo' && (entry.Status == "TodoCompleted" || !config['includeTodos'].Value)) {\r
+ log('skipping ' + entry.id );\r
counter--;\r
continue;\r
}\r
\r
// make sure that we don't include an event twice (useful for ToDos that might come up twice)\r
- if (eventIds[entry.id] == 1) {\r
- console.info('skipped (already included) ' + entry.id);\r
+ if (eventIds[entry.id] == 1 && entry.Type == 'ToDo') {\r
+ log('skipped (already included) ' + entry.id);\r
counter--;\r
continue;\r
} else\r
\r
// summary can be undefined!\r
var Summary = ((entry.Summary == null) ? '' : entry.Summary);\r
- if (entry.Type == 'Meeting' && entry.Location != '' && showLocation)\r
+ if (entry.Type == 'Meeting' && entry.Location != '' && config['showLocation'].Value)\r
Summary += ', ' + entry.Location;\r
\r
// fix by yves: determine start and end dates/times\r
\r
// Convert date/time string to Date object\r
var date = parseDate(entryDate);\r
- console.info('date: ' + date);\r
+ log('date: ' + date);\r
var endDate = ((entryEndTime == null) ? null : parseDate(entryEndTime));\r
- console.info('endDate: ' + endDate);\r
+ log('endDate: ' + endDate);\r
\r
// check if meeting event has already passed\r
if (entry.Type == 'Meeting') {\r
var compareTime = ((endDate == null) ? date.getTime() : endDate.getTime());\r
if (now.getTime() > compareTime) {\r
- console.info('skipping Meeting (already passed) ' + entry.id);\r
+ log('skipping Meeting (already passed) ' + entry.id);\r
counter--;\r
eventIds[entry.id] = 0;\r
continue;\r
if (entry.Type == 'Anniversary') {\r
var tmp = new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0,0,0);\r
if (date.getTime() < tmp.getTime()) {\r
- console.info('skipping Anniversary (already passed) ' + entry.id);\r
+ log('skipping Anniversary (already passed) ' + entry.id);\r
counter--;\r
eventIds[entry.id] = 0;\r
continue;\r
// fix DayEvents end time. End times are off by 1 Second. It's possible that the event has already passed\r
if (entry.Type == 'DayEvent' && endDate != null) {\r
endDate.setMinutes(endDate.getMinutes() - 1);\r
- console.info('fixing DayEvent endDate: ' + endDate);\r
+ log('fixing DayEvent endDate: ' + endDate);\r
if (now.getTime() > endDate.getTime()) {\r
- console.info('event already passed ' + entry.id);\r
+ log('event already passed ' + entry.id);\r
counter--;\r
eventIds[entry.id] = 0;\r
continue;\r
// check if we are between start and endtime\r
if ((date.getTime() < now.getTime()) && (now.getTime() < endDate.getTime())) {\r
date = now; // change appointment date/time to now\r
- console.info('event is currently taking place: ' + date);\r
+ log('event is currently taking place: ' + date);\r
}\r
}\r
\r
- // skip events for the first panel in case this is the second one\r
- if (panelNum == 1 && counter < eventsPerWidget + 1) {\r
- console.info('skipping (already in first widget) ' + entry.id);\r
+ // skip events for the first panel in case this is the second one and we're not in fullscreen mode\r
+ if (mode == 0 && panelNum > 0 && counter < panelNum * config['eventsPerWidget'].Value + 1) {\r
+ log('skipping (already in first widget) ' + entry.id);\r
continue;\r
}\r
+ \r
+ // mark overdue todos\r
+ var overdue = false;\r
+ if (entry.Type == 'ToDo' && date != null) {\r
+ var tmp1 = new Date(date.getFullYear(), date.getMonth(), date.getDate(), 0,0,0);\r
+ var tmp2 = new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0,0,0);\r
+ if (tmp1.getTime() < tmp2.getTime()) {\r
+ overdue = true;\r
+ }\r
+ }\r
\r
// generate html output\r
- entriesHtml += '<tr><td><img class="icon" src="' + entry.Type + '.png" /></td>';\r
+ entriesHtml += '<tr>';\r
+ if (config['showCalendarIndicator'].Value && calendarList.length - config['excludedCalendars'].Value.length > 1) {\r
+ entriesHtml += '<td><span class="calendar' + calendarColors[entry.CalendarName] + '"> </span></td>';\r
+ }\r
+ entriesHtml += '<td><img class="icon" src="' + entry.Type + '.png" /></td>';\r
if(date == null) {\r
// some languages have very strange locale date formats, can't parse all those. Also some todos don't have dates at all.\r
entriesHtml += '<td colspan="4"><span class="date">' + entryDate + '</span> ';\r
} else {\r
- var weekDay = date.toLocaleDateString().substr(0,weekDayLength);\r
+ var weekDay = date.toLocaleDateString().substr(0,config['weekDayLength'].Value);\r
var time = formatTime(date);\r
var dateStr = formatDate(date, entryDate);\r
- if (entry.Type == 'ToDo' || entry.Type == 'Anniversary' || entry.Type == 'DayEvent' || entry.Type == 'Reminder') {\r
- if (isToday(date))\r
- entriesHtml += '<td colspan="4"><span class="date">' + dateStr + '</span> ';\r
+ if (entry.Type == 'ToDo' && overdue && config['markOverdueTodos'].Value) {\r
+ dateStr = '<span class="overdue">' + config['overdueText'].Value + '</span>';\r
+ entriesHtml += '<td colspan="4" width="1px"><span class="date">' + dateStr + '</span> ';\r
+ } else if (entry.Type == 'ToDo' || entry.Type == 'Anniversary' || entry.Type == 'DayEvent' || entry.Type == 'Reminder') {\r
+ if ((isToday(date) || isTomorrow(date)) && config['showTodayAsText'].Value) // show weekday if the date string is not text. looks odd otherwise\r
+ entriesHtml += '<td colspan="4" width="1px"><span class="date">' + dateStr + '</span> ';\r
else\r
- entriesHtml += '<td><span class="weekDay">' + weekDay + '</span></td><td><span class="date">' + dateStr + '</span></td><td colspan="2">';\r
+ entriesHtml += '<td class="weekDay" width="1px">' + weekDay + '</td><td width="1px" class="date">' + dateStr + '</td><td colspan="2">';\r
} else if (entry.Type == 'Meeting') {\r
- if (showCombinedDateTime) {\r
+ if (config['showCombinedDateTime'].Value) {\r
if (isToday(date))\r
- entriesHtml += '<td colspan="4"><span class="today">' + time + '</span> ';\r
+ entriesHtml += '<td width="1px" colspan="4"><span class="today">' + time + '</span> ';\r
+ else if (isTomorrow(date))\r
+ entriesHtml += '<td width="1px" colspan="4"><span class="tomorrow">' + dateStr + '</span> <span class="time">' + time + '</span> ';\r
else\r
- entriesHtml += '<td><span class="weekDay">' + weekDay + '</span></td><td><span class="date">' + dateStr + '</span></td><td colspan="2">';\r
+ entriesHtml += '<td width="1px" class="weekDay">' + weekDay + '</td><td width="1px" class="date">' + dateStr + '</td><td colspan="2">';\r
} else {\r
- if (isToday(date))\r
- entriesHtml += '<td colspan="4"><span class="date">' + dateStr + '</span> <span class="time">' + time + '</span> ';\r
+ if ((isToday(date) || isTomorrow(date)) && config['showTodayAsText'].Value)\r
+ entriesHtml += '<td colspan="4" width="1px"><span class="today">' + dateStr + '</span> <span class="time">' + time + '</span> ';\r
else\r
- entriesHtml += '<td><span class="weekDay">' + weekDay + '</span></td><td><span class="date">' + dateStr + '</span></td><td width="1px"><span class="time">' + time + '</span></td><td>';\r
+ entriesHtml += '<td width="1px" class="weekDay">' + weekDay + '</td><td width="1px" class="date">' + dateStr + '</td><td width="1px" class="time">' + time + '</td><td>';\r
}\r
}\r
}\r
}\r
}\r
entriesHtml += '</table>';\r
- if (showNothingText && entriesHtml == '<table></table>')\r
- entriesHtml = '<div style="width:295px; height:75px; text-align:center; line-height:75px; overflow:visible;">' + nothingText + '</div>';\r
+ if (config['showNothingText'].Value && entriesHtml == '<table></table>') {\r
+ var text = config['nothingText'].Value.replace(/%d/, config['monthRange'].Value);\r
+ entriesHtml = '<div style="width:295px; height:75px; text-align:center; line-height:75px; overflow:visible;">' + text + '</div>';\r
+ }\r
if (cacheEntriesHtml != entriesHtml) {\r
- document.getElementById('calendarList').innerHTML = entriesHtml;\r
+ if (mode == 0)\r
+ document.getElementById('calendarList').innerHTML = entriesHtml;\r
+ else\r
+ document.getElementById('fullscreenCalendarList').innerHTML = entriesHtml;\r
cacheEntriesHtml = entriesHtml;\r
}\r
+ \r
+ lastUpdateTime = new Date();\r
} catch(e) {\r
error('displaying list:' + e + ', line ' + e.line);\r
return;\r
}\r
}\r
\r
+// called by handleOnShow() and onResize events\r
function updateScreen()\r
{\r
+ log('updateScreen()');\r
+\r
// check if opening fullscreen\r
- if( window.innerHeight > 91)\r
- launchCalendar();\r
+ if( window.innerHeight > 91 && mode == 0) {\r
+ mode = 1;\r
+ cacheEntriesHtml = '';\r
+ document.getElementById('body').style.backgroundImage = "";\r
+ showFullscreen();\r
+ }\r
+ else if (window.innerHeight <= 91 && mode != 0) {\r
+ mode = 0;\r
+ cacheEntriesHtml = '';\r
+ showHomescreen();\r
+ }\r
+ \r
+ if (mode == 0)\r
+ updateHomescreen(); // check for screen rotation\r
+ else if (mode == 1)\r
+ updateFullscreen();\r
+}\r
+\r
+function handleOnShow()\r
+{\r
+ updateScreen();\r
+\r
+ var time = new Date();\r
+ if (time.getTime() - lastUpdateTime.getTime() > config['updateDataInterval'].Value * 60 * 1000) {\r
+ log('updateScreen(): force updateData() because last update was too long ago (' + (time.getTime() - lastUpdateTime.getTime()) / 1000 + 's)');\r
+ clearUpdateTimer();\r
+ updateData();\r
+ setUpdateTimer(); // reinitialize update timer\r
+ }\r
+}\r
+\r
+function launchCalendar()\r
+{\r
+ try {\r
+ widget.openApplication(config['calendarApp'].Value, "");\r
+ if (config['hideWidgetOnCalendarOpen'].Value)\r
+ window.close();\r
+ } catch(e) {\r
+ error('starting Calendar App');\r
+ return;\r
+ }\r
+}\r
+\r
+function init()\r
+{\r
+ log('New widget instance starting up...');\r
+ \r
+ try {\r
+ // call calendar service\r
+ if (device != "undefined")\r
+ calendarService = device.getServiceObject("Service.Calendar", "IDataSource");\r
+ else\r
+ throw('device object does not exist');\r
+ } catch(e) {\r
+ error('loading Calendar service: ' + e + ', line ' + e.line + '<br /><a onclick="widget.openURL(\'http://comingnext.sf.net/help\'); return false;" href="http://comingnext.sf.net/help">' + getLocalizedText('menu.help') + '</a>');\r
+ //return;\r
+ }\r
+\r
+ calendarList = listCalendars();\r
+ loadSettings();\r
+ updateCalendarColors();\r
+ collectLocales();\r
+ //updateData();\r
+ requestNotification();\r
+ document.getElementById("settingsTitle").innerHTML = getLocalizedText('menu.settings');\r
+ setUpdateTimer();\r
+ if (window.innerHeight > 91) {\r
+ mode = 0; // we're starting fullscreen, we set mode to homescreen in order to let updateScreen() do all the work for us\r
+ }\r
+ else {\r
+ mode = 1;\r
+ }\r
+ log("init(): updateScreen()");\r
+ updateScreen();\r
+ if (config['useBackgroundImage'].Value)\r
+ // check for screen rotation every 1 secs\r
+ screenRotationTimer = window.setInterval('checkOrientation()', 1000 * 1);\r
+ \r
+ // call updateScreen() when widget changes from background to forground\r
+ window.widget.onshow = handleOnShow;\r
+\r
+ log("init(): finished...");\r
+}\r
+\r
+function checkOrientation()\r
+{\r
+ //updateScreen();\r
+ if (mode == 0)\r
+ updateHomescreen(); // check for screen rotation\r
+}\r
+\r
+function setUpdateTimer()\r
+{\r
+ updateTimer = window.setInterval('updateTimerCallback()', 1000 * 60 * config['updateDataInterval'].Value);\r
+}\r
+\r
+function clearUpdateTimer() \r
+{\r
+ window.clearInterval(updateTimer);\r
+}\r
+\r
+function updateTimerCallback()\r
+{\r
+ log("updateTimerCallback()");\r
+ updateData();\r
+}\r
\r
- if (useBackgroundImage) {\r
+function createMenu()\r
+{\r
+ window.menu.setLeftSoftkeyLabel("",null);\r
+ window.menu.setRightSoftkeyLabel("",null);\r
+ var id = 0;\r
+ var menuSettings = new MenuItem(getLocalizedText('menu.settings'), id++);\r
+ var menuCallApp = new MenuItem(getLocalizedText('menu.openCalendarApp'), id++);\r
+ var menuHelp = new MenuItem(getLocalizedText('menu.help'), id++);\r
+ var menuUpdate = new MenuItem(getLocalizedText('menu.update'), id++);\r
+ var menuAbout = new MenuItem(getLocalizedText('menu.about'), id++);\r
+ menuSettings.onSelect = showSettings;\r
+ menuAbout.onSelect = showAbout;\r
+ menuCallApp.onSelect = launchCalendar;\r
+ menuUpdate.onSelect = showUpdate;\r
+ menuHelp.onSelect = showHelp;\r
+ window.menu.clear();\r
+ window.menu.append(menuCallApp);\r
+ window.menu.append(menuSettings);\r
+ window.menu.append(menuHelp);\r
+ window.menu.append(menuUpdate);\r
+ window.menu.append(menuAbout); \r
+}\r
+\r
+function showSettings()\r
+{\r
+ mode = 2;\r
+ hideViews();\r
+ document.getElementById("settingsView").style.display = "block";\r
+ document.onclick = null;\r
+ \r
+ window.menu.setLeftSoftkeyLabel(getLocalizedText('settings.save'), function()\r
+ {\r
+ for (var key in config) {\r
+ if (config[key].Type == 'String')\r
+ config[key].Value = document.forms[0].elements["settings." + key].value;\r
+ else if (config[key].Type == 'Int') {\r
+ config[key].Value = parseInt(document.forms[0].elements["settings." + key].value);\r
+ if (config[key].Value < 0 || isNaN(config[key].Value))\r
+ config[key].Value = config[key].Default;\r
+ }\r
+ else if (config[key].Type == 'Bool')\r
+ config[key].Value = document.forms[0].elements["settings." + key].checked;\r
+ else if (config[key].Type == 'UID') {\r
+ config[key].Value = parseInt(document.forms[0].elements["settings." + key].value);\r
+ if (isNaN(config[key].Value))\r
+ config[key].Value = config[key].Default;\r
+ }\r
+ else if (config[key].Type == 'Enum') {\r
+ config[key].Value = document.forms[0].elements["settings." + key].value;\r
+ if (config[key].ValidValues.indexOf(config[key].Value) == -1)\r
+ config[key].Value = config[key].Default;\r
+ }\r
+ else if (config[key].Type == 'Array') {\r
+ if (key == 'excludedCalendars') {\r
+ config[key].Value = new Array();\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ var element = document.forms[0].elements["settings." + key + "." + calendarList[i]];\r
+ if (element != null && element.checked == false)\r
+ config[key].Value.push(calendarList[i]);\r
+ }\r
+ }\r
+ }\r
+ }\r
+ \r
+ updateCssClasses();\r
+ \r
+ saveSettings();\r
+ \r
+ mode = 1;\r
+ showFullscreen();\r
+ });\r
+ window.menu.setRightSoftkeyLabel(getLocalizedText('settings.cancel'), function()\r
+ {\r
+ mode = 1;\r
+ showFullscreen();\r
+ });\r
+ \r
+ var settingsHtml = '<form>';\r
+ for (var key in config) {\r
+ if (config[key].Type == 'String') {\r
+ var prefix = "";\r
+ if (key.substring(0,9) == "cssStyle_")\r
+ prefix = getLocalizedText('settings.cssStyle_prefix');\r
+ settingsHtml += '<table><tr><td>' + prefix + getLocalizedText('settings.name.' + key) + '<br /><input class="textInput" name="settings.' + key + '" type="text" value="' + config[key].Value + '" /></td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ }\r
+ else if (config[key].Type == 'Int')\r
+ settingsHtml += '<table><tr><td>' + getLocalizedText('settings.name.' + key) + '<br /><input class="textInput" name="settings.' + key + '" type="text" value="' + config[key].Value + '" /></td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ else if (config[key].Type == 'Bool')\r
+ settingsHtml += '<table><tr><td>' + getLocalizedText('settings.name.' + key) + '<br /><input name="settings.' + key + '" type="checkbox" value="true" ' + (config[key].Value ? 'checked="checked"' : '') + '/></td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ else if (config[key].Type == 'UID')\r
+ settingsHtml += '<table><tr><td>' + getLocalizedText('settings.name.' + key) + '<br /><input class="textInput" name="settings.' + key + '" type="text" value="0x' + config[key].Value.toString(16) + '" /></td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ else if (config[key].Type == 'Enum') {\r
+ settingsHtml += '<table><tr><td>' + getLocalizedText('settings.name.' + key) + '<br /><select name="settings.' + key + '" size="1">';\r
+ for(var i = 0; i < config[key].ValidValues.length; i++)\r
+ settingsHtml += '<option value="' + config[key].ValidValues[i] + '"' + (config[key].Value == config[key].ValidValues[i] ? ' selected="selected"' : '') + '>' + getLocalizedText('settings.validValues.' + key + '.' + config[key].ValidValues[i]) + '</option>';\r
+ settingsHtml += '</select></div></td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ }\r
+ else if (config[key].Type == 'Array') {\r
+ settingsHtml += '<table><tr><td>' + getLocalizedText('settings.name.' + key) + '<br />';\r
+ if (key == 'excludedCalendars') {\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ var checked = 'checked="checked"';\r
+ if (config[key].Value.indexOf(calendarList[i]) != -1)\r
+ checked = '';\r
+ settingsHtml += '<input name="settings.' + key + '.' + calendarList[i] + '" type="checkbox" value="' + calendarList[i] + '" ' + checked + '/> ' + calendarList[i] + '<br />';\r
+ }\r
+ }\r
+ settingsHtml += '</td>' + printHintBox(getLocalizedText('settings.info.' + key)) + '<hr />';\r
+ }\r
+ }\r
+ settingsHtml += '<input name="reset" type="button" value="' + getLocalizedText('settings.restoreDefaults') + '" onclick="javascript:restoreDefaultSettings();showSettings();" />';\r
+ settingsHtml += '</form>';\r
+ document.getElementById("settingsList").innerHTML = settingsHtml;\r
+}\r
+\r
+function changeCssClass(classname, properties)\r
+{\r
+ for(var i = 0; i < document.styleSheets[0]['cssRules'].length; i++)\r
+ {\r
+ if (document.styleSheets[0]['cssRules'][i].selectorText == classname) {\r
+ document.styleSheets[0].deleteRule(i);\r
+ document.styleSheets[0].insertRule(classname + ' { ' + properties + ' }', document.styleSheets[0]['cssRules'].length);\r
+ break;\r
+ }\r
+ }\r
+}\r
+\r
+function updateCssClasses()\r
+{\r
+ for(var key in config) {\r
+ changeCssClass(getLocalizedText('settings.name.' + key), config[key].Value);\r
+ }\r
+}\r
+\r
+function getSettingsCalEntryId()\r
+{\r
+ if (settingsCalEntryId == null) {\r
+ // check if entry already exists\r
+ var listFiltering = {\r
+ Type:'CalendarEntry', \r
+ Filter:{\r
+ StartRange: new Date(2000, 0, 1),\r
+ EndRange: new Date(2000, 0, 1),\r
+ SearchText: 'ComingNext Settings|',\r
+ Type: 'DayEvent'\r
+ }\r
+ }\r
+ var result = calendarService.IDataSource.GetList(listFiltering);\r
+ if (result.ErrorCode) {\r
+ error(result.ErrorMessage);\r
+ return;\r
+ }\r
+ var list = result.ReturnValue;\r
+ var entry = list.getNext();\r
+ if (entry != undefined) {\r
+ settingsCalEntryId = entry.LocalId;\r
+ log("settingsCalEntryId=" + settingsCalEntryId);\r
+ }\r
+ else { // create settings item\r
+ var item = new Object();\r
+ item.Type = "DayEvent";\r
+ item.StartTime = new Date(2000, 0, 1);\r
+ item.Summary = "ComingNext Settings|";\r
+ \r
+ var criteria = new Object();\r
+ criteria.Type = "CalendarEntry";\r
+ criteria.Item = item;\r
+\r
+ try {\r
+ var result = calendarService.IDataSource.Add(criteria);\r
+ if (result.ErrorCode)\r
+ error(result.ErrorMessage);\r
+ } catch (e) {\r
+ error("getSettingsCalEntryId: " + e + ', line ' + e.line);\r
+ }\r
+\r
+ getSettingsCalEntryId();\r
+ }\r
+ }\r
+}\r
+\r
+function restoreDefaultSettings()\r
+{\r
+ for (var key in config)\r
+ config[key].Value = config[key].Default;\r
+}\r
+\r
+function loadSettings()\r
+{\r
+ getSettingsCalEntryId();\r
+ var listFiltering = {\r
+ Type:'CalendarEntry', \r
+ Filter:{\r
+ LocalId: settingsCalEntryId\r
+ }\r
+ }\r
+ var result = calendarService.IDataSource.GetList(listFiltering);\r
+ if (result.ErrorCode) {\r
+ error(result.ErrorMessage);\r
+ return;\r
+ }\r
+ var entry = result.ReturnValue.getNext();\r
+ if (entry != undefined) {\r
+ log("Loading Settings...");\r
+ // only reload settings if they chanced since the last reload\r
+ if (settingsCache != entry.Summary)\r
+ {\r
+ restoreDefaultSettings();\r
+ var stringlist = entry.Summary.split("|");\r
+ // skip the first two entries, those contain header and version info\r
+ for(var i = 2; i < stringlist.length - 1; i++) {\r
+ var pair = stringlist[i].split('=');\r
+ var key = pair[0];\r
+ var value = pair[1];\r
+ log('stringlist: ' + key + '=\'' + value + '\'');\r
+ if (config[key].Type == 'Int') {\r
+ config[key].Value = Number(value);\r
+ if (isNaN(config[key].Value))\r
+ config[key].Value = config[key].Default;\r
+ }\r
+ else if (config[key].Type == 'String')\r
+ config[key].Value = value;\r
+ else if (config[key].Type == 'Bool')\r
+ config[key].Value = (value == 'true')\r
+ else if (config[key].Type == 'Enum')\r
+ config[key].Value = value;\r
+ else if (config[key].Type == 'UID') {\r
+ config[key].Value = Number(value);\r
+ if (isNaN(config[key].Value))\r
+ config[key].Value = config[key].Default;\r
+ }\r
+ else if (config[key].Type == 'Array') {\r
+ config[key].Value = value.split("^");\r
+ if (config[key].Value.length == 1 && config[key].Value[0] == "") {\r
+ config[key].Value = [];\r
+ }\r
+ }\r
+ }\r
+ settingsCache = entry.Summary;\r
+ updateCssClasses();\r
+ }\r
+ else {\r
+ log("Settings already cached and did not change");\r
+ }\r
+ }\r
+ else {\r
+ error("Failed to load settings, calendar entry could not be found");\r
+ }\r
+}\r
+\r
+function saveSettings()\r
+{\r
+ getSettingsCalEntryId();\r
+ var item = new Object();\r
+ item.Type = "DayEvent";\r
+ item.StartTime = new Date(2000, 0, 1);\r
+ item.LocalId = settingsCalEntryId;\r
+ item.Summary = "ComingNext Settings|" + version + "|";\r
+\r
+ for (var key in config) {\r
+ if (config[key].Type == 'Int')\r
+ item.Summary += key + "=" + config[key].Value.toString() + "|";\r
+ else if (config[key].Type == 'String')\r
+ item.Summary += key + "=" + config[key].Value + "|";\r
+ else if (config[key].Type == 'Bool')\r
+ item.Summary += key + "=" + (config[key].Value ? 'true' : 'false') + "|";\r
+ else if (config[key].Type == 'Enum')\r
+ item.Summary += key + "=" + config[key].Value + "|";\r
+ else if (config[key].Type == 'UID')\r
+ item.Summary += key + "=" + config[key].Value.toString() + "|";\r
+ else if (config[key].Type == 'Array')\r
+ item.Summary += key + "=" + config[key].Value.join("^") + "|";\r
+ }\r
+ settingsCache = item.Summary;\r
+ \r
+ var criteria = new Object();\r
+ criteria.Type = "CalendarEntry";\r
+ criteria.Item = item;\r
+\r
+ log("Saving settings to calendar entry: " + item.Summary);\r
+ try {\r
+ var result = calendarService.IDataSource.Add(criteria);\r
+ if (result.ErrorCode)\r
+ error(result.ErrorMessage);\r
+ } catch (e) {\r
+ error("saveSettings: " + e + ', line ' + e.line);\r
+ }\r
+ \r
+ lastReloadTime = null; // force calendar data reload on next update\r
+ clearUpdateTimer();\r
+ setUpdateTimer();\r
+}\r
+\r
+function toggleVisibility(elementId)\r
+{\r
+ if (document.getElementById(elementId).style.display == "none")\r
+ document.getElementById(elementId).style.display = "block";\r
+ else\r
+ document.getElementById(elementId).style.display = "none";\r
+}\r
+\r
+var uniqueId = 0;\r
+function printHintBox(text)\r
+{\r
+ uniqueId++;\r
+ return '<td width="1%" align="right" onclick="javascript:toggleVisibility(\'info' + uniqueId + '\')">' + getLocalizedText('settings.help') + '</td></tr></table>'+\r
+ '<div class="settingsInfo" id="info' + uniqueId + '">' + text + '</div>';\r
+}\r
+\r
+function showAbout()\r
+{\r
+ mode = 3;\r
+ hideViews();\r
+ document.getElementById("aboutView").style.display = "block";\r
+ document.onclick = null;\r
+ \r
+ window.menu.setLeftSoftkeyLabel(" ", function(){});\r
+ window.menu.setRightSoftkeyLabel(getLocalizedText('softkey.back'), function()\r
+ {\r
+ mode = 1;\r
+ showFullscreen();\r
+ });\r
+ \r
+ //document.getElementById("aboutView").innerHTML = 'aboutView';\r
+ document.getElementById("name").innerHTML = "Coming Next " + version;\r
+}\r
+\r
+function showHelp() {\r
+ widget.openURL('http://comingnext.sf.net/help');\r
+}\r
+\r
+function updateFullscreen()\r
+{\r
+}\r
+\r
+function showFullscreen()\r
+{\r
+ log("showFullscreen()");\r
+ hideViews();\r
+ document.getElementById("fullscreenView").style.display = "block";\r
+ document.getElementById('body').className = "backgroundFullscreen";\r
+ if (!errorOccured)\r
+ document.onclick = launchCalendar;\r
+ createMenu();\r
+ updateData();\r
+}\r
+\r
+function getBackgroundImage()\r
+{\r
+ if (errorOccured)\r
+ return '';\r
+ var bgImage;\r
+ if (config['backgroundImageLocation'].Value == config['backgroundImageLocation'].ValidValues[0]) // internal\r
+ bgImage = 'background_' + orientation + '.png';\r
+ else\r
+ bgImage = 'C:/Data/background_' + panelNum + '_' + orientation + '.png';\r
+ return bgImage;\r
+}\r
+\r
+function updateHomescreen()\r
+{\r
+ if (config['useBackgroundImage'].Value) {\r
// check for screen rotation\r
if (orientation != 'portrait' && screen.width == 360 && screen.height == 640) {\r
window.widget.prepareForTransition("fade");\r
orientation = 'portrait';\r
- document.getElementById('body').style.backgroundImage = 'url(background_' + orientation + '.png)';\r
+ document.getElementById('body').style.backgroundImage = 'url(' + getBackgroundImage() + ')';\r
document.getElementById('body').style.backgroundColor = 'none';\r
window.widget.performTransition();\r
} else if (orientation != 'landscape' && screen.width == 640 && screen.height == 360) {\r
window.widget.prepareForTransition("fade");\r
orientation = 'landscape';\r
- document.getElementById('body').style.backgroundImage = 'url(background_' + orientation + '.png)';\r
+ document.getElementById('body').style.backgroundImage = 'url(' + getBackgroundImage() + ')';\r
document.getElementById('body').style.backgroundColor = 'none';\r
window.widget.performTransition();\r
}\r
+ else if (document.getElementById('body').style.backgroundImage == "")\r
+ {\r
+ document.getElementById('body').style.backgroundImage = 'url(' + getBackgroundImage() + ')';\r
+ }\r
}\r
}\r
\r
-function launchCalendar()\r
+function showHomescreen()\r
{\r
- try {\r
- widget.openApplication(calendarApp, "");\r
- window.close();\r
- } catch(e) {\r
- error('starting Calendar App');\r
- return;\r
+ log("showHomescreen()");\r
+ hideViews();\r
+ document.getElementById("homescreenView").style.display = "block";\r
+ document.getElementById('body').className = "background";\r
+ document.onclick = null;\r
+ updateData();\r
+}\r
+\r
+function getLocalizedText(p_Txt)\r
+{\r
+ if (localizedText[p_Txt])\r
+ return localizedText[p_Txt];\r
+ else \r
+ return 'ERROR: missing translation for ' + p_Txt;\r
+}\r
+\r
+function showUpdate()\r
+{\r
+ mode = 4;\r
+ hideViews();\r
+ document.getElementById("updateView").style.display = "block";\r
+ document.onclick = null;\r
+ \r
+ window.menu.setLeftSoftkeyLabel(getLocalizedText('update.checknow'), function(){\r
+ checkForUpdate();\r
+ });\r
+ window.menu.setRightSoftkeyLabel(getLocalizedText('softkey.back'), function()\r
+ {\r
+ mode = 1;\r
+ showFullscreen();\r
+ });\r
+ \r
+ document.getElementById("currentVersion").innerHTML = getLocalizedText("update.current") + version;\r
+ checkForUpdate();\r
+}\r
+\r
+function checkForUpdate()\r
+{\r
+ // asynch XHR to server url\r
+ reqV = new XMLHttpRequest();\r
+ reqV.onreadystatechange = checkForUpdateCallback;\r
+ document.getElementById("updateDiv").innerHTML = getLocalizedText("update.checking");\r
+ reqV.open("GET", versionURL, true);\r
+ reqV.setRequestHeader( "If-Modified-Since", "Sat, 1 Jan 2000 00:00:00 GMT" ); // disable caching\r
+ reqV.send(null);\r
+}\r
+\r
+function checkForUpdateCallback()\r
+{ \r
+ if (reqV.readyState == 4) {\r
+ if (reqV.status == 200) {\r
+ var resultXml = reqV.responseText;\r
+ if (resultXml) {\r
+ var div = document.getElementById("tmp");\r
+ div.innerHTML = resultXml;\r
+ var newVersion = div.getElementsByTagName('version')[0].innerHTML;\r
+ var newVersionURL = div.getElementsByTagName('url')[0].innerHTML;\r
+ div.innerHTML = "";\r
+ if (version != newVersion) {\r
+ document.getElementById("updateDiv").innerHTML = getLocalizedText("update.download").replace(/%1/, newVersion).replace(/%2/, newVersionURL);\r
+ }\r
+ else {\r
+ document.getElementById("updateDiv").innerHTML = getLocalizedText("update.nonewversion");\r
+ }\r
+ }\r
+ }\r
+ else {\r
+ document.getElementById("updateDiv").innerHTML = getLocalizedText("update.error") + reqV.status + " " + reqV.responseText;\r
+ }\r
}\r
}\r
\r
-function init()\r
+function hideViews()\r
{\r
+ document.getElementById("homescreenView").style.display = "none";\r
+ document.getElementById("fullscreenView").style.display = "none";\r
+ document.getElementById("aboutView").style.display = "none";\r
+ document.getElementById("settingsView").style.display = "none";\r
+ document.getElementById("updateView").style.display = "none";\r
+}\r
+\r
+function listCalendars()\r
+{\r
+ if (errorOccured) {\r
+ return null;\r
+ }\r
+\r
try {\r
- // call calendar service\r
- calendarService = device.getServiceObject("Service.Calendar", "IDataSource");\r
+ var criteria = {\r
+ Type:'Calendar', \r
+ Filter:{\r
+ DefaultCalendar: false\r
+ }\r
+ }\r
+ \r
+ var calendarsResult = calendarService.IDataSource.GetList(criteria);\r
+ if (calendarsResult.ErrorCode != 0)\r
+ throw("Error fetching list of calendars: " + calendarsResult.ErrorCode + ': ' + calendarsResult.ErrorMessage);\r
+ var calendarListIterator = calendarsResult.ReturnValue;\r
+ \r
+ var calendars = [];\r
+ var count = 0;\r
+ var item;\r
+ while (( item = calendarListIterator.getNext()) != undefined ) {\r
+ calendars[count++] = item;\r
+ }\r
+ log("Available Calendars: " + calendars.join(", "));\r
+ return calendars;\r
} catch(e) {\r
- error('loading Calendar service');\r
- return;\r
+ error('listing calendars:' + e + ', line ' + e.line);\r
+ return null;\r
}\r
+}\r
\r
- collectLocales();\r
- updateData();\r
- requestNotification();\r
- window.setInterval('updateData()', 1000 * 60 * updateDataInterval);\r
+// Copies all objects and their properties to an array. Data is copied so nothing gets lost when the reference is removed\r
+// Note: this will also set the "CalendarName" property of every entry to the name specified by the calendarName parameter if it has not been defined already\r
+function listToArray(list, calendarName)\r
+{\r
+ var array = new Array();\r
+ var item;\r
+ var txt = "";\r
+ while (( item = list.getNext()) != undefined ) {\r
+ var itemCopy = new Object();\r
+ for(var i=0; i < entryFields.length; i++) {\r
+ itemCopy[entryFields[i]] = item[entryFields[i]];\r
+ }\r
+ // for some reason, the CalendarName property is never correctly queried, so we assign it manually here\r
+ if (!itemCopy['CalendarName']) {\r
+ itemCopy['CalendarName'] = calendarName;\r
+ }\r
+ array.push(itemCopy);\r
+ txt += array[array.length - 1].Summary + ", ";\r
+ }\r
+ log("listToArray(): " + txt);\r
+ return array;\r
+}\r
\r
- updateScreen();\r
- if (useBackgroundImage)\r
- // check for screen rotation every 3 secs\r
- window.setInterval('updateScreen()', 1000 * 3);\r
+function sortCalendarEntries(a, b)\r
+{\r
+ var atime, btime;\r
+ log("sortCalendarEntries(" + a.Summary + "," + b.Summary + ")");\r
+ \r
+ if (a.InstanceStartTime != null) {\r
+ atime = a.InstanceStartTime;\r
+ }\r
+ else if (a.StartTime != null) {\r
+ atime = a.StartTime;\r
+ }\r
+ else if (a.InstanceEndTime != null) {\r
+ atime = a.InstanceEndTime;\r
+ }\r
+ else if (a.EndTime != null) {\r
+ atime = a.EndTime;\r
+ }\r
+ \r
+ if (b.InstanceStartTime != null) {\r
+ btime = b.InstanceStartTime;\r
+ }\r
+ else if (b.StartTime != null) {\r
+ btime = b.StartTime;\r
+ }\r
+ else if (b.InstanceEndTime != null) {\r
+ btime = b.InstanceEndTime;\r
+ }\r
+ else if (b.EndTime != null) {\r
+ btime = b.EndTime;\r
+ }\r
+ \r
+ if (atime && btime) {\r
+ \r
+ atime = parseDate(atime);\r
+ btime = parseDate(btime);\r
+ \r
+ // sort by date & time\r
+ if (atime < btime) {\r
+ return -1;\r
+ }\r
+ else if (atime > btime) {\r
+ return 1;\r
+ }\r
+ // sort by type\r
+ else if (a.Type != b.Type) {\r
+ if (a.Type < b.Type) {\r
+ return -1;\r
+ }\r
+ else if (a.Type > b.Type) {\r
+ return 1;\r
+ }\r
+ }\r
+ // sort by description\r
+ else if (a.Summary && b.Summary && a.Summary != b.Summary) {\r
+ if (a.Summary < b.Summary) {\r
+ return -1;\r
+ }\r
+ else if (a.Summary > b.Summary) {\r
+ return 1;\r
+ }\r
+ }\r
+ }\r
+ // NOTE: events my have no date information at all. In that case, we list events without date first\r
+ else if (atime && !btime) {\r
+ return 1;\r
+ }\r
+ else if (!atime && btime) {\r
+ return -1;\r
+ }\r
+ else if (!atime && !btime) {\r
+ // sort by type\r
+ if (a.Type != b.Type) {\r
+ if (a.Type < b.Type) {\r
+ return -1;\r
+ }\r
+ else if (a.Type > b.Type) {\r
+ return 1;\r
+ }\r
+ }\r
+ // sort by description\r
+ else if (a.Summary && b.Summary && a.Summary != b.Summary) {\r
+ if (a.Summary < b.Summary) {\r
+ return -1;\r
+ }\r
+ else if (a.Summary > b.Summary) {\r
+ return 1;\r
+ }\r
+ }\r
+ }\r
+\r
+ return 0;\r
+}\r
+\r
+function updateCalendarColors()\r
+{\r
+ var maxColors = 6;\r
+ calendarColors = [];\r
+ if (calendarList.length > maxColors) {\r
+ log("updateCalendarColors(): Warning: more calendars than available indicator colors");\r
+ }\r
+ for(var i=0; i < calendarList.length; i++) {\r
+ calendarColors[calendarList[i]] = (i % maxColors) + 1;\r
+ }\r
+}\r
+\r
+function log(message) {\r
+ if (config['enableLogging'].Value) {\r
+ console.info(message);\r
+ }\r
}\r
\r
</script>\r
\r
<style type="text/css">\r
+a { color:#aaccff }\r
table { margin:0px; padding:0px; border-spacing:0px; }\r
td { padding:0px 5px 0px 0px; white-space:nowrap; overflow:hidden; }\r
+hr { color:#ffffff; background-color:#ffffff; height:1px; text-align:left; border-style:none; }\r
+.settingsInfo { display:none; font-style:italic; }\r
+.title { font-weight:bold; font-size:14pt; }\r
+.textInput { width:90%; }\r
+.credits { margin-left:40px; text-indent: -20px; margin-bottom:0px; }\r
#homescreenView { width: 315px; height:91px; overflow:hidden; }\r
-#calendarList { position:absolute; left:10px; top:4px; width:295px; height:75px; overflow:hidden; }\r
+#calendarList { position:absolute; left:5px; top:4px; width:295px; height:75px; overflow:hidden; }\r
+#name { text-align:center; }\r
+#appicon { display: block; margin-left: auto; margin-right: auto; margin-top: 10px; }\r
+#smallappicon { width:22px; height:22px; margin-right:10px; float:left; }\r
</style>\r
\r
</head>\r
\r
-<body id="body" class="background">\r
+<body onload="javascript:setTimeout('init()', 10)" onresize="javascript:updateScreen()" id="body" class="background">\r
<div id="homescreenView">\r
- <div id="calendarList"></div>\r
+ <div id="calendarList">loading...</div>\r
+</div>\r
+<div id="fullscreenView" style="display:none;">\r
+ <img src="Icon.png" id="smallappicon">\r
+ <h1 class="title">Coming Next</h1>\r
+ <hr />\r
+ <div id="fullscreenCalendarList">loading...</div>\r
+</div>\r
+<div id="settingsView" style="display:none">\r
+ <img src="Icon.png" id="smallappicon">\r
+ <h1 id="settingsTitle" class="title">Settings</h1>\r
+ <hr />\r
+ <div id="settingsList"></div>\r
+</div>\r
+<div id="aboutView" style="display:none">\r
+ <img src="Icon.png" id="appicon">\r
+ <h1 id="name">Coming Next</h1>\r
+ <hr />\r
+ <p>Created by Dr. Cochambre and Michael Prager.</p>\r
+ <p>Contributions:</p>\r
+ <p class="credits">Paul Moore (bug fixes, new features and code cleanup)</p>\r
+ <p class="credits">Manfred Hanselmann (DST support)</p>\r
+ <p class="credits">Christophe Milsent (translation support & french translation)</p>\r
+ <p class="credits">Flavio Nathan (portuguese-brazilian translation)</p>\r
+ <p class="credits">Tokeda (russian translation)</p>\r
+ <p class="credits">Marcella Ferrari (italian translation)</p>\r
+ <p class="credits">Venos (italian translation)</p>\r
+ <p>This software is open source and licensed under the GPLv3.</p>\r
+ <p>Visit <a onclick="widget.openURL('http://comingnext.sf.net/'); return false;" href="http://comingnext.sf.net/">comingnext.sf.net</a> for free updates.</p>\r
+ <hr />\r
+</div>\r
+<div id="updateView" style="display:none">\r
+ <img src="Icon.png" id="smallappicon">\r
+ <h1 class="title">Check for update</h1>\r
+ <hr />\r
+ <div id="currentVersion">Coming Next ??</div>\r
+ <div id="updateDiv"></div>\r
+ <div id="tmp" style="display:none;"></div>\r
</div>\r
</body>\r
\r