+function Log(prefix: string = "vp-") {
+ var next_log_index = 0;
+ return {
+ apply: function (entry: string) {
+ const [timestamp, command, data] = splitN(entry, " ", 2);
+ if (command == "Create") {
+ Model.addTask(timestamp, data);
+ }
+ if (command == "Destroy") {
+ Model.destroyTask(data.split(" ", 1)[0]);
+ }
+ if (command == "State") {
+ const [createTimestamp, state] = splitN(data, " ", 1);
+ Model.setState(timestamp, createTimestamp, state);
+ }
+ },
+
+ record: function (entry: string) {
+ window.localStorage.setItem(`${prefix}${next_log_index++}`, entry);
+ },
+
+ recordAndApply: function (entry: string) {
+ this.record(entry);
+ this.apply(entry);
+ },
+
+ replay: function () {
+ while (true) {
+ const entry = window.localStorage.getItem(`${prefix}${next_log_index}`);
+ if (entry === null) {
+ break;
+ }
+ this.apply(entry);
+ next_log_index++;
+ }
+ },
+ };
+}
+const log = Log();
+
+const UI = {