Overall timer
This commit is contained in:
40
main.go
40
main.go
@@ -32,6 +32,11 @@ type adminRequest struct {
|
|||||||
PublicClientId string `json:"public_client_id"`
|
PublicClientId string `json:"public_client_id"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type resetRequest struct {
|
||||||
|
RoomId string `json:"room_id"`
|
||||||
|
AdminSecret string `json:"admin_secret"`
|
||||||
|
}
|
||||||
|
|
||||||
type announceRequest struct {
|
type announceRequest struct {
|
||||||
RoomId string `json:"room_id"`
|
RoomId string `json:"room_id"`
|
||||||
ClientId string `json:"client_id"`
|
ClientId string `json:"client_id"`
|
||||||
@@ -79,6 +84,7 @@ type adminEvent struct {
|
|||||||
|
|
||||||
type standardEvent struct {
|
type standardEvent struct {
|
||||||
Active bool `json:"active"`
|
Active bool `json:"active"`
|
||||||
|
TimerStart int64 `json:"timer_start"`
|
||||||
AdminSecret string `json:"admin_secret"`
|
AdminSecret string `json:"admin_secret"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -88,6 +94,7 @@ type controlEvent struct {
|
|||||||
|
|
||||||
type room struct {
|
type room struct {
|
||||||
roomId string
|
roomId string
|
||||||
|
timerStart time.Time
|
||||||
clientById map[string]*client
|
clientById map[string]*client
|
||||||
clientByPublicId map[string]*client
|
clientByPublicId map[string]*client
|
||||||
present map[*presentState]bool
|
present map[*presentState]bool
|
||||||
@@ -140,6 +147,7 @@ func main() {
|
|||||||
http.HandleFunc("/api/create", create)
|
http.HandleFunc("/api/create", create)
|
||||||
http.HandleFunc("/api/present", present)
|
http.HandleFunc("/api/present", present)
|
||||||
http.HandleFunc("/api/remove", remove)
|
http.HandleFunc("/api/remove", remove)
|
||||||
|
http.HandleFunc("/api/reset", reset)
|
||||||
http.HandleFunc("/api/watch", watch)
|
http.HandleFunc("/api/watch", watch)
|
||||||
|
|
||||||
server := http.Server{
|
server := http.Server{
|
||||||
@@ -386,6 +394,29 @@ func remove(w http.ResponseWriter, r *http.Request) {
|
|||||||
c.remove()
|
c.remove()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func reset(w http.ResponseWriter, r *http.Request) {
|
||||||
|
mu.Lock()
|
||||||
|
defer mu.Unlock()
|
||||||
|
|
||||||
|
req := &resetRequest{}
|
||||||
|
|
||||||
|
err := json.NewDecoder(r.Body).Decode(req)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
rm := getRoom(req.RoomId)
|
||||||
|
|
||||||
|
if req.AdminSecret != rm.adminSecret() {
|
||||||
|
http.Error(w, "invalid admin_secret", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
rm.timerStart = time.Now()
|
||||||
|
rm.updateAllClients()
|
||||||
|
}
|
||||||
|
|
||||||
func watch(w http.ResponseWriter, r *http.Request) {
|
func watch(w http.ResponseWriter, r *http.Request) {
|
||||||
ws := newWatchState(w, r)
|
ws := newWatchState(w, r)
|
||||||
if ws == nil {
|
if ws == nil {
|
||||||
@@ -436,6 +467,7 @@ func (c *client) update() {
|
|||||||
e := &event{
|
e := &event{
|
||||||
StandardEvent: &standardEvent{
|
StandardEvent: &standardEvent{
|
||||||
Active: c.Active,
|
Active: c.Active,
|
||||||
|
TimerStart: c.room.timerStart.Unix(),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
if c.Admin {
|
if c.Admin {
|
||||||
@@ -447,6 +479,7 @@ func (c *client) update() {
|
|||||||
func newRoom(roomId string) *room {
|
func newRoom(roomId string) *room {
|
||||||
return &room{
|
return &room{
|
||||||
roomId: roomId,
|
roomId: roomId,
|
||||||
|
timerStart: time.Now(),
|
||||||
clientById: map[string]*client{},
|
clientById: map[string]*client{},
|
||||||
clientByPublicId: map[string]*client{},
|
clientByPublicId: map[string]*client{},
|
||||||
present: map[*presentState]bool{},
|
present: map[*presentState]bool{},
|
||||||
@@ -505,6 +538,12 @@ func (rm *room) sendControlEvent(ce *controlEvent) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (rm *room) updateAllClients() {
|
||||||
|
for _, client := range rm.clientById {
|
||||||
|
client.update()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func newWatchState(w http.ResponseWriter, r *http.Request) *watchState {
|
func newWatchState(w http.ResponseWriter, r *http.Request) *watchState {
|
||||||
mu.Lock()
|
mu.Lock()
|
||||||
defer mu.Unlock()
|
defer mu.Unlock()
|
||||||
@@ -539,6 +578,7 @@ func newWatchState(w http.ResponseWriter, r *http.Request) *watchState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ws.client.eventChan = ws.eventChan
|
ws.client.eventChan = ws.eventChan
|
||||||
|
ws.client.update()
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "text/event-stream")
|
w.Header().Set("Content-Type", "text/event-stream")
|
||||||
w.Header().Set("Cache-Control", "no-cache")
|
w.Header().Set("Cache-Control", "no-cache")
|
||||||
|
|||||||
@@ -78,6 +78,11 @@ tfoot tr {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.action {
|
||||||
|
cursor: pointer;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
.github {
|
.github {
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
color: var(--subtle-color);
|
color: var(--subtle-color);
|
||||||
|
|||||||
@@ -63,6 +63,7 @@ function watch(roomId, clientId, adminSecret, prnt) {
|
|||||||
}
|
}
|
||||||
const es = new EventSource(url.toString());
|
const es = new EventSource(url.toString());
|
||||||
renderControls(roomId, clientId, adminSecret, prnt, es);
|
renderControls(roomId, clientId, adminSecret, prnt, es);
|
||||||
|
renderTimers(roomId, adminSecret, prnt, es);
|
||||||
if (adminSecret) {
|
if (adminSecret) {
|
||||||
renderAdmin(roomId, adminSecret, prnt, es);
|
renderAdmin(roomId, adminSecret, prnt, es);
|
||||||
}
|
}
|
||||||
@@ -100,11 +101,47 @@ function renderControls(roomId, clientId, adminSecret, prnt, es) {
|
|||||||
controls.classList.remove("enable");
|
controls.classList.remove("enable");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
const clockDiv = create(prnt, "div", "\u00a0Time: ");
|
}
|
||||||
|
function renderTimers(roomId, adminSecret, prnt, es) {
|
||||||
|
let overallStart = null;
|
||||||
|
es.addEventListener("message", (e) => {
|
||||||
|
const event = JSON.parse(e.data);
|
||||||
|
if (!event.standard_event) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
overallStart = parseInt(event.standard_event.timer_start || "0", 10) || null;
|
||||||
|
});
|
||||||
|
const width = 10;
|
||||||
|
const clockDiv = create(prnt, "div", "Clock: ".padStart(width, "\u00a0"));
|
||||||
const clock = create(clockDiv, "span");
|
const clock = create(clockDiv, "span");
|
||||||
|
const overallDiv = create(prnt, "div", "Overall: ".padStart(width, "\u00a0"));
|
||||||
|
const overall = create(overallDiv, "span");
|
||||||
|
if (adminSecret) {
|
||||||
|
const reset = create(overallDiv, "span", "↺", ["action"]);
|
||||||
|
reset.addEventListener("click", () => {
|
||||||
|
const req = {
|
||||||
|
room_id: roomId,
|
||||||
|
admin_secret: adminSecret,
|
||||||
|
};
|
||||||
|
fetch("/api/reset", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
body: JSON.stringify(req),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
setInterval(() => {
|
setInterval(() => {
|
||||||
const now = new Date();
|
const now = new Date();
|
||||||
clock.innerText = `${now.getHours().toString().padStart(2, "0")}:${now.getMinutes().toString().padStart(2, "0")}:${now.getSeconds().toString().padStart(2, "0")}`;
|
clock.innerText = `${now.getHours().toString().padStart(2, "0")}h${now.getMinutes().toString().padStart(2, "0")}m${now.getSeconds().toString().padStart(2, "0")}s`;
|
||||||
|
if (overallStart) {
|
||||||
|
const o = Math.trunc(now.getTime() / 1000 - overallStart);
|
||||||
|
overall.innerText = `${Math.trunc(o / 3600).toString().padStart(2, "0")}h${Math.trunc(o % 3600 / 60).toString().padStart(2, "0")}m${Math.trunc(o % 60).toString().padStart(2, "0")}s`;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
overall.innerText = "";
|
||||||
|
}
|
||||||
}, 250);
|
}, 250);
|
||||||
}
|
}
|
||||||
function renderAdmin(roomId, adminSecret, prnt, es) {
|
function renderAdmin(roomId, adminSecret, prnt, es) {
|
||||||
|
|||||||
@@ -34,12 +34,18 @@ interface RemoveRequest {
|
|||||||
client_id: string;
|
client_id: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface ResetRequest {
|
||||||
|
room_id: string;
|
||||||
|
admin_secret: string;
|
||||||
|
}
|
||||||
|
|
||||||
interface Event {
|
interface Event {
|
||||||
standard_event?: StandardEvent;
|
standard_event?: StandardEvent;
|
||||||
admin_event?: AdminEvent;
|
admin_event?: AdminEvent;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface StandardEvent {
|
interface StandardEvent {
|
||||||
|
timer_start?: string;
|
||||||
active?: boolean;
|
active?: boolean;
|
||||||
admin_secret?: string;
|
admin_secret?: string;
|
||||||
}
|
}
|
||||||
@@ -135,6 +141,8 @@ function watch(roomId: string, clientId: string, adminSecret: string | null, prn
|
|||||||
|
|
||||||
renderControls(roomId, clientId, adminSecret, prnt, es);
|
renderControls(roomId, clientId, adminSecret, prnt, es);
|
||||||
|
|
||||||
|
renderTimers(roomId, adminSecret, prnt, es);
|
||||||
|
|
||||||
if (adminSecret) {
|
if (adminSecret) {
|
||||||
renderAdmin(roomId, adminSecret, prnt, es);
|
renderAdmin(roomId, adminSecret, prnt, es);
|
||||||
}
|
}
|
||||||
@@ -180,12 +188,57 @@ function renderControls(roomId: string, clientId: string, adminSecret: string |
|
|||||||
controls.classList.remove("enable");
|
controls.classList.remove("enable");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const clockDiv = create(prnt, "div", "\u00a0Time: ");
|
function renderTimers(roomId: string, adminSecret: string | null, prnt: HTMLElement, es: EventSource) {
|
||||||
|
let overallStart: number | null = null;
|
||||||
|
|
||||||
|
es.addEventListener("message", (e) => {
|
||||||
|
const event = JSON.parse(e.data) as Event;
|
||||||
|
|
||||||
|
if (!event.standard_event) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
overallStart = parseInt(event.standard_event.timer_start || "0", 10) || null;
|
||||||
|
});
|
||||||
|
|
||||||
|
const width = 10;
|
||||||
|
|
||||||
|
const clockDiv = create(prnt, "div", "Clock: ".padStart(width, "\u00a0"));
|
||||||
const clock = create(clockDiv, "span");
|
const clock = create(clockDiv, "span");
|
||||||
|
|
||||||
|
const overallDiv = create(prnt, "div", "Overall: ".padStart(width, "\u00a0"));
|
||||||
|
const overall = create(overallDiv, "span");
|
||||||
|
|
||||||
|
if (adminSecret) {
|
||||||
|
const reset = create(overallDiv, "span", "↺", ["action"]);
|
||||||
|
reset.addEventListener("click", () => {
|
||||||
|
const req: ResetRequest = {
|
||||||
|
room_id: roomId,
|
||||||
|
admin_secret: adminSecret,
|
||||||
|
};
|
||||||
|
|
||||||
|
fetch("/api/reset", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
body: JSON.stringify(req),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
setInterval(() => {
|
setInterval(() => {
|
||||||
const now = new Date();
|
const now = new Date();
|
||||||
clock.innerText = `${now.getHours().toString().padStart(2, "0")}:${now.getMinutes().toString().padStart(2, "0")}:${now.getSeconds().toString().padStart(2, "0")}`;
|
clock.innerText = `${now.getHours().toString().padStart(2, "0")}h${now.getMinutes().toString().padStart(2, "0")}m${now.getSeconds().toString().padStart(2, "0")}s`;
|
||||||
|
|
||||||
|
if (overallStart) {
|
||||||
|
const o = Math.trunc(now.getTime() / 1000 - overallStart);
|
||||||
|
overall.innerText = `${Math.trunc(o / 3600).toString().padStart(2, "0")}h${Math.trunc(o % 3600 / 60).toString().padStart(2, "0")}m${Math.trunc(o % 60).toString().padStart(2, "0")}s`;
|
||||||
|
} else {
|
||||||
|
overall.innerText = "";
|
||||||
|
}
|
||||||
}, 250);
|
}, 250);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user