Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a context menu in webmap #33

Open
wants to merge 27 commits into
base: v3
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
af631b9
Add a context menu in webmap
jedrek0429 Mar 29, 2024
3d07eca
Merge branch 'v3' into feat/hijack-context-menu
granny Mar 30, 2024
0077e66
whitespace
granny Mar 30, 2024
7abf48e
Merge branch 'granny:v3' into feat/hijack-context-menu
jedrek0429 Mar 30, 2024
94fa8ce
Add more configurability and custom html for the context menus
jedrek0429 Mar 31, 2024
a85b820
Apply suggestions from code review
jedrek0429 Apr 2, 2024
fb379ea
Merge branch 'v3' into feat/hijack-context-menu
granny Apr 4, 2024
9af1026
make the context menu items a button
granny Apr 4, 2024
a8dc6a6
make contextmenu a flexbox
granny Apr 4, 2024
693aa6f
make each coordinate a variable
granny Apr 4, 2024
ce169e1
take inspiration from LiveAtlas' contextmenu
granny Apr 4, 2024
3b859f5
remove custom html
granny Apr 4, 2024
1378fef
remove unused onRemove parameter
granny Apr 4, 2024
6d2c66c
remove unused import
granny Apr 4, 2024
3b0cf8b
Merge branch 'v3' into feat/hijack-context-menu
granny Apr 4, 2024
adb03e1
update copy-coords value in lang file
granny Apr 4, 2024
c6b7bba
update copy-coords value in lang file x2
granny Apr 4, 2024
5649794
Merge branch 'v3' into feat/hijack-context-menu
granny Apr 5, 2024
b2e3077
move display flex into leaflet-control-contextmenu class
granny Apr 5, 2024
d8aa273
use String constructor
granny Apr 5, 2024
26450a2
remove unused import
granny Apr 5, 2024
c6bdd76
remove dash from id
granny Apr 5, 2024
eff01b7
content -> context
granny Apr 5, 2024
ac770d0
bleugh
granny Apr 5, 2024
78ef803
Merge branch 'v3' into feat/hijack-context-menu
granny Apr 12, 2024
cdd801c
replace the switch case with a map that "gets" the item if set in config
granny Apr 12, 2024
c1ba1fb
whitespace
granny Apr 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions core/src/main/java/net/pl3x/map/core/configuration/Lang.java
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,14 @@ public final class Lang extends AbstractConfig {
public static String UI_BLOCKINFO_UNKNOWN_BLOCK = "Unknown block";
@Key("ui.blockinfo.unknown.biome")
public static String UI_BLOCKINFO_UNKNOWN_BIOME = "Unknown biome";
@Key("ui.contextmenu.label")
public static String UI_CONTEXTMENU_LABEL = "ContextMenu";
@Key("ui.contextmenu.copy-coords")
public static String UI_CONTEXTMENU_COPY_COORDS = "X: <x> Y: <y> Z: <z>";
@Key("ui.contextmenu.copy-link")
public static String UI_CONTEXTMENU_COPY_LINK = "Copy link to here";
@Key("ui.contextmenu.center-map")
public static String UI_CONTEXTMENU_CENTER_MAP = "Center map here";
@Key("ui.coords.label")
public static String UI_COORDS_LABEL = "Coordinates";
@Key("ui.coords.value")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,54 @@ public final class WorldConfig extends AbstractConfig {
@Comment("""
The display position for the link box""")
public String UI_LINK = "bottomright";

@Key("ui.context-menu.enabled")
@Comment("""
Enable the context menu.""")
public boolean UI_CONTEXT_MENU_ENABLED = true;

@Key("ui.context-menu.items")
@Comment("""
Items to show in the context menu.
Available items are:
copy-coords, copy-link, center-map""")
public List<@NotNull String> UI_CONTEXT_MENU_ITEMS = new ArrayList<>() {{
add("copy-coords");
add("copy-link");
add("center-map");
}};

@Key("ui.context-menu.css")
@Comment("""
Custom css for the context menu.""")
public static String UI_CONTEXT_MENU_CSS = """
.leaflet-control-contextmenu {
display: none;
position: absolute;
display: flex;
flex-direction: column;
font-family: monospace;
top: 0;
left: 0;
text-align: left;
white-space: pre;
background-color: var(--ui-background);
border: var(--ui-border);
border-radius: var(--ui-border-radius);
overflow: hidden;
z-index: 10000; /* Ensure the menu appears over other map controls */
}

.leaflet-control-contextmenu-item {
padding: 5px;
color: var(--ui-text);
transition: background-color 0.3s ease-in-out;

&:hover {
background-color: var(--ui-background-hover);
color: var(--ui-text-hover);
}
}""";

@Key("center.x")
@Comment("""
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,11 @@ public void run() {
ui.put("coords", config.UI_COORDS);
ui.put("blockinfo", config.UI_BLOCKINFO);
ui.put("attribution", config.UI_ATTRIBUTION);
ui.put("contextMenu", Map.of(
"enabled", config.UI_CONTEXT_MENU_ENABLED,
"items", config.UI_CONTEXT_MENU_ITEMS,
"css", config.UI_CONTEXT_MENU_CSS
));

Map<String, Object> settings = new LinkedHashMap<>();
settings.put("name", world.getName().replace(":", "-"));
Expand Down Expand Up @@ -166,6 +171,12 @@ private void parseSettings() {
"value", Lang.UI_BLOCKINFO_VALUE,
"unknown", Map.of("block", Lang.UI_BLOCKINFO_UNKNOWN_BLOCK, "biome", Lang.UI_BLOCKINFO_UNKNOWN_BIOME))
);
lang.put("contextMenu", Map.of(
"label", Lang.UI_CONTEXTMENU_LABEL,
"copyCoords", Lang.UI_CONTEXTMENU_COPY_COORDS,
"copyLink", Lang.UI_CONTEXTMENU_COPY_LINK,
"centerMap", Lang.UI_CONTEXTMENU_CENTER_MAP
));
lang.put("coords", Map.of("label", Lang.UI_COORDS_LABEL, "value", Lang.UI_COORDS_VALUE));
lang.put("layers", Map.of("label", Lang.UI_LAYERS_LABEL, "value", Lang.UI_LAYERS_VALUE));
lang.put("link", Map.of("label", Lang.UI_LINK_LABEL, "value", Lang.UI_LINK_VALUE));
Expand Down
5 changes: 5 additions & 0 deletions core/src/main/resources/locale/lang-pl.yml
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,11 @@ ui:
unknown:
block: Nieznany blok
biome: Nieznany biom
contextmenu:
label: ContextMenu
copy-coords: 'X: <x> Y: <y> Z: <z>'
copy-link: Skopiuj link do tego miejsca
center-map: Wyśrodkuj mapę w tym miejscu
coords:
label: Współrzędne
value: <x>, <y>, <z>
Expand Down
127 changes: 127 additions & 0 deletions webmap/src/control/ContextMenuControl.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,127 @@
import * as L from "leaflet";
import {Pl3xMap} from "../Pl3xMap";
import {ContextMenuItemType} from "../settings/WorldSettings";
import {insertCss, removeCss} from "../util/Util";

type ContextMenuCallback = {
label: () => string;
callback: (e: L.LeafletMouseEvent) => void;
};

export default class ContextMenuControl extends L.Control {
private readonly _pl3xmap: Pl3xMap;
private _dom: HTMLDivElement = L.DomUtil.create('div');
private _id: string = 'pl3xmap-contextmenu';
private _items: Map<ContextMenuItemType, ContextMenuCallback> = new Map([
[ContextMenuItemType.copyCoords, {
label: () => {
const {x, y, z} = this._pl3xmap.controlManager.coordsControl ?? {x: 0, y: 0, z: 0};
return this._pl3xmap.settings!.lang.contextMenu.copyCoords
.replace(/<x>/g, x.toString())
.replace(/<y>/g, y?.toString() ?? '???')
.replace(/<z>/g, z.toString())
},
callback: () => {
const {x, y, z} = this._pl3xmap.controlManager.coordsControl ?? {x: 0, y: 0, z: 0};
const coords = `(${x}, ${y ?? '???'}, ${z})`;
navigator.clipboard.writeText(coords)
},
}],
[ContextMenuItemType.copyLink, {
label: () => this._pl3xmap.settings!.lang.contextMenu.copyLink,
callback : () => {
const {x, z} = this._pl3xmap.controlManager.coordsControl ?? {x: 0, y: 0, z: 0};
const world = this._pl3xmap.worldManager.currentWorld;
navigator.clipboard.writeText(
window.location.href +
this._pl3xmap.controlManager.linkControl?.getUrlFromCoords(
x, z,
this._pl3xmap.map.getCurrentZoom(),
world
)
)
},
}],
[ContextMenuItemType.centerMap, {
label: () => this._pl3xmap.settings!.lang.contextMenu.centerMap,
callback: (event: L.LeafletMouseEvent) => {
this._pl3xmap.map.panTo(event.latlng);
},
}]

]);

constructor(pl3xmap: Pl3xMap) {
super();
this._pl3xmap = pl3xmap;
if (this._pl3xmap.worldManager.currentWorld?.settings.ui.contextMenu.enabled) {
this._init();
}
}

private _init(): void {
this._pl3xmap.map.on('contextmenu', this._show, this);
this._pl3xmap.map.on('click', this._hide, this);

const css = this._pl3xmap.worldManager.currentWorld?.settings.ui.contextMenu.css;
if (css !== undefined) {
insertCss(css, this._id);
}
}

onAdd(): HTMLDivElement {
this._dom = L.DomUtil.create('div', 'leaflet-control leaflet-control-contextmenu');
this._dom.dataset.label = this._pl3xmap.settings!.lang.contextMenu.label;
return this._dom;
}


onRemove(): void {
removeCss(this._id);
}

private _show(event: L.LeafletMouseEvent): void {
// Ignore right-clicks on controls (https://github.com/JLyne/LiveAtlas/blob/0819cdf2728b49d361f9adfda09ff08311a59337/src/components/map/MapContextMenu.vue#L188-L194)
granny marked this conversation as resolved.
Show resolved Hide resolved
if(event.originalEvent.target && (event.originalEvent.target as HTMLElement).closest('.leaflet-control')) {
return;
}

event.originalEvent.stopImmediatePropagation();
event.originalEvent.preventDefault();

this._dom.style.visibility = 'visible';

this._dom.innerHTML = '';

const world = this._pl3xmap.worldManager.currentWorld;
world?.settings.ui.contextMenu?.items?.forEach(itemType => {
const item: ContextMenuCallback | undefined = this._items.get(itemType);
if (item === undefined) return;

const menuItem = L.DomUtil.create('button', 'leaflet-control-contextmenu-item', this._dom);
menuItem.innerHTML = item.label();
L.DomEvent.on(menuItem, 'click', (ev) => {
L.DomEvent.stopPropagation(ev);
item.callback(event);
this._hide();
});
});

// Don't position offscreen (https://github.com/JLyne/LiveAtlas/blob/0819cdf2728b49d361f9adfda09ff08311a59337/src/components/map/MapContextMenu.vue#L123-L135)
const x = Math.min(
window.innerWidth - this._dom.offsetWidth - 10,
event.originalEvent.clientX
),
y = Math.min(
window.innerHeight - this._dom.offsetHeight - 10,
event.originalEvent.clientY
);

this._dom.style.transform = `translate(${x}px, ${y}px)`;
}

private _hide(): void {
this._dom.style.visibility = 'hidden';
this._dom.style.left = '-1000';
granny marked this conversation as resolved.
Show resolved Hide resolved
}
}
12 changes: 12 additions & 0 deletions webmap/src/control/ControlManager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,13 @@ import {Pl3xMap} from "../Pl3xMap";
import {BlockInfoControl} from "./BlockInfoControl";
import {CoordsControl} from "./CoordsControl";
import {LinkControl} from "./LinkControl";
import ContextMenuControl from "./ContextMenuControl";
import SidebarControl from "./SidebarControl";

export class ControlManager {
private readonly _pl3xmap: Pl3xMap;

private _contextMenuControl?: ContextMenuControl;
private _sidebarControl?: SidebarControl
private _blockInfoControl?: BlockInfoControl;
private _coordsControl?: CoordsControl;
Expand All @@ -16,6 +18,16 @@ export class ControlManager {
this._pl3xmap = pl3xmap;
}

get contextMenuControl(): ContextMenuControl | undefined {
return this._contextMenuControl;
}

set contextMenuControl(menu: ContextMenuControl | undefined) {
this._contextMenuControl?.remove();
this._contextMenuControl = menu;
this._contextMenuControl?.addTo(this._pl3xmap.map);
}

get sidebarControl(): SidebarControl | undefined {
return this._sidebarControl;
}
Expand Down
4 changes: 4 additions & 0 deletions webmap/src/control/LinkControl.ts
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,10 @@ export class LinkControl extends ControlBox {
const zoom: number = this._pl3xmap.map.getCurrentZoom();
const x: number = Math.floor(center[0]);
const z: number = Math.floor(center[1]);
return this.getUrlFromCoords(x, z, zoom, world);
}

public getUrlFromCoords(x: number, z: number, zoom: number, world?: World): string {
let url: string = `?`;
if (world !== undefined) {
url += `world=${world.name}&renderer=${world.currentRenderer?.label ?? 'basic'}`;
Expand Down
38 changes: 37 additions & 1 deletion webmap/src/settings/Lang.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
export class Lang {
private readonly _title: string;
private readonly _langFile: string;
private readonly _contextMenu: ContextMenu;
private readonly _coords: Label;
private readonly _blockInfo: BlockInfo;
private readonly _layers: Label;
Expand All @@ -12,9 +13,10 @@ export class Lang {
private readonly _players: Label;
private readonly _worlds: Label;

constructor(title: string, langFile: string, coords: Label, blockInfo: BlockInfo, layers: Label, link: Label, markers: Label, players: Label, worlds: Label) {
constructor(title: string, langFile: string, contextMenu: ContextMenu, coords: Label, blockInfo: BlockInfo, layers: Label, link: Label, markers: Label, players: Label, worlds: Label) {
this._title = title;
this._langFile = langFile;
this._contextMenu = contextMenu;
this._coords = coords;
this._blockInfo = blockInfo;
this._layers = layers;
Expand All @@ -32,6 +34,10 @@ export class Lang {
return this._langFile;
}

get contextMenu(): ContextMenu {
return this._contextMenu;
}

get coords(): Label {
return this._coords;
}
Expand Down Expand Up @@ -118,3 +124,33 @@ export class BlockInfo extends Label {
return this._unknown;
}
}

export class ContextMenu {
private readonly _label: string;
private readonly _copyCoords: string;
private readonly _copyLink: string;
private readonly _centerMap: string;

constructor(label: string, copyCoords: string, copyLink: string, centerMap: string) {
this._label = label;
this._copyCoords = copyCoords;
this._copyLink = copyLink;
this._centerMap = centerMap;
}

get label(): string {
return this._label;
}

get copyCoords(): string {
return this._copyCoords;
}

get copyLink(): string {
return this._copyLink;
}

get centerMap(): string {
return this._centerMap;
}
}
Loading
Loading