Skip to content

Instantly share code, notes, and snippets.

@VenkataRaju
Created May 12, 2015 13:12
Show Gist options
  • Save VenkataRaju/623e99544d7a8414c45e to your computer and use it in GitHub Desktop.
Save VenkataRaju/623e99544d7a8414c45e to your computer and use it in GitHub Desktop.
Simple reminder application written in HTML (Makes use of Local Storage API)
<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" />&nbsp;s&nbsp;&nbsp;
<input id="mTimeIntervalTF" type="text" size="4" title="Minutes" />&nbsp;m&nbsp;&nbsp;
<input id="hTimeIntervalTF" type="text" size="4" title="Hours" />&nbsp;h&nbsp;&nbsp;
</td>
</tr>
<tr>
<td>&nbsp;
</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