Refactor map handling
- Map definitions now have their own class - A map's projection is now a private property of the instance. Coordinates are converted via map instance methods. - Moved map icon handling to instance getter - Removed currentProjection and related mutations from store - Changed currentProjection watchers to currentMap
This commit is contained in:
parent
709e365944
commit
ef26d78c19
17
src/api.ts
17
src/api.ts
@ -33,7 +33,8 @@ import {
|
|||||||
} from "@/dynmap";
|
} from "@/dynmap";
|
||||||
import {useStore} from "@/store";
|
import {useStore} from "@/store";
|
||||||
import ChatError from "@/errors/ChatError";
|
import ChatError from "@/errors/ChatError";
|
||||||
import {LiveAtlasDimension, LiveAtlasServerMessageConfig, LiveAtlasWorld} from "@/index";
|
import {LiveAtlasDimension, LiveAtlasServerMessageConfig, LiveAtlasWorldDefinition} from "@/index";
|
||||||
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
const titleColours = /§[0-9a-f]/ig,
|
const titleColours = /§[0-9a-f]/ig,
|
||||||
netherWorldName = /_?nether(_|$)/i,
|
netherWorldName = /_?nether(_|$)/i,
|
||||||
@ -72,8 +73,8 @@ function buildMessagesConfig(response: any): LiveAtlasServerMessageConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function buildWorlds(response: any): Array<LiveAtlasWorld> {
|
function buildWorlds(response: any): Array<LiveAtlasWorldDefinition> {
|
||||||
const worlds: Map<string, LiveAtlasWorld> = new Map<string, LiveAtlasWorld>();
|
const worlds: Map<string, LiveAtlasWorldDefinition> = new Map<string, LiveAtlasWorldDefinition>();
|
||||||
|
|
||||||
//Get all the worlds first so we can handle append_to_world properly
|
//Get all the worlds first so we can handle append_to_world properly
|
||||||
(response.worlds || []).forEach((world: any) => {
|
(response.worlds || []).forEach((world: any) => {
|
||||||
@ -111,7 +112,7 @@ function buildWorlds(response: any): Array<LiveAtlasWorld> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
w.maps.set(map.name, {
|
w.maps.set(map.name, new LiveAtlasMapDefinition({
|
||||||
world: world, //Ignore append_to_world here otherwise things break
|
world: world, //Ignore append_to_world here otherwise things break
|
||||||
background: map.background || '#000000',
|
background: map.background || '#000000',
|
||||||
backgroundDay: map.backgroundday || '#000000',
|
backgroundDay: map.backgroundday || '#000000',
|
||||||
@ -123,11 +124,11 @@ function buildWorlds(response: any): Array<LiveAtlasWorld> {
|
|||||||
prefix: map.prefix || '',
|
prefix: map.prefix || '',
|
||||||
protected: map.protected || false,
|
protected: map.protected || false,
|
||||||
title: map.title || '',
|
title: map.title || '',
|
||||||
mapToWorld: map.maptoworld || [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
mapToWorld: map.maptoworld || undefined,
|
||||||
worldToMap: map.worldtomap || [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
worldToMap: map.worldtomap || undefined,
|
||||||
nativeZoomLevels: map.mapzoomout || 1,
|
nativeZoomLevels: map.mapzoomout || 1,
|
||||||
extraZoomLevels: map.mapzoomin || 0,
|
extraZoomLevels: map.mapzoomin || 0
|
||||||
});
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -81,7 +81,6 @@ export default defineComponent({
|
|||||||
|
|
||||||
currentWorld = computed(() => store.state.currentWorld),
|
currentWorld = computed(() => store.state.currentWorld),
|
||||||
currentMap = computed(() => store.state.currentMap),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
|
||||||
mapBackground = computed(() => store.getters.mapBackground),
|
mapBackground = computed(() => store.getters.mapBackground),
|
||||||
|
|
||||||
followTarget = computed(() => store.state.followTarget),
|
followTarget = computed(() => store.state.followTarget),
|
||||||
@ -114,7 +113,6 @@ export default defineComponent({
|
|||||||
|
|
||||||
currentWorld,
|
currentWorld,
|
||||||
currentMap,
|
currentMap,
|
||||||
currentProjection,
|
|
||||||
|
|
||||||
scheduledPan,
|
scheduledPan,
|
||||||
scheduledZoom,
|
scheduledZoom,
|
||||||
@ -142,7 +140,7 @@ export default defineComponent({
|
|||||||
this.updateFollow(newValue, false);
|
this.updateFollow(newValue, false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
currentProjection(newValue, oldValue) {
|
currentMap(newValue, oldValue) {
|
||||||
if(this.leaflet && newValue && oldValue) {
|
if(this.leaflet && newValue && oldValue) {
|
||||||
const panTarget = this.scheduledPan || oldValue.latLngToLocation(this.leaflet.getCenter(), 64);
|
const panTarget = this.scheduledPan || oldValue.latLngToLocation(this.leaflet.getCenter(), 64);
|
||||||
|
|
||||||
@ -196,12 +194,12 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
parsedUrl: {
|
parsedUrl: {
|
||||||
handler(newValue) {
|
handler(newValue) {
|
||||||
if(!newValue || !this.currentWorld || !this.leaflet) {
|
if(!newValue || !this.currentMap || !this.leaflet) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
//URL points to different map
|
//URL points to different map
|
||||||
if(newValue.world !== this.currentWorld.name || newValue.map !== this.currentMap!.name) {
|
if(newValue.world !== this.currentWorld!.name || newValue.map !== this.currentMap!.name) {
|
||||||
//Set scheduled pan for after map change
|
//Set scheduled pan for after map change
|
||||||
this.scheduledPan = newValue.location;
|
this.scheduledPan = newValue.location;
|
||||||
this.scheduledZoom = newValue.zoom;
|
this.scheduledZoom = newValue.zoom;
|
||||||
@ -225,7 +223,7 @@ export default defineComponent({
|
|||||||
animate: false,
|
animate: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.leaflet.panTo(this.currentProjection.locationToLatLng(newValue.location), {
|
this.leaflet.panTo(this.currentMap.locationToLatLng(newValue.location), {
|
||||||
animate: false,
|
animate: false,
|
||||||
noMoveStart: true,
|
noMoveStart: true,
|
||||||
});
|
});
|
||||||
@ -259,7 +257,10 @@ export default defineComponent({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
this.leaflet.on('moveend', () => {
|
this.leaflet.on('moveend', () => {
|
||||||
useStore().commit(MutationTypes.SET_CURRENT_LOCATION, this.currentProjection.latLngToLocation(this.leaflet!.getCenter(), 64));
|
if(this.currentMap) {
|
||||||
|
useStore().commit(MutationTypes.SET_CURRENT_LOCATION, this.currentMap
|
||||||
|
.latLngToLocation(this.leaflet!.getCenter(), 64));
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.leaflet.on('zoomend', () => {
|
this.leaflet.on('zoomend', () => {
|
||||||
@ -327,7 +328,7 @@ export default defineComponent({
|
|||||||
console.log(`Switching map to match player ${targetWorld.name} ${map.name}`);
|
console.log(`Switching map to match player ${targetWorld.name} ${map.name}`);
|
||||||
store.commit(MutationTypes.SET_CURRENT_MAP, {worldName: targetWorld.name, mapName: map.name});
|
store.commit(MutationTypes.SET_CURRENT_MAP, {worldName: targetWorld.name, mapName: map.name});
|
||||||
} else {
|
} else {
|
||||||
this.leaflet!.panTo(store.state.currentProjection.locationToLatLng(player.location));
|
this.leaflet!.panTo(store.state.currentMap?.locationToLatLng(player.location));
|
||||||
|
|
||||||
if(newFollow) {
|
if(newFollow) {
|
||||||
console.log(`Setting zoom for new follow ${store.state.configuration.followZoom}`);
|
console.log(`Setting zoom for new follow ${store.state.configuration.followZoom}`);
|
||||||
|
@ -59,18 +59,17 @@ export default defineComponent({
|
|||||||
menuElement = ref<HTMLInputElement | null>(null),
|
menuElement = ref<HTMLInputElement | null>(null),
|
||||||
menuVisible = computed(() => !!event.value),
|
menuVisible = computed(() => !!event.value),
|
||||||
|
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
|
||||||
currentWorld = computed(() => store.state.currentWorld),
|
currentWorld = computed(() => store.state.currentWorld),
|
||||||
currentMap = computed(() => store.state.currentMap),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
currentZoom = computed(() => store.state.currentZoom),
|
currentZoom = computed(() => store.state.currentZoom),
|
||||||
mapCount = computed(() => currentWorld.value ? currentWorld.value.maps.size : 0),
|
mapCount = computed(() => currentWorld.value ? currentWorld.value.maps.size : 0),
|
||||||
|
|
||||||
location = computed(() => {
|
location = computed(() => {
|
||||||
if (!event.value) {
|
if (!event.value || !currentMap.value) {
|
||||||
return {x: 0, y: 0, z: 0}
|
return {x: 0, y: 0, z: 0}
|
||||||
}
|
}
|
||||||
|
|
||||||
return currentProjection.value.latLngToLocation(event.value.latlng, 64);
|
return currentMap.value.latLngToLocation(event.value.latlng, 64);
|
||||||
}),
|
}),
|
||||||
|
|
||||||
//Label for location button
|
//Label for location button
|
||||||
@ -85,12 +84,12 @@ export default defineComponent({
|
|||||||
|
|
||||||
//Url to copy
|
//Url to copy
|
||||||
url = computed(() => {
|
url = computed(() => {
|
||||||
if (!currentWorld.value || !currentMap.value) {
|
if (!currentMap.value) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = new URL(window.location.href);
|
const url = new URL(window.location.href);
|
||||||
url.hash = getUrlForLocation(currentWorld.value, currentMap.value, location.value, currentZoom.value);
|
url.hash = getUrlForLocation(currentMap.value, location.value, currentZoom.value);
|
||||||
|
|
||||||
return url;
|
return url;
|
||||||
}),
|
}),
|
||||||
|
@ -18,11 +18,10 @@
|
|||||||
import {defineComponent, onUnmounted, computed, watch} from "@vue/runtime-core";
|
import {defineComponent, onUnmounted, computed, watch} from "@vue/runtime-core";
|
||||||
import {Map} from 'leaflet';
|
import {Map} from 'leaflet';
|
||||||
import {useStore} from "@/store";
|
import {useStore} from "@/store";
|
||||||
import {MutationTypes} from "@/store/mutation-types";
|
|
||||||
import {ActionTypes} from "@/store/action-types";
|
import {ActionTypes} from "@/store/action-types";
|
||||||
import {getMinecraftTime} from "@/util";
|
import {getMinecraftTime} from "@/util";
|
||||||
import {DynmapTileLayer} from "@/leaflet/tileLayer/DynmapTileLayer";
|
import {DynmapTileLayer} from "@/leaflet/tileLayer/DynmapTileLayer";
|
||||||
import {LiveAtlasWorldMap} from "@/index";
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
@ -31,7 +30,7 @@ export default defineComponent({
|
|||||||
required: true
|
required: true
|
||||||
},
|
},
|
||||||
map: {
|
map: {
|
||||||
type: Object as () => LiveAtlasWorldMap,
|
type: Object as () => LiveAtlasMapDefinition,
|
||||||
required: true
|
required: true
|
||||||
},
|
},
|
||||||
leaflet: {
|
leaflet: {
|
||||||
@ -55,7 +54,6 @@ export default defineComponent({
|
|||||||
active = computed(() => props.map === store.state.currentMap),
|
active = computed(() => props.map === store.state.currentMap),
|
||||||
|
|
||||||
enableLayer = () => {
|
enableLayer = () => {
|
||||||
useStore().commit(MutationTypes.SET_CURRENT_PROJECTION, layer.getProjection());
|
|
||||||
props.leaflet.addLayer(layer);
|
props.leaflet.addLayer(layer);
|
||||||
|
|
||||||
stopUpdateWatch = watch(pendingUpdates, (newValue, oldValue) => {
|
stopUpdateWatch = watch(pendingUpdates, (newValue, oldValue) => {
|
||||||
|
@ -39,7 +39,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
const store = useStore(),
|
const store = useStore(),
|
||||||
componentSettings = computed(() => store.state.components.playerMarkers),
|
componentSettings = computed(() => store.state.components.playerMarkers),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
currentWorld = computed(() => store.state.currentWorld),
|
currentWorld = computed(() => store.state.currentWorld),
|
||||||
chatBalloonsEnabled = computed(() => store.state.components.chatBalloons),
|
chatBalloonsEnabled = computed(() => store.state.components.chatBalloons),
|
||||||
|
|
||||||
@ -147,8 +147,8 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
enableLayer = () => {
|
enableLayer = () => {
|
||||||
if(!markerVisible.value) {
|
if(currentMap.value && !markerVisible.value) {
|
||||||
const latLng = currentProjection.value.locationToLatLng(props.player.location);
|
const latLng = currentMap.value.locationToLatLng(props.player.location);
|
||||||
|
|
||||||
props.layerGroup.addLayer(marker);
|
props.layerGroup.addLayer(marker);
|
||||||
marker.setLatLng(latLng);
|
marker.setLatLng(latLng);
|
||||||
@ -174,7 +174,7 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
if(currentWorld.value && currentWorld.value.name === props.player.location.world) {
|
if(currentMap.value && currentWorld.value!.name === props.player.location.world) {
|
||||||
enableLayer();
|
enableLayer();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -183,7 +183,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
componentSettings,
|
componentSettings,
|
||||||
currentProjection,
|
currentMap,
|
||||||
currentWorld,
|
currentWorld,
|
||||||
chatBalloonsEnabled,
|
chatBalloonsEnabled,
|
||||||
|
|
||||||
@ -203,11 +203,11 @@ export default defineComponent({
|
|||||||
player: {
|
player: {
|
||||||
deep: true,
|
deep: true,
|
||||||
handler(newValue) {
|
handler(newValue) {
|
||||||
if(this.currentWorld && newValue.location.world === this.currentWorld.name) {
|
if(this.currentMap && newValue.location.world === this.currentWorld!.name) {
|
||||||
if(!this.markerVisible) {
|
if(!this.markerVisible) {
|
||||||
this.enableLayer();
|
this.enableLayer();
|
||||||
} else {
|
} else {
|
||||||
const latLng = this.currentProjection.locationToLatLng(newValue.location);
|
const latLng = this.currentMap.locationToLatLng(newValue.location);
|
||||||
|
|
||||||
this.marker.setLatLng(latLng);
|
this.marker.setLatLng(latLng);
|
||||||
this.chatBalloon.setLatLng(latLng);
|
this.chatBalloon.setLatLng(latLng);
|
||||||
@ -232,12 +232,14 @@ export default defineComponent({
|
|||||||
this.disableLayer();
|
this.disableLayer();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
currentProjection() {
|
currentMap(newValue) {
|
||||||
const latLng = this.currentProjection.locationToLatLng(this.player.location);
|
if(newValue) {
|
||||||
|
const latLng = newValue.locationToLatLng(this.player.location);
|
||||||
|
|
||||||
this.marker.setLatLng(latLng);
|
this.marker.setLatLng(latLng);
|
||||||
this.chatBalloon.setLatLng(latLng);
|
this.chatBalloon.setLatLng(latLng);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
@ -41,7 +41,7 @@ export default defineComponent({
|
|||||||
let updateFrame = 0;
|
let updateFrame = 0;
|
||||||
|
|
||||||
const store = useStore(),
|
const store = useStore(),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
pendingUpdates = computed(() => {
|
pendingUpdates = computed(() => {
|
||||||
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
||||||
|
|
||||||
@ -102,12 +102,14 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
//FIXME: Prevent unnecessary repositioning when changing worlds
|
//FIXME: Prevent unnecessary repositioning when changing worlds
|
||||||
watch(currentProjection, () => {
|
watch(currentMap, (newValue) => {
|
||||||
|
if(newValue) {
|
||||||
const converter = getPointConverter();
|
const converter = getPointConverter();
|
||||||
|
|
||||||
for (const [id, area] of props.set.areas) {
|
for (const [id, area] of props.set.areas) {
|
||||||
updateArea(layers.get(id), area, converter);
|
updateArea(layers.get(id), area, converter);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
watch(pendingUpdates, (newValue, oldValue) => {
|
watch(pendingUpdates, (newValue, oldValue) => {
|
||||||
|
@ -41,7 +41,7 @@ export default defineComponent({
|
|||||||
let updateFrame = 0;
|
let updateFrame = 0;
|
||||||
|
|
||||||
const store = useStore(),
|
const store = useStore(),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
pendingUpdates = computed(() => {
|
pendingUpdates = computed(() => {
|
||||||
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
||||||
|
|
||||||
@ -102,12 +102,14 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
//FIXME: Prevent unnecessary repositioning when changing worlds
|
//FIXME: Prevent unnecessary repositioning when changing worlds
|
||||||
watch(currentProjection, () => {
|
watch(currentMap, (newValue) => {
|
||||||
|
if(newValue) {
|
||||||
const converter = getPointConverter();
|
const converter = getPointConverter();
|
||||||
|
|
||||||
for (const [id, circle] of props.set.circles) {
|
for (const [id, circle] of props.set.circles) {
|
||||||
updateCircle(layers.get(id), circle, converter);
|
updateCircle(layers.get(id), circle, converter);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
watch(pendingUpdates, (newValue, oldValue) => {
|
watch(pendingUpdates, (newValue, oldValue) => {
|
||||||
|
@ -40,7 +40,7 @@ export default defineComponent({
|
|||||||
let updateFrame = 0;
|
let updateFrame = 0;
|
||||||
|
|
||||||
const store = useStore(),
|
const store = useStore(),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
pendingUpdates = computed(() => {
|
pendingUpdates = computed(() => {
|
||||||
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
||||||
|
|
||||||
@ -101,12 +101,14 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
//FIXME: Prevent unnecessary repositioning when changing worlds
|
//FIXME: Prevent unnecessary repositioning when changing worlds
|
||||||
watch(currentProjection, () => {
|
watch(currentMap, (newValue) => {
|
||||||
|
if(newValue) {
|
||||||
const converter = getPointConverter();
|
const converter = getPointConverter();
|
||||||
|
|
||||||
for (const [id, line] of props.set.lines) {
|
for (const [id, line] of props.set.lines) {
|
||||||
updateLine(layers.get(id), line, converter);
|
updateLine(layers.get(id), line, converter);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
watch(pendingUpdates, (newValue, oldValue) => {
|
watch(pendingUpdates, (newValue, oldValue) => {
|
||||||
|
@ -22,6 +22,7 @@ import {DynmapMarker, DynmapMarkerSet} from "@/dynmap";
|
|||||||
import {ActionTypes} from "@/store/action-types";
|
import {ActionTypes} from "@/store/action-types";
|
||||||
import {createMarker, updateMarker} from "@/util/markers";
|
import {createMarker, updateMarker} from "@/util/markers";
|
||||||
import DynmapLayerGroup from "@/leaflet/layer/DynmapLayerGroup";
|
import DynmapLayerGroup from "@/leaflet/layer/DynmapLayerGroup";
|
||||||
|
import {getPointConverter} from "@/util";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
@ -39,7 +40,7 @@ export default defineComponent({
|
|||||||
let updateFrame = 0;
|
let updateFrame = 0;
|
||||||
|
|
||||||
const store = useStore(),
|
const store = useStore(),
|
||||||
currentProjection = computed(() => store.state.currentProjection),
|
currentMap = computed(() => store.state.currentMap),
|
||||||
pendingUpdates = computed(() => {
|
pendingUpdates = computed(() => {
|
||||||
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
const markerSetUpdates = store.state.pendingSetUpdates.get(props.set.id);
|
||||||
|
|
||||||
@ -48,10 +49,10 @@ export default defineComponent({
|
|||||||
layers = Object.freeze(new Map()) as Map<string, Marker>,
|
layers = Object.freeze(new Map()) as Map<string, Marker>,
|
||||||
|
|
||||||
createMarkers = () => {
|
createMarkers = () => {
|
||||||
const projection = currentProjection.value;
|
const converter = getPointConverter();
|
||||||
|
|
||||||
props.set.markers.forEach((marker: DynmapMarker, id: string) => {
|
props.set.markers.forEach((marker: DynmapMarker, id: string) => {
|
||||||
const layer = createMarker(marker, projection);
|
const layer = createMarker(marker, converter);
|
||||||
|
|
||||||
layers.set(id, layer);
|
layers.set(id, layer);
|
||||||
props.layerGroup.addLayer(layer);
|
props.layerGroup.addLayer(layer);
|
||||||
@ -75,13 +76,13 @@ export default defineComponent({
|
|||||||
amount: 10,
|
amount: 10,
|
||||||
});
|
});
|
||||||
|
|
||||||
const projection = currentProjection.value;
|
const converter = getPointConverter();
|
||||||
|
|
||||||
for(const update of updates) {
|
for(const update of updates) {
|
||||||
if(update.removed) {
|
if(update.removed) {
|
||||||
deleteMarker(update.id);
|
deleteMarker(update.id);
|
||||||
} else {
|
} else {
|
||||||
const layer = updateMarker(layers.get(update.id), update.payload as DynmapMarker, projection);
|
const layer = updateMarker(layers.get(update.id), update.payload as DynmapMarker, converter);
|
||||||
|
|
||||||
if(!layers.has(update.id)) {
|
if(!layers.has(update.id)) {
|
||||||
props.layerGroup.addLayer(layer);
|
props.layerGroup.addLayer(layer);
|
||||||
@ -100,11 +101,11 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
//FIXME: Prevent unnecessary repositioning when changing worlds
|
//FIXME: Prevent unnecessary repositioning when changing worlds
|
||||||
watch(currentProjection, () => {
|
watch(currentMap, (newValue) => {
|
||||||
const projection = currentProjection.value;
|
if(newValue) {
|
||||||
|
|
||||||
for (const [id, marker] of props.set.markers) {
|
for (const [id, marker] of props.set.markers) {
|
||||||
updateMarker(layers.get(id), marker, projection);
|
updateMarker(layers.get(id), marker, getPointConverter());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
v-bind:value="[world.name,map.name]" v-model="currentMap"
|
v-bind:value="[world.name,map.name]" v-model="currentMap"
|
||||||
:aria-labelledby="`${name}-${world.name}-${key}-label`">
|
:aria-labelledby="`${name}-${world.name}-${key}-label`">
|
||||||
<label :id="`${name}-${world.name}-${key}-label`" class="map" :for="`${name}-${world.name}-${key}`" :title="`${world.title} - ${map.title}`">
|
<label :id="`${name}-${world.name}-${key}-label`" class="map" :for="`${name}-${world.name}-${key}`" :title="`${world.title} - ${map.title}`">
|
||||||
<SvgIcon :name="getMapIcon(map)"></SvgIcon>
|
<SvgIcon :name="map.getIcon()"></SvgIcon>
|
||||||
</label>
|
</label>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
@ -46,14 +46,14 @@ import "@/assets/icons/block_the_end_surface.svg";
|
|||||||
import "@/assets/icons/block_other.svg";
|
import "@/assets/icons/block_other.svg";
|
||||||
import "@/assets/icons/block_other_flat.svg";
|
import "@/assets/icons/block_other_flat.svg";
|
||||||
import "@/assets/icons/block_skylands.svg";
|
import "@/assets/icons/block_skylands.svg";
|
||||||
import {LiveAtlasWorld, LiveAtlasWorldMap} from "@/index";
|
import {LiveAtlasWorldDefinition} from "@/index";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'WorldListItem',
|
name: 'WorldListItem',
|
||||||
components: {SvgIcon},
|
components: {SvgIcon},
|
||||||
props: {
|
props: {
|
||||||
world: {
|
world: {
|
||||||
type: Object as () => LiveAtlasWorld,
|
type: Object as () => LiveAtlasWorldDefinition,
|
||||||
required: true
|
required: true
|
||||||
},
|
},
|
||||||
name: {
|
name: {
|
||||||
@ -72,33 +72,6 @@ export default defineComponent({
|
|||||||
useStore().commit(MutationTypes.SET_CURRENT_MAP, {worldName: value[0], mapName: value[1]});
|
useStore().commit(MutationTypes.SET_CURRENT_MAP, {worldName: value[0], mapName: value[1]});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
|
||||||
getMapIcon(map: LiveAtlasWorldMap): string {
|
|
||||||
let worldType: string,
|
|
||||||
mapType: string;
|
|
||||||
|
|
||||||
switch(this.world.dimension) {
|
|
||||||
case 'nether':
|
|
||||||
worldType = 'nether';
|
|
||||||
mapType = ['surface', 'nether'].includes(map.name) ? 'surface' : 'flat';
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'end':
|
|
||||||
worldType = 'the_end';
|
|
||||||
mapType = ['surface', 'the_end'].includes(map.name) ? 'surface' : 'flat';
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'overworld':
|
|
||||||
default:
|
|
||||||
worldType = 'world';
|
|
||||||
mapType = ['surface', 'flat', 'biome', 'cave'].includes(map.name) ? map.name : 'flat';
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return `block_${worldType}_${mapType}`;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
4
src/dynmap.d.ts
vendored
4
src/dynmap.d.ts
vendored
@ -22,7 +22,7 @@ import {
|
|||||||
Coordinate,
|
Coordinate,
|
||||||
LiveAtlasLocation,
|
LiveAtlasLocation,
|
||||||
LiveAtlasServerMessageConfig,
|
LiveAtlasServerMessageConfig,
|
||||||
LiveAtlasWorld,
|
LiveAtlasWorldDefinition,
|
||||||
LiveAtlasWorldState
|
LiveAtlasWorldState
|
||||||
} from "@/index";
|
} from "@/index";
|
||||||
|
|
||||||
@ -102,7 +102,7 @@ interface DynmapChatSendingConfig {
|
|||||||
interface DynmapConfigurationResponse {
|
interface DynmapConfigurationResponse {
|
||||||
config: DynmapServerConfig,
|
config: DynmapServerConfig,
|
||||||
messages: LiveAtlasServerMessageConfig,
|
messages: LiveAtlasServerMessageConfig,
|
||||||
worlds: Array<LiveAtlasWorld>,
|
worlds: Array<LiveAtlasWorldDefinition>,
|
||||||
components: DynmapComponentConfig,
|
components: DynmapComponentConfig,
|
||||||
loggedIn: boolean,
|
loggedIn: boolean,
|
||||||
}
|
}
|
||||||
|
25
src/index.d.ts
vendored
25
src/index.d.ts
vendored
@ -1,5 +1,6 @@
|
|||||||
import {State} from "@/store";
|
import {State} from "@/store";
|
||||||
import {DynmapPlayer, DynmapUrlConfig, LiveAtlasWorldMap} from "@/dynmap";
|
import {DynmapPlayer, DynmapUrlConfig} from "@/dynmap";
|
||||||
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
declare module "*.png" {
|
declare module "*.png" {
|
||||||
const value: any;
|
const value: any;
|
||||||
@ -111,7 +112,7 @@ interface LiveAtlasSortedPlayers extends Array<DynmapPlayer> {
|
|||||||
dirty?: boolean;
|
dirty?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LiveAtlasWorld {
|
interface LiveAtlasWorldDefinition {
|
||||||
seaLevel: number;
|
seaLevel: number;
|
||||||
name: string;
|
name: string;
|
||||||
dimension: LiveAtlasDimension;
|
dimension: LiveAtlasDimension;
|
||||||
@ -119,7 +120,7 @@ interface LiveAtlasWorld {
|
|||||||
title: string;
|
title: string;
|
||||||
height: number;
|
height: number;
|
||||||
center: Coordinate;
|
center: Coordinate;
|
||||||
maps: Map<string, LiveAtlasWorldMap>;
|
maps: Map<string, LiveAtlasMapDefinition>;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LiveAtlasWorldState {
|
interface LiveAtlasWorldState {
|
||||||
@ -128,24 +129,6 @@ interface LiveAtlasWorldState {
|
|||||||
timeOfDay: number;
|
timeOfDay: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LiveAtlasWorldMap {
|
|
||||||
world: LiveAtlasWorld;
|
|
||||||
name: string;
|
|
||||||
icon: string;
|
|
||||||
title: string;
|
|
||||||
background: string;
|
|
||||||
nightAndDay: boolean;
|
|
||||||
backgroundDay?: string;
|
|
||||||
backgroundNight?: string;
|
|
||||||
imageFormat: string;
|
|
||||||
prefix: string;
|
|
||||||
protected: boolean;
|
|
||||||
mapToWorld: [number, number, number, number, number, number, number, number, number];
|
|
||||||
worldToMap: [number, number, number, number, number, number, number, number, number];
|
|
||||||
nativeZoomLevels: number;
|
|
||||||
extraZoomLevels: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface LiveAtlasParsedUrl {
|
interface LiveAtlasParsedUrl {
|
||||||
world?: string;
|
world?: string;
|
||||||
map?: string;
|
map?: string;
|
||||||
|
@ -90,11 +90,11 @@ export class CoordinatesControl extends Control {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_onMouseMove(event: LeafletMouseEvent) {
|
_onMouseMove(event: LeafletMouseEvent) {
|
||||||
if (!this._map || !store.state.currentWorld) {
|
if (!this._map || !store.state.currentMap) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this._location = store.state.currentProjection.latLngToLocation(event.latlng, store.state.currentWorld.seaLevel + 1);
|
this._location = store.state.currentMap.latLngToLocation(event.latlng, store.state.currentWorld!.seaLevel + 1);
|
||||||
|
|
||||||
if(!this._locationChanged) {
|
if(!this._locationChanged) {
|
||||||
this._locationChanged = true;
|
this._locationChanged = true;
|
||||||
|
@ -18,20 +18,19 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {TileLayer, Coords, DoneCallback, TileLayerOptions, DomUtil, Util, LatLng} from 'leaflet';
|
import {TileLayer, Coords, DoneCallback, TileLayerOptions, DomUtil, Util, LatLng} from 'leaflet';
|
||||||
import {DynmapProjection} from "@/leaflet/projection/DynmapProjection";
|
|
||||||
import {store} from "@/store";
|
import {store} from "@/store";
|
||||||
import {Coordinate, LiveAtlasWorldMap} from "@/index";
|
import {Coordinate} from "@/index";
|
||||||
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
export interface DynmapTileLayerOptions extends TileLayerOptions {
|
export interface DynmapTileLayerOptions extends TileLayerOptions {
|
||||||
mapSettings: LiveAtlasWorldMap;
|
mapSettings: LiveAtlasMapDefinition;
|
||||||
errorTileUrl: string;
|
errorTileUrl: string;
|
||||||
night?: boolean;
|
night?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DynmapTileLayer extends TileLayer {
|
export interface DynmapTileLayer extends TileLayer {
|
||||||
options: DynmapTileLayerOptions;
|
options: DynmapTileLayerOptions;
|
||||||
_projection: DynmapProjection;
|
_mapSettings: LiveAtlasMapDefinition;
|
||||||
_mapSettings: LiveAtlasWorldMap;
|
|
||||||
_cachedTileUrls: Map<string, string>;
|
_cachedTileUrls: Map<string, string>;
|
||||||
_namedTiles: Map<string, DynmapTileElement>;
|
_namedTiles: Map<string, DynmapTileElement>;
|
||||||
_tileTemplate: DynmapTileElement;
|
_tileTemplate: DynmapTileElement;
|
||||||
@ -87,11 +86,6 @@ export class DynmapTileLayer extends TileLayer {
|
|||||||
throw new TypeError("mapSettings missing");
|
throw new TypeError("mapSettings missing");
|
||||||
}
|
}
|
||||||
|
|
||||||
this._projection = new DynmapProjection({
|
|
||||||
mapToWorld: this._mapSettings.mapToWorld,
|
|
||||||
worldToMap: this._mapSettings.worldToMap,
|
|
||||||
nativeZoomLevels: this._mapSettings.nativeZoomLevels,
|
|
||||||
});
|
|
||||||
this._cachedTileUrls = Object.seal(new Map());
|
this._cachedTileUrls = Object.seal(new Map());
|
||||||
this._namedTiles = Object.seal(new Map());
|
this._namedTiles = Object.seal(new Map());
|
||||||
this._loadQueue = [];
|
this._loadQueue = [];
|
||||||
@ -276,10 +270,6 @@ export class DynmapTileLayer extends TileLayer {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
getProjection(): DynmapProjection {
|
|
||||||
return this._projection;
|
|
||||||
}
|
|
||||||
|
|
||||||
setNight(night: boolean) {
|
setNight(night: boolean) {
|
||||||
if(this.options.night !== night) {
|
if(this.options.night !== night) {
|
||||||
this.options.night = night;
|
this.options.night = night;
|
||||||
|
118
src/model/LiveAtlasMapDefinition.ts
Normal file
118
src/model/LiveAtlasMapDefinition.ts
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {Coordinate, LiveAtlasWorldDefinition} from "@/index";
|
||||||
|
import {LatLng} from "leaflet";
|
||||||
|
import {LiveAtlasProjection} from "@/model/LiveAtlasProjection";
|
||||||
|
|
||||||
|
export interface LiveAtlasMapDefinitionOptions {
|
||||||
|
world: LiveAtlasWorldDefinition;
|
||||||
|
name: string;
|
||||||
|
title?: string;
|
||||||
|
icon?: string;
|
||||||
|
background?: string;
|
||||||
|
nightAndDay?: boolean;
|
||||||
|
backgroundDay?: string;
|
||||||
|
backgroundNight?: string;
|
||||||
|
imageFormat: string;
|
||||||
|
prefix?: string;
|
||||||
|
protected?: boolean;
|
||||||
|
mapToWorld?: [number, number, number, number, number, number, number, number, number];
|
||||||
|
worldToMap?: [number, number, number, number, number, number, number, number, number];
|
||||||
|
nativeZoomLevels: number;
|
||||||
|
extraZoomLevels: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class LiveAtlasMapDefinition {
|
||||||
|
readonly world: LiveAtlasWorldDefinition;
|
||||||
|
readonly name: string;
|
||||||
|
readonly icon?: string;
|
||||||
|
readonly title: string;
|
||||||
|
readonly background: string;
|
||||||
|
readonly nightAndDay: boolean;
|
||||||
|
readonly backgroundDay?: string;
|
||||||
|
readonly backgroundNight?: string;
|
||||||
|
readonly imageFormat: string;
|
||||||
|
readonly prefix: string;
|
||||||
|
readonly protected: boolean;
|
||||||
|
private readonly projection?: Readonly<LiveAtlasProjection>;
|
||||||
|
readonly nativeZoomLevels: number;
|
||||||
|
readonly extraZoomLevels: number;
|
||||||
|
readonly scale: number;
|
||||||
|
|
||||||
|
constructor(options: LiveAtlasMapDefinitionOptions) {
|
||||||
|
this.world = options.world; //Ignore append_to_world here otherwise things break
|
||||||
|
this.name = options.name;
|
||||||
|
this.icon = options.icon || undefined;
|
||||||
|
this.title = options.title || '';
|
||||||
|
|
||||||
|
this.background = options.background || '#000000';
|
||||||
|
this.nightAndDay = options.nightAndDay || false;
|
||||||
|
this.backgroundDay = options.backgroundDay || '#000000';
|
||||||
|
this.backgroundNight = options.backgroundNight || '#000000';
|
||||||
|
|
||||||
|
this.imageFormat = options.imageFormat;
|
||||||
|
this.prefix = options.prefix || '';
|
||||||
|
this.protected = options.protected || false;
|
||||||
|
|
||||||
|
this.nativeZoomLevels = options.nativeZoomLevels || 1;
|
||||||
|
this.extraZoomLevels = options.extraZoomLevels || 0;
|
||||||
|
this.scale = (1 / Math.pow(2, this.nativeZoomLevels));
|
||||||
|
|
||||||
|
if(options.mapToWorld || options.worldToMap) {
|
||||||
|
this.projection = Object.freeze(new LiveAtlasProjection({
|
||||||
|
mapToWorld: options.mapToWorld || [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
|
worldToMap: options.worldToMap || [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
|
nativeZoomLevels: this.nativeZoomLevels,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
locationToLatLng(location: Coordinate): LatLng {
|
||||||
|
return this.projection ? this.projection.locationToLatLng(location)
|
||||||
|
: new LatLng(-location.z * this.scale, location.x * this.scale);
|
||||||
|
}
|
||||||
|
|
||||||
|
latLngToLocation(latLng: LatLng, y: number): Coordinate {
|
||||||
|
return this.projection ? this.projection.latLngToLocation(latLng, y)
|
||||||
|
: {x: latLng.lng / this.scale, y: y, z: -latLng.lat / this.scale};
|
||||||
|
}
|
||||||
|
|
||||||
|
getIcon(): string {
|
||||||
|
let worldType: string,
|
||||||
|
mapType: string;
|
||||||
|
|
||||||
|
switch(this.world.dimension) {
|
||||||
|
case 'nether':
|
||||||
|
worldType = 'nether';
|
||||||
|
mapType = ['surface', 'nether'].includes(this.name) ? 'surface' : 'flat';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'end':
|
||||||
|
worldType = 'the_end';
|
||||||
|
mapType = ['surface', 'the_end'].includes(this.name) ? 'surface' : 'flat';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'overworld':
|
||||||
|
default:
|
||||||
|
worldType = 'world';
|
||||||
|
mapType = ['surface', 'flat', 'biome', 'cave'].includes(this.name) ? this.name : 'flat';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `block_${worldType}_${mapType}`;
|
||||||
|
}
|
||||||
|
}
|
@ -17,27 +17,24 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Util, LatLng, Class} from 'leaflet';
|
import {LatLng} from 'leaflet';
|
||||||
import {Coordinate} from "@/index";
|
import {Coordinate} from "@/index";
|
||||||
|
|
||||||
export interface DynmapProjectionOptions {
|
export interface LiveAtlasProjectionOptions {
|
||||||
mapToWorld: [number, number, number, number, number, number, number, number, number],
|
mapToWorld: [number, number, number, number, number, number, number, number, number],
|
||||||
worldToMap: [number, number, number, number, number, number, number, number, number],
|
worldToMap: [number, number, number, number, number, number, number, number, number],
|
||||||
nativeZoomLevels: number
|
nativeZoomLevels: number
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DynmapProjection {
|
export class LiveAtlasProjection {
|
||||||
options: DynmapProjectionOptions
|
private readonly options: LiveAtlasProjectionOptions
|
||||||
locationToLatLng(location: Coordinate): LatLng;
|
|
||||||
latLngToLocation(latLng: LatLng, y: number): Coordinate;
|
|
||||||
}
|
|
||||||
|
|
||||||
// noinspection JSUnusedGlobalSymbols
|
constructor(options: LiveAtlasProjectionOptions) {
|
||||||
export class DynmapProjection extends Class {
|
this.options = {
|
||||||
|
mapToWorld: options.mapToWorld || [0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
constructor(options: DynmapProjectionOptions) {
|
worldToMap: options.worldToMap || [0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
super();
|
nativeZoomLevels: options.nativeZoomLevels || 1
|
||||||
Util.setOptions(this, options);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
locationToLatLng(location: Coordinate): LatLng {
|
locationToLatLng(location: Coordinate): LatLng {
|
@ -28,7 +28,7 @@ import {
|
|||||||
DynmapUpdateResponse
|
DynmapUpdateResponse
|
||||||
} from "@/dynmap";
|
} from "@/dynmap";
|
||||||
import {getAPI} from "@/util";
|
import {getAPI} from "@/util";
|
||||||
import {LiveAtlasWorld} from "@/index";
|
import {LiveAtlasWorldDefinition} from "@/index";
|
||||||
|
|
||||||
type AugmentedActionContext = {
|
type AugmentedActionContext = {
|
||||||
commit<K extends keyof Mutations>(
|
commit<K extends keyof Mutations>(
|
||||||
@ -119,7 +119,7 @@ export const actions: ActionTree<State, State> & Actions = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(worldName) {
|
if(worldName) {
|
||||||
const world = state.worlds.get(worldName) as LiveAtlasWorld;
|
const world = state.worlds.get(worldName) as LiveAtlasWorldDefinition;
|
||||||
|
|
||||||
// Use config default map if it exists
|
// Use config default map if it exists
|
||||||
if(config.config.defaultMap && world.maps.has(config.config.defaultMap)) {
|
if(config.config.defaultMap && world.maps.has(config.config.defaultMap)) {
|
||||||
|
@ -72,7 +72,7 @@ export const getters: GetterTree<State, State> & Getters = {
|
|||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
return getUrlForLocation(state.currentWorld, state.currentMap, {x,y,z}, zoom);
|
return getUrlForLocation(state.currentMap, {x,y,z}, zoom);
|
||||||
},
|
},
|
||||||
|
|
||||||
serverConfig(state: State): LiveAtlasDynmapServerDefinition {
|
serverConfig(state: State): LiveAtlasDynmapServerDefinition {
|
||||||
|
@ -44,7 +44,6 @@ export enum MutationTypes {
|
|||||||
|
|
||||||
SET_CURRENT_SERVER = 'setCurrentServer',
|
SET_CURRENT_SERVER = 'setCurrentServer',
|
||||||
SET_CURRENT_MAP = 'setCurrentMap',
|
SET_CURRENT_MAP = 'setCurrentMap',
|
||||||
SET_CURRENT_PROJECTION = 'setCurrentProjection',
|
|
||||||
SET_CURRENT_LOCATION = 'setCurrentLocation',
|
SET_CURRENT_LOCATION = 'setCurrentLocation',
|
||||||
SET_CURRENT_ZOOM = 'setCurrentZoom',
|
SET_CURRENT_ZOOM = 'setCurrentZoom',
|
||||||
SET_PARSED_URL = 'setParsedUrl',
|
SET_PARSED_URL = 'setParsedUrl',
|
||||||
|
@ -28,14 +28,13 @@ import {
|
|||||||
DynmapServerConfig, DynmapTileUpdate,
|
DynmapServerConfig, DynmapTileUpdate,
|
||||||
DynmapChat
|
DynmapChat
|
||||||
} from "@/dynmap";
|
} from "@/dynmap";
|
||||||
import {DynmapProjection} from "@/leaflet/projection/DynmapProjection";
|
|
||||||
import {
|
import {
|
||||||
Coordinate,
|
Coordinate,
|
||||||
LiveAtlasWorldState,
|
LiveAtlasWorldState,
|
||||||
LiveAtlasSidebarSection,
|
LiveAtlasSidebarSection,
|
||||||
LiveAtlasSortedPlayers,
|
LiveAtlasSortedPlayers,
|
||||||
LiveAtlasUIElement,
|
LiveAtlasUIElement,
|
||||||
LiveAtlasWorld,
|
LiveAtlasWorldDefinition,
|
||||||
LiveAtlasParsedUrl,
|
LiveAtlasParsedUrl,
|
||||||
LiveAtlasGlobalConfig,
|
LiveAtlasGlobalConfig,
|
||||||
LiveAtlasGlobalMessageConfig,
|
LiveAtlasGlobalMessageConfig,
|
||||||
@ -53,12 +52,12 @@ export type Mutations<S = State> = {
|
|||||||
[MutationTypes.SET_SERVER_CONFIGURATION_HASH](state: S, hash: number): void
|
[MutationTypes.SET_SERVER_CONFIGURATION_HASH](state: S, hash: number): void
|
||||||
[MutationTypes.CLEAR_SERVER_CONFIGURATION_HASH](state: S): void
|
[MutationTypes.CLEAR_SERVER_CONFIGURATION_HASH](state: S): void
|
||||||
[MutationTypes.SET_SERVER_MESSAGES](state: S, messages: LiveAtlasServerMessageConfig): void
|
[MutationTypes.SET_SERVER_MESSAGES](state: S, messages: LiveAtlasServerMessageConfig): void
|
||||||
[MutationTypes.SET_WORLDS](state: S, worlds: Array<LiveAtlasWorld>): void
|
[MutationTypes.SET_WORLDS](state: S, worlds: Array<LiveAtlasWorldDefinition>): void
|
||||||
[MutationTypes.CLEAR_WORLDS](state: S): void
|
[MutationTypes.CLEAR_WORLDS](state: S): void
|
||||||
[MutationTypes.SET_COMPONENTS](state: S, worlds: DynmapComponentConfig): void
|
[MutationTypes.SET_COMPONENTS](state: S, worlds: DynmapComponentConfig): void
|
||||||
[MutationTypes.SET_MARKER_SETS](state: S, worlds: Map<string, DynmapMarkerSet>): void
|
[MutationTypes.SET_MARKER_SETS](state: S, worlds: Map<string, DynmapMarkerSet>): void
|
||||||
[MutationTypes.CLEAR_MARKER_SETS](state: S): void
|
[MutationTypes.CLEAR_MARKER_SETS](state: S): void
|
||||||
[MutationTypes.ADD_WORLD](state: S, world: LiveAtlasWorld): void
|
[MutationTypes.ADD_WORLD](state: S, world: LiveAtlasWorldDefinition): void
|
||||||
[MutationTypes.SET_WORLD_STATE](state: S, worldState: LiveAtlasWorldState): void
|
[MutationTypes.SET_WORLD_STATE](state: S, worldState: LiveAtlasWorldState): void
|
||||||
[MutationTypes.SET_UPDATE_TIMESTAMP](state: S, time: Date): void
|
[MutationTypes.SET_UPDATE_TIMESTAMP](state: S, time: Date): void
|
||||||
[MutationTypes.ADD_MARKER_SET_UPDATES](state: S, updates: Map<string, DynmapMarkerSetUpdates>): void
|
[MutationTypes.ADD_MARKER_SET_UPDATES](state: S, updates: Map<string, DynmapMarkerSetUpdates>): void
|
||||||
@ -77,7 +76,6 @@ export type Mutations<S = State> = {
|
|||||||
[MutationTypes.CLEAR_PLAYERS](state: S): void
|
[MutationTypes.CLEAR_PLAYERS](state: S): void
|
||||||
[MutationTypes.SET_CURRENT_SERVER](state: S, server: string): void
|
[MutationTypes.SET_CURRENT_SERVER](state: S, server: string): void
|
||||||
[MutationTypes.SET_CURRENT_MAP](state: S, payload: CurrentMapPayload): void
|
[MutationTypes.SET_CURRENT_MAP](state: S, payload: CurrentMapPayload): void
|
||||||
[MutationTypes.SET_CURRENT_PROJECTION](state: S, payload: DynmapProjection): void
|
|
||||||
[MutationTypes.SET_CURRENT_LOCATION](state: S, payload: Coordinate): void
|
[MutationTypes.SET_CURRENT_LOCATION](state: S, payload: Coordinate): void
|
||||||
[MutationTypes.SET_CURRENT_ZOOM](state: S, payload: number): void
|
[MutationTypes.SET_CURRENT_ZOOM](state: S, payload: number): void
|
||||||
[MutationTypes.SET_PARSED_URL](state: S, payload: LiveAtlasParsedUrl): void
|
[MutationTypes.SET_PARSED_URL](state: S, payload: LiveAtlasParsedUrl): void
|
||||||
@ -181,7 +179,7 @@ export const mutations: MutationTree<State> & Mutations = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
//Sets the list of worlds, and their settings, from the initial config fetch
|
//Sets the list of worlds, and their settings, from the initial config fetch
|
||||||
[MutationTypes.SET_WORLDS](state: State, worlds: Array<LiveAtlasWorld>) {
|
[MutationTypes.SET_WORLDS](state: State, worlds: Array<LiveAtlasWorldDefinition>) {
|
||||||
state.worlds.clear();
|
state.worlds.clear();
|
||||||
state.maps.clear();
|
state.maps.clear();
|
||||||
|
|
||||||
@ -250,7 +248,7 @@ export const mutations: MutationTree<State> & Mutations = {
|
|||||||
state.pendingSetUpdates.clear();
|
state.pendingSetUpdates.clear();
|
||||||
},
|
},
|
||||||
|
|
||||||
[MutationTypes.ADD_WORLD](state: State, world: LiveAtlasWorld) {
|
[MutationTypes.ADD_WORLD](state: State, world: LiveAtlasWorldDefinition) {
|
||||||
state.worlds.set(world.name, world);
|
state.worlds.set(world.name, world);
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -521,11 +519,6 @@ export const mutations: MutationTree<State> & Mutations = {
|
|||||||
state.currentMap = state.maps.get(mapName);
|
state.currentMap = state.maps.get(mapName);
|
||||||
},
|
},
|
||||||
|
|
||||||
//Sets the projection to use for coordinate conversion in the current map
|
|
||||||
[MutationTypes.SET_CURRENT_PROJECTION](state: State, projection) {
|
|
||||||
state.currentProjection = projection;
|
|
||||||
},
|
|
||||||
|
|
||||||
//Sets the current location the map is showing. This is called by the map itself, and calling elsewhere will not update the map.
|
//Sets the current location the map is showing. This is called by the map itself, and calling elsewhere will not update the map.
|
||||||
[MutationTypes.SET_CURRENT_LOCATION](state: State, payload: Coordinate) {
|
[MutationTypes.SET_CURRENT_LOCATION](state: State, payload: Coordinate) {
|
||||||
state.currentLocation = payload;
|
state.currentLocation = payload;
|
||||||
|
@ -20,7 +20,6 @@ import {
|
|||||||
DynmapServerConfig, DynmapTileUpdate,
|
DynmapServerConfig, DynmapTileUpdate,
|
||||||
DynmapChat
|
DynmapChat
|
||||||
} from "@/dynmap";
|
} from "@/dynmap";
|
||||||
import {DynmapProjection} from "@/leaflet/projection/DynmapProjection";
|
|
||||||
import {
|
import {
|
||||||
Coordinate,
|
Coordinate,
|
||||||
LiveAtlasWorldState,
|
LiveAtlasWorldState,
|
||||||
@ -28,11 +27,11 @@ import {
|
|||||||
LiveAtlasSidebarSection,
|
LiveAtlasSidebarSection,
|
||||||
LiveAtlasSortedPlayers,
|
LiveAtlasSortedPlayers,
|
||||||
LiveAtlasUIElement,
|
LiveAtlasUIElement,
|
||||||
LiveAtlasWorld,
|
LiveAtlasWorldDefinition,
|
||||||
LiveAtlasWorldMap,
|
|
||||||
LiveAtlasParsedUrl,
|
LiveAtlasParsedUrl,
|
||||||
LiveAtlasMessageConfig
|
LiveAtlasMessageConfig
|
||||||
} from "@/index";
|
} from "@/index";
|
||||||
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
export type State = {
|
export type State = {
|
||||||
version: string;
|
version: string;
|
||||||
@ -44,8 +43,8 @@ export type State = {
|
|||||||
|
|
||||||
loggedIn: boolean;
|
loggedIn: boolean;
|
||||||
|
|
||||||
worlds: Map<string, LiveAtlasWorld>;
|
worlds: Map<string, LiveAtlasWorldDefinition>;
|
||||||
maps: Map<string, LiveAtlasWorldMap>;
|
maps: Map<string, LiveAtlasMapDefinition>;
|
||||||
players: Map<string, DynmapPlayer>;
|
players: Map<string, DynmapPlayer>;
|
||||||
sortedPlayers: LiveAtlasSortedPlayers;
|
sortedPlayers: LiveAtlasSortedPlayers;
|
||||||
markerSets: Map<string, DynmapMarkerSet>;
|
markerSets: Map<string, DynmapMarkerSet>;
|
||||||
@ -63,11 +62,10 @@ export type State = {
|
|||||||
|
|
||||||
currentServer?: LiveAtlasServerDefinition;
|
currentServer?: LiveAtlasServerDefinition;
|
||||||
currentWorldState: LiveAtlasWorldState;
|
currentWorldState: LiveAtlasWorldState;
|
||||||
currentWorld?: LiveAtlasWorld;
|
currentWorld?: LiveAtlasWorldDefinition;
|
||||||
currentMap?: LiveAtlasWorldMap;
|
currentMap?: LiveAtlasMapDefinition;
|
||||||
currentLocation: Coordinate;
|
currentLocation: Coordinate;
|
||||||
currentZoom: number;
|
currentZoom: number;
|
||||||
currentProjection: DynmapProjection;
|
|
||||||
|
|
||||||
updateRequestId: number;
|
updateRequestId: number;
|
||||||
updateTimestamp: Date;
|
updateTimestamp: Date;
|
||||||
@ -214,12 +212,6 @@ export const state: State = {
|
|||||||
z: 0,
|
z: 0,
|
||||||
},
|
},
|
||||||
currentZoom: 0,
|
currentZoom: 0,
|
||||||
|
|
||||||
currentProjection: new DynmapProjection({
|
|
||||||
mapToWorld: [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
|
||||||
worldToMap: [0, 0, 0, 0, 0, 0, 0, 0, 0],
|
|
||||||
nativeZoomLevels: 1
|
|
||||||
}), //Projection for converting location <-> latlg. Object itself isn't reactive for performance reasons
|
|
||||||
currentWorldState: {
|
currentWorldState: {
|
||||||
raining: false,
|
raining: false,
|
||||||
thundering: false,
|
thundering: false,
|
||||||
|
18
src/util.ts
18
src/util.ts
@ -17,7 +17,7 @@
|
|||||||
import API from '@/api';
|
import API from '@/api';
|
||||||
import {DynmapPlayer} from "@/dynmap";
|
import {DynmapPlayer} from "@/dynmap";
|
||||||
import {useStore} from "@/store";
|
import {useStore} from "@/store";
|
||||||
import {LiveAtlasWorld, LiveAtlasWorldMap} from "@/index";
|
import LiveAtlasMapDefinition from "@/model/LiveAtlasMapDefinition";
|
||||||
|
|
||||||
interface HeadQueueEntry {
|
interface HeadQueueEntry {
|
||||||
cacheKey: string;
|
cacheKey: string;
|
||||||
@ -115,11 +115,17 @@ export const concatURL = (base: string, addition: string) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const getPointConverter = () => {
|
export const getPointConverter = () => {
|
||||||
const projection = useStore().state.currentProjection;
|
const map = useStore().state.currentMap;
|
||||||
|
|
||||||
|
if(map) {
|
||||||
return (x: number, y: number, z: number) => {
|
return (x: number, y: number, z: number) => {
|
||||||
return projection.locationToLatLng({x, y, z});
|
return map.locationToLatLng({x, y, z});
|
||||||
};
|
};
|
||||||
|
} else {
|
||||||
|
return (x: number, y: number, z: number) => {
|
||||||
|
return LiveAtlasMapDefinition.defaultProjection.locationToLatLng({x, y, z});
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const parseUrl = () => {
|
export const parseUrl = () => {
|
||||||
@ -215,7 +221,7 @@ export const getAPI = () => {
|
|||||||
return API;
|
return API;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getUrlForLocation = (world: LiveAtlasWorld, map: LiveAtlasWorldMap, location: {
|
export const getUrlForLocation = (map: LiveAtlasMapDefinition, location: {
|
||||||
x: number,
|
x: number,
|
||||||
y: number,
|
y: number,
|
||||||
z: number }, zoom: number): string => {
|
z: number }, zoom: number): string => {
|
||||||
@ -224,11 +230,11 @@ export const getUrlForLocation = (world: LiveAtlasWorld, map: LiveAtlasWorldMap,
|
|||||||
z = Math.round(location.z),
|
z = Math.round(location.z),
|
||||||
locationString = `${x},${y},${z}`;
|
locationString = `${x},${y},${z}`;
|
||||||
|
|
||||||
if(!world || !map) {
|
if(!map) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
return `#${world.name};${map.name};${locationString};${zoom}`;
|
return `#${map.world.name};${map.name};${locationString};${zoom}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const focus = (selector: string) => {
|
export const focus = (selector: string) => {
|
||||||
|
@ -20,11 +20,10 @@
|
|||||||
import {LeafletMouseEvent, Marker} from "leaflet";
|
import {LeafletMouseEvent, Marker} from "leaflet";
|
||||||
import {DynmapMarker} from "@/dynmap";
|
import {DynmapMarker} from "@/dynmap";
|
||||||
import {DynmapIcon} from "@/leaflet/icon/DynmapIcon";
|
import {DynmapIcon} from "@/leaflet/icon/DynmapIcon";
|
||||||
import {DynmapProjection} from "@/leaflet/projection/DynmapProjection";
|
|
||||||
import {GenericMarker} from "@/leaflet/marker/GenericMarker";
|
import {GenericMarker} from "@/leaflet/marker/GenericMarker";
|
||||||
|
|
||||||
export const createMarker = (options: DynmapMarker, projection: DynmapProjection): Marker => {
|
export const createMarker = (options: DynmapMarker, converter: Function): Marker => {
|
||||||
const marker = new GenericMarker(projection.locationToLatLng(options.location), {
|
const marker = new GenericMarker(converter(options.location.x, options.location.y, options.location.z), {
|
||||||
icon: new DynmapIcon({
|
icon: new DynmapIcon({
|
||||||
icon: options.icon,
|
icon: options.icon,
|
||||||
label: options.label,
|
label: options.label,
|
||||||
@ -46,13 +45,13 @@ export const createMarker = (options: DynmapMarker, projection: DynmapProjection
|
|||||||
return marker;
|
return marker;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const updateMarker = (marker: Marker | undefined, options: DynmapMarker, projection: DynmapProjection): Marker => {
|
export const updateMarker = (marker: Marker | undefined, options: DynmapMarker, converter: Function): Marker => {
|
||||||
if (!marker) {
|
if (!marker) {
|
||||||
return createMarker(options, projection);
|
return createMarker(options, converter);
|
||||||
}
|
}
|
||||||
|
|
||||||
const oldLocation = marker.getLatLng(),
|
const oldLocation = marker.getLatLng(),
|
||||||
newLocation = projection.locationToLatLng(options.location);
|
newLocation = converter(options.location.x, options.location.y, options.location.z);
|
||||||
|
|
||||||
if(!oldLocation.equals(newLocation)) {
|
if(!oldLocation.equals(newLocation)) {
|
||||||
marker.setLatLng(newLocation);
|
marker.setLatLng(newLocation);
|
||||||
|
Loading…
Reference in New Issue
Block a user