2022-04-17 20:50:04 +03:00
|
|
|
import { writable } from "svelte/store";
|
2022-04-17 04:08:40 +03:00
|
|
|
import gateway, { GatewayEventType } from "./gateway";
|
2022-04-19 21:21:55 +03:00
|
|
|
import request from "./request";
|
|
|
|
import { apiRoute } from "./storage";
|
2022-04-17 04:08:40 +03:00
|
|
|
|
2022-04-17 20:23:20 +03:00
|
|
|
class Store {
|
2022-04-19 02:22:32 +03:00
|
|
|
constructor(value=null) {
|
2022-04-17 20:23:20 +03:00
|
|
|
this._handlers = [];
|
2022-04-19 02:22:32 +03:00
|
|
|
this.value = value;
|
2022-04-17 20:23:20 +03:00
|
|
|
}
|
|
|
|
|
2022-04-17 04:08:40 +03:00
|
|
|
subscribe(handler) {
|
|
|
|
const newLength = this._handlers.push(handler);
|
|
|
|
const handlerIndex = newLength - 1;
|
2022-04-19 02:17:24 +03:00
|
|
|
handler(this.value);
|
2022-04-17 04:08:40 +03:00
|
|
|
return () => {
|
|
|
|
this._handlers.splice(handlerIndex, 1);
|
|
|
|
};
|
2022-04-17 20:23:20 +03:00
|
|
|
}
|
|
|
|
|
2022-04-19 02:17:24 +03:00
|
|
|
updated() {
|
2022-04-17 04:08:40 +03:00
|
|
|
this._handlers.forEach(e => {
|
2022-04-19 02:17:24 +03:00
|
|
|
e(this.value);
|
2022-04-17 04:08:40 +03:00
|
|
|
});
|
2022-04-17 20:23:20 +03:00
|
|
|
}
|
|
|
|
}
|
2022-04-17 04:08:40 +03:00
|
|
|
|
2022-04-17 20:23:20 +03:00
|
|
|
class ChannelsStore extends Store {
|
|
|
|
constructor() {
|
2022-04-19 02:22:32 +03:00
|
|
|
super(gateway.channels || []);
|
2022-04-17 04:08:40 +03:00
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Ready, ({ channels }) => {
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value = channels;
|
|
|
|
this.updated();
|
2022-04-17 04:08:40 +03:00
|
|
|
});
|
|
|
|
gateway.subscribe(GatewayEventType.ChannelCreate, (channel) => {
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value.push(channel);
|
|
|
|
this.updated();
|
2022-04-17 04:08:40 +03:00
|
|
|
});
|
|
|
|
gateway.subscribe(GatewayEventType.ChannelDelete, ({ id }) => {
|
2022-04-19 02:17:24 +03:00
|
|
|
const index = this.value.findIndex(e => e.id === id);
|
2022-04-19 02:19:24 +03:00
|
|
|
if (index === -1)
|
2022-04-17 04:08:40 +03:00
|
|
|
return;
|
|
|
|
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value.splice(index, 1);
|
|
|
|
this.updated();
|
2022-04-17 04:08:40 +03:00
|
|
|
});
|
|
|
|
gateway.subscribe(GatewayEventType.ChannelUpdate, (data) => {
|
2022-04-19 02:17:24 +03:00
|
|
|
const index = this.value.findIndex(e => e.id === data.id);
|
2022-04-19 02:19:24 +03:00
|
|
|
if (index === -1)
|
2022-04-17 04:08:40 +03:00
|
|
|
return;
|
2022-04-19 02:17:24 +03:00
|
|
|
if (!this.value[index])
|
2022-04-17 04:08:40 +03:00
|
|
|
return;
|
|
|
|
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value[index] = data;
|
|
|
|
this.updated();
|
2022-04-17 04:08:40 +03:00
|
|
|
});
|
|
|
|
}
|
2022-04-17 20:23:20 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
class GatewayStatusStore extends Store {
|
|
|
|
constructor() {
|
2022-04-19 02:22:32 +03:00
|
|
|
super({ open: gateway.open, ready: gateway.authenticated });
|
2022-04-17 20:23:20 +03:00
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Open, () => {
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value.open = true;
|
|
|
|
this.updated();
|
2022-04-17 20:23:20 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Close, () => {
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value.open = false;
|
|
|
|
this.value.ready = false;
|
|
|
|
this.updated();
|
2022-04-17 20:23:20 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Ready, () => {
|
2022-04-19 02:17:24 +03:00
|
|
|
this.value.ready = true;
|
|
|
|
this.updated();
|
2022-04-17 20:23:20 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-20 03:44:48 +03:00
|
|
|
class UserInfoStore extends Store {
|
|
|
|
constructor() {
|
|
|
|
super(null);
|
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Ready, ({ user }) => {
|
|
|
|
this.value = user;
|
|
|
|
this.updated();
|
|
|
|
});
|
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.Close, () => {
|
|
|
|
this.value = null;
|
|
|
|
this.updated();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-19 21:21:55 +03:00
|
|
|
class MessageStore extends Store {
|
|
|
|
constructor(channelId) {
|
|
|
|
super([]);
|
|
|
|
this.channelId = channelId;
|
|
|
|
this.isCollectingOldMessages = true;
|
|
|
|
this.didDoInitialLoad = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
addMessage(message) {
|
|
|
|
this.value.push(message);
|
|
|
|
// only dispatch update if collectOldMessages didn't
|
|
|
|
if (!this.collectOldMessages()) {
|
|
|
|
this.updated();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-20 03:44:48 +03:00
|
|
|
updateId(oldId, newId) {
|
|
|
|
const index = this.value.findIndex(e => e.id === oldId);
|
|
|
|
if (index === -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this.value[index].id = newId;
|
|
|
|
|
|
|
|
this.updated();
|
|
|
|
}
|
|
|
|
|
2022-04-19 21:21:55 +03:00
|
|
|
updateMessage(message) {
|
|
|
|
const index = this.value.findIndex(e => e.id === message.id);
|
|
|
|
if (index === -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this.value[index] = message;
|
|
|
|
|
|
|
|
this.updated();
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteMessage({ id }) {
|
|
|
|
const index = this.value.findIndex(e => e.id === id);
|
|
|
|
if (index === -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this.value.splice(index, 1);
|
|
|
|
|
|
|
|
this.updated();
|
|
|
|
}
|
|
|
|
|
|
|
|
collectOldMessages() {
|
|
|
|
if (!this.isCollectingOldMessages)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const target = 50;
|
|
|
|
const delta = this.value.length - target;
|
|
|
|
if (delta >= 1) {
|
|
|
|
this.value.splice(0, delta);
|
|
|
|
this.updated();
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setIsCollectingOldMessages(isCollectingOldMessages) {
|
|
|
|
this.isCollectingOldMessages = isCollectingOldMessages;
|
|
|
|
this.collectOldMessages();
|
|
|
|
}
|
|
|
|
|
|
|
|
async loadOlderMessages() {
|
|
|
|
const oldestMessage = this.value[0];
|
|
|
|
const endpoint = oldestMessage ? `channels/${this.channelId}/messages/?before=${oldestMessage.id}` : `channels/${this.channelId}/messages`;
|
|
|
|
const res = await request("GET", apiRoute(endpoint), true, null);
|
|
|
|
if (res.success && res.ok && res.json && res.json.length > 0) {
|
2022-04-20 03:14:28 +03:00
|
|
|
res.json.reverse();
|
2022-04-19 21:21:55 +03:00
|
|
|
this.value = res.json.concat(this.value);
|
|
|
|
this.updated();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async doInitialLoad() {
|
|
|
|
await this.loadOlderMessages();
|
|
|
|
this.didDoInitialLoad = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MessagesStoreProvider {
|
|
|
|
constructor() {
|
|
|
|
this.storeByChannel = new Map();
|
|
|
|
|
|
|
|
gateway.subscribe(GatewayEventType.MessageCreate, (message) => {
|
2022-04-20 03:44:48 +03:00
|
|
|
// we currently don't care about our own messages
|
|
|
|
if (gateway.user && message.author_id === gateway.user.id)
|
|
|
|
return;
|
2022-04-19 21:21:55 +03:00
|
|
|
this.getStore(message.channel_id).addMessage(message);
|
|
|
|
});
|
|
|
|
gateway.subscribe(GatewayEventType.MessageUpdate, (message) => {
|
|
|
|
this.getStore(message.channel_id).updateMessage(message);
|
|
|
|
});
|
|
|
|
gateway.subscribe(GatewayEventType.MessageDelete, (message) => {
|
|
|
|
this.getStore(message.channel_id).deleteMessage(message);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
getStore(channelId) {
|
2022-04-20 03:14:28 +03:00
|
|
|
if (!this.storeByChannel.get(channelId)) {
|
2022-04-20 03:44:48 +03:00
|
|
|
const store = new MessageStore(channelId);
|
|
|
|
store.doInitialLoad();
|
|
|
|
this.storeByChannel.set(channelId, store);
|
2022-04-19 21:21:55 +03:00
|
|
|
}
|
2022-04-20 03:14:28 +03:00
|
|
|
return this.storeByChannel.get(channelId);
|
2022-04-19 21:21:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-17 20:23:20 +03:00
|
|
|
export const channels = new ChannelsStore();
|
|
|
|
export const gatewayStatus = new GatewayStatusStore();
|
2022-04-19 21:21:55 +03:00
|
|
|
export const messagesStoreProvider = new MessagesStoreProvider();
|
2022-04-20 03:44:48 +03:00
|
|
|
export const userInfoStore = new UserInfoStore();
|
2022-04-17 20:50:04 +03:00
|
|
|
export const selectedChannel = writable({ id: -1, name: "none", creator_id: -1 });
|