-
Notifications
You must be signed in to change notification settings - Fork 44
/
MMM-Traffic.js
158 lines (138 loc) · 4.92 KB
/
MMM-Traffic.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
/* Magic Mirror
* Module: MMM-Traffic
*
* By Sam Lewis https://github.com/SamLewis0602
* MIT Licensed.
*/
Module.register('MMM-Traffic', {
defaults: {
interval: 300000,
showSymbol: true,
firstLine: 'Current duration is {duration} mins',
loadingText: 'Loading...',
language: config.language,
mode: 'driving',
days: [0, 1, 2, 3, 4, 5, 6],
hoursStart: '00:00',
hoursEnd: '23:59'
},
start: function () {
console.log('Starting module: ' + this.name);
this.loading = true;
this.internalHidden = false;
this.firstResume = true;
this.errorMessage = undefined;
this.errorDescription = undefined;
this.updateCommute = this.updateCommute.bind(this);
this.getCommute = this.getCommute.bind(this);
this.getDom = this.getDom.bind(this);
if ([this.config.originCoords, this.config.destinationCoords, this.config.accessToken].includes(undefined)) {
this.errorMessage = 'Config error';
this.errorDescription = 'You must set originCoords, destinationCoords, and accessToken in your config';
this.updateDom();
} else {
this.updateCommute();
}
},
updateCommute: function () {
let mode = this.config.mode == 'driving' ? 'driving-traffic' : this.config.mode;
this.url = encodeURI(`https://api.mapbox.com/directions/v5/mapbox/${mode}/${this.config.originCoords};${this.config.destinationCoords}?access_token=${this.config.accessToken}`);
// only run getDom once at the start of a hidden period to remove the module from the screen, then just wait until time to unhide to run again
if (this.shouldHide() && !this.internalHidden) {
console.log('Hiding MMM-Traffic due to config options: days, hoursStart, hoursEnd');
this.internalHidden = true;
this.updateDom();
} else if (!this.shouldHide()) {
this.internalHidden = false;
this.getCommute(this.url);
}
// no network requests are made when the module is hidden, so check every 30 seconds during hidden
// period to see if it's time to unhide yet
setTimeout(this.updateCommute, this.internalHidden ? 3000 : this.config.interval);
},
getCommute: function (api_url) {
var self = this;
fetch(api_url)
.then(self.checkStatus)
.then(json => {
self.duration = Math.round(json.routes[0].duration / 60);
self.errorMessage = self.errorDescription = undefined;
self.loading = false;
self.updateDom();
})
.catch(e => {
self.errorMessage = payload.error.message;
self.errorDescription = payload.error.description;
self.loading = false;
self.updateDom();
});
},
checkStatus: function (res) {
if (res.ok) {
return res.json();
} else {
return res.json().then(json => {
throw new MMMTrafficError(`API Error - ${json.code}`, json.message);
});
}
},
getStyles: function () {
return ['traffic.css', 'font-awesome.css'];
},
getScripts: function () {
return ['moment.js'];
},
getDom: function () {
var wrapper = document.createElement("div");
// hide when desired (called once on first update during hidden period)
if (this.internalHidden) return wrapper;
// base divs
var firstLineDiv = document.createElement('div');
firstLineDiv.className = 'bright medium mmmtraffic-firstline';
var secondLineDiv = document.createElement('div');
secondLineDiv.className = 'normal small mmmtraffic-secondline';
// display any errors
if (this.errorMessage) {
firstLineDiv.innerHTML = this.errorMessage;
secondLineDiv.innerHTML = this.errorDescription;
wrapper.append(firstLineDiv);
wrapper.append(secondLineDiv);
return wrapper;
}
let symbolString = 'car';
if (this.config.mode == 'cycling') symbolString = 'bicycle';
if (this.config.mode == 'walking') symbolString = 'walking';
// symbol
if (this.config.showSymbol) {
var symbol = document.createElement('span');
symbol.className = `fa fa-${symbolString} symbol`;
firstLineDiv.appendChild(symbol);
}
// first line
var firstLineText = document.createElement('span');
firstLineText.innerHTML = this.loading ? this.config.loadingText : this.replaceTokens(this.config.firstLine)
firstLineDiv.appendChild(firstLineText);
wrapper.appendChild(firstLineDiv);
if (this.loading) return wrapper;
// second line
if (this.config.secondLine) {
secondLineDiv.innerHTML = this.replaceTokens(this.config.secondLine);
wrapper.appendChild(secondLineDiv);
}
return wrapper;
},
replaceTokens: function (text) {
return text.replace(/{duration}/g, this.duration);
},
shouldHide: function () {
let hide = true;
let now = moment();
if (this.config.days.includes(now.day()) &&
moment(this.config.hoursStart, 'HH:mm').isBefore(now) &&
moment(this.config.hoursEnd, 'HH:mm').isAfter(now)
) {
hide = false;
}
return hide;
},
});