1 #include "mainwindow.h"
2 #include "post_to_main_thread.h"
3 #include "ui_mainwindow.h"
6 #include <netinet/in.h>
7 #include <netinet/tcp.h>
10 #include <sys/socket.h>
14 string escape_html(const string &str)
20 } else if (ch == '>') {
22 } else if (ch == '&') {
31 string escape_unicode(const string &str)
34 for (size_t pos = 0; pos < str.size(); ) {
36 int len = mbtowc(&wc, str.data() + pos, str.size() - pos);
45 } else if (isprint(wc)) {
49 snprintf(buf, sizeof(buf), "\\u%04x", wc);
56 string escape_quotes(const string &str)
68 string serialize_as_json(const map<string, string> ¶m)
73 for (const auto &key_value : param) {
74 if (!first) s += ", ";
78 s += escape_quotes(escape_unicode(key_value.first));
80 s += escape_quotes(escape_unicode(key_value.second));
87 MainWindow::MainWindow(QWidget *parent) :
89 ui(new Ui::MainWindow)
92 ws = new WSServer("127.0.0.1", 5250);
93 ws->set_connection_callback([this](bool connected) {
94 string msg = connected ? "Connected" : "Not connected";
95 post_to_main_thread([this, msg]() {
96 ui->ws_connected_label->setText(QString::fromStdString(msg));
100 udp_thread = std::thread(&MainWindow::udp_thread_func, this, 6000);
101 udp_thread2 = std::thread(&MainWindow::udp_thread_func, this, 6001);
103 connect(ui->ws_disconnect_btn, &QPushButton::clicked, this, &MainWindow::ws_disconnect_clicked);
104 connect(ui->set_initials_btn, &QPushButton::clicked, this, &MainWindow::set_initials_clicked);
105 connect(ui->set_match_2_initials_btn, &QPushButton::clicked, this, &MainWindow::set_match_2_initials_clicked);
106 connect(ui->set_color_btn, &QPushButton::clicked, this, &MainWindow::set_color_clicked);
107 connect(ui->set_score_btn, &QPushButton::clicked, this, &MainWindow::set_score_clicked);
108 connect(ui->set_all_scorebug_btn, &QPushButton::clicked, this, &MainWindow::set_all_scorebug_clicked);
109 connect(ui->goal_1_btn, &QPushButton::clicked, this, [this]() { add_goal(ui->score_1_box, 1); });
110 connect(ui->ungoal_1_btn, &QPushButton::clicked, this, [this]() { add_goal(ui->score_1_box, -1); });
111 connect(ui->goal_2_btn, &QPushButton::clicked, this, [this]() { add_goal(ui->score_2_box, 1); });
112 connect(ui->ungoal_2_btn, &QPushButton::clicked, this, [this]() { add_goal(ui->score_2_box, -1); });
114 connect(ui->set_clock_btn, &QPushButton::clicked, this, &MainWindow::set_clock_clicked);
115 connect(ui->set_clock_limit_btn, &QPushButton::clicked, this, &MainWindow::set_clock_limit_clicked);
116 connect(ui->start_and_show_clock_btn, &QPushButton::clicked, this, &MainWindow::start_and_show_clock_clicked);
117 connect(ui->stop_clock_btn, &QPushButton::clicked, this, &MainWindow::stop_clock_clicked);
118 connect(ui->show_clock_btn, &QPushButton::clicked, this, &MainWindow::show_clock_clicked);
119 connect(ui->hide_clock_btn, &QPushButton::clicked, this, &MainWindow::hide_clock_clicked);
120 connect(ui->show_match_2_btn, &QPushButton::clicked, this, &MainWindow::show_match_2_clicked);
121 connect(ui->hide_match_2_btn, &QPushButton::clicked, this, &MainWindow::hide_match_2_clicked);
123 connect(ui->set_comment_btn, &QPushButton::clicked, this, &MainWindow::set_comment_clicked);
124 connect(ui->set_and_show_comment_btn, &QPushButton::clicked, this, &MainWindow::set_and_show_comment_clicked);
125 connect(ui->hide_comment_btn, &QPushButton::clicked, this, &MainWindow::hide_comment_clicked);
126 connect(ui->set_and_show_autocomment_btn, &QPushButton::clicked, this, &MainWindow::set_and_show_autocomment_clicked);
127 connect(ui->autoshow_autocomment, &QCheckBox::stateChanged, this, &MainWindow::autocomment_update);
129 connect(ui->show_lower_third_btn, &QPushButton::clicked, this, &MainWindow::show_lower_third_clicked);
130 connect(ui->hide_lower_third_btn, &QPushButton::clicked, this, &MainWindow::hide_lower_third_clicked);
132 connect(ui->quick_lower_third_edit, &QLineEdit::returnPressed, this, &MainWindow::quick_lower_third_activate);
133 connect(ui->show_quick_lower_third_btn, &QPushButton::clicked, this, &MainWindow::quick_lower_third_activate);
135 connect(ui->show_scorebug_btn, &QPushButton::clicked, this, &MainWindow::show_scorebug_clicked);
136 connect(ui->show_group_a_btn, &QPushButton::clicked, this, [this]() { show_group_clicked("Group A"); });
137 connect(ui->show_group_b_btn, &QPushButton::clicked, this, [this]() { show_group_clicked("Group B"); });
138 connect(ui->show_group_c_btn, &QPushButton::clicked, this, [this]() { show_group_clicked("Group C"); });
139 connect(ui->show_schedule_btn, &QPushButton::clicked, this, &MainWindow::show_schedule_clicked);
140 connect(ui->show_carousel_btn, &QPushButton::clicked, this, &MainWindow::show_carousel_clicked);
141 connect(ui->show_nothing_btn, &QPushButton::clicked, this, &MainWindow::show_nothing_clicked);
142 connect(ui->show_roster_1_btn, &QPushButton::clicked, this, [this]() { show_roster_clicked(ui->initials_1_edit->text().toStdString()); });
143 connect(ui->show_roster_2_btn, &QPushButton::clicked, this, [this]() { show_roster_clicked(ui->initials_2_edit->text().toStdString()); });
144 connect(ui->show_roster_carousel_btn, &QPushButton::clicked, this, &MainWindow::show_roster_carousel_clicked);
146 autocomment_update();
148 const set<pair<unsigned, unsigned>> usb{{ 0x0e8f, 0x0041 }};
149 event_device = new EventDevice(usb, ui->quick_lower_third_edit);
150 event_device->start_thread();
153 MainWindow::~MainWindow()
158 void MainWindow::ws_disconnect_clicked()
160 ws->change_port(stoi(ui->ws_port_box->text().toStdString()));
163 void MainWindow::set_initials_clicked()
165 map<string, string> param;
166 param["team1"] = escape_html(ui->initials_1_edit->text().toStdString());
167 param["team2"] = escape_html(ui->initials_2_edit->text().toStdString());
168 ws->send_command("update " + serialize_as_json(param));
169 ws->send_command("eval setteams()");
172 void MainWindow::set_match_2_initials_clicked()
174 map<string, string> param;
175 param["team1"] = escape_html(ui->match_2_initials_1_edit->text().toStdString());
176 param["team2"] = escape_html(ui->match_2_initials_2_edit->text().toStdString());
177 ws->send_command("update " + serialize_as_json(param));
178 ws->send_command("eval setteams2()");
181 void MainWindow::set_color_clicked()
183 map<string, string> param;
184 param["team1color"] = ui->color_1_edit->text().toStdString(); // Should maybe be escaped, but meh.
185 param["team2color"] = ui->color_2_edit->text().toStdString();
186 ws->send_command("update " + serialize_as_json(param));
187 ws->send_command("eval setcolors()");
190 void MainWindow::set_score_clicked()
192 map<string, string> param;
193 param["score1"] = to_string(ui->score_1_box->value());
194 param["score2"] = to_string(ui->score_2_box->value());
195 ws->send_command("update " + serialize_as_json(param));
196 ws->send_command("eval setscore()");
197 autocomment_update();
200 void MainWindow::set_all_scorebug_clicked()
202 set_initials_clicked();
207 void MainWindow::add_goal(QSpinBox *box, int delta)
209 box->setValue(box->value() + delta);
213 void MainWindow::set_clock_clicked()
215 map<string, string> param;
216 param["clock_min"] = to_string(ui->clock_min_box->value());
217 param["clock_sec"] = to_string(ui->clock_sec_box->value());
218 ws->send_command("update " + serialize_as_json(param));
219 ws->send_command("eval setclockfromstate()");
222 void MainWindow::set_clock_limit_clicked()
224 map<string, string> param;
225 param["clock_limit_min"] = to_string(ui->clock_limit_min_box->value());
226 param["clock_limit_sec"] = to_string(ui->clock_limit_sec_box->value());
227 ws->send_command("update " + serialize_as_json(param));
228 ws->send_command("eval setclocklimitfromstate()");
231 void MainWindow::start_and_show_clock_clicked()
233 ws->send_command("eval startclock()"); // Also shows.
236 void MainWindow::stop_clock_clicked()
238 ws->send_command("eval stopclock()");
241 void MainWindow::show_clock_clicked()
243 ws->send_command("eval showclock()");
246 void MainWindow::hide_clock_clicked()
248 ws->send_command("eval hideclock()");
251 void MainWindow::show_match_2_clicked()
253 ws->send_command("eval showmatch2()");
256 void MainWindow::hide_match_2_clicked()
258 ws->send_command("eval hidematch2()");
261 void MainWindow::set_comment_clicked()
263 map<string, string> param;
264 param["comment"] = ui->comment_edit->text().toStdString();
265 ws->send_command("update " + serialize_as_json(param));
266 ws->send_command("eval setcomment()");
269 void MainWindow::set_and_show_comment_clicked()
271 set_comment_clicked();
272 ws->send_command("eval showcomment()");
275 void MainWindow::set_and_show_autocomment_clicked()
277 ui->comment_edit->setText(ui->autocomment_edit->text());
278 set_and_show_comment_clicked();
281 void MainWindow::hide_comment_clicked()
283 ws->send_command("eval hidecomment()");
286 void MainWindow::show_lower_third_clicked()
288 map<string, string> param;
289 param["text1"] = ui->lowerthird_heading_edit->text().toStdString();
290 param["text2"] = ui->lowerthird_subheading_edit->text().toStdString();
291 ws->send_command("update " + serialize_as_json(param));
292 ws->send_command("eval setandshowlowerthird()");
295 void MainWindow::hide_lower_third_clicked()
297 ws->send_command("eval hidelowerthird()");
300 void MainWindow::quick_lower_third_activate()
302 string code = ui->quick_lower_third_edit->text().toUpper().toStdString();
304 add_goal(ui->score_1_box, 1);
305 } else if (code == "B") {
306 add_goal(ui->score_2_box, 1);
307 } else if (code == "C") {
308 ws->send_command("eval hidelowerthird()");
310 map<string, string> param;
311 param["code"] = code;
312 ws->send_command("update " + serialize_as_json(param));
313 ws->send_command("eval quicklowerthird()");
315 ui->quick_lower_third_edit->clear();
318 void MainWindow::autocomment_update()
320 int score1 = ui->score_1_box->value();
321 int score2 = ui->score_2_box->value();
323 if (abs(score1 - score2) >= 3) {
324 msg = "Game ends after this point";
326 int cap = max(score1, score2) + 1;
327 if (score1 == score2) ++cap;
330 msg = "Point cap: First to 13";
333 snprintf(buf, sizeof(buf), "Pagacap: First to %d", cap);
337 ui->autocomment_edit->setText(QString::fromStdString(msg));
339 map<string, string> param;
340 param["autocomment_on_clock_limit"] = ui->autoshow_autocomment->isChecked() ? "1" : "0";
341 param["autocomment"] = msg;
342 ws->send_command("update " + serialize_as_json(param));
345 void MainWindow::show_scorebug_clicked()
347 ws->send_command("eval stopcarousel()");
348 ws->send_command("eval hidetable()");
349 ws->send_command("eval showscorebug()");
352 void MainWindow::show_group_clicked(const std::string &group_name)
354 map<string, string> param;
355 param["group_name"] = group_name;
356 ws->send_command("eval stopcarousel()");
357 ws->send_command("update " + serialize_as_json(param));
358 ws->send_command("eval showgroup_from_state()");
361 void MainWindow::show_roster_clicked(const std::string &team_code)
363 map<string, string> param;
364 param["team_code"] = team_code;
365 ws->send_command("eval stopcarousel()");
366 ws->send_command("update " + serialize_as_json(param));
367 ws->send_command("eval showroster_from_state()");
370 void MainWindow::show_schedule_clicked()
372 ws->send_command("eval stopcarousel()");
373 ws->send_command("eval showschedule()");
376 void MainWindow::show_carousel_clicked()
378 ws->send_command("eval stopcarousel()");
379 ws->send_command("eval showcarousel()");
382 void MainWindow::show_roster_carousel_clicked()
384 map<string, string> param;
385 param["team1"] = escape_html(ui->initials_1_edit->text().toStdString());
386 param["team2"] = escape_html(ui->initials_2_edit->text().toStdString());
387 ws->send_command("eval stopcarousel()");
388 ws->send_command("update " + serialize_as_json(param));
389 ws->send_command("eval showrostercarousel_from_state()");
392 void MainWindow::show_nothing_clicked()
394 ws->send_command("eval hidescorebug()");
395 ws->send_command("eval stopcarousel()");
396 ws->send_command("eval hidetable()");
399 void MainWindow::udp_thread_func(int port)
401 int sock = socket(PF_INET6, SOCK_DGRAM, IPPROTO_UDP);
408 if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one)) == -1) {
409 perror("setsockopt");
414 memset(&saddr6, 0, sizeof(saddr6));
415 saddr6.sin6_family = AF_INET6;
416 inet_pton(AF_INET6, "::", &saddr6.sin6_addr);
417 saddr6.sin6_port = htons(port);
418 if (bind(sock, (sockaddr *)&saddr6, sizeof(saddr6)) == -1) {
425 int err = recv(sock, buf, sizeof(buf), 0);
431 post_to_main_thread([buf, err, port, this] {
432 bt6000_message_received(string(buf, err), port);
437 int parse_digit(char ch)
439 if (ch >= '0' && ch <= '9') {
445 int parse_clock(char ch1, char ch2)
447 int s1 = parse_digit(ch1);
448 int s2 = parse_digit(ch2);
452 int parse_score(char ch1, char ch2, char ch3)
454 int s1 = parse_digit(ch1);
455 int s2 = parse_digit(ch2);
456 int s3 = parse_digit(ch3);
457 return s1 * 100 + s2 * 10 + s3;
460 void MainWindow::bt6000_message_received(const string &msg, int port)
462 fprintf(stderr, "BT6000 message: '%s' (port %d)\n", msg.c_str(), port);
464 if (!ui->bt6000_2_enable->isChecked()) {
468 if (!ui->bt6000_enable->isChecked()) {
473 if (msg.size() >= 9 && msg[0] == 'G' && msg[1] == '0' && msg[2] == '1') {
474 // G01: Game clock, period number, and number of time-outs.
475 bool clock_running = !(msg[3] & 0x02);
476 // bool klaxon = (msg[3] & 0x04);
477 int minutes = parse_clock(msg[5], msg[6]);
478 int seconds = parse_clock(msg[7], msg[8]);
480 map<string, string> param;
481 param["clock_min"] = to_string(minutes);
482 param["clock_sec"] = to_string(seconds);
483 ws->send_command("update " + serialize_as_json(param));
486 ws->send_command("eval adjustclock2fromstate()");
488 ws->send_command("eval startclock2()");
490 ws->send_command("eval stopclock2()");
493 ws->send_command("eval adjustclockfromstate()");
495 ws->send_command("eval startclock()");
497 ws->send_command("eval stopclock()");
501 if (msg.size() >= 10 && msg[0] == 'G' && msg[1] == '0' && msg[2] == '2') {
502 int score1 = parse_score(msg[4], msg[5], msg[6]);
503 int score2 = parse_score(msg[7], msg[8], msg[9]);
505 map<string, string> param;
506 param["score1"] = to_string(score1);
507 param["score2"] = to_string(score2);
508 ws->send_command("update " + serialize_as_json(param));
509 ws->send_command("eval setscore2()");
511 ui->score_1_box->setValue(score1);
512 ui->score_2_box->setValue(score2);
517 // Ignore type 3 (penalties) and type 4 (timeouts).