1 #include <QMediaPlayer>
3 #include <QApplication>
5 #include <QVideoWidget>
13 #include "mainwindow.h"
14 #include "ui_mainwindow.h"
18 string format_timestamp(uint64_t pos)
28 snprintf(buf, sizeof(buf), "%d:%02d:%02d.%03d", hour, min, sec, ms);
32 class EventsModel : public QAbstractTableModel
35 EventsModel(sqlite3 *db) : db(db) {}
37 int rowCount(const QModelIndex &parent) const override
42 int columnCount(const QModelIndex &column) const override
46 QVariant headerData(int section, Qt::Orientation orientation, int role) const override;
47 QVariant data(const QModelIndex &index, int role) const override;
49 int insert_event(uint64_t t, int player_id);
57 mutable map<int, Player> players;
61 optional<int> player_id;
64 mutable vector<Event> events;
65 mutable bool stale = true;
69 void refresh_if_needed() const;
72 QVariant EventsModel::headerData(int section, Qt::Orientation orientation, int role) const
74 if (role != Qt::DisplayRole) {
77 if (orientation == Qt::Horizontal) {
80 } else if (section == 1) {
90 QVariant EventsModel::data(const QModelIndex &index, int role) const
92 if (role != Qt::DisplayRole) {
96 if (index.column() == 0) {
97 return QString::fromUtf8(format_timestamp(events[index.row()].t));
98 } else if (index.column() == 1) {
99 optional<int> player_id = events[index.row()].player_id;
101 const Player &p = players[*player_id];
102 return QString::fromUtf8(p.name + " (" + p.number + ")");
106 } else if (index.column() == 2) {
107 return QString::fromUtf8(events[index.row()].type);
112 void EventsModel::refresh_if_needed() const
124 int ret = sqlite3_prepare_v2(db, "SELECT player, number, name FROM player", -1, &stmt, 0);
125 if (ret != SQLITE_OK) {
126 fprintf(stderr, "SELECT prepare: %s\n", sqlite3_errmsg(db));
130 ret = sqlite3_step(stmt);
131 if (ret == SQLITE_ROW) {
133 p.player_id = sqlite3_column_int(stmt, 0);
134 p.number = (const char *)sqlite3_column_text(stmt, 1);
135 p.name = (const char *) sqlite3_column_text(stmt, 2);
136 players[p.player_id] = std::move(p);
137 } else if (ret == SQLITE_DONE) {
140 fprintf(stderr, "SELECT step: %s\n", sqlite3_errmsg(db));
144 ret = sqlite3_finalize(stmt);
145 if (ret != SQLITE_OK) {
146 fprintf(stderr, "SELECT finalize: %s\n", sqlite3_errmsg(db));
151 ret = sqlite3_prepare_v2(db, "SELECT t, player, type FROM event", -1, &stmt, 0);
152 if (ret != SQLITE_OK) {
153 fprintf(stderr, "SELECT prepare: %s\n", sqlite3_errmsg(db));
157 ret = sqlite3_step(stmt);
158 if (ret == SQLITE_ROW) {
160 e.t = sqlite3_column_int(stmt, 0);
161 e.player_id = sqlite3_column_int(stmt, 1);
162 e.type = (const char *)sqlite3_column_text(stmt, 2);
163 events.push_back(std::move(e));
164 } else if (ret == SQLITE_DONE) {
167 fprintf(stderr, "SELECT step: %s\n", sqlite3_errmsg(db));
171 ret = sqlite3_finalize(stmt);
172 if (ret != SQLITE_OK) {
173 fprintf(stderr, "SELECT finalize: %s\n", sqlite3_errmsg(db));
177 // TODO what if data changes externally?
178 //emit dataChanged(QModelIndex(
181 int EventsModel::insert_event(uint64_t t, int player_id)
183 auto it = lower_bound(events.begin(), events.end(), t,
184 [](const Event &e, uint64_t t) { return e.t < t; });
185 int pos = distance(events.begin(), it);
186 beginInsertRows(QModelIndex(), pos, pos + 1);
190 e.player_id = player_id;
192 events.insert(events.begin() + pos, e);
201 MainWindow::MainWindow()
203 player = new QMediaPlayer;
204 //player->setSource(QUrl::fromLocalFile("/home/sesse/dev/stats/ultimate.mkv"));
205 player->setSource(QUrl::fromLocalFile("/home/sesse/dev/stats/ultimate-prores.mkv"));
208 ui = new Ui::MainWindow;
211 connect(player, &QMediaPlayer::positionChanged, [this](uint64_t pos) {
212 ui->timestamp->setText(QString::fromUtf8(format_timestamp(pos)));
214 player->setPosition(*buffered_seek);
215 buffered_seek.reset();
218 player->pause(); // We only played to get a picture.
222 player->setVideoOutput(ui->video);
224 connect(ui->minus10s, &QPushButton::clicked, [this]() { seek(-10000); });
225 connect(ui->plus10s, &QPushButton::clicked, [this]() { seek(10000); });
227 connect(ui->minus2s, &QPushButton::clicked, [this]() { seek(-2000); });
228 connect(ui->plus2s, &QPushButton::clicked, [this]() { seek(2000); });
230 // TODO: Would be nice to actually have a frame...
231 connect(ui->minus1f, &QPushButton::clicked, [this]() { seek(-20); });
232 connect(ui->plus1f, &QPushButton::clicked, [this]() { seek(20); });
234 connect(ui->play_pause, &QPushButton::clicked, [this]() {
237 ui->play_pause->setText("Play (space)");
239 player->setPlaybackRate(1.0);
241 ui->play_pause->setText("Pause (space)");
245 // Needs to be set anew when we modify setText(), evidently.
246 ui->play_pause->setShortcut(QCoreApplication::translate("MainWindow", "Space", nullptr));
249 connect(ui->player_1, &QPushButton::clicked, [this]() {
250 ui->event_view->selectRow(model->insert_event(player->position(), 1));
252 connect(ui->player_2, &QPushButton::clicked, [this]() {
253 ui->event_view->selectRow(model->insert_event(player->position(), 2));
255 connect(ui->player_3, &QPushButton::clicked, [this]() {
256 ui->event_view->selectRow(model->insert_event(player->position(), 3));
258 connect(ui->player_4, &QPushButton::clicked, [this]() {
259 ui->event_view->selectRow(model->insert_event(player->position(), 4));
261 connect(ui->player_5, &QPushButton::clicked, [this]() {
262 ui->event_view->selectRow(model->insert_event(player->position(), 5));
264 connect(ui->player_6, &QPushButton::clicked, [this]() {
265 ui->event_view->selectRow(model->insert_event(player->position(), 6));
267 connect(ui->player_7, &QPushButton::clicked, [this]() {
268 ui->event_view->selectRow(model->insert_event(player->position(), 7));
272 void MainWindow::setModel(EventsModel *model)
274 ui->event_view->setModel(model);
278 void MainWindow::seek(int64_t delta_ms)
280 int64_t current_pos = buffered_seek ? *buffered_seek : player->position();
281 uint64_t pos = max<int64_t>(current_pos + delta_ms, 0);
284 player->setPlaybackRate(0.01);
285 player->play(); // Or Qt won't show the seek.
289 sqlite3 *open_db(const char *filename)
292 int ret = sqlite3_open(filename, &db);
293 if (ret != SQLITE_OK) {
294 fprintf(stderr, "%s: %s\n", filename, sqlite3_errmsg(db));
299 CREATE TABLE IF NOT EXISTS player (player INTEGER PRIMARY KEY, number VARCHAR, name VARCHAR);
300 )", nullptr, nullptr, nullptr); // Ignore errors.
303 CREATE TABLE IF NOT EXISTS event (t INTEGER, player INTEGER, type VARCHAR, FOREIGN KEY (player) REFERENCES player(player));
304 )", nullptr, nullptr, nullptr); // Ignore errors.
306 sqlite3_exec(db, "PRAGMA journal_mode=WAL", nullptr, nullptr, nullptr); // Ignore errors.
307 sqlite3_exec(db, "PRAGMA synchronous=NORMAL", nullptr, nullptr, nullptr); // Ignore errors.
311 int main(int argc, char *argv[])
313 QApplication app(argc, argv);
314 sqlite3 *db = open_db("ultimate.db");
316 MainWindow mainWindow;
317 mainWindow.setModel(new EventsModel(db));
318 mainWindow.resize(QSize(1280, 720));