mirror of
https://github.com/Azgaar/Fantasy-Map-Generator.git
synced 2025-12-16 17:31:24 +01:00
* fix: rework texture save format * refactor: move textures to a local folder * fix: export texture * fix: don't export empty texture * chore: pump version * fix: rescale water mask on screen size change --------- Co-authored-by: Azgaar <azgaar.fmg@yandex.com>
554 lines
21 KiB
JavaScript
554 lines
21 KiB
JavaScript
"use strict";
|
||
// Module to store general UI functions
|
||
|
||
// fit full-screen map if window is resized
|
||
window.addEventListener("resize", function (e) {
|
||
if (stored("mapWidth") && stored("mapHeight")) return;
|
||
mapWidthInput.value = window.innerWidth;
|
||
mapHeightInput.value = window.innerHeight;
|
||
changeMapSize();
|
||
});
|
||
|
||
if (location.hostname !== "localhost" && location.hostname !== "127.0.0.1") {
|
||
window.onbeforeunload = () => "Are you sure you want to navigate away?";
|
||
}
|
||
|
||
// Tooltips
|
||
const tooltip = document.getElementById("tooltip");
|
||
|
||
// show tip for non-svg elemets with data-tip
|
||
document.getElementById("dialogs").addEventListener("mousemove", showDataTip);
|
||
document.getElementById("optionsContainer").addEventListener("mousemove", showDataTip);
|
||
document.getElementById("exitCustomization").addEventListener("mousemove", showDataTip);
|
||
|
||
const tipBackgroundMap = {
|
||
info: "linear-gradient(0.1turn, #ffffff00, #5e5c5c80, #ffffff00)",
|
||
success: "linear-gradient(0.1turn, #ffffff00, #127912cc, #ffffff00)",
|
||
warn: "linear-gradient(0.1turn, #ffffff00, #be5d08cc, #ffffff00)",
|
||
error: "linear-gradient(0.1turn, #ffffff00, #e11d1dcc, #ffffff00)"
|
||
};
|
||
|
||
function tip(tip = "Tip is undefined", main = false, type = "info", time = 0) {
|
||
tooltip.innerHTML = tip;
|
||
tooltip.style.background = tipBackgroundMap[type];
|
||
|
||
if (main) {
|
||
tooltip.dataset.main = tip;
|
||
tooltip.dataset.color = tooltip.style.background;
|
||
}
|
||
if (time) setTimeout(clearMainTip, time);
|
||
}
|
||
|
||
function showMainTip() {
|
||
tooltip.style.background = tooltip.dataset.color;
|
||
tooltip.innerHTML = tooltip.dataset.main;
|
||
}
|
||
|
||
function clearMainTip() {
|
||
tooltip.dataset.color = "";
|
||
tooltip.dataset.main = "";
|
||
tooltip.innerHTML = "";
|
||
}
|
||
|
||
// show tip at the bottom of the screen, consider possible translation
|
||
function showDataTip(event) {
|
||
if (!event.target) return;
|
||
|
||
let dataTip = event.target.dataset.tip;
|
||
if (!dataTip && event.target.parentNode.dataset.tip) dataTip = event.target.parentNode.dataset.tip;
|
||
if (!dataTip) return;
|
||
|
||
const shortcut = event.target.dataset.shortcut;
|
||
if (shortcut && !MOBILE) dataTip += `. Shortcut: ${shortcut}`;
|
||
|
||
//const tooltip = lang === "en" ? dataTip : translate(e.target.dataset.t || e.target.parentNode.dataset.t, dataTip);
|
||
tip(dataTip);
|
||
}
|
||
|
||
function showElementLockTip(event) {
|
||
const locked = event?.target?.classList?.contains("icon-lock");
|
||
if (locked) {
|
||
tip("Click to unlock the element and allow it to be changed by regeneration tools");
|
||
} else {
|
||
tip("Click to lock the element and prevent changes to it by regeneration tools");
|
||
}
|
||
}
|
||
|
||
const onMouseMove = debounce(handleMouseMove, 100);
|
||
function handleMouseMove() {
|
||
const point = d3.mouse(this);
|
||
const i = findCell(point[0], point[1]); // pack cell id
|
||
if (i === undefined) return;
|
||
|
||
showNotes(d3.event);
|
||
const gridCell = findGridCell(point[0], point[1], grid);
|
||
if (tooltip.dataset.main) showMainTip();
|
||
else showMapTooltip(point, d3.event, i, gridCell);
|
||
if (cellInfo?.offsetParent) updateCellInfo(point, i, gridCell);
|
||
}
|
||
|
||
let currentNoteId = null; // store currently displayed node to not rerender to often
|
||
|
||
// show note box on hover (if any)
|
||
function showNotes(e) {
|
||
if (notesEditor?.offsetParent) return;
|
||
let id = e.target.id || e.target.parentNode.id || e.target.parentNode.parentNode.id;
|
||
if (e.target.parentNode.parentNode.id === "burgLabels") id = "burg" + e.target.dataset.id;
|
||
else if (e.target.parentNode.parentNode.id === "burgIcons") id = "burg" + e.target.dataset.id;
|
||
|
||
const note = notes.find(note => note.id === id);
|
||
if (note !== undefined && note.legend !== "") {
|
||
if (currentNoteId === id) return;
|
||
currentNoteId = id;
|
||
|
||
document.getElementById("notes").style.display = "block";
|
||
document.getElementById("notesHeader").innerHTML = note.name;
|
||
document.getElementById("notesBody").innerHTML = note.legend;
|
||
} else if (!options.pinNotes && !markerEditor?.offsetParent && !e.shiftKey) {
|
||
document.getElementById("notes").style.display = "none";
|
||
document.getElementById("notesHeader").innerHTML = "";
|
||
document.getElementById("notesBody").innerHTML = "";
|
||
currentNoteId = null;
|
||
}
|
||
}
|
||
|
||
// show viewbox tooltip if main tooltip is blank
|
||
function showMapTooltip(point, e, i, g) {
|
||
tip(""); // clear tip
|
||
const path = e.composedPath ? e.composedPath() : getComposedPath(e.target); // apply polyfill
|
||
if (!path[path.length - 8]) return;
|
||
const group = path[path.length - 7].id;
|
||
const subgroup = path[path.length - 8].id;
|
||
const land = pack.cells.h[i] >= 20;
|
||
|
||
// specific elements
|
||
if (group === "armies") return tip(e.target.parentNode.dataset.name + ". Click to edit");
|
||
|
||
if (group === "emblems" && e.target.tagName === "use") {
|
||
const parent = e.target.parentNode;
|
||
const [g, type] =
|
||
parent.id === "burgEmblems"
|
||
? [pack.burgs, "burg"]
|
||
: parent.id === "provinceEmblems"
|
||
? [pack.provinces, "province"]
|
||
: [pack.states, "state"];
|
||
const i = +e.target.dataset.i;
|
||
if (event.shiftKey) highlightEmblemElement(type, g[i]);
|
||
|
||
d3.select(e.target).raise();
|
||
d3.select(parent).raise();
|
||
|
||
const name = g[i].fullName || g[i].name;
|
||
tip(`${name} ${type} emblem. Click to edit. Hold Shift to show associated area or place`);
|
||
return;
|
||
}
|
||
|
||
if (group === "rivers") {
|
||
const river = +e.target.id.slice(5);
|
||
const r = pack.rivers.find(r => r.i === river);
|
||
const name = r ? r.name + " " + r.type : "";
|
||
tip(name + ". Click to edit");
|
||
if (riversOverview?.offsetParent) highlightEditorLine(riversOverview, river, 5000);
|
||
return;
|
||
}
|
||
|
||
if (group === "routes") return tip("Click to edit the Route");
|
||
|
||
if (group === "terrain") return tip("Click to edit the Relief Icon");
|
||
|
||
if (subgroup === "burgLabels" || subgroup === "burgIcons") {
|
||
const burg = +path[path.length - 10].dataset.id;
|
||
const b = pack.burgs[burg];
|
||
const population = si(b.population * populationRate * urbanization);
|
||
tip(`${b.name}. Population: ${population}. Click to edit`);
|
||
if (burgsOverview?.offsetParent) highlightEditorLine(burgsOverview, burg, 5000);
|
||
return;
|
||
}
|
||
if (group === "labels") return tip("Click to edit the Label");
|
||
|
||
if (group === "markers") return tip("Click to edit the Marker. Hold Shift to not close the assosiated note");
|
||
|
||
if (group === "ruler") {
|
||
const tag = e.target.tagName;
|
||
const className = e.target.getAttribute("class");
|
||
if (tag === "circle" && className === "edge")
|
||
return tip("Drag to adjust. Hold Ctrl and drag to add a point. Click to remove the point");
|
||
if (tag === "circle" && className === "control")
|
||
return tip("Drag to adjust. Hold Shift and drag to keep axial direction. Click to remove the point");
|
||
if (tag === "circle") return tip("Drag to adjust the measurer");
|
||
if (tag === "polyline") return tip("Click on drag to add a control point");
|
||
if (tag === "path") return tip("Drag to move the measurer");
|
||
if (tag === "text") return tip("Drag to move, click to remove the measurer");
|
||
}
|
||
|
||
if (subgroup === "burgIcons") return tip("Click to edit the Burg");
|
||
|
||
if (subgroup === "burgLabels") return tip("Click to edit the Burg");
|
||
|
||
if (group === "lakes" && !land) {
|
||
const lakeId = +e.target.dataset.f;
|
||
const name = pack.features[lakeId]?.name;
|
||
const fullName = subgroup === "freshwater" ? name : name + " " + subgroup;
|
||
tip(`${fullName} lake. Click to edit`);
|
||
return;
|
||
}
|
||
if (group === "coastline") return tip("Click to edit the coastline");
|
||
|
||
if (group === "zones") {
|
||
const zone = path[path.length - 8];
|
||
tip(zone.dataset.description);
|
||
if (zonesEditor?.offsetParent) highlightEditorLine(zonesEditor, zone.id, 5000);
|
||
return;
|
||
}
|
||
|
||
if (group === "ice") return tip("Click to edit the Ice");
|
||
|
||
// covering elements
|
||
if (layerIsOn("togglePrec") && land) tip("Annual Precipitation: " + getFriendlyPrecipitation(i));
|
||
else if (layerIsOn("togglePopulation")) tip(getPopulationTip(i));
|
||
else if (layerIsOn("toggleTemp")) tip("Temperature: " + convertTemperature(grid.cells.temp[g]));
|
||
else if (layerIsOn("toggleBiomes") && pack.cells.biome[i]) {
|
||
const biome = pack.cells.biome[i];
|
||
tip("Biome: " + biomesData.name[biome]);
|
||
if (biomesEditor?.offsetParent) highlightEditorLine(biomesEditor, biome);
|
||
} else if (layerIsOn("toggleReligions") && pack.cells.religion[i]) {
|
||
const religion = pack.cells.religion[i];
|
||
const r = pack.religions[religion];
|
||
const type = r.type === "Cult" || r.type == "Heresy" ? r.type : r.type + " religion";
|
||
tip(type + ": " + r.name);
|
||
if (byId("religionsEditor")?.offsetParent) highlightEditorLine(religionsEditor, religion);
|
||
} else if (pack.cells.state[i] && (layerIsOn("toggleProvinces") || layerIsOn("toggleStates"))) {
|
||
const state = pack.cells.state[i];
|
||
const stateName = pack.states[state].fullName;
|
||
const province = pack.cells.province[i];
|
||
const prov = province ? pack.provinces[province].fullName + ", " : "";
|
||
tip(prov + stateName);
|
||
if (document.getElementById("statesEditor")?.offsetParent) highlightEditorLine(statesEditor, state);
|
||
if (document.getElementById("diplomacyEditor")?.offsetParent) highlightEditorLine(diplomacyEditor, state);
|
||
if (document.getElementById("militaryOverview")?.offsetParent) highlightEditorLine(militaryOverview, state);
|
||
if (document.getElementById("provincesEditor")?.offsetParent) highlightEditorLine(provincesEditor, province);
|
||
} else if (layerIsOn("toggleCultures") && pack.cells.culture[i]) {
|
||
const culture = pack.cells.culture[i];
|
||
tip("Culture: " + pack.cultures[culture].name);
|
||
if (document.getElementById("culturesEditor")?.offsetParent) highlightEditorLine(culturesEditor, culture);
|
||
} else if (layerIsOn("toggleHeight")) tip("Height: " + getFriendlyHeight(point));
|
||
}
|
||
|
||
function highlightEditorLine(editor, id, timeout = 10000) {
|
||
Array.from(editor.getElementsByClassName("states hovered")).forEach(el => el.classList.remove("hovered")); // clear all hovered
|
||
const hovered = Array.from(editor.querySelectorAll("div")).find(el => el.dataset.id == id);
|
||
if (hovered) hovered.classList.add("hovered"); // add hovered class
|
||
if (timeout)
|
||
setTimeout(() => {
|
||
hovered && hovered.classList.remove("hovered");
|
||
}, timeout);
|
||
}
|
||
|
||
// get cell info on mouse move
|
||
function updateCellInfo(point, i, g) {
|
||
const cells = pack.cells;
|
||
const x = (infoX.innerHTML = rn(point[0]));
|
||
const y = (infoY.innerHTML = rn(point[1]));
|
||
const f = cells.f[i];
|
||
infoLat.innerHTML = toDMS(getLatitude(y, 4), "lat");
|
||
infoLon.innerHTML = toDMS(getLongitude(x, 4), "lon");
|
||
|
||
infoCell.innerHTML = i;
|
||
infoArea.innerHTML = cells.area[i] ? si(getArea(cells.area[i])) + " " + getAreaUnit() : "n/a";
|
||
infoEvelation.innerHTML = getElevation(pack.features[f], pack.cells.h[i]);
|
||
infoDepth.innerHTML = getDepth(pack.features[f], point);
|
||
infoTemp.innerHTML = convertTemperature(grid.cells.temp[g]);
|
||
infoPrec.innerHTML = cells.h[i] >= 20 ? getFriendlyPrecipitation(i) : "n/a";
|
||
infoRiver.innerHTML = cells.h[i] >= 20 && cells.r[i] ? getRiverInfo(cells.r[i]) : "no";
|
||
infoState.innerHTML =
|
||
cells.h[i] >= 20
|
||
? cells.state[i]
|
||
? `${pack.states[cells.state[i]].fullName} (${cells.state[i]})`
|
||
: "neutral lands (0)"
|
||
: "no";
|
||
infoProvince.innerHTML = cells.province[i]
|
||
? `${pack.provinces[cells.province[i]].fullName} (${cells.province[i]})`
|
||
: "no";
|
||
infoCulture.innerHTML = cells.culture[i] ? `${pack.cultures[cells.culture[i]].name} (${cells.culture[i]})` : "no";
|
||
infoReligion.innerHTML = cells.religion[i]
|
||
? `${pack.religions[cells.religion[i]].name} (${cells.religion[i]})`
|
||
: "no";
|
||
infoPopulation.innerHTML = getFriendlyPopulation(i);
|
||
infoBurg.innerHTML = cells.burg[i] ? pack.burgs[cells.burg[i]].name + " (" + cells.burg[i] + ")" : "no";
|
||
infoFeature.innerHTML = f ? pack.features[f].group + " (" + f + ")" : "n/a";
|
||
infoBiome.innerHTML = biomesData.name[cells.biome[i]];
|
||
}
|
||
|
||
// convert coordinate to DMS format
|
||
function toDMS(coord, c) {
|
||
const degrees = Math.floor(Math.abs(coord));
|
||
const minutesNotTruncated = (Math.abs(coord) - degrees) * 60;
|
||
const minutes = Math.floor(minutesNotTruncated);
|
||
const seconds = Math.floor((minutesNotTruncated - minutes) * 60);
|
||
const cardinal = c === "lat" ? (coord >= 0 ? "N" : "S") : coord >= 0 ? "E" : "W";
|
||
return degrees + "° " + minutes + "′ " + seconds + "″ " + cardinal;
|
||
}
|
||
|
||
// get surface elevation
|
||
function getElevation(f, h) {
|
||
if (f.land) return getHeight(h) + " (" + h + ")"; // land: usual height
|
||
if (f.border) return "0 " + heightUnit.value; // ocean: 0
|
||
if (f.type === "lake") return getHeight(f.height) + " (" + f.height + ")"; // lake: defined on river generation
|
||
}
|
||
|
||
// get water depth
|
||
function getDepth(f, p) {
|
||
if (f.land) return "0 " + heightUnit.value; // land: 0
|
||
|
||
// lake: difference between surface and bottom
|
||
const gridH = grid.cells.h[findGridCell(p[0], p[1], grid)];
|
||
if (f.type === "lake") {
|
||
const depth = gridH === 19 ? f.height / 2 : gridH;
|
||
return getHeight(depth, "abs");
|
||
}
|
||
|
||
return getHeight(gridH, "abs"); // ocean: grid height
|
||
}
|
||
|
||
// get user-friendly (real-world) height value from map data
|
||
function getFriendlyHeight([x, y]) {
|
||
const packH = pack.cells.h[findCell(x, y)];
|
||
const gridH = grid.cells.h[findGridCell(x, y, grid)];
|
||
const h = packH < 20 ? gridH : packH;
|
||
return getHeight(h);
|
||
}
|
||
|
||
function getHeight(h, abs) {
|
||
const unit = heightUnit.value;
|
||
let unitRatio = 3.281; // default calculations are in feet
|
||
if (unit === "m") unitRatio = 1; // if meter
|
||
else if (unit === "f") unitRatio = 0.5468; // if fathom
|
||
|
||
let height = -990;
|
||
if (h >= 20) height = Math.pow(h - 18, +heightExponentInput.value);
|
||
else if (h < 20 && h > 0) height = ((h - 20) / h) * 50;
|
||
|
||
if (abs) height = Math.abs(height);
|
||
return rn(height * unitRatio) + " " + unit;
|
||
}
|
||
|
||
function getPrecipitation(prec) {
|
||
return prec * 100 + " mm";
|
||
}
|
||
|
||
// get user-friendly (real-world) precipitation value from map data
|
||
function getFriendlyPrecipitation(i) {
|
||
const prec = grid.cells.prec[pack.cells.g[i]];
|
||
return getPrecipitation(prec);
|
||
}
|
||
|
||
function getRiverInfo(id) {
|
||
const r = pack.rivers.find(r => r.i == id);
|
||
return r ? `${r.name} ${r.type} (${id})` : "n/a";
|
||
}
|
||
|
||
function getCellPopulation(i) {
|
||
const rural = pack.cells.pop[i] * populationRate;
|
||
const urban = pack.cells.burg[i] ? pack.burgs[pack.cells.burg[i]].population * populationRate * urbanization : 0;
|
||
return [rural, urban];
|
||
}
|
||
|
||
// get user-friendly (real-world) population value from map data
|
||
function getFriendlyPopulation(i) {
|
||
const [rural, urban] = getCellPopulation(i);
|
||
return `${si(rural + urban)} (${si(rural)} rural, urban ${si(urban)})`;
|
||
}
|
||
|
||
function getPopulationTip(i) {
|
||
const [rural, urban] = getCellPopulation(i);
|
||
return `Cell population: ${si(rural + urban)}; Rural: ${si(rural)}; Urban: ${si(urban)}`;
|
||
}
|
||
|
||
function highlightEmblemElement(type, el) {
|
||
const i = el.i,
|
||
cells = pack.cells;
|
||
const animation = d3.transition().duration(1000).ease(d3.easeSinIn);
|
||
|
||
if (type === "burg") {
|
||
const {x, y} = el;
|
||
debug
|
||
.append("circle")
|
||
.attr("cx", x)
|
||
.attr("cy", y)
|
||
.attr("r", 0)
|
||
.attr("fill", "none")
|
||
.attr("stroke", "#d0240f")
|
||
.attr("stroke-width", 1)
|
||
.attr("opacity", 1)
|
||
.transition(animation)
|
||
.attr("r", 20)
|
||
.attr("opacity", 0.1)
|
||
.attr("stroke-width", 0)
|
||
.remove();
|
||
return;
|
||
}
|
||
|
||
const [x, y] = el.pole || pack.cells.p[el.center];
|
||
const obj = type === "state" ? cells.state : cells.province;
|
||
const borderCells = cells.i.filter(id => obj[id] === i && cells.c[id].some(n => obj[n] !== i));
|
||
const data = Array.from(borderCells)
|
||
.filter((c, i) => !(i % 2))
|
||
.map(i => cells.p[i])
|
||
.map(i => [i[0], i[1], Math.hypot(i[0] - x, i[1] - y)]);
|
||
|
||
debug
|
||
.selectAll("line")
|
||
.data(data)
|
||
.enter()
|
||
.append("line")
|
||
.attr("x1", x)
|
||
.attr("y1", y)
|
||
.attr("x2", d => d[0])
|
||
.attr("y2", d => d[1])
|
||
.attr("stroke", "#d0240f")
|
||
.attr("stroke-width", 0.5)
|
||
.attr("opacity", 0.2)
|
||
.attr("stroke-dashoffset", d => d[2])
|
||
.attr("stroke-dasharray", d => d[2])
|
||
.transition(animation)
|
||
.attr("stroke-dashoffset", 0)
|
||
.attr("opacity", 1)
|
||
.transition(animation)
|
||
.delay(1000)
|
||
.attr("stroke-dashoffset", d => d[2])
|
||
.attr("opacity", 0)
|
||
.remove();
|
||
}
|
||
|
||
// assign lock behavior
|
||
document.querySelectorAll("[data-locked]").forEach(function (e) {
|
||
e.addEventListener("mouseover", function (event) {
|
||
if (this.className === "icon-lock")
|
||
tip("Click to unlock the option and allow it to be randomized on new map generation");
|
||
else tip("Click to lock the option and always use the current value on new map generation");
|
||
event.stopPropagation();
|
||
});
|
||
|
||
e.addEventListener("click", function () {
|
||
const id = this.id.slice(5);
|
||
if (this.className === "icon-lock") unlock(id);
|
||
else lock(id);
|
||
});
|
||
});
|
||
|
||
// lock option
|
||
function lock(id) {
|
||
const input = document.querySelector('[data-stored="' + id + '"]');
|
||
if (input) store(id, input.value);
|
||
const el = document.getElementById("lock_" + id);
|
||
if (!el) return;
|
||
el.dataset.locked = 1;
|
||
el.className = "icon-lock";
|
||
}
|
||
|
||
// unlock option
|
||
function unlock(id) {
|
||
localStorage.removeItem(id);
|
||
const el = document.getElementById("lock_" + id);
|
||
if (!el) return;
|
||
el.dataset.locked = 0;
|
||
el.className = "icon-lock-open";
|
||
}
|
||
|
||
// check if option is locked
|
||
function locked(id) {
|
||
const lockEl = document.getElementById("lock_" + id);
|
||
return lockEl.dataset.locked === "1";
|
||
}
|
||
|
||
// return key value stored in localStorage or null
|
||
function stored(key) {
|
||
return localStorage.getItem(key) || null;
|
||
}
|
||
|
||
// store key value in localStorage
|
||
function store(key, value) {
|
||
return localStorage.setItem(key, value);
|
||
}
|
||
|
||
// assign skeaker behaviour
|
||
Array.from(document.getElementsByClassName("speaker")).forEach(el => {
|
||
const input = el.previousElementSibling;
|
||
el.addEventListener("click", () => speak(input.value));
|
||
});
|
||
|
||
function speak(text) {
|
||
const speaker = new SpeechSynthesisUtterance(text);
|
||
const voices = speechSynthesis.getVoices();
|
||
if (voices.length) {
|
||
const voiceId = +document.getElementById("speakerVoice").value;
|
||
speaker.voice = voices[voiceId];
|
||
}
|
||
speechSynthesis.speak(speaker);
|
||
}
|
||
|
||
// apply drop-down menu option. If the value is not in options, add it
|
||
function applyOption($select, value, name = value) {
|
||
const isExisting = Array.from($select.options).some(o => o.value === value);
|
||
if (!isExisting) $select.options.add(new Option(name, value));
|
||
$select.value = value;
|
||
}
|
||
|
||
// show info about the generator in a popup
|
||
function showInfo() {
|
||
const Discord = link("https://discordapp.com/invite/X7E84HU", "Discord");
|
||
const Reddit = link("https://www.reddit.com/r/FantasyMapGenerator", "Reddit");
|
||
const Patreon = link("https://www.patreon.com/azgaar", "Patreon");
|
||
const Armoria = link("https://azgaar.github.io/Armoria", "Armoria");
|
||
const Deorum = link("https://deorum.vercel.app", "Deorum");
|
||
|
||
const QuickStart = link(
|
||
"https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Quick-Start-Tutorial",
|
||
"Quick start tutorial"
|
||
);
|
||
const QAA = link("https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Q&A", "Q&A page");
|
||
const VideoTutorial = link("https://youtube.com/playlist?list=PLtgiuDC8iVR2gIG8zMTRn7T_L0arl9h1C", "Video tutorial");
|
||
|
||
alertMessage.innerHTML = /* html */ `<b>Fantasy Map Generator</b> (FMG) is a free open-source application. It means that you own all created maps and can use them as
|
||
you wish.
|
||
|
||
<p>
|
||
The development is community-backed, you can donate on ${Patreon}. You can also help creating overviews, tutorials and spreding the word about the
|
||
Generator.
|
||
</p>
|
||
|
||
<p>
|
||
The best way to get help is to contact the community on ${Discord} and ${Reddit}. Before asking questions, please check out the ${QuickStart}, the ${QAA},
|
||
and ${VideoTutorial}.
|
||
</p>
|
||
|
||
<ul style="columns:2">
|
||
<li>${link("https://github.com/Azgaar/Fantasy-Map-Generator", "GitHub repository")}</li>
|
||
<li>${link("https://github.com/Azgaar/Fantasy-Map-Generator/blob/master/LICENSE", "License")}</li>
|
||
<li>${link("https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Changelog", "Changelog")}</li>
|
||
<li>${link("https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Hotkeys", "Hotkeys")}</li>
|
||
<li>${link("https://trello.com/b/7x832DG4/fantasy-map-generator", "Devboard")}</li>
|
||
<li><a href="mailto:azgaar.fmg@yandex.by" target="_blank">Contact Azgaar</a></li>
|
||
</ul>
|
||
|
||
<p>Check out our other projects:
|
||
<ul>
|
||
<li>${Armoria}: a tool for creating heraldic coats of arms</li>
|
||
<li>${Deorum}: a vast gallery of customizable fantasy characters</li>
|
||
</ul>
|
||
</p>
|
||
|
||
<p>Chinese localization: <a href="https://www.8desk.top" target="_blank">8desk.top</a></p>`;
|
||
|
||
$("#alert").dialog({
|
||
resizable: false,
|
||
title: document.title,
|
||
width: "28em",
|
||
buttons: {
|
||
OK: function () {
|
||
$(this).dialog("close");
|
||
}
|
||
},
|
||
position: {my: "center", at: "center", of: "svg"}
|
||
});
|
||
}
|