Skip to content

Instantly share code, notes, and snippets.

Show Gist options
  • Save kuddl/d999cefb73d7132be637 to your computer and use it in GitHub Desktop.
Save kuddl/d999cefb73d7132be637 to your computer and use it in GitHub Desktop.

Creating the QLOCKTWO in HTML5, CSS and Javascript

Just a little side project to recreate the QLOCKTWO. I had an old version using canvas and images but it was ugly. This is plain HTML, CSS and javascript. The aim of this rewrite was to create an interface for creating multiply languages.

Added the option to change the color of the clock.

A Pen by Fabrice Weinberg on CodePen.

License.

<div id="clock">
<span id="e0">●</span>
<span id="e1">●</span>
<span id="e2">●</span>
<span id="e3">●</span>
<div id="letters">
</div>
</div>
<div id="settings">
<div class="button selected" id="darkblue"></div>
<div class="button" id="darkred"></div>
<div class="button" id="green"></div>
<div class="button" id="purple"></div>
<div class="button" id="orange"></div>
<p><a href="#" id="english">en</a> | <a href="#" id="german">de</a> | <a href="#" id="french">fr</a> | <a href="#" id="italian">it</a> | <a href="#" id="spanish">es</a></p>
<p><a href="#" id="state">show seconds</a></p>
</div>
<div id="desc">
<h1>QLOCKTWO idea from <a href="http://qlocktwo.com/">Biegert & Funk</a></h1>
<p>Build in pure html, css and javascript. No image were used. It supports five languages (english, german, french, italian and spanish) and you can change the color.</p>
</div>
/*
2012 by Fabrice Weinberg
fabrice.weinberg.me
Thanks to:
Hugo Giraudel (@HugoGiraudel) for the french translation.
Kseso (@Kseso) for the spanish translation.
*/
var language = {};
language['spanish'] = {
map : [["E", "S", "O", "N", "F", "L", "A", "S", "U", "N", "A"],
["D", "O", "S", "T", "R", "E", "S", "O", "C", "H", "O"],
["C", "U", "A", "T", "R", "O", "C", "I", "N", "C", "O"],
["S", "I", "E", "T", "E", "R", "N", "U", "E", "V", "E"],
["O", "N", "C", "E", "V", "D", "I", "E", "Z", "F", "G"],
["S", "E", "I", "S", "D", "O", "C", "E", "Y", "E", "N"],
["P", "U", "N", "T", "O", "G", "M", "E", "N", "O", "S"],
["M", "E", "D", "I", "A", "C", "U", "A", "R", "T", "O"],
["D", "I", "E", "Z", "L", "V", "E", "I", "N", "T", "E"],
["V", "E", "I", "N", "T", "I", "C", "I", "N", "C", "O"]],
sentences : {
start : { all : [[0,1,3],[0,5,7]], 1 : [[0,0,1], [0, 5, 6]] },
end : { all : [[5, 9, 10], [6,0,4]] },
hours : {
0 : [[[5,4,7]]],
1 : [[[0,8,10]]],
2 : [[[1,0,2]]],
3 : [[[1,3,6]]],
4 : [[[2,0,5]]],
5 : [[[2,6,10]]],
6 : [[[5,0,3]]],
7 : [[[3,0,4]]],
8 : [[[1,7,10]]],
9 : [[[3,6,10]]],
10: [[[4,5,8]]],
11: [[[4,0,3]]],
12: [[[5,4,7]]]
},
minutes :{
5 : [[5,8], [9,6,10]],
10 : [[5,8], [8, 0, 3]],
15 : [[5,8], [7, 5, 10]],
20 : [[5,8], [8, 5, 10]],
25 : [[5,8], [9, 0, 10]],
30 : [[5,8], [7,0,4]],
35 : [[6,6,10], [9, 0, 10]],
40 : [[6,6,10], [8, 5, 10] ],
45 : [[6,6,10], [7, 5, 10]],
50 : [[6,6,10], [8, 0, 3]],
55 : [[6,6,10], [9,6,10]]
},
incrementHourAt : 30
}
};
language['italian'] = {
map : [["S","O","N","O","E","L","E","H","I","J","K"],
["È","M","L’","U","N","A","S","D","U","E","B"],
["T","R","E","N","O","V","E","O","T","T","O"],
["S","E","T","T","E","C","I","N","Q","U","E"],
["D","O","D","I","C","B","D","I","E","C","I"],
["Q","U","A","T","T","R","O","F","S","E","I"],
["U","N","D","I","C","I","F","M","E","N","O"],
["E","K","U","N","L","Q","U","A","R","T","O"],
["V","E","N","T","I","C","I","N","Q","U","E"],
["D","I","E","C","I","L","M","E","Z","Z","O"]],
sentences : {
start : { all : [[0,0,3],[0,5,6]], 1 : [[1,0], [1, 2, 5]] },
end : { all : [] },
hours : {
0 : [[[4,0,4]]],
1 : [[[]]],
2 : [[[1,7,9]]],
3 : [[[2,0,2]]],
4 : [[[4,0,4]]],
5 : [[[3,5,10]]],
6 : [[[5,8,10]]],
7 : [[[3,0,4]]],
8 : [[[2,7,10]]],
9 : [[[2,3,6]]],
10: [[[4,6,10]]],
11: [[[6,0,5]]],
12: [[[4,0,4]]]
},
minutes :{
5 : [[7,0], [7,2,3], [8, 5, 10]],
10 : [[7,0], [7,2,3], [9, 0, 5]],
15 : [[7,0], [7,2,3], [7, 5, 10]],
20 : [[7,0], [7,2,3], [8, 0, 4]],
25 : [[7,0], [7,2,3], [8,0,10]],
30 : [[7,0], [9,6,10] ],
35 : [[6,7,10], [7,2,3], [8,0,10]],
40 : [[6,7,10], [7,2,3], [8,0,4] ],
45 : [[6,7,10], [7,2,3], [7, 5, 10]],
50 : [[6,7,10], [7,2,3], [9,0,5]],
55 : [[6,7,10], [7,2,3], [8, 5, 10]]
},
incrementHourAt : 30
}
};
language['french'] = {
map : [["I", "I", "C", "E", "S", "T", "G", "H", "U", "N", "E"],
["D", "E", "U", "X", "P", "Q", "T", "R", "O", "I", "S"],
["Q", "U", "A", "T", "R", "E", "C", "D", "S", "I", "X"],
["S", "E", "P", "T", "D", "I", "X", "C", "I", "N", "Q"],
["N", "E", "U", "F", "W", "H", "U", "I", "T", "B", "C"],
["O", "N", "Z", "E", "H", "D", "O", "U", "Z", "E", "O"],
["P", "Q", "R", "S", "T", "H", "E", "U", "R", "E", "S"],
["M", "O", "I", "N", "S", "F", "E", "T", "D", "I", "X"],
["Q", "U", "A", "R", "T", "D", "E", "M", "I", "U", "V"],
["W", "V", "I", "N", "G", "T", "–", "C", "I", "N", "Q"]],
sentences : {
start : { all : [[0,0,1],[0,3,5]]},
end : { all : [[6,5,9]] },
hours : {
0 : [[[5,5,9]]],
1 : [[[0,8,10]]],
2 : [[[1,0,3]]],
3 : [[[1,6,10]]],
4 : [[[2,0,5]]],
5 : [[[3,7,10]]],
6 : [[[2,8,10]]],
7 : [[[3,0,3]]],
8 : [[[4,0,3]]],
9 : [[[4,5,8]]],
10: [[[3,4,6]]],
11: [[[5,0,3]]],
12: [[[5,5,9]]]
},
minutes :{
5 : [[6,5,10], [9,7,10] ],
10 : [[6,5,10], [7,8,10] ],
15 : [[6,5,10], [8,0,4] ],
20 : [[6,5,10], [9,1,5] ],
25 : [[6,5,10], [9,1,10] ],
30 : [[6,5,10], [7,6,7], [8,5,8] ],
35 : [[6,5,10], [7,0,4], [9,1,10]],
40 : [[6,5,10], [7,0,4], [9,1,5] ],
45 : [[6,5,10], [7,0,4], [8,0,4] ],
50 : [[6,5,10], [7,0,4], [7,8,10]],
55 : [[6,5,10], [7,0,4], [9,7,10]]
},
incrementHourAt : 30
}
};
language['english'] = {
map : [["I","T","L","I","S","B","F","A","M","P","M"],
["A","C","Q","U","A","R","T","E","R","D","C"],
["T","W","E","N","T","Y","F","I","V","E","X"],
["H","A","L","F","B","T","E","N","F","T","O"],
["P","A","S","T","E","R","U","N","I","N","E"],
["O","N","E","S","I","X","T","H","R","E","E"],
["F","O","U","R","F","I","V","E","T","W","O"],
["E","I","G","H","T","E","L","E","V","E","N"],
["S","E","V","E","N","T","W","E","L","V","E"],
["T","E","N","S","E","O'","C","L","O","C","K"]],
sentences : {
start : { all : [[0,0,1],[0,3,4]]},
end : { all : [[9,5,10]] },
hours : {
0 : [[[8,5,10]]],
1 : [[[5,0,2]]],
2 : [[[6,8,10]]],
3 : [[[5,6,10]]] ,
4 : [[[6,0,3]]],
5 : [[[6,4,7]]] ,
6 : [[[5,3,5]]],
7 : [[[8,0,4]]] ,
8 : [[[7,0,4]]],
9 : [[[4,7,10]]] ,
10: [[[9,0,2]]] ,
11: [[[7,5,10]]],
12: [[[8,5,10]]]
},
minutes :{
5 : [ [2,6,9] , [4,0,3] ],
10 : [ [3,5,7] , [4,0,3] ],
15 : [ [1,2,8] , [4,0,3] ],
20 : [ [2,0,5] , [4,0,3] ],
25 : [ [2,0,9] , [4,0,3] ],
30 : [ [3,0,3] , [4,0,3] ],
35 : [ [2,0,9] , [3,9,10]],
40 : [ [2,0,5] , [3,9,10]],
45 : [ [1,2,8] , [3,9,10]],
50 : [ [3,5,7] , [3,9,10] ],
55 : [ [2,6,9] , [3,9,10] ]
},
incrementHourAt : 29
}
};
language['german'] = {
map : [["E","S","M","I","S","T","E","F","Ü","N","F"],
["Z","E","H","N","Z","W","A","N","Z","I","G"],
["N","A","C","H","V","I","E","R","T","E","L"],
["V","O","R","N","A","C","H","H","A","L","B"],
["E","I","N","S","I","N","K","Z","W","E","I"],
["D","R","E","I","E","A","N","V","I","E","R"],
["F","Ü","N","F","N","I","S","E","C","H","S"],
["S","I","E","B","E","N","I","A","C","H","T"],
["N","E","U","N","Z","E","H","N","E","L","F"],
["Z","W","Ö","L","F","K","A","B","U","H","R"]],
sentences : {
start : { all :[[0,0,1], [0,3,5]] },
end : { all : [[9,8,10]] },
hours : {
0 : [[[9,0,4]]],
1 : [[[4,0,2]] , [[4,0,3]] ],
2 : [[[4,7,10]]],
3 : [[[5,0,3]]] ,
4 : [[[5,7,10]]],
5 : [[[6,0,3]]] ,
6 : [[[6,6,10]]],
7 : [[[7,0,5]]] ,
8 : [[[7,7,10]]],
9 : [[[8,0,3]]] ,
10: [[[8,4,7]]] ,
11: [[[8,8,10]]],
12: [[[9,0,4]]]
},
minutes :{
5 : [ [0,7,10] , [2,0,3] ],
10 : [ [1,0,3] , [2,0,3] ],
15 : [ [2,4,10] , [3,3,6] ],
20 : [ [1,4,10] , [3,3,6] ],
25 : [ [0,7,10] , [3,0,2] , [3,7,10] ],
30 : [ [3,7,10] ],
35 : [ [0,7,10] , [2,0,3] , [3,7,10] ],
40 : [ [1,4,10] , [3,0,2] ],
45 : [ [2,4,10] , [3,0,2] ],
50 : [ [1,0,3] , [3,0,2] ],
55 : [ [0,7,10] , [3,0,2] ]
},
incrementHourAt : 24
}
};
var wordClock = {
seconds : {
0 : [[[2,1,3],[3,0],[3,4],[4,0],[4,3],[4,4],[5,0],[5,2],[5,4],[6,0],[6,1],[6,4],[7,0],[7,4],[8,1,3]]],
1 : [[[2,2],[3,1],[3,2],[4,2],[5,2],[6,2],[7,2],[8,1,3]]],
2 : [[[2,1,3],[3,0],[3,4],[4,4],[5,3],[6,2],[7,1],[8,0,4]]],
3 : [[[2,0,4],[3,3],[4,2],[5,3],[6,4],[7,0],[7,4],[8,1,3]]],
4 : [[[2,3],[3,2],[3,3],[4,1],[4,3],[5,0],[5,3],[6,0,4],[7,3],[8,3]]],
5 : [[[2,0,4],[3,0],[4,0,3],[5,4],[6,4],[7,0,0],[7,4],[8,1,3]]],
6 : [[[2,2,3],[3,1],[4,0],[5,0,3],[6,0],[6,4],[7,0],[7,4],[8,1,3]]],
7 : [[[2,0,4],[3,4],[4,3],[5,2],[6,1],[7,1],[8,1]]],
8 : [[[2,1,3],[3,0],[3,4],[4,0],[4,4],[5,1,3],[6,0],[6,4],[7,0],[7,4],[8,1,3]]],
9 : [[[2,1,3],[3,0],[3,4],[4,0],[4,4],[5,1,4],[6,4],[7,3],[8,1,2]]]
},
letterMap : null,
prevSentence : [],
prevSecond : null,
language : null,
letters : null,
state : 0, // 0 = Normal Clock, 1 = Show Seconds + Minutes
init : function(language){
this.letters = document.getElementById('letters');
this.language = language;
this.buildLetters();
this.startClock();
},
startClock : function(){
this.hideLetters(this.prevSentence);
this.hideNumber(this.prevSecond);
this.showMinutes();
if (this.state === 0){
this.showText();
}else{
this.showSeconds();
}
},
switchState : function(state){
if (state !== undefined){
this.state = state;
}else{
this.state = ++this.state % 2;
}
this.startClock();
},
buildLetters : function(){
//Clear Letters
this.letters.innerHTML = "";
// Create 2D Array to store the span elements.
this.letterMap = [];
for (var i=0;i<this.language.map.length;i++){
this.letterMap.push([]);
}
//"Draw" Letters
for (var x=0;x<this.language.map.length;x++){
for (var y=0;y<this.language.map[x].length;y++){
var span = document.createElement('span');
span.innerHTML = this.language.map[x][y];
// Add the letter to the lettrMap
this.letterMap[x].push(span);
this.letters.appendChild(span);
}
}
},
showText: function(){
if (this.state !== 0)
return;
// Hide last Sentence
this.hideLetters(this.prevSentence);
var showClock = false;
var plural = true;
var hours = this.getHours();
var minutes = this.getMinutes();
var showArr = [];
var startLetters = (this.language.sentences.start[hours] !== undefined ) ? this.language.sentences.start[hours] : this.language.sentences.start.all;
showArr.push(startLetters);
if (minutes < 5){
showClock = true;
plural = false;
}else{
minutes -= minutes % 5;
showArr.push(this.language.sentences.minutes[minutes]);
if (minutes > this.language.sentences.incrementHourAt){
// Increment hour
hours = (hours+1) % 12;
}
}
if (plural && (this.language.sentences.hours[hours].length === 2)){
showArr.push(this.language.sentences.hours[hours][1]);
}else{
showArr.push(this.language.sentences.hours[hours][0]);
}
if (showClock){
var endLetters = (this.language.sentences.end[hours] !== undefined) ? this.language.sentences.end[hours] : this.language.sentences.end.all;
showArr.push(endLetters);
}
this.showLetters(showArr);
this.prevSentence = showArr;
var that = this;
// Wait for the rest of the 5min.
setTimeout(function(){that.showText();}, (5 - (this.getMinutes() % 5)) * 60000 - this.getSeconds()*1000 - this.getMilliseconds());
},
showMinutes : function(){
var minutes = this.getMinutes() % 5;
if (minutes === 0){
this.hideMinutesDot(4);
}else{
this.showMinutesDot(minutes);
}
var that = this;
setTimeout(function(){that.showMinutes();}, (60-this.getSeconds())*1000 - this.getMilliseconds());
},
showSeconds : function(){
if (this.state != 1)
return;
var seconds = this.getSeconds();
this.hideNumber(this.prevSecond);
this.showNumber(seconds);
this.prevSecond = seconds;
var that = this;
setTimeout(function(){that.showSeconds();},1000-this.getMilliseconds());
},
getHours : function(){
var hour = (new Date()).getHours();
return (hour > 11) ? hour - 12 : hour;
},
getMinutes : function(){
return (new Date()).getMinutes();
},
getSeconds: function(){
return (new Date()).getSeconds();
},
getMilliseconds: function(){
return (new Date()).getMilliseconds();
},
hideNumber : function(number){
if (number === undefined || number === null) return;
var tens = this.seconds[~~(number/10)];
var ones = this.seconds[(number % 10)];
this.changeClassLetters(tens, "", 0);
this.changeClassLetters(ones, "", 6);
},
showNumber : function(number){
tens = this.seconds[~~(number/10)];
ones = this.seconds[(number % 10)];
this.changeClassLetters(tens, "light", 0);
this.changeClassLetters(ones, "light", 6);
},
hideLetters : function(la){
this.changeClassLetters(la, "", 0);
},
showLetters : function(la){
this.changeClassLetters(la, "light", 0);
},
hideMinutesDot : function(count){
this.changeClassMinutes(count, '');
},
showMinutesDot : function(count){
this.changeClassMinutes(count, 'light');
},
changeClassLetters: function(la, value, offset){
for (var s=0;s<la.length;s++){
for (var d=0;d<la[s].length;d++){
var x = la[s][d][0];
var y = la[s][d][1]+ offset;
if (la[s][d].length === 3){
var end = la[s][d][2] + offset;
for (var i=y;i<=end;i++){
this.letterMap[x][i].className = value;
}
}else{
this.letterMap[x][y].className = value;
}
}
}
},
changeClassMinutes : function(count, value){
for (var i=0;i<count;i++){
document.getElementById("e"+i).className = value;
}
},
_inArray : function(arr, obj){
for (var i=0;i<arr.length;i++){
if (arr[i] == obj) return true;
}
return false;
}
};
wordClock.init(language['english']);
$('#settings a').on('click', function(e){
if (e.target.id === "state"){
wordClock.switchState();
if (wordClock.state){
$(e.target).html("show text");
}else{
$(e.target).html("show seconds");
}
}else{
wordClock.init(language[e.target.id]);
}
e.preventDefault();
});
var clock = $("#clock");
var buttons = $("#settings .button");
$('#settings').on("click", ".button", function(e){
var target = $(e.target);
buttons.removeClass("selected");
target.toggleClass("selected");
clock.css({backgroundColor: target.css("backgroundColor")});
});
@import url(http://fonts.googleapis.com/css?family=Josefin+Sans);
* { box-sizing: border-box; }
body{
font-family:'Josefin Sans', Arial, "Trebuchet MS", sans-serif;
background-color:black;
/* Codepen flash fix*/
transform:translateZ(0);
}
#clock{
position:relative;
display:block;
color:#000;
font-size:30px;
background: rgb(0,0,139);
background: rgb(0,0,139) linear-gradient(140deg, rgba(0,0,0,0) 50%, rgba(255,255,255,0.05) 50%, rgba(255,255,255,0.0) 100%);
width:550px;
padding:70px;
margin:0 auto;
transition:background-color 0.5s linear;
}
#letters, #letters span{
display:inline-block;
line-height:41px;
text-align:center;
transition:all 0.5s linear;
}
#letters span{
width:35px;
}
.light{
color:#fff !important;
text-shadow: 0px 0px 5px #fff;
}
#settings, #settings a{
color:#555;
font-size:13px;
width:160px;
margin:10px auto;
}
#settings a:hover{
text-shadow: 0 0 1px #fff;
}
#settings p{
margin:0;
width:100%;
text-align:center;
}
.button{
position:relative;
display:inline-block;
margin:0 4px;
background: linear-gradient(-75deg, rgba(0,0,0,0) 50%, rgba(255,255,255,0.05) 50%, rgba(255,255,255,0.0) 100%);
height:20px;
opacity:0.3;
-webkit-transition:opacity 0.5s linear;
-moz-transition:opacity 0.5s linear;
-ms-transition:opacity 0.5s linear;
-o-transition:opacity 0.5s linear;
transition:opacity 0.5s linear;
width:20px;
}
#settings:hover .button{
opacity:1;
}
.button:hover{
box-shadow:0 0 3px 0 white inset;
}
.selected:before{
content:'✔';
position:absolute;
color:rgba(255,255,255,0.4);
left:5px;
top:3px;
}
/* Background colors for the color Setting*/
#darkblue{
background-color:rgb(0,0,139);
}
#darkred{
background-color:rgb(139,0,0);
}
#green{
background-color:rgb(100,120,0);
}
#purple{
background-color:rgb(70,40,120);
}
#orange{
background-color:rgb(150,80,0);
}
#desc{
margin:50px auto;
width:420px;
height:520px;
color:white;
font-size:18px;
}
/* dots in the edge for showing the minutes */
#e0,#e1,#e2,#e3{
position:absolute;
color:black;
font-size:15px;
}
#e0, #e1{
top:15px;
}
#e2, #e1{
right:25px;
}
#e0, #e3{
left:25px;
}
#e2, #e3{
bottom:20px;
}
/* Just some basic styling */
h1{
display:block;
color:white;
width:100%;
text-align:center;
}
a{
color:#555;
text-decoration:none;
}
a:hover{
text-shadow:0 0 2px #555;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment