Skip to content

Commit 10e71b7

Browse files
authoredDec 20, 2021
the LOS lines are now put at the top of the wall instead of at the middle (Roll20#1409)
1 parent 6b9dc3d commit 10e71b7

File tree

3 files changed

+382
-8
lines changed

3 files changed

+382
-8
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,373 @@
1+
// This script allows you to import Dungeon Alchemist maps into Roll20.
2+
// By: Karel Crombecq, Briganti
3+
// Contact: info@briganti.be
4+
5+
const DungeonAlchemistImporter = (() => {
6+
// eslint-disable-line no-unused-vars
7+
8+
const scriptName = "DungeonAlchemistImporter";
9+
const version = "0.0.4";
10+
const lastUpdate = 20211220;
11+
const schemaVersion = 0.1;
12+
const defaultGridSize = 70;
13+
const clearURL = 'https://s3.amazonaws.com/files.d20.io/images/4277467/iQYjFOsYC5JsuOPUCI9RGA/thumb.png?1401938659';
14+
15+
16+
const regex = {
17+
colors: /^(?:#[0-9a-fA-F]{3}(?:[0-9a-fA-F]{3})?|transparent)$/,
18+
};
19+
20+
const _h = {
21+
outer: (...o) =>
22+
`<div style="border: 1px solid black; background-color: white; padding: 3px 3px;">${o.join(
23+
" "
24+
)}</div>`,
25+
title: (t, v) =>
26+
`<div style="font-weight: bold; border-bottom: 1px solid black;font-size: 130%;">${t} v${v}</div>`,
27+
subhead: (...o) => `<b>${o.join(" ")}</b>`,
28+
minorhead: (...o) => `<u>${o.join(" ")}</u>`,
29+
optional: (...o) => `${ch("[")}${o.join(` ${ch("|")} `)}${ch("]")}`,
30+
required: (...o) => `${ch("<")}${o.join(` ${ch("|")} `)}${ch(">")}`,
31+
header: (...o) =>
32+
`<div style="padding-left:10px;margin-bottom:3px;">${o.join(" ")}</div>`,
33+
section: (s, ...o) => `${_h.subhead(s)}${_h.inset(...o)}`,
34+
paragraph: (...o) => `<p>${o.join(" ")}</p>`,
35+
group: (...o) => `${o.join(" ")}`,
36+
items: (o) => `<li>${o.join("</li><li>")}</li>`,
37+
ol: (...o) => `<ol>${_h.items(o)}</ol>`,
38+
ul: (...o) => `<ul>${_h.items(o)}</ul>`,
39+
clearBoth: () => `<div style="clear:both;"></div>`,
40+
grid: (...o) =>
41+
`<div style="padding: 12px 0;">${o.join("")}${_h.clearBoth()}</div>`,
42+
cell: (o) =>
43+
`<div style="width: 130px; padding: 0 3px; float: left;">${o}</div>`,
44+
inset: (...o) =>
45+
`<div style="padding-left: 10px;padding-right:20px">${o.join(" ")}</div>`,
46+
join: (...o) => o.join(" "),
47+
configRow: (...o) =>
48+
`<div ${css(defaults.css.configRow)}>${o.join(" ")}</div>`,
49+
makeButton: (c, l, bc, color) =>
50+
`<a ${css({
51+
...defaults.css.button,
52+
...{ color, "background-color": bc },
53+
})} href="${c}">${l}</a>`,
54+
floatRight: (...o) => `<div style="float:right;">${o.join(" ")}</div>`,
55+
pre: (...o) =>
56+
`<div style="border:1px solid #e1e1e8;border-radius:4px;padding:8.5px;margin-bottom:9px;font-size:12px;white-space:normal;word-break:normal;word-wrap:normal;background-color:#f7f7f9;font-family:monospace;overflow:auto;">${o.join(
57+
" "
58+
)}</div>`,
59+
preformatted: (...o) => _h.pre(o.join("<br>").replace(/\s/g, ch(" "))),
60+
code: (...o) => `<code>${o.join(" ")}</code>`,
61+
attr: {
62+
bare: (o) => `${ch("@")}${ch("{")}${o}${ch("}")}`,
63+
selected: (o) => `${ch("@")}${ch("{")}selected${ch("|")}${o}${ch("}")}`,
64+
target: (o) => `${ch("@")}${ch("{")}target${ch("|")}${o}${ch("}")}`,
65+
char: (o, c) =>
66+
`${ch("@")}${ch("{")}${c || "CHARACTER UniversalVTTImporter"}${ch(
67+
"|"
68+
)}${o}${ch("}")}`,
69+
},
70+
bold: (...o) => `<b>${o.join(" ")}</b>`,
71+
italic: (...o) => `<i>${o.join(" ")}</i>`,
72+
font: {
73+
command: (...o) =>
74+
`<b><span style="font-family:serif;">${o.join(" ")}</span></b>`,
75+
},
76+
};
77+
78+
const helpParts = {
79+
helpBody: (context) =>
80+
_h.join(
81+
_h.header(
82+
_h.paragraph(
83+
`${scriptName} allows you to import Dungeon Alchemist map data (walls and lights) into Roll20.`
84+
)
85+
),
86+
_h.section(
87+
"How to import?",
88+
_h.paragraph(
89+
`In order to import a Dungeon Alchemist map into Roll20, you need to export it with the Roll20 setting. This will give you two files: your map image with extension .jpg, and .txt file with the same file name.`
90+
),
91+
_h.paragraph(
92+
`Just copy all the text in the .txt file, and paste it in the chat window. It should import all your walls and lights. If you go to the "Dynamic Lighting" layer, they should be there now.`
93+
),
94+
_h.paragraph(`That's it! Enjoy your adventure.`)
95+
)
96+
),
97+
helpConfig: (context) =>
98+
_h.outer(
99+
_h.title(scriptName, version),
100+
playerIsGM(context.playerid)
101+
? _h.group(_h.subhead("Configuration"), getAllConfigOptions())
102+
: ""
103+
),
104+
helpDoc: (context) =>
105+
_h.join(_h.title(scriptName, version), helpParts.helpBody(context)),
106+
107+
helpChat: (context) =>
108+
_h.outer(_h.title(scriptName, version), helpParts.helpBody(context)),
109+
};
110+
111+
const assureHelpHandout = (create = true) => {
112+
const helpIcon =
113+
"https://s3.amazonaws.com/files.d20.io/images/127392204/tAiDP73rpSKQobEYm5QZUw/thumb.png?15878425385";
114+
115+
// find handout
116+
let props = { type: "handout", name: `Help: ${scriptName}` };
117+
let hh = findObjs(props)[0];
118+
if (!hh) {
119+
log(`Create handout for ${scriptName}`);
120+
hh = createObj("handout", Object.assign(props, { avatar: helpIcon }));
121+
create = true;
122+
}
123+
if (create || version !== state[scriptName].lastHelpVersion) {
124+
hh.set({
125+
notes: helpParts.helpDoc({ who: "handout", playerid: "handout" }),
126+
});
127+
state[scriptName].lastHelpVersion = version;
128+
log(" > Updated Help Handout to v" + version + " <");
129+
}
130+
};
131+
132+
const checkInstall = () => {
133+
log("-=> Dungeon Alchemist importer, version " + version);
134+
if (!state[scriptName]) {
135+
state[scriptName] = {};
136+
}
137+
assureHelpHandout();
138+
};
139+
140+
const createWall = (page, map, wall, originalGridSize, gridSize) => {
141+
let x1 = wall.wall3D.p1.top.x * gridSize / originalGridSize;
142+
let y1 = wall.wall3D.p1.top.y * gridSize / originalGridSize;
143+
let x2 = wall.wall3D.p2.top.x * gridSize / originalGridSize;
144+
let y2 = wall.wall3D.p2.top.y * gridSize / originalGridSize;
145+
146+
const xCenter = (x1 + x2) * 0.5;
147+
const yCenter = (y1 + y2) * 0.5;
148+
149+
const xMin = Math.min(x1, x2);
150+
const yMin = Math.min(y1, y2);
151+
const xMax = Math.max(x1, x2);
152+
const yMax = Math.max(y1, y2);
153+
154+
const width = xMax - xMin;
155+
const height = yMax - yMin;
156+
157+
x1 -= xMin;
158+
x2 -= xMin;
159+
y1 -= yMin;
160+
y2 -= yMin;
161+
162+
const path = [
163+
["M", x1, y1],
164+
["L", x2, y2],
165+
];
166+
167+
// different wall types have different colors - we use a color scheme compatible with WOTC modules and DoorKnocker
168+
let color = "#000000";
169+
if (wall.type == 0) {
170+
color = "#0000ff"; // normal wall (blue)
171+
}
172+
else if (wall.type == 1) {
173+
color = "#00ff00"; // door (green)
174+
}
175+
else if (wall.type == 2) {
176+
color = "#00ffff"; // window (light blue)
177+
}
178+
else {
179+
return; // other types are not supported by roll20
180+
}
181+
182+
createObj("path", {
183+
pageid: page.get("_id"),
184+
stroke: color,
185+
fill: "transparent",
186+
left: xCenter,
187+
top: yCenter,
188+
width: width,
189+
height: height,
190+
rotation: 0,
191+
scaleX: 1,
192+
scaleY: 1,
193+
stroke_width: 5,
194+
layer: "walls",
195+
path: JSON.stringify(path),
196+
controller_by: map.get("_id")
197+
});
198+
};
199+
200+
const createLight = (page, light, originalGridSize, gridSize) => {
201+
202+
const x = light.position.x * gridSize / originalGridSize;
203+
const y = light.position.y * gridSize / originalGridSize;
204+
205+
206+
const range = light.range * 1.0;
207+
let dim_radius = range;
208+
let bright_radius = range / 2;
209+
210+
// convert to the local scale value
211+
const scale_number = page.get("scale_number");
212+
//log("Go from dim_radius " + dim_radius + " which has range " + range + " to per tile " + (dim_radius/originalGridSize) + " from original grid size " + originalGridSize + " and scale_number is " + scale_number);
213+
dim_radius *= scale_number / originalGridSize;
214+
bright_radius *= scale_number / originalGridSize;
215+
216+
217+
var newObj = createObj('graphic',{
218+
imgsrc: clearURL,
219+
subtype: 'token',
220+
name: '',
221+
aura1_radius: 0.5,
222+
aura1_color: "#" + light.color.substring(0, 6),
223+
224+
// UDL
225+
emits_bright_light: true,
226+
emits_low_light: true,
227+
bright_light_distance: bright_radius,
228+
low_light_distance: dim_radius,
229+
230+
width:70,
231+
height:70,
232+
top: y,
233+
left: x,
234+
layer: "walls",
235+
pageid: page.get("_id")
236+
});
237+
238+
//log("New obj light distance: " + newObj.get("bright_light_distance") + " / " + newObj.get("low_light_distance"));
239+
};
240+
241+
const getMap = (page, msg) => {
242+
243+
// simplest case - get the ONLY map graphic and use that one
244+
var mapGraphics = findObjs({
245+
_pageid: page.get("_id"),
246+
_type: "graphic",
247+
layer: "map",
248+
});
249+
250+
// filter them all so we only consider the layer=map graphics
251+
if (mapGraphics.length === 1) {
252+
return mapGraphics[0];
253+
}
254+
255+
// no map
256+
if (mapGraphics.length == 0) {
257+
sendChat(
258+
"Dungeon Alchemist",
259+
"You need to upload your map image and put it in the Map Layer before importing the line-of-sight data. Make sure that your map is in the background layer by right clicking on it, selecting \"Layer\" and choosing \"Map Layer\"."
260+
);
261+
return null;
262+
}
263+
264+
// otherwise, see if we selected one
265+
var selected = msg.selected;
266+
if (selected === undefined || selected.length == 0) {
267+
sendChat(
268+
"Dungeon Alchemist",
269+
"If you have more than one image in the map layer, you need to select the one that contains the Dungeon Alchemist map image before running the command."
270+
);
271+
return null;
272+
}
273+
else {
274+
return getObj("graphic", selected[0]._id);
275+
}
276+
};
277+
278+
const resizeMap = (gridSize, grid, page, map) => {
279+
280+
const mapWidth = grid.x * gridSize;
281+
const mapHeight = grid.y * gridSize;
282+
283+
page.set({
284+
width: (grid.x * gridSize) / defaultGridSize,
285+
height: (grid.y * gridSize) / defaultGridSize,
286+
});
287+
288+
map.set({
289+
width: mapWidth,
290+
height: mapHeight,
291+
top: mapHeight / 2,
292+
left: mapWidth / 2,
293+
layer: "map",
294+
});
295+
};
296+
297+
const handleInput = (msg) => {
298+
if (
299+
"api" === msg.type &&
300+
/^!dungeonalchemist\b/i.test(msg.content) &&
301+
playerIsGM(msg.playerid)
302+
) {
303+
const s = msg.content;
304+
const endOfHeader = s.indexOf(" ");
305+
306+
try {
307+
// parse the data
308+
const json = s.substring(endOfHeader);
309+
const data = JSON.parse(json);
310+
311+
// load the player
312+
const player = getObj("player", msg.playerid);
313+
//log("PLAYER:");
314+
//log(player);
315+
const lastPageId = player.get("_lastpage");
316+
//log(lastPageId);
317+
318+
// load the page
319+
const pageId = Campaign().get("playerpageid");
320+
//log("Page id: " + pageId + " vs last page " + lastPageId);
321+
const page = getObj("page", lastPageId);
322+
//log("PAGE:");
323+
//log(page);
324+
325+
// calculate the REAL grid size
326+
const gridSize = defaultGridSize * page.get("snapping_increment");
327+
328+
// load the map
329+
const map = getMap(page, msg);
330+
//log("MAP:");
331+
//log(map);
332+
333+
// we are done - no map found
334+
if (map === null) return;
335+
336+
// resize the map properly
337+
resizeMap(gridSize, data.grid, page, map);
338+
339+
// spawn the walls & lights
340+
for (const wall of data.walls) {
341+
createWall(page, map, wall, data.pixelsPerTile, gridSize);
342+
}
343+
344+
for (const light of data.lights) {
345+
createLight(page, light, data.pixelsPerTile, gridSize);
346+
}
347+
348+
sendChat(
349+
"Dungeon Alchemist",
350+
"Succesfully imported map data!"
351+
);
352+
} catch (err) {
353+
sendChat(
354+
"Dungeon Alchemist",
355+
"Failed to import Dungeon Alchemist map data: " + err
356+
);
357+
}
358+
}
359+
};
360+
361+
const registerEventHandlers = () => {
362+
on("chat:message", handleInput);
363+
};
364+
365+
on("ready", () => {
366+
checkInstall();
367+
registerEventHandlers();
368+
});
369+
370+
return {
371+
/* public interface */
372+
};
373+
})();

‎DungeonAlchemistImporter/DungeonAlchemist.js

+6-6
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ const DungeonAlchemistImporter = (() => {
66
// eslint-disable-line no-unused-vars
77

88
const scriptName = "DungeonAlchemistImporter";
9-
const version = "0.0.3";
10-
const lastUpdate = 20210908;
9+
const version = "0.0.4";
10+
const lastUpdate = 20211220;
1111
const schemaVersion = 0.1;
1212
const defaultGridSize = 70;
1313
const clearURL = 'https://s3.amazonaws.com/files.d20.io/images/4277467/iQYjFOsYC5JsuOPUCI9RGA/thumb.png?1401938659';
@@ -138,10 +138,10 @@ const DungeonAlchemistImporter = (() => {
138138
};
139139

140140
const createWall = (page, map, wall, originalGridSize, gridSize) => {
141-
let x1 = (wall.wall3D.p1.bottom.x+wall.wall3D.p1.top.x) * 0.5 * gridSize / originalGridSize;
142-
let y1 = (wall.wall3D.p1.bottom.y+wall.wall3D.p1.top.y) * 0.5 * gridSize / originalGridSize;
143-
let x2 = (wall.wall3D.p2.bottom.x+wall.wall3D.p2.top.x) * 0.5 * gridSize / originalGridSize;
144-
let y2 = (wall.wall3D.p2.bottom.y+wall.wall3D.p2.top.y) * 0.5 * gridSize / originalGridSize;
141+
let x1 = wall.wall3D.p1.top.x * gridSize / originalGridSize;
142+
let y1 = wall.wall3D.p1.top.y * gridSize / originalGridSize;
143+
let x2 = wall.wall3D.p2.top.x * gridSize / originalGridSize;
144+
let y2 = wall.wall3D.p2.top.y * gridSize / originalGridSize;
145145

146146
const xCenter = (x1 + x2) * 0.5;
147147
const yCenter = (y1 + y2) * 0.5;

‎DungeonAlchemistImporter/script.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
{
22
"name": "Dungeon Alchemist Importer",
33
"script": "DungeonAlchemist.js",
4-
"version": "0.0.3",
4+
"version": "0.0.4",
55
"description": "Imports Dungeon Alchemist maps. Read the guide at https://www.dungeonalchemist.com/import-to-roll20 for more info on how to use this script.",
66
"authors": "Karel Crombecq, Briganti",
77
"roll20userid": "7633819",
@@ -13,6 +13,7 @@
1313
"conflicts": [],
1414
"previousversions": [
1515
"0.0.1",
16-
"0.0.2"
16+
"0.0.2",
17+
"0.0.3"
1718
]
1819
}

0 commit comments

Comments
 (0)
Please sign in to comment.