-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathMMM-MinecraftStatus.js
executable file
·278 lines (234 loc) · 10.7 KB
/
MMM-MinecraftStatus.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
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
/* global Module */
/* MMM-MinecraftStatus.js
*
* Magic Mirror
* Module: MMM-MinecraftStatus
* MIT Licensed.
*
* See README.md for details on this.
*/
Module.register("MMM-MinecraftStatus", {
defaults: {
banner: "Minecraft Server",
hostname: "localhost",
port: 12345,
intervalSeconds: 30,
alertSoundWhenPingFails: null,
alertSoundOnFirstPlayer: null
},
/*
* Remembers the number of players the server last sent back on the previous ping call.
* Needed below to tell when we cross the threshold from zero players to more than zero
* players: need to play the alert sound if configured.
*/
lastPingNumPlayers: 0,
getStyles: function() {
return [ "MMM-MinecraftStatus.css", "font-awesome.css" ];
},
/*
* Don't hate, these came from Google translate. If you don't like them, send
* a pull request on Github, please!
*/
getTranslations: function() {
return {
cn: "translations/cn.json",
de: "translations/de.json",
en: "translations/en.json",
es: "translations/es.json",
fr: "translations/fr.json",
it: "translations/it.json",
jp: "translations/jp.json",
ru: "translations/ru.json",
sv: "translations/sv.json"
};
},
/*
* OVERRIDES parent function. Called when MM wants to draw this widget on the browser
* page's DOM.
* This creates "this.playersDiv" and "this.latencyDiv" for use below in function
* socketNotificationReceived that overwrites their HTML with whatever the server
* (node_helper.js) sends back from Minecraft.
*
* The rest of the code in this module makes sure exactly 1 of successDiv and errorDiv
* is "visibility: hidden" and the other is "visibility: visible".
*/
getDom: function() {
var topDiv = this.createEle(null, "div", "minecraftStatus"); // top level that holds everything
this.createEle(topDiv, "div", "title", this.config.banner);
if (this.config.alertSoundWhenPingFails) {
this.audioPlayerPingFailed = this.createAudioPlayer(topDiv, this.config.alertSoundWhenPingFails);
}
if (this.config.alertSoundOnFirstPlayer) {
this.audioPlayerFirstPlayer = this.createAudioPlayer(topDiv, this.config.alertSoundOnFirstPlayer);
}
var table = this.createEle(topDiv, "table");
// successTbody holds the number of active players and latency to the server
this.successTbody = this.createEle(table, "tbody", "successSection");
this.playersDiv = this.createMetricPanel(this.successTbody, "user", this.translate("players"));
this.latencyDiv = this.createMetricPanel(this.successTbody, "clock-o", this.translate("latency"));
// errorTbody holds the reason why we can't ping the server
this.errorTbody = this.createEle(table, "tbody", "errorSection");
this.errorText = this.createErrorSection(this.errorTbody);
this.errorText.colSpan = 2;
return topDiv;
},
createAudioPlayer: function(parent, audioUri) {
return this.createEle(parent, "audio", null,
"<source type='audio/mpeg' src='/modules/MMM-MinecraftStatus/public/sounds/" +
audioUri + "'/>" + "(" + this.translate("no_audio_support") + ")</audio>");
},
/*
* Helper method called exclusively by getDom() above. This draws either the
* Players row with icon and number, or the Latency row with number and icon.
*/
createMetricPanel: function(parent, fontAwesomeName, label) {
var row = this.createEle(parent, "tr");
this.createEle(row, "td", "iconbox",
"<i class='fa fa-" + fontAwesomeName + " fa-fw'></i>");
var value = this.createEle(row, "td", "value", "?");
this.createEle(row, "td", "label", label.replace(" ", " "));
return value;
},
createErrorSection: function(parent) {
var tr1 = this.createEle(parent, "tr");
this.createEle(tr1, "td", "iconbox",
"<i class='fa fa-exclamation-triangle fa-fw'></i>");
this.createEle(tr1, "td", "errorLabel", this.translate("error"));
var tr2 = this.createEle(parent, "tr");
return this.createEle(tr2, "td", "errorText");
},
/*
* Create a single HTML Element (something with < and > around it and children
* html things).
*/
createEle: function(parentEle, eleType, name, innerHtml) {
var div = document.createElement(eleType);
if (name) {
div.className = name;
}
if (innerHtml) {
div.innerHTML = innerHtml;
}
if (parentEle) {
parentEle.appendChild(div);
}
return div;
},
/*
* OVERRIDES notificationReceived() from parent.
* This handles the DOM_OBJECTS_CREATED in-browser message sent from MagicMirror.
* This single alert is sent (delivered to) every widget so they can take actions
* once the MM screen is setup and available, and after getDom() has been called
* on every module.
* we respond to this event here by starting the recurring timeer that will issue
* pings, and not before: if the response from the server-side helper arrives before
* getDom() was done (or before getDom() was invoked!), there would be no elements
* on the browser matching that needs to be present.
*/
notificationReceived: function(notification, payload, sender) {
switch(notification) {
case "DOM_OBJECTS_CREATED":
this.triggerHelper(); // Ask the server-side helper right away
this.startTimer(); // wake up every once and a while and try again
break;
}
},
/*
* Send a message to our node_helper.js, running on the server. Tell it that it is time
* to query a Minecraft server. This isn't a round-trip event where we ask the server.
* It is a notification from us to the node_helper. When and if the helper completes,
* we hope it will deliver a push message to us that we can react to.
* Because of the nature of send and receive socket notification functions, our reuquest
* is delivered to our node_helper, but there could be multiple instances of this mod
* running on different browsers. We pass this.identifier to the server, so the server
* can send it back in th response. This way, each instance of MMM-MinecraftStatus
* object will know to which of them a message pertains.
*/
triggerHelper: function() {
//mclog("Asking server to mcping " + this.config.hostname + ":" + this.config.port);
this.sendSocketNotification("MINECRAFT_PING", {
identifier: this.identifier, // sent to helper and sent back to us below
hostname: this.config.hostname,
port: this.config.port
});
},
/*
* Create a single timer for this module that fires on a cadence. Each "tick",
* call triggerHelper(). This sends a message to our node_helper.js code on the
* server with the host and port to MinecraftPing. Nothing waits for a reply here,
* this function ends immediately. The server sends results to us asynchronously.
* Just to be safe in case something gets really out of sync, this won't start
* a timer if there is already one running in this module.
*/
startTimer: function() {
var self = this; // "this" means something else in the setInterval() function below
if (! this.timer) {
this.timer = setInterval(
function() { self.triggerHelper(); }, // "self" is "this" for MMM-MinecraftStatus object
self.config.intervalSeconds * 1000 // setInterval() runs in milliseconds
);
//this.mclog("Now watching server " + this.config.hostname + ":" + this.config.port);
}
},
/*
* Callback function invoked by MagicMirror whenever our node_helper.js sends a message.
* NOTE: it's up to us to be careful, since there is one node_helper.js running on the
* server, but there could be many instances of us running in the browser (we could be
* watching multiple Minecraft servers). This function only takes action if the message
* sent from the helper has the same intended "this.identifier" as us: we send in our
* identifier in triggerHelper() above.
*/
socketNotificationReceived: function(notification, payload) {
// this.mclog("Got " + notification + ": " +
// JSON.stringify(payload, null, 2));
if (payload.identifier === this.identifier) {
switch(notification) {
case "MINECRAFT_UPDATE":
var currPlayers = parseInt(payload.players); // just being safe
this.playersDiv.innerHTML = currPlayers;
this.latencyDiv.innerHTML = payload.latency;
this.successTbody.style.display = "block";
this.errorTbody.style.display = "none";
if (this.audioPlayerFirstPlayer && this.lastPingNumPlayers == 0 && currPlayers > 0) {
this.audioPlayerFirstPlayer.play();
}
this.lastPingNumPlayers = currPlayers;
break;
case "MINECRAFT_ERROR":
this.errorText.innerHTML = payload.message;
this.successTbody.style.display = "none";
this.errorTbody.style.display = "block";
this.sendNotification("SHOW_ALERT", { // picked-up by default module "alert"
type: "notification",
title: this.config.banner,
message: payload.message
});
if (this.audioPlayerPingFailed) {
this.audioPlayerPingFailed.play();
}
break;
}
}
},
/*
* Callback function invoked by MagicMirror whenever we are removed from the screen
* visually. When we're not visible, there isn't really a reason for us to continue
* pinging the Minecraft server. Stop the timer here so we go "idle" for the time-being.
*/
suspend: function() {
if (!!this.timer) {
clearInterval(this.timer);
this.timer = null;
//this.mclog("Not watching a server anymore");
}
},
/*
* Callback function invoked by MagicMirror whenever we were suspended and are now
* visible again: turn the timer back on. Since the timer will wait a while before
* triggering the helper, force-trigger it now so the user gets an update quickly.
*/
resume: function() {
this.triggerHelper();
this.startTimer();
}
});