LiveAtlas/src/store/actions.ts

266 lines
8.7 KiB
TypeScript
Raw Normal View History

2020-12-16 16:54:41 +00:00
/*
* Copyright 2020 James Lyne
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2020-11-23 16:16:26 +00:00
import {MutationTypes} from "@/store/mutation-types";
import {ActionContext, ActionTree} from "vuex";
import {State} from "@/store/state";
import {ActionTypes} from "@/store/action-types";
import API from '@/api';
import {Mutations} from "@/store/mutations";
import {
DynmapAreaUpdate, DynmapCircleUpdate,
DynmapConfigurationResponse, DynmapLineUpdate,
DynmapMarkerSet,
DynmapMarkerUpdate,
DynmapPlayer, DynmapTileUpdate,
DynmapUpdateResponse, DynmapWorld
} from "@/dynmap";
2020-11-23 16:16:26 +00:00
type AugmentedActionContext = {
commit<K extends keyof Mutations>(
key: K,
payload: Parameters<Mutations[K]>[1]
):ReturnType<Mutations[K]>;
} & Omit<ActionContext<State, State>, "commit">
export interface Actions {
[ActionTypes.LOAD_CONFIGURATION](
{commit}: AugmentedActionContext,
):Promise<DynmapConfigurationResponse>
[ActionTypes.GET_UPDATE](
{commit}: AugmentedActionContext,
):Promise<DynmapUpdateResponse>
2020-12-01 23:20:38 +00:00
[ActionTypes.GET_MARKER_SETS](
{commit}: AugmentedActionContext,
):Promise<Map<string, DynmapMarkerSet>>
[ActionTypes.SET_PLAYERS](
{commit}: AugmentedActionContext,
payload: Set<DynmapPlayer>
):Promise<Map<string, DynmapMarkerSet>>
[ActionTypes.POP_MARKER_UPDATES](
{commit}: AugmentedActionContext,
payload: {markerSet: string, amount: number}
): Promise<DynmapMarkerUpdate[]>
[ActionTypes.POP_AREA_UPDATES](
{commit}: AugmentedActionContext,
payload: {markerSet: string, amount: number}
): Promise<DynmapAreaUpdate[]>
[ActionTypes.POP_CIRCLE_UPDATES](
{commit}: AugmentedActionContext,
payload: {markerSet: string, amount: number}
): Promise<DynmapCircleUpdate[]>
[ActionTypes.POP_LINE_UPDATES](
{commit}: AugmentedActionContext,
payload: {markerSet: string, amount: number}
): Promise<DynmapLineUpdate[]>
[ActionTypes.POP_TILE_UPDATES](
{commit}: AugmentedActionContext,
payload: number
): Promise<DynmapTileUpdate[]>
2021-01-07 22:40:05 +00:00
[ActionTypes.SEND_CHAT_MESSAGE](
{commit}: AugmentedActionContext,
payload: string
): Promise<void>
2020-11-23 16:16:26 +00:00
}
export const actions: ActionTree<State, State> & Actions = {
2021-05-18 17:22:34 +00:00
async [ActionTypes.LOAD_CONFIGURATION]({commit, state}): Promise<DynmapConfigurationResponse> {
2021-05-17 23:19:51 +00:00
//Clear any existing has to avoid triggering a second config load, after this load changes the hash
commit(MutationTypes.CLEAR_CONFIGURATION_HASH, undefined);
2021-05-17 02:39:25 +00:00
2021-05-18 17:22:34 +00:00
const config = await API.getConfiguration();
2020-11-23 16:16:26 +00:00
2021-05-18 17:22:34 +00:00
commit(MutationTypes.SET_CONFIGURATION, config.config);
commit(MutationTypes.SET_MESSAGES, config.messages);
commit(MutationTypes.SET_WORLDS, config.worlds);
commit(MutationTypes.SET_COMPONENTS, config.components);
commit(MutationTypes.SET_LOGGED_IN, config.loggedIn);
2021-05-17 23:19:51 +00:00
2021-05-18 17:22:34 +00:00
//Skip default map/ui visibility logic if we already have a map selected (i.e config reload after hash change)
if(state.currentMap) {
return config;
}
2021-05-18 17:22:34 +00:00
if(state.configuration.expandUI && !state.ui.smallScreen) {
commit(MutationTypes.SET_UI_ELEMENT_VISIBILITY, {element: 'players', state: true});
commit(MutationTypes.SET_UI_ELEMENT_VISIBILITY, {element: 'maps', state: true});
}
2021-05-18 17:22:34 +00:00
let worldName, mapName;
2021-05-18 17:22:34 +00:00
// Use config default world if it exists
if(config.config.defaultWorld && state.worlds.has(config.config.defaultWorld)) {
worldName = config.config.defaultWorld;
}
2021-05-18 17:22:34 +00:00
// Prefer world from parsed url if present and it exists
if(state.parsedUrl.world && state.worlds.has(state.parsedUrl.world)) {
worldName = state.parsedUrl.world;
}
2021-05-18 17:22:34 +00:00
// Use first world, if any, if neither of the above exist
if(!worldName) {
worldName = state.worlds.size ? state.worlds.entries().next().value[1].name : undefined;
}
2021-05-18 17:22:34 +00:00
if(worldName) {
const world = state.worlds.get(worldName) as DynmapWorld;
2021-05-18 17:22:34 +00:00
// Use config default map if it exists
if(config.config.defaultMap && world.maps.has(config.config.defaultMap)) {
mapName = config.config.defaultMap;
}
2021-05-18 17:22:34 +00:00
// Prefer map from parsed url if present and it exists
if(state.parsedUrl.map && world.maps.has(state.parsedUrl.map)) {
mapName = state.parsedUrl.map;
}
2021-05-18 17:22:34 +00:00
// Use first map, if any, if neither of the above exist
if(!mapName) {
mapName = world.maps.size ? world.maps.entries().next().value[1].name : undefined;
2020-11-23 16:16:26 +00:00
}
2021-05-18 17:22:34 +00:00
}
2020-11-23 16:16:26 +00:00
2021-05-18 17:22:34 +00:00
if(worldName && mapName) {
commit(MutationTypes.SET_CURRENT_MAP, {
worldName, mapName
});
}
return config;
2020-11-23 16:16:26 +00:00
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.GET_UPDATE]({commit, dispatch, state}) {
2020-11-23 16:16:26 +00:00
if(!state.currentWorld) {
return Promise.reject("No current world");
}
2021-05-18 17:22:34 +00:00
const update = await API.getUpdate(state.updateRequestId, state.currentWorld.name, state.updateTimestamp.valueOf())
commit(MutationTypes.SET_WORLD_STATE, update.worldState);
commit(MutationTypes.SET_UPDATE_TIMESTAMP, new Date(update.timestamp));
commit(MutationTypes.INCREMENT_REQUEST_ID, undefined);
commit(MutationTypes.ADD_MARKER_SET_UPDATES, update.updates.markerSets);
commit(MutationTypes.ADD_TILE_UPDATES, update.updates.tiles);
commit(MutationTypes.ADD_CHAT, update.updates.chat);
commit(MutationTypes.SET_CONFIGURATION_HASH, update.configHash);
await dispatch(ActionTypes.SET_PLAYERS, update.players);
return update;
2020-12-01 23:20:38 +00:00
},
2021-05-18 17:22:34 +00:00
2020-12-01 23:20:38 +00:00
[ActionTypes.SET_PLAYERS]({commit, state}, players: Set<DynmapPlayer>) {
const keep: Set<string> = new Set();
for(const player of players) {
keep.add(player.account);
}
//Remove any players that aren't in the set
commit(MutationTypes.SYNC_PLAYERS, keep);
const processQueue = (players: Set<DynmapPlayer>, resolve: Function) => {
commit(MutationTypes.SET_PLAYERS_ASYNC, players);
if(!players.size) {
resolve();
} else {
requestAnimationFrame(() => processQueue(players, resolve));
}
}
//Set players every frame until done
return new Promise((resolve) => {
requestAnimationFrame(() => processQueue(players, resolve));
});
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.GET_MARKER_SETS]({commit, state}) {
2020-12-01 23:20:38 +00:00
if(!state.currentWorld) {
2021-05-18 17:22:34 +00:00
throw new Error("No current world");
2020-12-01 23:20:38 +00:00
}
2021-05-18 17:22:34 +00:00
const markerSets = await API.getMarkerSets(state.currentWorld.name)
commit(MutationTypes.SET_MARKER_SETS, markerSets);
2020-12-01 23:20:38 +00:00
2021-05-18 17:22:34 +00:00
return markerSets;
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.POP_MARKER_UPDATES]({commit, state}, {markerSet, amount}: {markerSet: string, amount: number}): Promise<DynmapMarkerUpdate[]> {
if(!state.markerSets.has(markerSet)) {
2020-12-18 16:12:14 +00:00
console.warn(`POP_MARKER_UPDATES: Marker set ${markerSet} doesn't exist`);
2021-05-18 17:22:34 +00:00
return [];
}
const updates = state.pendingSetUpdates.get(markerSet)!.markerUpdates.slice(0, amount);
commit(MutationTypes.POP_MARKER_UPDATES, {markerSet, amount});
2021-05-18 17:22:34 +00:00
return updates;
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.POP_AREA_UPDATES]({commit, state}, {markerSet, amount}: {markerSet: string, amount: number}): Promise<DynmapAreaUpdate[]> {
if(!state.markerSets.has(markerSet)) {
2020-12-18 16:12:14 +00:00
console.warn(`POP_AREA_UPDATES: Marker set ${markerSet} doesn't exist`);
2021-05-18 17:22:34 +00:00
return [];
}
const updates = state.pendingSetUpdates.get(markerSet)!.areaUpdates.slice(0, amount);
commit(MutationTypes.POP_AREA_UPDATES, {markerSet, amount});
2021-05-18 17:22:34 +00:00
return updates;
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.POP_CIRCLE_UPDATES]({commit, state}, {markerSet, amount}: {markerSet: string, amount: number}): Promise<DynmapCircleUpdate[]> {
if(!state.markerSets.has(markerSet)) {
2020-12-18 16:12:14 +00:00
console.warn(`POP_CIRCLE_UPDATES: Marker set ${markerSet} doesn't exist`);
2021-05-18 17:22:34 +00:00
return [];
}
const updates = state.pendingSetUpdates.get(markerSet)!.circleUpdates.slice(0, amount);
commit(MutationTypes.POP_CIRCLE_UPDATES, {markerSet, amount});
2021-05-18 17:22:34 +00:00
return updates;
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.POP_LINE_UPDATES]({commit, state}, {markerSet, amount}: {markerSet: string, amount: number}): Promise<DynmapLineUpdate[]> {
if(!state.markerSets.has(markerSet)) {
2020-12-18 16:12:14 +00:00
console.warn(`POP_LINE_UPDATES: Marker set ${markerSet} doesn't exist`);
2021-05-18 17:22:34 +00:00
return [];
}
const updates = state.pendingSetUpdates.get(markerSet)!.lineUpdates.slice(0, amount);
commit(MutationTypes.POP_LINE_UPDATES, {markerSet, amount});
2021-05-18 17:22:34 +00:00
return updates;
},
2021-05-18 17:22:34 +00:00
async [ActionTypes.POP_TILE_UPDATES]({commit, state}, amount: number): Promise<Array<DynmapTileUpdate>> {
const updates = state.pendingTileUpdates.slice(0, amount);
commit(MutationTypes.POP_TILE_UPDATES, amount);
2021-05-18 17:22:34 +00:00
return updates;
},
2021-01-07 22:40:05 +00:00
2021-05-18 17:22:34 +00:00
async [ActionTypes.SEND_CHAT_MESSAGE]({commit, state}, message: string): Promise<void> {
await API.sendChatMessage(message);
2021-01-07 22:40:05 +00:00
},
2020-11-23 16:16:26 +00:00
}