Created
May 12, 2015 13:12
-
-
Save VenkataRaju/623e99544d7a8414c45e to your computer and use it in GitHub Desktop.
Simple reminder application written in HTML (Makes use of Local Storage API)
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<html> | |
<head> | |
<meta charset="UTF-8"> | |
<title>Task Tracker</title> | |
<!-- Version: 0.5 --> | |
<style> | |
table { | |
border-collapse: collapse; | |
} | |
.roundedCorner { | |
border: #AAAAAA solid; | |
border-radius: 4px; | |
display: inline-block; | |
padding: 4px; | |
} | |
.errorBorder { | |
border-color: red; | |
} | |
.trackerTbl tbody tr:nth-child(odd) { | |
background-color: #eee; | |
} | |
.trackerTbl { | |
border-width: 1px; | |
border-style: solid; | |
width: 90%; | |
min-width: 1000px; | |
} | |
.th0 { | |
width: 2%; | |
} | |
.th1 { | |
width: 61%; | |
} | |
.th2 { | |
width: 25%; | |
} | |
.th3 { | |
width: 6%; | |
} | |
.timeExpired { | |
color: #DD0000; | |
font-weight: bold; | |
} | |
</style> | |
<script> | |
const PAGE_TITLE = document.title; | |
const ALART_SOUND = "data:AUDIO/mpeg;base64,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"; | |
const AUDIO = new Audio(ALART_SOUND); | |
AUDIO.loop = true; | |
const NOTIFICAION_IMAGE = ""; | |
var tbody, templateRow; | |
var taskTF, infLbl, errorLbl; | |
var infTimer, updateTimer; | |
var userChosenVolume = 0.5; | |
// Contains {taskName: "Task Name", expirationTime: 3232323, expirationIndicated: true} objects | |
var tasks = [], focussed = true; | |
function $(id) document.getElementById(id); | |
function init() | |
{ | |
templateRow = $("templateRow"); | |
templateRow.removeAttribute("id"); | |
tbody = templateRow.parentNode; | |
tbody.removeChild(templateRow); | |
templateRow.style.display = "table-row"; | |
taskTF = $("taskTF"); | |
infLbl = $("infLbl"); | |
errorLbl = $("errorLbl"); | |
if (!($("volumeTF").disabled = $("playMusicCB").disabled = AUDIO.error !== null)) | |
updateVolume($("volumeTF")); | |
// localStorage["TaskTracker"] = null; | |
addEventListener("beforeunload", () => { localStorage["TaskTracker"] = JSON.stringify(tasks); }); | |
var tasksFromStorage = JSON.parse(localStorage["TaskTracker"]); | |
if (tasksFromStorage && tasksFromStorage.length) | |
{ | |
tasks = tasksFromStorage; | |
for (var i = 0; i < tasks.length; i++) | |
addRow(tasks[i]); | |
restartUpdateTimer(); | |
} | |
} | |
function updateVolume(tf) | |
{ | |
var volume = +tf.value; | |
if(isNaN(volume) || volume < 0 || volume > 20) | |
{ | |
tf.className = "errorBorder"; | |
return; | |
} | |
tf.className = ""; | |
userChosenVolume = volume / 20; | |
tf.nextElementSibling.textContent = volume + "/20"; | |
} | |
function add() | |
{ | |
clearTimeout(infTimer); | |
infLbl.textContent = errorLbl.textContent = ""; | |
var taskName = taskTF.value.trim(); | |
if (!taskName) | |
{ | |
errorLbl.textContent = "Task name is required"; | |
taskTF.value = ""; | |
taskTF.focus(); | |
return; | |
} | |
var expirationTime = +(+$("hTimeIntervalTF").value * 60 * 60) + (+$("mTimeIntervalTF").value * 60) + (+$("sTimeIntervalTF").value); | |
$("hTimeIntervalTF").value = $("mTimeIntervalTF").value = $("sTimeIntervalTF").value = ""; | |
if (isNaN(expirationTime) || expirationTime < 1) | |
{ | |
errorLbl.textContent = ($("hTimeIntervalTF").value + $("mTimeIntervalTF").value + $("sTimeIntervalTF").value).trim().length === 0 | |
? "Time interval is required" | |
: "Invalid time interval"; | |
$("sTimeIntervalTF").focus(); | |
return; | |
} | |
taskTF.value = ""; | |
taskTF.focus(); | |
expirationTime += millisToSeconds(Date.now()); | |
var task = | |
{ | |
taskName, | |
expirationTime, | |
expirationIndicated: false | |
}; | |
for (var i = 0; i < tasks.length; i++) | |
{ | |
if (tasks[i].taskName !== taskName) | |
continue; | |
tasks[i] = task; | |
tbody.rows[i].classList.remove("timeExpired"); | |
infLbl.textContent = "Reset Task at row no: " + (i + 1); | |
infTimer = setTimeout(() => { infLbl.textContent = ""; }, 4000); | |
restartUpdateTimer(); | |
return; | |
} | |
// Need to add new row | |
tasks.push(task); | |
addRow(task); | |
restartUpdateTimer(); | |
} | |
function millisToSeconds(timeInMillis) ~~(timeInMillis/1000); | |
function addRow(task) | |
{ | |
var clone = templateRow.cloneNode(true); | |
tbody.appendChild(clone); | |
clone.cells[0].textContent = tbody.childElementCount; | |
clone.cells[1].textContent = task.taskName; | |
if(task.expirationTime <= millisToSeconds(Date.now())) | |
clone.classList.add("timeExpired"); | |
} | |
function restartUpdateTimer() | |
{ | |
clearTimeout(updateTimer); | |
update(); // Immediate refresh | |
updateTimer = setInterval(update, 1000); | |
function update() | |
{ | |
var now = millisToSeconds(Date.now()); | |
for (var i = 0, diff, rt; i < tasks.length; i++) | |
{ | |
diff = tasks[i].expirationTime - now; | |
rt = getReadableTime(Math.abs(diff)); | |
if (diff > 0) | |
{ | |
tbody.rows[i].cells[2].textContent = rt; | |
continue; | |
} | |
tbody.rows[i].cells[2].textContent = "Expired " + rt + " ago"; | |
if (!tasks[i].expirationIndicated) | |
{ | |
if (taskTF.value.length === 0) | |
{ | |
taskTF.value = tbody.rows[i].cells[1].textContent; | |
taskTF.select(); | |
} | |
tbody.rows[i].classList.add("timeExpired"); | |
if (!focussed) | |
notifyUser("Time expired for Task", tbody.rows[i].cells[1].textContent); | |
tasks[i].expirationIndicated = true; | |
} | |
} | |
} | |
} | |
function removeRow(el) | |
{ | |
var rowToBeRemoved = el.parentNode.parentNode; | |
for (var i = 0, rowNo = 1, rows = tbody.rows, len = rows.length; i < len;) | |
{ | |
var row = rows[i]; | |
row.cells[0].textContent = rowNo; | |
if (row == rowToBeRemoved) | |
{ | |
if (row.cells[1].textContent == taskTF.value) | |
taskTF.value = ""; | |
tbody.removeChild(row); | |
tasks.splice(i, 1); | |
if (len == 1) | |
clearInterval(updateTimer); | |
len--; | |
continue; // Not breaking as we have to fix the remaining row numbers | |
} | |
i++; | |
rowNo++; | |
} | |
} | |
function edit(el) | |
{ | |
taskTF.value = el.parentNode.parentNode.cells[1].textContent; | |
$("sTimeIntervalTF").focus(); | |
} | |
const getReadableTime = (function() | |
{ | |
const MINUTE_IN_SECONDS = 60, HOUR_IN_SECONDS = MINUTE_IN_SECONDS * 60; | |
return function(timeInSecs) | |
{ | |
var timeStr = ""; | |
var time = ~~(timeInSecs / HOUR_IN_SECONDS); | |
if (time > 0) | |
{ | |
timeStr = twoDigit(time) + "h:"; | |
timeInSecs %= HOUR_IN_SECONDS; | |
} | |
time = ~~(timeInSecs / MINUTE_IN_SECONDS); | |
if (timeStr || time > 0) | |
{ | |
timeStr += twoDigit(time) + "m:"; | |
timeInSecs %= MINUTE_IN_SECONDS; | |
} | |
return timeStr + twoDigit(timeInSecs) + "s"; | |
} | |
function twoDigit(num) (num <= 9 ? "0" : "") + num; | |
})(); | |
const notifyUser = (function() | |
{ | |
var timer, audioTimer; | |
return function(title, msg) | |
{ | |
if ($("desktopNotificationRadio").checked) | |
{ | |
if (Notification.permission === "granted") | |
new Notification(title, {icon: NOTIFICAION_IMAGE, body: msg}); | |
else if (Notification.permission !== "denied") | |
Notification.requestPermission(function (permission) | |
{ | |
if (permission === "granted") | |
new Notification(title, {icon: NOTIFICAION_IMAGE, body: msg}); | |
}); | |
} | |
else if ($("popupNotificationRadio").checked) | |
alert(msg); | |
else if($("titleChangeNotificationRadio").checked) | |
{ | |
var i = 0, t = msg + " "; | |
clearInterval(timer); | |
timer = setInterval(function() | |
{ | |
if (focussed) | |
{ | |
clearInterval(timer); | |
document.title = PAGE_TITLE; | |
} | |
else | |
{ | |
if (i === t.length) | |
i = 0; | |
document.title = t.substr(i) + t.substr(0, i++); | |
} | |
}, 500); | |
} | |
if ($("playMusicCB").checked) | |
{ | |
clearInterval(audioTimer); | |
AUDIO.volume = userChosenVolume; | |
AUDIO.play(); | |
const endTime = Date.now() + 15000; | |
audioTimer = setInterval(function() | |
{ | |
if (focussed || Date.now() > endTime) | |
{ | |
AUDIO.pause(); | |
clearInterval(audioTimer); | |
} | |
}, 500); | |
} | |
} | |
})(); | |
</script> | |
</head> | |
<body onload="init()" onfocus="focussed = true" onblur="focussed = false"> | |
<form onsubmit="add(); return false;" style="margin-top: 2%;"> | |
<table align="center"> | |
<tr> | |
<td class="align"> | |
<label>Task Name: </label> | |
</td> | |
<td> | |
<input id="taskTF" type="text" size="70" autofocus /> | |
</td> | |
</tr> | |
<tr> | |
<td class="align"> | |
<label>Time Interval: </label> | |
</td> | |
<td> | |
<input id="sTimeIntervalTF" type="text" size="4" title="Seconds" /> s | |
<input id="mTimeIntervalTF" type="text" size="4" title="Minutes" /> m | |
<input id="hTimeIntervalTF" type="text" size="4" title="Hours" /> h | |
</td> | |
</tr> | |
<tr> | |
<td> | |
</td> | |
<td> | |
<label id="infLbl" style="float: left; color: green; font-weight: bold"></label> | |
<label id="errorLbl" style="float: left; color: red; font-weight: bold"></label> | |
</td> | |
</tr> | |
<tr> | |
<td colspan="2" align="center" style="padding-top: 4px"> | |
<input type="submit" style="width: 30%" value="Add" /> | |
</td> | |
</tr> | |
</table> | |
</form> | |
<br /> | |
<table align="center"> | |
<tr> | |
<td> | |
<table> | |
<tr> | |
<td> | |
<form onsubmit="return false;"> | |
<fieldset class="roundedCorner"> | |
<legend style="font-weight: bold;">If the timer expires when i'm in another page</legend> | |
<table align="left" style="margin: 10px;"> | |
<tr> | |
<td> | |
<input type="radio" id="desktopNotificationRadio" name="notificationType" checked /> | |
<label for="desktopNotificationRadio">Desktop Notification</label> | |
</td> | |
</tr> | |
<tr> | |
<td> | |
<input type="radio" id="popupNotificationRadio" name="notificationType" /> | |
<label for="popupNotificationRadio">Show Popup Alert</label> | |
</td> | |
</tr> | |
<tr> | |
<td> | |
<input type="radio" id="titleChangeNotificationRadio" name="notificationType" /> | |
<label for="titleChangeNotificationRadio">Scroll Page Title</label> | |
</td> | |
</tr> | |
<tr> | |
<td> | |
<input id="playMusicCB" type="checkbox" checked style="margin-top: 2px" | |
onchange='$("volumeTF").disabled=!$( "playMusicCB").checked;' /> | |
<label for="playMusicCB">Play music</label> | |
</td> | |
</tr> | |
<tr> | |
<td> | |
<label for="volumeTF">Volume: </label> | |
<input id="volumeTF" type="range" onchange="updateVolume(this)" onkeyup="updateVolume(this)" value="4" | |
min="0" max="20" title="0 to 20" /> | |
<label style="font-size:small"></label> | |
</td> | |
</tr> | |
</table> | |
</fieldset> | |
</form> | |
</td> | |
</tr> | |
</table> | |
</td> | |
</tr> | |
</table> | |
<br /> | |
<table align="center" border="1" class="trackerTbl"> | |
<thead> | |
<th class="th0">No.</th> | |
<th class="th1">Task Name</th> | |
<th class="th2">Time Left</th> | |
<th class="th3">Remove</th> | |
<th class="th3">Edit</th> | |
</thead> | |
<tbody> | |
<tr id="templateRow" style="display: none;"> | |
<td></td> | |
<td></td> | |
<td style="font-size: 1.2em; font-family: monospace;"></td> | |
<td align="center"> | |
<button onclick="removeRow(this)">Remove</button> | |
</td> | |
<td align="center"> | |
<button onclick="edit(this)">Edit</button> | |
</td> | |
</tr> | |
</tbody> | |
</table> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment