|
| 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 | +})(); |
0 commit comments