Skip to content

Instantly share code, notes, and snippets.

@Azgaar
Last active November 22, 2023 14:26
Show Gist options
  • Star 21 You must be signed in to star a gist
  • Fork 3 You must be signed in to fork a gist
  • Save Azgaar/b845ce22ea68090d43a4ecfb914f51bd to your computer and use it in GitHub Desktop.
Save Azgaar/b845ce22ea68090d43a4ecfb914f51bd to your computer and use it in GitHub Desktop.
Fantasy Map Generator
license: gpl-3.0
height: 570
border: no

Azgaar's Fantasy Map Generator demo, v. 0.53b. Based on D3 Voronoi diagram rendered to svg.

Project goal is a procedurally generated map for my Medieval Dynasty simulator. Map should be interactive, scalable, fast and plausible. There should be enought space to place at least 500 manors within 7 regions. The imagined area is about 200.000 km2.

Click on the arrow to open the Options. Click on New map to genarate a random map based on options setup. Check out the project wiki for guidance.

This is a demo version, some new cool features are developed, but not yet deployed. Details are covered in my blog Fantasy Maps for fun and glory, development board is here. Comments and ideas are highly welcomed, kindly contact me via email. I would also like to see your completed or work in progress maps. For bug reports and change requests please use the main project issues page.

Inspiration:

@font-face {
font-family: 'Architects Daughter';
font-style: normal;
font-weight: 400;
src: local('Architects Daughter Regular'), local('ArchitectsDaughter-Regular'), url(https://fonts.gstatic.com/s/architectsdaughter/v8/RXTgOOQ9AAtaVOHxx0IUBM3t7GjCYufj5TXV5VnA2p8.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Bangers';
font-style: normal;
font-weight: 400;
src: local('Bangers Regular'), local('Bangers-Regular'), url(https://fonts.gstatic.com/s/bangers/v10/yJQgrSMUoqRj-0SbnQsv4g.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Bitter';
font-style: normal;
font-weight: 400;
src: local('Bitter Regular'), local('Bitter-Regular'), url(https://fonts.gstatic.com/s/bitter/v12/zfs6I-5mjWQ3nxqccMoL2A.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Chewy';
font-style: normal;
font-weight: 400;
src: local('Chewy Regular'), local('Chewy-Regular'), url(https://fonts.gstatic.com/s/chewy/v9/rb3O4cUMVLYzfgbaJOdJHw.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Cinzel';
font-style: normal;
font-weight: 400;
src: local('Cinzel Regular'), local('Cinzel-Regular'), url(https://fonts.gstatic.com/s/cinzel/v7/zOdksD_UUTk1LJF9z4tURA.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Comfortaa';
font-style: normal;
font-weight: 700;
src: local('Comfortaa Bold'), local('Comfortaa-Bold'), url(https://fonts.gstatic.com/s/comfortaa/v12/fND5XPYKrF2tQDwwfWZJI-gdm0LZdjqr5-oayXSOefg.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Dancing Script';
font-style: normal;
font-weight: 700;
src: local('Dancing Script Bold'), local('DancingScript-Bold'), url(https://fonts.gstatic.com/s/dancingscript/v9/KGBfwabt0ZRLA5W1ywjowUHdOuSHeh0r6jGTOGdAKHA.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Gloria Hallelujah';
font-style: normal;
font-weight: 400;
src: local('Gloria Hallelujah'), local('GloriaHallelujah'), url(https://fonts.gstatic.com/s/gloriahallelujah/v9/CA1k7SlXcY5kvI81M_R28cNDay8z-hHR7F16xrcXsJw.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Great Vibes';
font-style: normal;
font-weight: 400;
src: local('Great Vibes'), local('GreatVibes-Regular'), url(https://fonts.gstatic.com/s/greatvibes/v5/6q1c0ofG6NKsEhAc2eh-3Y4P5ICox8Kq3LLUNMylGO4.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'IM Fell English';
font-style: normal;
font-weight: 400;
src: local('IM FELL English Roman'), local('IM_FELL_English_Roman'), url(https://fonts.gstatic.com/s/imfellenglish/v7/xwIisCqGFi8pff-oa9uSVAkYLEKE0CJQa8tfZYc_plY.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Josefin Sans';
font-style: normal;
font-weight: 400;
src: local('Josefin Sans Regular'), local('JosefinSans-Regular'), url(https://fonts.gstatic.com/s/josefinsans/v12/xgzbb53t8j-Mo-vYa23n5ugdm0LZdjqr5-oayXSOefg.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Kaushan Script';
font-style: normal;
font-weight: 400;
src: local('Kaushan Script'), local('KaushanScript-Regular'), url(https://fonts.gstatic.com/s/kaushanscript/v6/qx1LSqts-NtiKcLw4N03IEd0sm1ffa_JvZxsF_BEwQk.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Lobster';
font-style: normal;
font-weight: 400;
src: local('Lobster Regular'), local('Lobster-Regular'), url(https://fonts.gstatic.com/s/lobster/v20/cycBf3mfbGkh66G5NhszPQ.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Montez';
font-style: normal;
font-weight: 400;
src: local('Montez Regular'), local('Montez-Regular'), url(https://fonts.gstatic.com/s/montez/v8/aq8el3-0osHIcFK6bXAPkw.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Orbitron';
font-style: normal;
font-weight: 400;
src: local('Orbitron Regular'), local('Orbitron-Regular'), url(https://fonts.gstatic.com/s/orbitron/v9/HmnHiRzvcnQr8CjBje6GQvesZW2xOQ-xsNqO47m55DA.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Satisfy';
font-style: normal;
font-weight: 400;
src: local('Satisfy Regular'), local('Satisfy-Regular'), url(https://fonts.gstatic.com/s/satisfy/v8/2OzALGYfHwQjkPYWELy-cw.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Shadows Into Light';
font-style: normal;
font-weight: 400;
src: local('Shadows Into Light'), local('ShadowsIntoLight'), url(https://fonts.gstatic.com/s/shadowsintolight/v7/clhLqOv7MXn459PTh0gXYFK2TSYBz0eNcHnp4YqE4Ts.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'Yellowtail';
font-style: normal;
font-weight: 400;
src: local('Yellowtail Regular'), local('Yellowtail-Regular'), url(https://fonts.gstatic.com/s/yellowtail/v8/GcIHC9QEwVkrA19LJU1qlPk_vArhqVIZ0nv9q090hN8.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
font-family: 'icons';
src: url('data:application/font-woff2;base64,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') format('woff2');
font-weight: normal;
font-style: normal;
}
[class^="icon-"]:before, [class*=" icon-"]:before {
font-family: "icons";
font-style: normal;
font-weight: normal;
speak: none;
display: inline-block;
text-decoration: inherit;
width: 1em;
text-align: center;
font-size: 1em;
margin: -1px;
padding: 0;
/* For safety - reset parent styles, that can break glyph codes*/
font-variant: normal;
text-transform: none;
line-height: 1em;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
/* text-shadow: 1px 1px 1px rgba(127, 127, 127, 0.3); */
}
.icon-pencil:before { content: '\e800'; } /* '' */
.icon-font:before { content: '\e801'; } /* '' */
.icon-arrows-cw:before { content: '\e802'; } /* '' */
.icon-doc:before { content: '\e803'; } /* '' */
.icon-trash-empty:before { content: '\e804'; } /* '' */
.icon-ok:before { content: '\e805'; } /* '' */
.icon-ok-circled:before { content: '\e806'; } /* '' */
.icon-ok-circled2:before { content: '\e807'; } /* '' */
.icon-link:before { content: '\e808'; } /* '' */
.icon-globe:before { content: '\e809'; } /* '' */
.icon-plus:before { content: '\e80a'; } /* '' */
.icon-plus-circled:before { content: '\e80b'; } /* '' */
.icon-minus-circled:before { content: '\e80c'; } /* '' */
.icon-minus:before { content: '\e80d'; } /* '' */
.icon-text-height:before { content: '\e80e'; } /* '' */
.icon-adjust:before { content: '\e80f'; } /* '' */
.icon-tag:before { content: '\e810'; } /* '' */
.icon-tags:before { content: '\e811'; } /* '' */
.icon-logout:before { content: '\e812'; } /* '' */
.icon-download:before { content: '\e813'; } /* '' */
.icon-down-circled2:before { content: '\e814'; } /* '' */
.icon-upload:before { content: '\e815'; } /* '' */
.icon-up-circled2:before { content: '\e816'; } /* '' */
.icon-cancel-circled2:before { content: '\e817'; } /* '' */
.icon-cancel-circled:before { content: '\e818'; } /* '' */
.icon-cancel:before { content: '\e819'; } /* '' */
.icon-check:before { content: '\e81a'; } /* '' */
.icon-align-left:before { content: '\e81b'; } /* '' */
.icon-align-center:before { content: '\e81c'; } /* '' */
.icon-align-right:before { content: '\e81d'; } /* '' */
.icon-align-justify:before { content: '\e81e'; } /* '' */
.icon-star:before { content: '\e81f'; } /* '' */
.icon-star-empty:before { content: '\e820'; } /* '' */
.icon-search:before { content: '\e821'; } /* '' */
.icon-mail:before { content: '\e822'; } /* '' */
.icon-eye:before { content: '\e823'; } /* '' */
.icon-eye-off:before { content: '\e824'; } /* '' */
.icon-pin:before { content: '\e825'; } /* '' */
.icon-lock-open:before { content: '\e826'; } /* '' */
.icon-lock:before { content: '\e827'; } /* '' */
.icon-attach:before { content: '\e828'; } /* '' */
.icon-home:before { content: '\e829'; } /* '' */
.icon-info-circled:before { content: '\e82a'; } /* '' */
.icon-help-circled:before { content: '\e82b'; } /* '' */
.icon-shuffle:before { content: '\e82c'; } /* '' */
.icon-ccw:before { content: '\e82d'; } /* '' */
.icon-cw:before { content: '\e82e'; } /* '' */
.icon-play:before { content: '\e82f'; } /* '' */
.icon-play-circled2:before { content: '\e830'; } /* '' */
.icon-down-big:before { content: '\e831'; } /* '' */
.icon-left-big:before { content: '\e832'; } /* '' */
.icon-right-big:before { content: '\e833'; } /* '' */
.icon-up-big:before { content: '\e834'; } /* '' */
.icon-up-open:before { content: '\e835'; } /* '' */
.icon-right-open:before { content: '\e836'; } /* '' */
.icon-left-open:before { content: '\e837'; } /* '' */
.icon-down-open:before { content: '\e838'; } /* '' */
.icon-cloud:before { content: '\e839'; } /* '' */
.icon-text-width:before { content: '\e83a'; } /* '' */
.icon-italic:before { content: '\e83b'; } /* '' */
.icon-bold:before { content: '\e83c'; } /* '' */
.icon-move:before { content: '\f047'; } /* '' */
.icon-link-ext:before { content: '\f08e'; } /* '' */
.icon-check-empty:before { content: '\f096'; } /* '' */
.icon-docs:before { content: '\f0c5'; } /* '' */
.icon-list-bullet:before { content: '\f0ca'; } /* '' */
.icon-mail-alt:before { content: '\f0e0'; } /* '' */
.icon-sitemap:before { content: '\f0e8'; } /* '' */
.icon-exchange:before { content: '\f0ec'; } /* '' */
.icon-download-cloud:before { content: '\f0ed'; } /* '' */
.icon-upload-cloud:before { content: '\f0ee'; } /* '' */
.icon-plus-squared:before { content: '\f0fe'; } /* '' */
.icon-unlink:before { content: '\f127'; } /* '' */
.icon-help:before { content: '\f128'; } /* '' */
.icon-info:before { content: '\f129'; } /* '' */
.icon-eraser:before { content: '\f12d'; } /* '' */
.icon-rocket:before { content: '\f135'; } /* '' */
.icon-lock-open-alt:before { content: '\f13e'; } /* '' */
.icon-play-circled:before { content: '\f144'; } /* '' */
.icon-minus-squared:before { content: '\f146'; } /* '' */
.icon-minus-squared-alt:before { content: '\f147'; } /* '' */
.icon-level-up:before { content: '\f148'; } /* '' */
.icon-level-down:before { content: '\f149'; } /* '' */
.icon-ok-squared:before { content: '\f14a'; } /* '' */
.icon-expand:before { content: '\f150'; } /* '' */
.icon-collapse:before { content: '\f151'; } /* '' */
.icon-expand-right:before { content: '\f152'; } /* '' */
.icon-sort-alt-up:before { content: '\f160'; } /* '' */
.icon-sort-alt-down:before { content: '\f161'; } /* '' */
.icon-right-circled2:before { content: '\f18e'; } /* '' */
.icon-left-circled2:before { content: '\f190'; } /* '' */
.icon-collapse-left:before { content: '\f191'; } /* '' */
.icon-plus-squared-alt:before { content: '\f196'; } /* '' */
.icon-history:before { content: '\f1da'; } /* '' */
.icon-header:before { content: '\f1dc'; } /* '' */
.icon-trash:before { content: '\f1f8'; } /* '' */
.icon-brush:before { content: '\f1fc'; } /* '' */
.icon-clone:before { content: '\f24d'; } /* '' */
.icon-hourglass-1:before { content: '\f251'; } /* '' */
.icon-hand-grab-o:before { content: '\f255'; } /* '' */
.icon-hand-paper-o:before { content: '\f256'; } /* '' */
.icon-calendar-check-o:before { content: '\f274'; } /* '' */
.icon-map-pin:before { content: '\f276'; } /* '' */
.icon-w:before { font-style: italic; content: 'w:'; }
.icon-n:before { font-style: italic; content: 'n:'; }
.icon-i:before { font-style: italic; content: 'i:'; }
.icon-s:before { font-style: italic; content: 's:'; }
.icon-a:before { font-style: italic; content: 'a:'; }
.icon-smooth:before {font-weight: bold; content: '∼'; }
.icon-disrupt:before {font-weight: bold; content: '෴'; }
.icon-if:before {font-style: italic; font-weight: bold; content: 'if'; }
@font-face {
font-family: 'Amatic SC';
font-style: normal;
font-weight: 700;
src: local('Amatic SC Bold'), local('AmaticSC-Bold'), url('data:application/font-woff2;base64,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') format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@media print {
div, canvas {display: none;}
}
svg {
background-color: #5E4FA2;
border: 1px solid #5e4fa2;
position: absolute;
}
canvas {
position: absolute;
pointer-events: none;
}
button {
cursor: pointer;
}
input, a {
outline: none;
}
#terrs {
stroke-width: 0.7;
stroke-linejoin: round;
mask: url(#shape);
}
#cults {
stroke-width: 0.7;
stroke-linejoin: round;
mask: url(#shape);
pointer-events: none;
}
#grid {
display: none;
fill: none;
}
#landmass {
fill-rule: evenodd;
stroke: none;
}
#lakes, #oceanLayers {
fill-rule: evenodd;
}
#coastline {
fill: none;
stroke-linejoin: round;
}
#regions {
stroke-width: 0;
fill-rule: evenodd;
stroke-linejoin: round;
mask: url(#shape);
pointer-events: none;
}
#rivers {
stroke: none;
mask: url(#shape);
cursor: pointer;
}
#burgs {
fill-opacity: 0.6;
cursor: pointer;
}
#terrain {
mask: url(#shape);
}
#hills {
stroke-width: 0.1px;
fill: #999999;
}
#mounts {
stroke-width: 0.1px;
fill: white;
}
#strokes {
stroke-width: 0.08px;
width: 2px;
stroke: #5c5c70;
stroke-dasharray: 0.5, 0.7;
stroke-linecap: round;
}
#routes, #borders {
fill: none;
}
#roads, #trails {
mask: url(#shape);
}
#swamps {
stroke-width: 0.05px;
fill: none;
stroke: #5c5c70;
}
#forests {
stroke-width: 0.1px;
stroke: #5c5c70;
}
#options .pressed {
background-color: #916e7f;
font-style: italic;
}
.editTrigger {
display: none;
position: relative;
width: 60px;
}
.editTrigger[type="number"] {
width: 44px;
height: 14px;
}
.editTrigger[type="range"] {
width: 132px;
cursor: pointer;
}
#editGroupSelect {
width: 165px;
}
#editGroupInput {
display: none;
width: 161px;
}
#editSizeIcon, #editOpacityIcon, #editShadowIcon {
display: none;
}
#editText {
width: 160px;
}
#editFontSelect {
width: 129px;
}
#editFontInput {
width: 125px;
}
#editColor {
height: 20px;
width: 50px;
padding: 0;
}
#riverScale {
width: 50px;
}
#riverAngle, #riverWidthInput, #riverIncrement {
width: 79px;
}
.editButtonS {
display: none;
cursor: pointer;
}
i.dialog-icon {
display: none;
margin: 0 -1px 0 4px;
}
.editValue {
display: none;
cursor: default;
font-size: small;
width: 34px;
}
#labels {
text-anchor: middle;
dominant-baseline: alphabetic;
text-shadow: 0 0 4px white;
cursor: pointer;
}
#countries {
dominant-baseline: central;
}
.tag {
fill: #fffa90;
stroke: #333333;
stroke-width: 1.4px;
}
.line {
stroke: #666666;
stroke-width: 1px;
}
.circle {
stroke: #666666;
stroke-width: 1px;
fill: none;
}
.drag {
text-shadow: 0 0 6px red;
}
.draggable {
cursor: move;
}
.ui-dialog, #optionsContainer {
user-select: none;
}
#options {
margin: 10px;
display: none;
font-size: smaller;
font-family: monospace;
position: absolute;
background-color: rgba(168, 130, 147, 0.85);
border: solid 1px #5e4fa2;
}
.tab {
overflow: hidden;
border-bottom: 1px solid #5d4651;;
}
button.options {
background-color: #997c89;
font-family: monospace;
font-weight: bold;
float: left;
border: none;
outline: none;
padding: 8px 16px;
transition: 0.1s;
font-size: 1em;
}
#options p {
font-style: italic;
font-weight: bold;
}
#options input[type="color"], #convertImageDialog input[type="color"] {
width: 38px;
padding: 0;
border: 0;
background: none;
cursor: pointer;
}
#options input[type="checkbox"] {
width: 8;
padding: 0;
border: 0;
background: none;
cursor: pointer;
}
#options input[type="range"] {
outline: none;
width: 120px;
height: 2px;
background: #ffffff;
top: -2px;
position: relative;
-webkit-appearance: none;
appearance: none;
}
#options input[type="range"]::-webkit-slider-thumb {
-webkit-appearance: none;
border-radius: 15%;
width: 10px;
height: 10px;
background: #916e7f;
border: 1px solid #5d4651;
cursor: pointer;
}
#options input[type="range"]::-moz-range-thumb {
appearance: none;
border-radius: 15%;
width: 10px;
height: 10px;
background: #916e7f;
border: 1px solid #5d4651;
cursor: pointer;
}
#options select {
height: 14px;
width: 122px;
border: 0;
outline: none;
font-size: smaller;
font-family: monospace;
cursor: pointer;
}
#options .buttonoff {
background-color: #b6b4b440;
color: grey;
}
#sticked {
padding: 0 5px;
}
#sticked button {
background-color: rgba(153, 124, 137, 0);
padding: 0;
margin: 1px 10px;
}
#collapsible {
padding: 10.5px;
position: absolute;
z-index: 2;
}
#optionsTrigger {
display: block;
padding: 5.5px 5.5px;
opacity: 0.6;
}
#regenerate {
display: none;
opacity: 0.9;
padding: 7px 10px;
}
button.options:hover {
background-color: #806070;
color: white;
}
button.active {
background-color: #916e7f;
color: white;
}
#layoutTab {
margin-left: 19px;
}
.tabcontent {
display: none;
padding: 0 6px 0 12px;
opacity: 0.8;
max-width: 290px;
}
.tabcontent button {
background-color: #997c89;
font-family: monospace;
border: none;
outline: none;
padding: 5px 8px;
margin: 4px 0;
transition: 0.1s;
font-size: 1em;
}
.tabcontent button:hover {
background-color: #a8879d;
}
#mapLayers {
display: inline-block;
}
.tabcontent li {
list-style-type: none;
background-color: #997c89;
cursor: pointer;
padding: 5px 8px;
margin: 4px;
transition: 0.1s;
float: left;
}
.tabcontent li:hover {
background-color: #a8879d;
}
.tabcontent li.solid {
color: #42383f;
}
p {
margin-bottom: 0;
}
#optionsContainer span {
cursor: default;
}
#statusbar {
font-family: monospace;
position: absolute;
top: 548px;
}
table {
table-layout: fixed;
width: 288px;
}
table td:nth-of-type(1) {
text-decoration: underline dotted gray;
cursor: help;
width: 126px;
}
table td:nth-of-type(3) {
text-align: right;
}
#icons {
stroke: #0d0d0d;
fill: grey;
}
#fileToLoad {
display: none;
}
#customizationMenu {
display: none;
}
.setColors {
display: inline-block;
}
body button.noicon {
width: 24px;
height: 20px;
margin: 1px;
padding: 1px 6px;
float: left;
font-family: Copperplate, monospace;
}
#brushesPanel > div, #templateEditor > div {
margin: 2px 0;
}
#templateEditor #templateTools {
display: inline-block;
margin-bottom: -3px;
}
#templateSelect {
width: 150px;
}
#templateBody > div {
border: 1px solid #a3a3a3;
border-radius: 1px;
background-image: linear-gradient(to right, #ffffff 0%, #fafafa 51%, #ebebeb 100%);
margin: 1px 1px;
width: 226px;
padding: 1px 2px;
font-size: 9px;
}
#templateBody > div:hover {
border-color: #808080;
background-image: linear-gradient(to right, #fcfcfc 0%, #ededed 51%, #dedede 100%);
}
#templateBody span {
display: inline-block;
margin: 0 1px;
float: right;
cursor: pointer;
}
#templateBody span:hover {
color: #297cb8;
}
#templateBody label {
float: right;
margin-right: 4px;
}
#templateBody label:first-of-type {
margin-right: 12px;
}
#templateBody input {
height: 4px;
width: 45px;
font-family: monospace;
height: 4px;
font-family: monospace;
}
#templateBody select {
border: 0;
background-color: rgba(255, 255, 255, 0);
width: 58px;
cursor: pointer;
}
.riverPoints {
fill: red;
stroke: none;
stroke-width: 0.1;
cursor: move;
}
.riverPoints circle:hover {
stroke: brown;
}
.drag-trigger{
border-left: 12px solid transparent;
border-right: 12px solid #916e7f;
border-top: 12px solid transparent;
position: absolute;
right: 0;
top: 100%;
margin-top: -12px;
}
.drag-trigger:hover{
cursor: move;
border-right-color: #5e4fa2;
}
#styleInputs div {
display: none;
line-height: 6px;
}
#styleInputs #styleOpacity, #styleInputs #styleFill, #styleInputs #styleFilter {
display: block;
}
#styleInputs button {
padding: 0 6px;
margin: 0 2px;
border: 1px #827c7f solid;
background-color: #ffffff;
}
.pureInput {
display: inline-block;
width: 50px;
height: 10px;
font-size: small;
font-size: smaller;
font-family: monospace;
}
.grayscale {
filter: grayscale(1);
}
.sepia {
filter: sepia(1) saturate(0.8);
}
.tint {
filter: sepia(1) hue-rotate(200deg);
}
.dingy {
filter: contrast(1) saturate(1.8) sepia(.6);
}
.color-div {
width: 32px;
height: 12px;
display: inline-block;
margin: 1px 2px;
border: 1px #c5c5c5 groove;
cursor: pointer;
}
#colorsSelect div {
height: 18px;
display: inline-block;
cursor: pointer;
}
.color-div:hover {
border-color: red;
}
.hoveredColor {
box-shadow: 0 0 1px 1px #717171;
}
.selectedColor {
border-color: red;
}
#colorScheme {
margin: 6px 1px 4px 1px;
}
#colorsSelectValue {
font-size: larger;
position: relative;
font-family: monospace;
font-weight: bold;
top: -3px;
}
.selectedCell {
stroke-width: 1;
stroke: #da3126;
}
body .ui-dialog {
padding: 1px;
font-size: 12px;
}
body .ui-dialog-titlebar {
font-size: 14px;
}
.ui-dialog input {
height: 14px;
}
.ui-dialog button.pressed {
box-shadow: inset 1px 1px 0 0 #ccc;
}
.ui-dialog input[type="range"] {
outline: none;
height: 2px;
background: #e9e9e9;
top: -4px;
position: relative;
-webkit-appearance: none;
appearance: none;
}
.ui-dialog input[type="range"]::-webkit-slider-thumb {
-webkit-appearance: none;
border-radius: 15%;
width: 10px;
height: 10px;
background: #e9e9e9;
border: 1px solid #9b9b9b;
cursor: pointer;
}
.ui-dialog input[type="range"]::-moz-range-thumb {
appearance: none;
border-radius: 15%;
width: 10px;
height: 10px;
background: #e9e9e9;
border: 1px solid #9b9b9b;
cursor: pointer;
}
.ui-dialog .disabled {
opacity: 0.2;
}
.ui-dialog .disabled::-webkit-slider-thumb {
opacity: 0.2;
}
.ui-dialog .disabled::-moz-range-thumb {
opacity: 0.2;
}
.ui-dialog:disabled {
cursor: default;
}
div.slider {
width: 40em;
margin-top: 0.2em;
}
div.slider .ui-slider-handle {
width: 3em;
height: 1.6em;
top: 50%;
margin-top: -.8em;
text-align: center;
line-height: 1.6em;
}
#saveDropdown {
display: none;
position: absolute;
left: 25%;
top: 100%;
border: 1px solid #5e4fa2;
background-color: #a4879b;
width: 44px;
}
#saveDropdown > div {
padding: 2px 4px;
cursor: pointer;
}
#saveDropdown > div:hover {
color: white;
}
#brushPower, #brushRadius {
width: 88px;
}
#rescaleHigher, #rescaleLower, #rescaleModifier {
width: 40px;
}
#rescaler {
width: 175px;
top: -2px;
}
.italic {
font-style: italic;
}
.hidden {
display: none;
}
<!DOCTYPE html>
<head>
<title>Azgaar's Fantasy Map Generator Demo</title>
<meta name="author" content="Azgaar (Max Ganiev)">
<meta charset="utf-8">
<meta name="description" content="Azgaar's Fantasy Map Generator demo. Based on D3 Voronoi diagram rendered to svg.">
<script src="https://code.jquery.com/jquery-3.1.1.min.js"></script>
<script src="https://code.jquery.com/ui/1.12.1/jquery-ui.min.js"></script>
<script src="https://d3js.org/d3.v4.min.js"></script>
<script src="https://d3js.org/d3-scale-chromatic.v1.min.js"></script>
<script src="https://mewo2.com/js/priority-queue.js"></script>
<script src="https://rawgit.com/LuisSevillano/9f6c9edd7f90ac6cca54ed744e28f3ee/raw/38f9774f83f00b286360db1ea97d851f79e594aa/polylabel.js"></script>
<script src="names.js"></script>
<link rel="stylesheet" type="text/css" href="index.css?version=0.53b"/>
<link rel="stylesheet" type="text/css" href="icons.css?version=0.53b"/>
<link rel="stylesheet" type="text/css" href="https://code.jquery.com/ui/1.12.1/themes/base/jquery-ui.css"/>
<script src="quantize.js"></script>
</head>
<body>
<svg xmlns="http://www.w3.org/2000/svg" width="960" height="540">
<defs>
<filter id="blurFilter" x="-1" y="-1" width="100" height="100">
<feGaussianBlur in="SourceGraphic" stdDeviation="0.2"/>
</filter>
<filter id="dropShadow">
<feGaussianBlur in="SourceAlpha" stdDeviation="2"/>
<feOffset dx="1" dy="2"/>
<feMerge>
<feMergeNode/>
<feMergeNode in="SourceGraphic"/>
</feMerge>
</filter>
<g id="deftemp">
<mask id="shape" x="0" y="0" width="100%" height="100%" fill="black"></mask>
</g>
<g id="defs-icons">
<symbol id="icon-anchor" viewBox="0 0 28 28">
<title>Anchor</title>
<path d="M15 4c0-0.547-0.453-1-1-1s-1 0.453-1 1 0.453 1 1 1 1-0.453 1-1zM28 18.5v5.5c0 0.203-0.125 0.391-0.313 0.469-0.063 0.016-0.125 0.031-0.187 0.031-0.125 0-0.25-0.047-0.359-0.141l-1.453-1.453c-2.453 2.953-6.859 4.844-11.688 4.844s-9.234-1.891-11.688-4.844l-1.453 1.453c-0.094 0.094-0.234 0.141-0.359 0.141-0.063 0-0.125-0.016-0.187-0.031-0.187-0.078-0.313-0.266-0.313-0.469v-5.5c0-0.281 0.219-0.5 0.5-0.5h5.5c0.203 0 0.391 0.125 0.469 0.313s0.031 0.391-0.109 0.547l-1.563 1.563c1.406 1.891 4.109 3.266 7.203 3.687v-10.109h-3c-0.547 0-1-0.453-1-1v-2c0-0.547 0.453-1 1-1h3v-2.547c-1.188-0.688-2-1.969-2-3.453 0-2.203 1.797-4 4-4s4 1.797 4 4c0 1.484-0.812 2.766-2 3.453v2.547h3c0.547 0 1 0.453 1 1v2c0 0.547-0.453 1-1 1h-3v10.109c3.094-0.422 5.797-1.797 7.203-3.687l-1.563-1.563c-0.141-0.156-0.187-0.359-0.109-0.547s0.266-0.313 0.469-0.313h5.5c0.281 0 0.5 0.219 0.5 0.5z"></path>
</symbol>
</g>
<pattern id="oceanPattern" width="100" height="100" patternUnits="userSpaceOnUse">
<filter id='image'>
<feImage x="0" y="0" width="100" height="100" xlink:href="">
</filter>
<rect width='100' height='100' filter="url(#image)" opacity='0.2'/>
</pattern>
<pattern id="mottling" width="16" height="9" patternUnits="userSpaceOnUse">
<filter id='turb'>
<feTurbulence type='fractalNoise' baseFrequency='.7' numOctaves='10' stitchTiles='stitch'/>
</filter>
<rect width='16' height='9' filter="url(#turb)"/>
</pattern>
</defs>
</svg>
<canvas id="canvas" width="960" height="540" style="opacity: 0;"></canvas>
<div id="optionsContainer">
<div id="collapsible">
<button id="optionsTrigger" class="options" title="Click to display Options">▶</button>
<button id="regenerate" class="options" title="Click to generate a new map">New Map!</button>
</div>
<div id="options">
<div class="drag-trigger"></div>
<div class="tab">
<button id="layoutTab" class="options">Layout</button>
<button id="styleTab" class="options">Style</button>
<button id="optionsTab" class="options">Options</button>
<button id="customizeTab" class="options">Customize</button>
</div>
<div id="layoutContent" class="tabcontent">
<p style="display: inline-block;">Select preset:</p>
<select id="layoutPreset">
<option value="layoutPolitical" selected>Political map</option>
<option value="layoutCultural">Cultural map</option>
<!-- <option value="layoutEconomical">Economical map</option> -->
<option value="layoutHeightmap">Heightmap</option>
<option value="layoutLandmass">Pure landmass</option>
</select>
<p>Displayed layers. Drag to move, click to toggle</p>
<div id="mapLayers">
<li title="Toggle Ocean, not movable" id="toggleOcean" onclick="$('#oceanPattern').fadeToggle()" class="solid">Ocean</li>
<li title="Toggle Landmass, not movable" id="toggleLandmass" onclick="$('#landmass').fadeToggle()" class="solid">Landmass</span></li>
<li title="Toggle Heightmap" id="toggleHeight" class="buttonoff">Heightmap</li>
<li title="Toggle Cultures map" id="toggleCultures" class="buttonoff">Cultures</li>
<li title="Toggle Routes" id="toggleRoutes" onclick="$('#routes').fadeToggle()">Routes</li>
<li title="Toggle Rivers" id="toggleRivers" onclick="$('#rivers').fadeToggle()">Rivers</li>
<li title="Toggle Countries" id="toggleCountries">Countries</li>
<li title="Toggle Borders" id="toggleBorders" onclick="$('#borders').fadeToggle()">Borders</li>
<li title="Toggle Relief icons" id="toggleRelief" onclick="$('#terrain').fadeToggle()">Relief</li>
<li title="Toggle Grid" id="toggleGrid" class="buttonoff" onclick="$('#grid').fadeToggle()">Grid</li>
<li title="Toggle Labels" id="toggleLabels" onclick="$('#labels').fadeToggle()">Labels</li>
<li title="Toggle Burg icons" id="toggleIcons" onclick="$('#burgs').fadeToggle()">Burgs</li>
</div>
</div>
<div id="styleContent" class="tabcontent">
<p style="display: inline-block;">Select element:</p>
<select id="styleElementSelect">
<option value="oceanBase" selected>Ocean</option>
<option value="landmass">Landmass</option>
<option value="terrs">Heightmap</option>
<option value="cults">Cultures</option>
<option value="roads">Roads</option>
<option value="trails">Trails</option>
<option value="searoutes">Searoutes</option>
<option value="rivers">Rivers</option>
<option value="terrain">Relief</option>
<option value="regions">Countries</option>
<option value="stateBorders">State Borders</option>
<option value="neutralBorders">Neutral Borders</option>
<option value="coastline">Coastline</option>
<option value="lakes">Lakes</option>
<option value="grid">Grid</option>
<option value="labels">Labels</option>
<option value="burgs">Burgs</option>
</select>
<div id="styleInputs">
<div id="styleFill">
Fill: <input id="styleFillInput" type="color" value="#5E4FA2"/>
<output id="styleFillOutput">#5E4FA2</output>
</div>
<div id="styleStroke">
Stroke: <input id="styleStrokeInput" type="color" value="#5E4FA2"/>
<output id="styleStrokeOutput">#5E4FA2</output>
</div>
<div id="styleMultiple">Colors:</div>
<div id="styleStrokeWidth">
<br>Stroke width: <input id="styleStrokeWidthInput" type="range" min="0" max="3" step="0.01" value="1">
<output id="styleStrokeWidthOutput">1</output>
</div>
<div id="styleStrokeDasharray">
<br>Stroke dasharray: <input id="styleStrokeDasharrayInput" class="pureInput" value="1 2">
</div>
<div id="styleStrokeLinecap">
<br>Stroke linecap: <select id="styleStrokeLinecapInput" class="pureInput">
<option value="inherit" selected>Inherit</option>
<option value="butt">Butt</option>
<option value="round">Round</option>
<option value="square">Square</option>
</select>
</div>
<div id="styleFontSize">
<br>Font size: <button title="Multiply all Fonts size by 1.1" id="styleFontPlus">+</button><button title="Multiply all Fonts size by 0.9" id="styleFontMinus">-</button>
</div>
<div id="styleSize">
<br>Radius: <button title="Multiply Radius by 1.1" id="styleFillPlus">+</button><button title="Multiply Radius by 0.9" id="styleFillMinus">-</button>
<span> Stroke: </span><button title="Multiply Stroke-width by 1.1" id="styleStrokePlus">+</button><button title="Multiply Stroke-width by 0.9" id="styleStrokeMinus">-</button>
</div>
<div id="styleOpacity">
<br>Opacity: <input id="styleOpacityInput" type="range" min="0" max="1" step="0.01" value="1">
<output id="styleOpacityOutput">1</output>
</div>
<div id="styleFilter">
<br>Filter: <select id="styleFilterInput" class="pureInput">
<option value="">None</option>
<option value="url(#blurFilter)" selected>Blur</option>
<option value="url(#dropShadow)" selected>Shadow</option>
</select>
</div>
<div id="styleScheme">
<br>Color scheme: <select id="styleSchemeInput" class="pureInput">
<option value="bright" selected>Bright</option>
<option value="light">Light</option>
<option value="green">Green</option>
<option value="monochrome">Monochrome</option>
</select>
</div>
</div>
<div id="mapFilters">
<p>Toggle filters:</p>
<button id="grayscale" class="radio">Grayscale</button>
<button id="sepia" class="radio">Sepia</button>
<button id="tint" class="radio">Tint</button>
<button id="dingy" class="radio">Dingy</button>
</div>
</div>
<div id="optionsContent" class="tabcontent">
<p>Generate new map to apply the options!</p>
<table>
<tr>
<td title="Select template to be used for a Heightmap generation">Heightmap template</td>
<td>
<select id="templateInput">
<option value="Random" selected>Random</option>
<option value="Volcano">Volcano</option>
<option value="High Island">High Island</option>
<option value="Low Island">Low Island</option>
<option value="Continents">Continents</option>
<option value="Archipelago">Archipelago</option>
<option value="Atoll">Atoll</option>
</select>
</td>
<td></td>
</tr>
<tr>
<td title="Set the graph size. Map on size 3 and 4 requires up to 1 minute to be generated!">Graph size</td>
<td>
<input id="sizeInput" type="range" min="1" max="4" value="1">
</td>
<td>
<output id="sizeOutput">1</output>
</td>
</tr>
<tr>
<td title="Allow options randomization">Randomize</td>
<td>
<input id="randomizeInput" type="range" min="0" max="1" value="1">
</td>
<td>
<output id="randomizeOutput">✓</output>
</td>
</tr>
<tr>
<td title="Define how many Settlements should be placed">Burgs count</td>
<td>
<input id="manorsInput" type="range" min="0" max="1000" value="500">
</td>
<td>
<output id="manorsOutput">500</output>
</td>
</tr>
<tr>
<td title="Define how many Countries should be created">Countries count</td>
<td>
<input id="regionsInput" type="range" min="0" max="100" value="13">
</td>
<td>
<output id="regionsOutput">13</output>
</td>
</tr>
<tr>
<td title="Define Countries size variety. Set to 0 to have all countries sized the same">Countries disbalance</td>
<td>
<input id="powerInput" type="range" min="0" max="10" step="0.2" value="5">
</td>
<td>
<output id="powerOutput">5</output><br>
</td>
</tr>
<tr>
<td title="Maximum distance to a closer manor to consider polygon as a Neutral Land">Burg influence radius</td>
<td>
<input id="neutralInput" type="range" min="1" max="500" step="1" value="200">
</td>
<td>
<output id="neutralOutput">200</output>
</td>
</tr>
<tr>
<td title="Set precipitation level. Controls river quantity and power">Precipitation</td>
<td>
<input id="precInput" type="range" min="0" max="50" value="15">
</td>
<td>
<output id="precOutput">15</output>
</td>
</tr>
<tr>
<td title="Define the land swampiness. Increase to see more marshes (turn on 'Relief' layer)">Swampness</td>
<td>
<input id="swampinessInput" type="range" min="0" max="100" value="10">
</td>
<td>
<output id="swampinessOutput">10</output>
</td>
</tr>
<tr>
<td title="Define the coastline sharpness. Decrease for a more round land shape">Coastline curvature</td>
<td>
<input id="sharpnessInput" type="range" min="0.1" max="0.2" value="0.2" step="0.05">
</td>
<td>
<output id="sharpnessOutput">0.2</output>
</td>
</tr>
<tr>
<td title="Define the Land outline layers scheme">Coast outline layers</td>
<td>
<select id="outlineLayers">
<option value="random">Random</option>
<option value="-6,-3,-1" selected>-6,-3,-1</option>
<option value="-9,-6,-3,-1">-9,-6,-3,-1</option>
<option value="-6,-5,-4,-3,-2,-1">-6,-5,-4,-3,-2,-1</option>
<option value="-9,-8,-7,-6,-5,-4,-3,-2,-1">-9,-8,-7,-6,-5,-4,-3,-2,-1</option>
<option value="-6,-4,-2">-6,-4,-2</option>
<option value="-8,-6,-4,-2">-8,-6,-4,-2</option>
</select>
</td>
<td></td>
</tr>
<tr>
<td title="Select the coastline rendering style">Coastline style</td>
<td>
<select id="curveType">
<option value="Catmull–Rom" selected>Catmull–Rom</option>
<option value="Linear">Linear</option>
<option value="Basis">Basis</option>
<option value="Cardinal">Cardinal</option>
<option value="Step">Step</option>
</select>
</td>
<td></td>
</tr>
</table>
</div>
<div id="customizeContent" class="tabcontent" style="display: block;">
<p title="Click &quot;Start&quot; to initiate customization, &quot;Complete&quot; to finalize the Heightmap">Heightmap customization:</p>
<div id="customizationMain">
<button title="Roll back to Heightmap customization" id="fromHeightmap">Roll back</button>
<button title="Start from scratch" id="fromScratch">Clear all</button>
<button class="buttonoff" title="Finalize the Heightmap. Not allowed if landmass area is insufficient" id="getMap" disabled="disabled">Complete</button>
</div>
<div id="customizationMenu" style="display: none;">
<div id="customizeTools">
<label title="Customization Tools">Tools:</label><br>
<button title="Display brushes panel" id="paintBrushes">Paint Brushes</button>
<button title="Open template editor" id="applyTemplate">Template Editor</button>
<button title="Open Image Converter" id="convertImage">Image Converter</button>
<button title="Show Heightmap in perspective" id="perspectiveView">Perspective View</button>
</div>
<label title="Count of Land cells and Land-Map ratio">Landmass: <span id="landmassCounter">0</span></label><hr>
</div>
<div id="addFeature">
<p>Click to add a Label:</p>
<button id="addLabel" class="radio">Label</button>
<button id="addBurg" class="radio">Burg</button>
<button id="addRiver" class="radio">River</button>
</div>
</div>
<div id="sticked">
<button id="randomMap" title="Generate new random map based on options being set" class="options">New Map</button>
<button id="saveButton" title="Select file format to save map" class="options">Save as</button>
<div id="saveDropdown">
<div id="saveMap" title="Save as fully functional map in .map format">.map</div>
<div id="saveSVG" title="Download the map as .svg image for later use in vector graphics editors">.svg</div>
<div id="savePNG" title="Download the visible part of the map as .png image">.png</div>
</div>
<button id="loadMap" title="Load fully functional map in a .map format" class="options">Load</button>
<input type="file" accept=".map" id="fileToLoad">
<button id="printMap" title="Print visible part of the map" class="options">Print</button>
<button id="zoomReset" title="Reset map zoom to default" class="options">Reset Zoom</button>
<!-- Zoom should be smooth and centrified, to be fixed later
<button id="zoomMinus" title="Zoom out" class="options">-</button>
<button id="zoomPlus" title="Zoom in" class="options">+</button>
-->
</div>
</div>
</div>
<div id="labelEditor" style="display: none">
<button id="editGroupButton" title="Edit label Group" class="editButton icon-list-bullet"></button>
<select id="editGroupSelect" title="Select Group for this label" class="editTrigger"/></select>
<input id="editGroupInput" placeholder="new name" title="Declare new Group for this label" class="editTrigger"/>
<span id="editGroupNew" title="Declare new Group for this label" class="editButtonS icon-plus-squared-alt"></span>
<span id="editGroupRemove" title="Remove the Group with all labels" class="editButtonS icon-trash-empty"></span>
<button id="editTextButton" title="Edit label Text" class="editButton icon-pencil"></button>
<input id="editText" class="editTrigger"/>
<span id="editTextRandom" title="Generate random name" class="editButtonS icon-shuffle"></span>
<button id="editFontButton" title="Select Font for the entire Group" class="editButton icon-font"></button>
<span id="editExternalFont" title="Fetch fonts by linking @font-face declaration" class="editButtonS icon-link"></span>
<select id="editFontSelect" class="editTrigger" title="Select one of the default Fonts"></select>
<input id="editFontInput" placeholder="link to @font-face" title="Fetch fonts by linking @font-face declaration" class="editTrigger"/>
<i id="editSizeIcon" class="icon-text-height"></i>
<input id="editSize" title="Change Font size for the entire Group" class="editTrigger" value="14" type="number" min="1" max="100" step=".5"/>
<button id="editStyleButton" title="Select Color for the entire Group" class="editButton icon-brush"></button>
<input id="editColor" type="color" class="editTrigger" value="#3e3e4b">
<i id="editOpacityIcon" class="icon-adjust"></i>
<input id="editOpacity" title="Change Opacity for the entire Group" class="editTrigger" value="1" type="number" min="0" max="1" step="0.02">
<i id="editShadowIcon" class="icon-clone"></i>
<input id="editShadow" title="Change Shadow for the entire Group" class="editTrigger" value="1" type="number" min="0" max="1" step="0.02" disabled="true">
<button id="editAngleButton" title="Rotate the label" class="editButton icon-ccw"></button>
<input id="editAngle" class="editTrigger" value="0" type="range" min="-180" max="180" step="0.2" oninput="editAngleValue.innerHTML = Math.abs(this.value)+'°'">
<span id="editAngleValue" class="editValue">0°</span>
<button id="editCopy" title="Copy the label" class="editButton icon-clone"></button>
<button id="editRemoveSingle" title="Remove the label" class="editButton icon-trash"></button>
</div>
<div id="riverEditor" style="display: none">
<button id="riverWidth" title="Change river width and widening" class="editButton icon-sort-alt-up"></button>
<i id="riverWidthIcon" title="Change river width" class="dialog-icon icon-w"></i>
<input id="riverWidthInput" title="Change river width" class="editTrigger" value="1" type="range" min="0.2" max="5" step="0.1">
<i id="riverIncrementIcon" title="Change river bed increment (widening speed)" class="dialog-icon icon-i"></i>
<input id="riverIncrement" title="Change river bed increment (widening speed)" class="editTrigger" value="1" type="range" min="0.1" max="10" step="0.2">
<button id="riverRegenerate" title="Regenerate river" class="editButton icon-shuffle"></button>
<button id="riverResize" title="Visually transform (rotate, scale) river" class="editButton icon-arrows-cw"></button>
<i id="riverAngleIcon" title="Rotate river (set angle)" class="dialog-icon icon-a"></i>
<input id="riverAngle" title="Rotate river (set angle)" class="editTrigger" value="0" type="range" min="-180" max="180" step="0.2">
<span id="riverAngleValue" class="editValue">0°</span>
<i id="riverScaleIcon" title="Change river scale" class="dialog-icon icon-s"></i>
<input id="riverScale" title="Change river scale" class="editTrigger" value="1" type="number" min="0.1" max="3" step="0.01">
<span id="riverReset" title="Reset transformation to default" class="editButtonS icon-ccw"></span>
<button id="riverAddPoint" title="Click to add a river point" class="editButton icon-plus-squared-alt"></button>
<button id="riverRemovePoint" title="Click on red circle to remove river point" class="editButton icon-minus-squared-alt"></button>
<button id="riverCopy" title="Copy river" class="editButton icon-clone"></button>
<button id="riverNew" title="Create new river clicking on map" class="editButton icon-pin"></button>
<button id="riverRemove" title="Remove river" class="editButton icon-trash"></button>
</div>
<div id="templateEditor" style="display: none">
<div id="templateTop">
Base template: <select id="templateSelect" data-prev="templateCustom" title="Select base template"/>
<option value="templateCustom" selected>Custom</option>
<option value="templateVolcano">Volcano</option>
<option value="templateHighIsland">High Island</option>
<option value="templateLowIsland">Low Island</option>
<option value="templateContinents">Continents</option>
<option value="templateArchipelago">Archipelago</option>
<option value="templateAtoll">Atoll</option>
</select>
</div>
<div id="templateTools">
<button id="templateMountain" title="Mountain: high big blob. Can be placed only once and only as a first step" class="noicon">M</button>
<button id="templateHill" title="Hill: small blob" class="noicon">H</button>
<button id="templatePit" title="Pit: round depression" class="noicon">P</button>
<button id="templateRange" title="Pit: elongated elevation" class="noicon">R</button>
<button id="templateTrough" title="Trough: elongated depression" class="noicon">T</button>
<button id="templateStrait" title="Strait: centered vertical depression" class="noicon">S</button>
<button id="templateAdd" title="Add or subtract value from all heights" class="noicon">+</button>
<button id="templateMultiply" title="Multiply all heights by factor" class="noicon">*</button>
<button id="templateSmooth" title="Smooth the map replacing cell heights by an average values of its neighbors" class="noicon">~</button>
</div>
<div id="templateBody" data-changed=0>
<div data-type="Mountain">Mountain
<span title="Remove step" class="icon-trash-empty" onclick="this.parentNode.parentNode.removeChild(this.parentNode)"></span>
</div>
</div>
<div id="templateBottom">
<button id="templateRun" title="Apply current template" class="icon-play-circled2"></button>
<button id="templateClear" title="Clear the map" class="icon-eraser"></button>
<button id="templateComplete" title="Finalize the Heightmap. Not allowed if insufficient land area available" class="icon-check"></button>
<button id="templateLoad" title="Open previously saved template" class="icon-upload"></button>
<input type="file" accept=".txt" id="templateToLoad" style="display: none;">
<button id="templateSave" title="Save template" class="icon-download"></button>
</div>
</div>
<div id="imageConverter" style="display: none">
<div id="convertImageButtons">
<input type="file" accept="image/*" id="imageToLoad" style="display: none;">
<button id="convertImageLoad" title="Load image to convert" class="icon-upload"></button>
<button id="convertAutoLum" title="Auto-assign colors based on liminosity" class="icon-adjust"></button>
<button id="convertAutoHue" title="Auto-assign colors based on hue" class="icon-brush"></button>
<button id="convertColorsMinus" title="Reduce the number of colors. Minimal number is 3" class="icon-minus-squared"></button>
<button id="convertColorsPlus" title="Increase the number of colors. Minimal number is 256" class="icon-plus-squared"></button>
<input id="convertColors" value="12" style="display: none;"/>
<button id="convertImageGrid" title="Toggle grid" class="icon-eye"></button>
<button id="convertOverlayButton" title="Change overlay opacity" class="icon-clone"></button>
<input id="convertOverlay" title="Change overlay opacity" type="range" min="0" max="1" step="0.01" value="0" style="display: none;">
<span id="convertOverlayValue" title="Overlay opacity" style="display: none;">0</span>
<button id="convertComplete" title="Complete conversion. All unassigned colors will be considered as ocean" class="icon-check"></button>
</div>
<div id="colorsSelect">
<div id="colorScheme"></div>
<span id="colorsSelectValue">0</span>
</div>
<div id="colorsAssigned" style="display: none">
<label>Assigned colors: </label><br>
</div>
<div id="colorsUnassigned" style="display: none">
<label>Unassigned colors: </label><br>
</div>
</div>
<div id="brushesPanel" style="display: none">
<div id="brushesButtons">
<button id="brushHill" title="Click on the map to place a Hill" class="feature noicon radio pressed">H</button>
<button id="brushPit" title="Click on the map to place a Pit" class="feature noicon radio">P</button>
<button id="brushRange" title="Select two points to place a Range" class="feature noicon radio">R</button>
<button id="brushTrough" title="Select two points to place a Trought" class="feature noicon radio">T</button>
<button id="brushElevate" title="Click and drag the map to increase cells elevation" class="noicon radio">↥</button>
<button id="brushDepress" title="Click and drag the map to decrease cells elevation" class="noicon radio">↧</button>
<button id="brushAlign" title="Click and drag the map to align cells elevation" class="noicon radio">=</button>
<button id="brushSmooth" title="Click and drag the map to smooth cells elevation" class="noicon radio">~</button>
</div>
<div style="display: table;">
<label title="Set the brush power" class="italic">H:</label>
<input id="brushPower" onchange="this.title = this.value" type="range" min="0.01" max="0.3" step="0.01" value="0.05">
<label id="brushRadiusLabel" title="Set the brush effective radius. Works only with 4 brushes above" class="disabled italic" disabled>R:</label>
<input id="brushRadius" onchange="this.title = this.value" type="range" min="1" max="10" value="3" class="disabled" disabled>
</div>
<div id="modifyButtons">
<button id="undo" title="Up-do the latest action" class="icon-ccw" disabled></button>
<button id="redo" title="Re-do the action" class="icon-cw" disabled></button>
<button id="rescaleButton" title="Show rescaler slider" class="icon-exchange"></button>
<input id="rescaler" title="Slide to change map height" type="range" min="1" max="10" step="0.1" value="5" class="hidden">
<button id="rescaleCondButton" title="Conditional rescaler" class="icon-if"></button>
<label class="condition hidden">h ≥</label>
<input id="rescaleLower" class="condition hidden" title="Set lower threshold" value="0.2" type="number" min="0" max="1" step="0.01">
<label class="condition hidden">≤</label>
<input id="rescaleHigher" class="condition hidden" title="Set higher threshold" value="1" type="number" min="0.01" max="1" step="0.01">
<label class="condition hidden">⇒</label>
<select class="condition hidden" id="conditionSign">
<option value="×" selected>×</option>
<option value="÷">÷</option>
<option value="+">+</option>
<option value="-">-</option>
<option value="^">^</option>
</select>
<input id="rescaleModifier" title="Set modifier value" type="number" value="0.9" min="0" max="1.5" step="0.01" class="condition hidden">
<button id="rescaleExecute" title="Run condition" class="icon-play-circled2 condition hidden"></button>
<button id="smoothHeights" title="Smooth all heights" class="icon-smooth"></button>
<button id="disruptHeights" title="Disrupt (randomize) heights" class="icon-disrupt"></button>
<button id="brushClear" title="Clear all height" class="icon-eraser"></button>
</div>
</div>
<div id="perspectivePanel" style="display: none">
<div id="lineSlider" class="slider">
<div id="lineHandle0" class="ui-slider-handle" data-value=240></div>
<div id="lineHandle1" class="ui-slider-handle" data-value=90></div>
</div><br>
<div id="ySlider" class="slider">
<div id="yHandle" class="ui-slider-handle" data-value=4></div>
</div><br>
<div id="scaleSlider" class="slider">
<div id="scaleHandle" class="ui-slider-handle" data-value=1></div>
</div><br>
<div id="heightSlider" class="slider">
<div id="heightHandle" class="ui-slider-handle" data-value=30></div>
</div><br>
<canvas id="perspective" width="480" height="200"></canvas>
</div>
<div id="alert" title="Warning!" style="display: none">
<p id="alertMessage">Warning!</p>
</div>
<div id="statusbar">
Coord: <span id="lx">0</span>/<span id="ly">0</span>;
Cell: <span id="cell">0</span>;
Height: <span id="height">0</span>;
Type: <span id="feature">no</span>;
Flux: <span id="flux">0</span>;
Region: <span id="region">no</span>;
Culture: <span id="culture">no</span>;
River: <span id="river">no</span>;
Path: <span id="path">no</span>;
Score: <span id="score">no</span>.
</div>
<script type="text/javascript" src="script.js?version=0.53b"></script>
</body>
// Names are getting procedurally generated based on Markov chain approach. Training data is below:
var cultures = ["Shwazen","Angshire","Luari","Latian","Toledi","Slovian","Varangian"];
var manorNames = [
["Achern","Aichhalden","Aitern","Albbruck","Alpirsbach","Altensteig","Althengstett","Appenweier","Auggen","Wildbad","Badenen","Badenweiler","Baiersbronn","Ballrechten","Bellingen","Berghaupten","Bernau","Biberach","Biederbach","Binzen","Birkendorf","Birkenfeld","Bischweier","Blumberg","Bollen","Bollschweil","Bonndorf","Bosingen","Braunlingen","Breisach","Breisgau","Breitnau","Brigachtal","Buchenbach","Buggingen","Buhl","Buhlertal","Calw","Dachsberg","Dobel","Donaueschingen","Dornhan","Dornstetten","Dottingen","Dunningen","Durbach","Durrheim","Ebhausen","Ebringen","Efringen","Egenhausen","Ehrenkirchen","Ehrsberg","Eimeldingen","Eisenbach","Elzach","Elztal","Emmendingen","Endingen","Engelsbrand","Enz","Enzklosterle","Eschbronn","Ettenheim","Ettlingen","Feldberg","Fischerbach","Fischingen","Fluorn","Forbach","Freiamt","Freiburg","Freudenstadt","Friedenweiler","Friesenheim","Frohnd","Furtwangen","Gaggenau","Geisingen","Gengenbach","Gernsbach","Glatt","Glatten","Glottertal","Gorwihl","Gottenheim","Grafenhausen","Grenzach","Griesbach","Gutach","Gutenbach","Hag","Haiterbach","Hardt","Harmersbach","Hasel","Haslach","Hausach","Hausen","Hausern","Heitersheim","Herbolzheim","Herrenalb","Herrischried","Hinterzarten","Hochenschwand","Hofen","Hofstetten","Hohberg","Horb","Horben","Hornberg","Hufingen","Ibach","Ihringen","Inzlingen","Kandern","Kappel","Kappelrodeck","Karlsbad","Karlsruhe","Kehl","Keltern","Kippenheim","Kirchzarten","Konigsfeld","Krozingen","Kuppenheim","Kussaberg","Lahr","Lauchringen","Lauf","Laufenburg","Lautenbach","Lauterbach","Lenzkirch","Liebenzell","Loffenau","Loffingen","Lorrach","Lossburg","Mahlberg","Malsburg","Malsch","March","Marxzell","Marzell","Maulburg","Monchweiler","Muhlenbach","Mullheim","Munstertal","Murg","Nagold","Neubulach","Neuenburg","Neuhausen","Neuried","Neuweiler","Niedereschach","Nordrach","Oberharmersbach","Oberkirch","Oberndorf","Oberbach","Oberried","Oberwolfach","Offenburg","Ohlsbach","Oppenau","Ortenberg","otigheim","Ottenhofen","Ottersweier","Peterstal","Pfaffenweiler","Pfalzgrafenweiler","Pforzheim","Rastatt","Renchen","Rheinau","Rheinfelden","Rheinmunster","Rickenbach","Rippoldsau","Rohrdorf","Rottweil","Rummingen","Rust","Sackingen","Sasbach","Sasbachwalden","Schallbach","Schallstadt","Schapbach","Schenkenzell","Schiltach","Schliengen","Schluchsee","Schomberg","Schonach","Schonau","Schonenberg","Schonwald","Schopfheim","Schopfloch","Schramberg","Schuttertal","Schwenningen","Schworstadt","Seebach","Seelbach","Seewald","Sexau","Simmersfeld","Simonswald","Sinzheim","Solden","Staufen","Stegen","Steinach","Steinen","Steinmauern","Straubenhardt","Stuhlingen","Sulz","Sulzburg","Teinach","Tiefenbronn","Tiengen","Titisee","Todtmoos","Todtnau","Todtnauberg","Triberg","Tunau","Tuningen","uhlingen","Unterkirnach","Reichenbach","Utzenfeld","Villingen","Villingendorf","Vogtsburg","Vohrenbach","Waldachtal","Waldbronn","Waldkirch","Waldshut","Wehr","Weil","Weilheim","Weisenbach","Wembach","Wieden","Wiesental","Wildberg","Winzeln","Wittlingen","Wittnau","Wolfach","Wutach","Wutoschingen","Wyhlen","Zavelstein"],
["Abingdon","Albrighton","Alcester","Almondbury","Altrincham","Amersham","Andover","Appleby","Ashboume","Atherstone","Aveton","Axbridge","Aylesbury","Baldock","Bamburgh","Barton","Basingstoke","Berden","Bere","Berkeley","Berwick","Betley","Bideford","Bingley","Birmingham","Blandford","Blechingley","Bodmin","Bolton","Bootham","Boroughbridge","Boscastle","Bossinney","Bramber","Brampton","Brasted","Bretford","Bridgetown","Bridlington","Bromyard","Bruton","Buckingham","Bungay","Burton","Calne","Cambridge","Canterbury","Carlisle","Castleton","Caus","Charmouth","Chawleigh","Chichester","Chillington","Chinnor","Chipping","Chisbury","Cleobury","Clifford","Clifton","Clitheroe","Cockermouth","Coleshill","Combe","Congleton","Crafthole","Crediton","Cuddenbeck","Dalton","Darlington","Dodbrooke","Drax","Dudley","Dunstable","Dunster","Dunwich","Durham","Dymock","Exeter","Exning","Faringdon","Felton","Fenny","Finedon","Flookburgh","Fowey","Frampton","Gateshead","Gatton","Godmanchester","Grampound","Grantham","Guildford","Halesowen","Halton","Harbottle","Harlow","Hatfield","Hatherleigh","Haydon","Helston","Henley","Hertford","Heytesbury","Hinckley","Hitchin","Holme","Hornby","Horsham","Kendal","Kenilworth","Kilkhampton","Kineton","Kington","Kinver","Kirby","Knaresborough","Knutsford","Launceston","Leighton","Lewes","Linton","Louth","Luton","Lyme","Lympstone","Macclesfield","Madeley","Malborough","Maldon","Manchester","Manningtree","Marazion","Marlborough","Marshfield","Mere","Merryfield","Middlewich","Midhurst","Milborne","Mitford","Modbury","Montacute","Mousehole","Newbiggin","Newborough","Newbury","Newenden","Newent","Norham","Northleach","Noss","Oakham","Olney","Orford","Ormskirk","Oswestry","Padstow","Paignton","Penkneth","Penrith","Penzance","Pershore","Petersfield","Pevensey","Pickering","Pilton","Pontefract","Portsmouth","Preston","Quatford","Reading","Redcliff","Retford","Rockingham","Romney","Rothbury","Rothwell","Salisbury","Saltash","Seaford","Seasalter","Sherston","Shifnal","Shoreham","Sidmouth","Skipsea","Skipton","Solihull","Somerton","Southam","Southwark","Standon","Stansted","Stapleton","Stottesdon","Sudbury","Swavesey","Tamerton","Tarporley","Tetbury","Thatcham","Thaxted","Thetford","Thornbury","Tintagel","Tiverton","Torksey","Totnes","Towcester","Tregoney","Trematon","Tutbury","Uxbridge","Wallingford","Wareham","Warenmouth","Wargrave","Warton","Watchet","Watford","Wendover","Westbury","Westcheap","Weymouth","Whitford","Wickwar","Wigan","Wigmore","Winchelsea","Winkleigh","Wiscombe","Witham","Witheridge","Wiveliscombe","Woodbury","Yeovil"],
["Adon","Aillant","Amilly","Andonville","Ardon","Artenay","Ascheres","Ascoux","Attray","Aubin","Audeville","Aulnay","Autruy","Auvilliers","Auxy","Aveyron","Baccon","Bardon","Barville","Batilly","Baule","Bazoches","Beauchamps","Beaugency","Beaulieu","Beaune","Bellegarde","Boesses","Boigny","Boiscommun","Boismorand","Boisseaux","Bondaroy","Bonnee","Bonny","Bordes","Bou","Bougy","Bouilly","Boulay","Bouzonville","Bouzy","Boynes","Bray","Breteau","Briare","Briarres","Bricy","Bromeilles","Bucy","Cepoy","Cercottes","Cerdon","Cernoy","Cesarville","Chailly","Chaingy","Chalette","Chambon","Champoulet","Chanteau","Chantecoq","Chapell","Charme","Charmont","Charsonville","Chateau","Chateauneuf","Chatel","Chatenoy","Chatillon","Chaussy","Checy","Chevannes","Chevillon","Chevilly","Chevry","Chilleurs","Choux","Chuelles","Clery","Coinces","Coligny","Combleux","Combreux","Conflans","Corbeilles","Corquilleroy","Cortrat","Coudroy","Coullons","Coulmiers","Courcelles","Courcy","Courtemaux","Courtempierre","Courtenay","Cravant","Crottes","Dadonville","Dammarie","Dampierre","Darvoy","Desmonts","Dimancheville","Donnery","Dordives","Dossainville","Douchy","Dry","Echilleuses","Egry","Engenville","Epieds","Erceville","Ervauville","Escrennes","Escrignelles","Estouy","Faverelles","Fay","Feins","Ferolles","Ferrieres","Fleury","Fontenay","Foret","Foucherolles","Freville","Gatinais","Gaubertin","Gemigny","Germigny","Gidy","Gien","Girolles","Givraines","Gondreville","Grangermont","Greneville","Griselles","Guigneville","Guilly","Gyleslonains","Huetre","Huisseau","Ingrannes","Ingre","Intville","Isdes","Jargeau","Jouy","Juranville","Bussiere","Laas","Ladon","Lailly","Langesse","Leouville","Ligny","Lombreuil","Lorcy","Lorris","Loury","Louzouer","Malesherbois","Marcilly","Mardie","Mareau","Marigny","Marsainvilliers","Melleroy","Menestreau","Merinville","Messas","Meung","Mezieres","Migneres","Mignerette","Mirabeau","Montargis","Montbarrois","Montbouy","Montcresson","Montereau","Montigny","Montliard","Mormant","Morville","Moulinet","Moulon","Nancray","Nargis","Nesploy","Neuville","Neuvy","Nevoy","Nibelle","Nogent","Noyers","Ocre","Oison","Olivet","Ondreville","Onzerain","Orleans","Ormes","Orville","Oussoy","Outarville","Ouzouer","Pannecieres","Pannes","Patay","Paucourt","Pers","Pierrefitte","Pithiverais","Pithiviers","Poilly","Potier","Prefontaines","Presnoy","Pressigny","Puiseaux","Quiers","Ramoulu","Rebrechien","Rouvray","Rozieres","Rozoy","Ruan","Sandillon","Santeau","Saran","Sceaux","Seichebrieres","Semoy","Sennely","Sermaises","Sigloy","Solterre","Sougy","Sully","Sury","Tavers","Thignonville","Thimory","Thorailles","Thou","Tigy","Tivernon","Tournoisis","Trainou","Treilles","Trigueres","Trinay","Vannes","Varennes","Vennecy","Vieilles","Vienne","Viglain","Vignes","Villamblain","Villemandeur","Villemoutiers","Villemurlin","Villeneuve","Villereau","Villevoques","Villorceau","Vimory","Vitry","Vrigny","Ivre"],
["Accumoli","Acquafondata","Acquapendente","Acuto","Affile","Agosta","Alatri","Albano","Allumiere","Alvito","Amaseno","Amatrice","Anagni","Anguillara","Anticoli","Antrodoco","Anzio","Aprilia","Aquino","Arce","Arcinazzo","Ardea","Ariccia","Arlena","Arnara","Arpino","Arsoli","Artena","Ascrea","Atina","Ausonia","Bagnoregio","Barbarano","Bassano","Bassiano","Bellegra","Belmonte","Blera","Bolsena","Bomarzo","Borbona","Borgo","Borgorose","Boville","Bracciano","Broccostella","Calcata","Camerata","Campagnano","Campodimele","Campoli","Canale","Canepina","Canino","Cantalice","Cantalupo","Canterano","Capena","Capodimonte","Capranica","Caprarola","Carbognano","Casalattico","Casalvieri","Casape","Casaprota","Casperia","Cassino","Castelforte","Castelliri","Castello","Castelnuovo","Castiglione","Castro","Castrocielo","Cave","Ceccano","Celleno","Cellere","Ceprano","Cerreto","Cervara","Cervaro","Cerveteri","Ciampino","Ciciliano","Cineto","Cisterna","Cittaducale","Cittareale","Civita","Civitavecchia","Civitella","Colfelice","Collalto","Colle","Colleferro","Collegiove","Collepardo","Collevecchio","Colli","Colonna","Concerviano","Configni","Contigliano","Corchiano","Coreno","Cori","Cottanello","Esperia","Fabrica","Faleria","Falvaterra","Fara","Farnese","Ferentino","Fiamignano","Fiano","Filacciano","Filettino","Fiuggi","Fiumicino","Fondi","Fontana","Fonte","Fontechiari","Forano","Formello","Formia","Frascati","Frasso","Frosinone","Fumone","Gaeta","Gallese","Gallicano","Gallinaro","Gavignano","Genazzano","Genzano","Gerano","Giuliano","Gorga","Gradoli","Graffignano","Greccio","Grottaferrata","Grotte","Guarcino","Guidonia","Ischia","Isola","Itri","Jenne","Labico","Labro","Ladispoli","Lanuvio","Lariano","Latera","Lenola","Leonessa","Licenza","Longone","Lubriano","Maenza","Magliano","Mandela","Manziana","Marano","Marcellina","Marcetelli","Marino","Marta","Mazzano","Mentana","Micigliano","Minturno","Mompeo","Montalto","Montasola","Monte","Montebuono","Montefiascone","Monteflavio","Montelanico","Monteleone","Montelibretti","Montenero","Monterosi","Monterotondo","Montopoli","Montorio","Moricone","Morlupo","Morolo","Morro","Nazzano","Nemi","Nepi","Nerola","Nespolo","Nettuno","Norma","Olevano","Onano","Oriolo","Orte","Orvinio","Paganico","Palestrina","Paliano","Palombara","Pastena","Patrica","Percile","Pescorocchiano","Pescosolido","Petrella","Piansano","Picinisco","Pico","Piedimonte","Piglio","Pignataro","Pisoniano","Pofi","Poggio","Poli","Pomezia","Pontecorvo","Pontinia","Ponza","Ponzano","Posta","Pozzaglia","Priverno","Proceno","Prossedi","Riano","Rieti","Rignano","Riofreddo","Ripi","Rivodutri","Rocca","Roccagiovine","Roccagorga","Roccantica","Roccasecca","Roiate","Ronciglione","Roviano","Sabaudia","Sacrofano","Salisano","Sambuci","Santa","Santi","Santopadre","Saracinesco","Scandriglia","Segni","Selci","Sermoneta","Serrone","Settefrati","Sezze","Sgurgola","Sonnino","Sora","Soriano","Sperlonga","Spigno","Stimigliano","Strangolagalli","Subiaco","Supino","Sutri","Tarano","Tarquinia","Terelle","Terracina","Tessennano","Tivoli","Toffia","Tolfa","Torre","Torri","Torrice","Torricella","Torrita","Trevi","Trevignano","Trivigliano","Turania","Tuscania","Vacone","Valentano","Vallecorsa","Vallemaio","Vallepietra","Vallerano","Vallerotonda","Vallinfreda","Valmontone","Varco","Vasanello","Vejano","Velletri","Ventotene","Veroli","Vetralla","Vicalvi","Vico","Vicovaro","Vignanello","Viterbo","Viticuso","Vitorchiano","Vivaro","Zagarolo"],
["Abanades","Ablanque","Adobes","Ajofrin","Alameda","Alaminos","Alarilla","Albalate","Albares","Albarreal","Albendiego","Alcabon","Alcanizo","Alcaudete","Alcocer","Alcolea","Alcoroches","Aldea","Aldeanueva","Algar","Algora","Alhondiga","Alique","Almadrones","Almendral","Almoguera","Almonacid","Almorox","Alocen","Alovera","Alustante","Angon","Anguita","Anover","Anquela","Arbancon","Arbeteta","Arcicollar","Argecilla","Arges","Armallones","Armuna","Arroyo","Atanzon","Atienza","Aunon","Azuqueca","Azutan","Baides","Banos","Banuelos","Barcience","Bargas","Barriopedro","Belvis","Berninches","Borox","Brihuega","Budia","Buenaventura","Bujalaro","Burguillos","Burujon","Bustares","Cabanas","Cabanillas","Calera","Caleruela","Calzada","Camarena","Campillo","Camunas","Canizar","Canredondo","Cantalojas","Cardiel","Carmena","Carranque","Carriches","Casa","Casarrubios","Casas","Casasbuenas","Caspuenas","Castejon","Castellar","Castilforte","Castillo","Castilnuevo","Cazalegas","Cebolla","Cedillo","Cendejas","Centenera","Cervera","Checa","Chequilla","Chillaron","Chiloeches","Chozas","Chueca","Cifuentes","Cincovillas","Ciruelas","Ciruelos","Cobeja","Cobeta","Cobisa","Cogollor","Cogolludo","Condemios","Congostrina","Consuegra","Copernal","Corduente","Corral","Cuerva","Domingo","Dosbarrios","Driebes","Duron","El","Embid","Erustes","Escalona","Escalonilla","Escamilla","Escariche","Escopete","Espinosa","Espinoso","Esplegares","Esquivias","Estables","Estriegana","Fontanar","Fuembellida","Fuensalida","Fuentelsaz","Gajanejos","Galve","Galvez","Garciotum","Gascuena","Gerindote","Guadamur","Henche","Heras","Herreria","Herreruela","Hijes","Hinojosa","Hita","Hombrados","Hontanar","Hontoba","Horche","Hormigos","Huecas","Huermeces","Huerta","Hueva","Humanes","Illan","Illana","Illescas","Iniestola","Irueste","Jadraque","Jirueque","Lagartera","Las","Layos","Ledanca","Lillo","Lominchar","Loranca","Los","Lucillos","Lupiana","Luzaga","Luzon","Madridejos","Magan","Majaelrayo","Malaga","Malaguilla","Malpica","Mandayona","Mantiel","Manzaneque","Maqueda","Maranchon","Marchamalo","Marjaliza","Marrupe","Mascaraque","Masegoso","Matarrubia","Matillas","Mazarete","Mazuecos","Medranda","Megina","Mejorada","Mentrida","Mesegar","Miedes","Miguel","Millana","Milmarcos","Mirabueno","Miralrio","Mocejon","Mochales","Mohedas","Molina","Monasterio","Mondejar","Montarron","Mora","Moratilla","Morenilla","Muduex","Nambroca","Navalcan","Negredo","Noblejas","Noez","Nombela","Noves","Numancia","Nuno","Ocana","Ocentejo","Olias","Olmeda","Ontigola","Orea","Orgaz","Oropesa","Otero","Palmaces","Palomeque","Pantoja","Pardos","Paredes","Pareja","Parrillas","Pastrana","Pelahustan","Penalen","Penalver","Pepino","Peralejos","Peralveche","Pinilla","Pioz","Piqueras","Polan","Portillo","Poveda","Pozo","Pradena","Prados","Puebla","Puerto","Pulgar","Quer","Quero","Quintanar","Quismondo","Rebollosa","Recas","Renera","Retamoso","Retiendas","Riba","Rielves","Rillo","Riofrio","Robledillo","Robledo","Romanillos","Romanones","Rueda","Sacecorbo","Sacedon","Saelices","Salmeron","San","Santa","Santiuste","Santo","Sartajada","Sauca","Sayaton","Segurilla","Selas","Semillas","Sesena","Setiles","Sevilleja","Sienes","Siguenza","Solanillos","Somolinos","Sonseca","Sotillo","Sotodosos","Talavera","Tamajon","Taragudo","Taravilla","Tartanedo","Tembleque","Tendilla","Terzaga","Tierzo","Tordellego","Tordelrabano","Tordesilos","Torija","Torralba","Torre","Torrecilla","Torrecuadrada","Torrejon","Torremocha","Torrico","Torrijos","Torrubia","Tortola","Tortuera","Tortuero","Totanes","Traid","Trijueque","Trillo","Turleque","Uceda","Ugena","Ujados","Urda","Utande","Valdarachas","Valdesotos","Valhermoso","Valtablado","Valverde","Velada","Viana","Vinuelas","Yebes","Yebra","Yelamos","Yeles","Yepes","Yuncler","Yunclillos","Yuncos","Yunquera","Zaorejas","Zarzuela","Zorita"],
["Belgorod","Beloberezhye","Belyi","Belz","Berestei","Berezhets","Berezovech","Berezutsk","Bobruisk","Bolonets","Borisov","Borovsk","Bozhesk","Bratslav","Bryansk","Brynsk","Buryn","Byhov","Chechersk","Chemesov","Cheremosh","Cherlen","Chern","Chernigov","Chernitsa","Chernobyl","Chernogorod","Chertoryesk","Chetvertnia","Demyansk","Derevesk","Devyagoresk","Dichin","Dmitrov","Dorogobuch","Dorogobuzh","Drestvin","Drokov","Drutsk","Dubechin","Dubichi","Dubki","Dubkov","Dveren","Galich","Glebovo","Glinsk","Goloty","Gomiy","Gorodets","Gorodische","Gorodno","Gorohovets","Goroshin","Gorval","Goryshon","Holm","Horobor","Hoten","Hotin","Hotmyzhsk","Ilovech","Ivan","Izborsk","Izheslavl","Kamenets","Kanev","Karachev","Karna","Kavarna","Klechesk","Klyapech","Kolomyya","Kolyvan","Kopyl","Korec","Kornik","Korochunov","Korshev","Korsun","Koshkin","Kotelno","Kovyla","Kozelsk","Kozelsk","Kremenets","Krichev","Krylatsk","Ksniatin","Kulatsk","Kursk","Kursk","Lebedev","Lida","Logosko","Lomihvost","Loshesk","Loshichi","Lubech","Lubno","Lubutsk","Lutsk","Luchin","Luki","Lukoml","Luzha","Lvov","Mtsensk","Mdin","Medniki","Melecha","Merech","Meretsk","Mescherskoe","Meshkovsk","Metlitsk","Mezetsk","Mglin","Mihailov","Mikitin","Mikulino","Miloslavichi","Mogilev","Mologa","Moreva","Mosalsk","Moschiny","Mozyr","Mstislav","Mstislavets","Muravin","Nemech","Nemiza","Nerinsk","Nichan","Novgorod","Novogorodok","Obolichi","Obolensk","Obolensk","Oleshsk","Olgov","Omelnik","Opoka","Opoki","Oreshek","Orlets","Osechen","Oster","Ostrog","Ostrov","Perelai","Peremil","Peremyshl","Pererov","Peresechen","Perevitsk","Pereyaslav","Pinsk","Ples","Polotsk","Pronsk","Proposhesk","Punia","Putivl","Rechitsa","Rodno","Rogachev","Romanov","Romny","Roslavl","Rostislavl","Rostovets","Rsha","Ruza","Rybchesk","Rylsk","Rzhavesk","Rzhev","Rzhischev","Sambor","Serensk","Serensk","Serpeysk","Shilov","Shuya","Sinech","Sizhka","Skala","Slovensk","Slutsk","Smedin","Sneporod","Snitin","Snovsk","Sochevo","Sokolec","Starica","Starodub","Stepan","Sterzh","Streshin","Sutesk","Svinetsk","Svisloch","Terebovl","Ternov","Teshilov","Teterin","Tiversk","Torchevsk","Toropets","Torzhok","Tripolye","Trubchevsk","Tur","Turov","Usvyaty","Uteshkov","Vasilkov","Velil","Velye","Venev","Venicha","Verderev","Vereya","Veveresk","Viazma","Vidbesk","Vidychev","Voino","Volodimer","Volok","Volyn","Vorobesk","Voronich","Voronok","Vorotynsk","Vrev","Vruchiy","Vselug","Vyatichsk","Vyatka","Vyshegorod","Vyshgorod","Vysokoe","Yagniatin","Yaropolch","Yasenets","Yuryev","Yuryevets","Zaraysk","Zhitomel","Zholvazh","Zizhech","Zubkov","Zudechev","Zvenigorod"],
["Akureyri","Aldra","Alftanes","Andenes","Austbo","Auvog","Bakkafjordur","Ballangen","Bardal","Beisfjord","Bifrost","Bildudalur","Bjerka","Bjerkvik","Bjorkosen","Bliksvaer","Blokken","Blonduos","Bolga","Bolungarvik","Borg","Borgarnes","Bosmoen","Bostad","Bostrand","Botsvika","Brautarholt","Breiddalsvik","Bringsli","Brunahlid","Budardalur","Byggdakjarni","Dalvik","Djupivogur","Donnes","Drageid","Drangsnes","Egilsstadir","Eiteroga","Elvenes","Engavogen","Ertenvog","Eskifjordur","Evenes","Eyrarbakki","Fagernes","Fallmoen","Fellabaer","Fenes","Finnoya","Fjaer","Fjelldal","Flakstad","Flateyri","Flostrand","Fludir","Gardabær","Gardur","Gimstad","Givaer","Gjeroy","Gladstad","Godoya","Godoynes","Granmoen","Gravdal","Grenivik","Grimsey","Grindavik","Grytting","Hafnir","Halsa","Hauganes","Haugland","Hauknes","Hella","Helland","Hellissandur","Hestad","Higrav","Hnifsdalur","Hofn","Hofsos","Holand","Holar","Holen","Holkestad","Holmavik","Hopen","Hovden","Hrafnagil","Hrisey","Husavik","Husvik","Hvammstangi","Hvanneyri","Hveragerdi","Hvolsvollur","Igeroy","Indre","Inndyr","Innhavet","Innnes","Isafjordur","Jarklaustur","Jarnsreykir","Junkerdal","Kaldvog","Kanstad","Karlsoy","Kavosen","Keflavik","Kjelde","Kjerstad","Klakk","Kopasker","Kopavogur","Korgen","Kristnes","Krutoga","Krystad","Kvina","Lande","Laugar","Laugaras","Laugarbakki","Laugarvatn","Laupstad","Leines","Leira","Leiren","Leland","Lenvika","Loding","Lodingen","Lonsbakki","Lopsmarka","Lovund","Luroy","Maela","Melahverfi","Meloy","Mevik","Misvaer","Mornes","Mosfellsbær","Moskenes","Myken","Naurstad","Nesberg","Nesjahverfi","Nesset","Nevernes","Obygda","Ofoten","Ogskardet","Okervika","Oknes","Olafsfjordur","Oldervika","Olstad","Onstad","Oppeid","Oresvika","Orsnes","Orsvog","Osmyra","Overdal","Prestoya","Raudalaekur","Raufarhofn","Reipo","Reykholar","Reykholt","Reykjahlid","Rif","Rinoya","Rodoy","Rognan","Rosvika","Rovika","Salhus","Sanden","Sandgerdi","Sandoker","Sandset","Sandvika","Saudarkrokur","Selfoss","Selsoya","Sennesvik","Setso","Siglufjordur","Silvalen","Skagastrond","Skjerstad","Skonland","Skorvogen","Skrova","Sleneset","Snubba","Softing","Solheim","Solheimar","Sorarnoy","Sorfugloy","Sorland","Sormela","Sorvaer","Sovika","Stamsund","Stamsvika","Stave","Stokka","Stokkseyri","Storjord","Storo","Storvika","Strand","Straumen","Strendene","Sudavik","Sudureyri","Sundoya","Sydalen","Thingeyri","Thorlakshofn","Thorshofn","Tjarnabyggd","Tjotta","Tosbotn","Traelnes","Trofors","Trones","Tverro","Ulvsvog","Unnstad","Utskor","Valla","Vandved","Varmahlid","Vassos","Vevelstad","Vidrek","Vik","Vikholmen","Vogar","Vogehamn","Vopnafjordur"]
];
// Forked from color-thief.js Copyright 2011 Lokesh Dhakar under MIT license
// var pixelArray = [[190,197,190], [202,204,200], [207,214,210]]; // ... etc;
// var cmap = MMCQ.quantize(pixelArray, colorCount);
// var palette = cmap ? cmap.palette() : null;
// Protovis. Copyright 2010 Stanford Visualization Group (http://mbostock.github.com/protovis/)
// Licensed under the BSD License: http://www.opensource.org/licenses/bsd-license.php
if (!pv) {
var pv = {
map: function(array, f) {
var o = {};
return f ? array.map(function(d, i) { o.index = i; return f.call(o, d); }) : array.slice();
},
naturalOrder: function(a, b) {
return (a < b) ? -1 : ((a > b) ? 1 : 0);
},
sum: function(array, f) {
var o = {};
return array.reduce(f ? function(p, d, i) { o.index = i; return p + f.call(o, d); } : function(p, d) { return p + d; }, 0);
},
max: function(array, f) {
return Math.max.apply(null, f ? pv.map(array, f) : array);
}
};
}
// MMCQ (Modified median cut quantization). Algorithm from the Leptonica library, modified by Nick Rabinowitz
// quantize.js Copyright 2008 Nick Rabinowitz under MIT license
var MMCQ = (function() {
// private constants
var sigbits = 5,
rshift = 8 - sigbits,
maxIterations = 1000,
fractByPopulations = 0.75;
// get reduced-space color index for a pixel
function getColorIndex(r, g, b) {
return (r << (2 * sigbits)) + (g << sigbits) + b;
}
// Simple priority queue
function PQueue(comparator) {
var contents = [],
sorted = false;
function sort() {
contents.sort(comparator);
sorted = true;
}
return {
push: function(o) {
contents.push(o);
sorted = false;
},
peek: function(index) {
if (!sorted) sort();
if (index===undefined) index = contents.length - 1;
return contents[index];
},
pop: function() {
if (!sorted) sort();
return contents.pop();
},
size: function() {
return contents.length;
},
map: function(f) {
return contents.map(f);
},
debug: function() {
if (!sorted) sort();
return contents;
}
};
}
// 3d color space box
function VBox(r1, r2, g1, g2, b1, b2, histo) {
var vbox = this;
vbox.r1 = r1;
vbox.r2 = r2;
vbox.g1 = g1;
vbox.g2 = g2;
vbox.b1 = b1;
vbox.b2 = b2;
vbox.histo = histo;
}
VBox.prototype = {
volume: function(force) {
var vbox = this;
if (!vbox._volume || force) {
vbox._volume = ((vbox.r2 - vbox.r1 + 1) * (vbox.g2 - vbox.g1 + 1) * (vbox.b2 - vbox.b1 + 1));
}
return vbox._volume;
},
count: function(force) {
var vbox = this,
histo = vbox.histo;
if (!vbox._count_set || force) {
var npix = 0,
index, i, j, k;
for (i = vbox.r1; i <= vbox.r2; i++) {
for (j = vbox.g1; j <= vbox.g2; j++) {
for (k = vbox.b1; k <= vbox.b2; k++) {
index = getColorIndex(i,j,k);
npix += (histo[index] || 0);
}
}
}
vbox._count = npix;
vbox._count_set = true;
}
return vbox._count;
},
copy: function() {
var vbox = this;
return new VBox(vbox.r1, vbox.r2, vbox.g1, vbox.g2, vbox.b1, vbox.b2, vbox.histo);
},
avg: function(force) {
var vbox = this,
histo = vbox.histo;
if (!vbox._avg || force) {
var ntot = 0,
mult = 1 << (8 - sigbits),
rsum = 0,
gsum = 0,
bsum = 0,
hval,
i, j, k, histoindex;
for (i = vbox.r1; i <= vbox.r2; i++) {
for (j = vbox.g1; j <= vbox.g2; j++) {
for (k = vbox.b1; k <= vbox.b2; k++) {
histoindex = getColorIndex(i,j,k);
hval = histo[histoindex] || 0;
ntot += hval;
rsum += (hval * (i + 0.5) * mult);
gsum += (hval * (j + 0.5) * mult);
bsum += (hval * (k + 0.5) * mult);
}
}
}
if (ntot) {
vbox._avg = [~~(rsum/ntot), ~~(gsum/ntot), ~~(bsum/ntot)];
} else {
// console.log('empty box');
vbox._avg = [
~~(mult * (vbox.r1 + vbox.r2 + 1) / 2),
~~(mult * (vbox.g1 + vbox.g2 + 1) / 2),
~~(mult * (vbox.b1 + vbox.b2 + 1) / 2)
];
}
}
return vbox._avg;
},
contains: function(pixel) {
var vbox = this,
rval = pixel[0] >> rshift;
gval = pixel[1] >> rshift;
bval = pixel[2] >> rshift;
return (rval >= vbox.r1 && rval <= vbox.r2 &&
gval >= vbox.g1 && gval <= vbox.g2 &&
bval >= vbox.b1 && bval <= vbox.b2);
}
};
// Color map
function CMap() {
this.vboxes = new PQueue(function(a,b) {
return pv.naturalOrder(
a.vbox.count()*a.vbox.volume(),
b.vbox.count()*b.vbox.volume()
);
});
}
CMap.prototype = {
push: function(vbox) {
this.vboxes.push({
vbox: vbox,
color: vbox.avg()
});
},
palette: function() {
return this.vboxes.map(function(vb) { return vb.color; });
},
size: function() {
return this.vboxes.size();
},
map: function(color) {
var vboxes = this.vboxes;
for (var i=0; i<vboxes.size(); i++) {
if (vboxes.peek(i).vbox.contains(color)) {
return vboxes.peek(i).color;
}
}
return this.nearest(color);
},
nearest: function(color) {
var vboxes = this.vboxes,
d1, d2, pColor;
for (var i=0; i<vboxes.size(); i++) {
d2 = Math.sqrt(
Math.pow(color[0] - vboxes.peek(i).color[0], 2) +
Math.pow(color[1] - vboxes.peek(i).color[1], 2) +
Math.pow(color[2] - vboxes.peek(i).color[2], 2)
);
if (d2 < d1 || d1 === undefined) {
d1 = d2;
pColor = vboxes.peek(i).color;
}
}
return pColor;
},
forcebw: function() {
// XXX: won't work yet
var vboxes = this.vboxes;
vboxes.sort(function(a,b) { return pv.naturalOrder(pv.sum(a.color), pv.sum(b.color));});
// force darkest color to black if everything < 5
var lowest = vboxes[0].color;
if (lowest[0] < 5 && lowest[1] < 5 && lowest[2] < 5)
vboxes[0].color = [0,0,0];
// force lightest color to white if everything > 251
var idx = vboxes.length-1,
highest = vboxes[idx].color;
if (highest[0] > 251 && highest[1] > 251 && highest[2] > 251)
vboxes[idx].color = [255,255,255];
}
};
// histo (1-d array, giving the number of pixels in
// each quantized region of color space), or null on error
function getHisto(pixels) {
var histosize = 1 << (3 * sigbits),
histo = new Array(histosize),
index, rval, gval, bval;
pixels.forEach(function(pixel) {
rval = pixel[0] >> rshift;
gval = pixel[1] >> rshift;
bval = pixel[2] >> rshift;
index = getColorIndex(rval, gval, bval);
histo[index] = (histo[index] || 0) + 1;
});
return histo;
}
function vboxFromPixels(pixels, histo) {
var rmin=1000000, rmax=0,
gmin=1000000, gmax=0,
bmin=1000000, bmax=0,
rval, gval, bval;
// find min/max
pixels.forEach(function(pixel) {
rval = pixel[0] >> rshift;
gval = pixel[1] >> rshift;
bval = pixel[2] >> rshift;
if (rval < rmin) rmin = rval;
else if (rval > rmax) rmax = rval;
if (gval < gmin) gmin = gval;
else if (gval > gmax) gmax = gval;
if (bval < bmin) bmin = bval;
else if (bval > bmax) bmax = bval;
});
return new VBox(rmin, rmax, gmin, gmax, bmin, bmax, histo);
}
function medianCutApply(histo, vbox) {
if (!vbox.count()) return;
var rw = vbox.r2 - vbox.r1 + 1,
gw = vbox.g2 - vbox.g1 + 1,
bw = vbox.b2 - vbox.b1 + 1,
maxw = pv.max([rw, gw, bw]);
// only one pixel, no split
if (vbox.count() == 1) {
return [vbox.copy()];
}
/* Find the partial sum arrays along the selected axis. */
var total = 0,
partialsum = [],
lookaheadsum = [],
i, j, k, sum, index;
if (maxw == rw) {
for (i = vbox.r1; i <= vbox.r2; i++) {
sum = 0;
for (j = vbox.g1; j <= vbox.g2; j++) {
for (k = vbox.b1; k <= vbox.b2; k++) {
index = getColorIndex(i,j,k);
sum += (histo[index] || 0);
}
}
total += sum;
partialsum[i] = total;
}
}
else if (maxw == gw) {
for (i = vbox.g1; i <= vbox.g2; i++) {
sum = 0;
for (j = vbox.r1; j <= vbox.r2; j++) {
for (k = vbox.b1; k <= vbox.b2; k++) {
index = getColorIndex(j,i,k);
sum += (histo[index] || 0);
}
}
total += sum;
partialsum[i] = total;
}
}
else { /* maxw == bw */
for (i = vbox.b1; i <= vbox.b2; i++) {
sum = 0;
for (j = vbox.r1; j <= vbox.r2; j++) {
for (k = vbox.g1; k <= vbox.g2; k++) {
index = getColorIndex(j,k,i);
sum += (histo[index] || 0);
}
}
total += sum;
partialsum[i] = total;
}
}
partialsum.forEach(function(d,i) {
lookaheadsum[i] = total-d;
});
function doCut(color) {
var dim1 = color + '1',
dim2 = color + '2',
left, right, vbox1, vbox2, d2, count2=0;
for (i = vbox[dim1]; i <= vbox[dim2]; i++) {
if (partialsum[i] > total / 2) {
vbox1 = vbox.copy();
vbox2 = vbox.copy();
left = i - vbox[dim1];
right = vbox[dim2] - i;
if (left <= right)
d2 = Math.min(vbox[dim2] - 1, ~~(i + right / 2));
else d2 = Math.max(vbox[dim1], ~~(i - 1 - left / 2));
// avoid 0-count boxes
while (!partialsum[d2]) d2++;
count2 = lookaheadsum[d2];
while (!count2 && partialsum[d2-1]) count2 = lookaheadsum[--d2];
// set dimensions
vbox1[dim2] = d2;
vbox2[dim1] = vbox1[dim2] + 1;
// console.log('vbox counts:', vbox.count(), vbox1.count(), vbox2.count());
return [vbox1, vbox2];
}
}
}
// determine the cut planes
return maxw == rw ? doCut('r') :
maxw == gw ? doCut('g') :
doCut('b');
}
function quantize(pixels, maxcolors) {
maxcolors++;
if (!pixels.length || maxcolors < 2 || maxcolors > 256) {return false;}
// XXX: check color content and convert to grayscale if insufficient
var histo = getHisto(pixels),
histosize = 1 << (3 * sigbits);
// check that we aren't below maxcolors already
var nColors = 0;
histo.forEach(function() { nColors++; });
if (nColors <= maxcolors) {
// XXX: generate the new colors from the histo and return
}
// get the beginning vbox from the colors
var vbox = vboxFromPixels(pixels, histo),
pq = new PQueue(function(a,b) { return pv.naturalOrder(a.count(), b.count()); });
pq.push(vbox);
// inner function to do the iteration
function iter(lh, target) {
var ncolors = 1,
niters = 0,
vbox;
while (niters < maxIterations) {
vbox = lh.pop();
if (!vbox.count()) { /* just put it back */
lh.push(vbox);
niters++;
continue;
}
// do the cut
var vboxes = medianCutApply(histo, vbox),
vbox1 = vboxes[0],
vbox2 = vboxes[1];
if (!vbox1) {
// console.log("vbox1 not defined; shouldn't happen!");
return;
}
lh.push(vbox1);
if (vbox2) { /* vbox2 can be null */
lh.push(vbox2);
ncolors++;
}
if (ncolors >= target) return;
if (niters++ > maxIterations) {
// console.log("infinite loop; perhaps too few pixels!");
return;
}
}
}
// first set of colors, sorted by population
iter(pq, fractByPopulations * maxcolors);
// Re-sort by the product of pixel occupancy times the size in color space.
var pq2 = new PQueue(function(a,b) {
return pv.naturalOrder(a.count()*a.volume(), b.count()*b.volume());
});
while (pq.size()) {
pq2.push(pq.pop());
}
// next set - generate the median cuts using the (npix * vol) sorting.
iter(pq2, maxcolors - pq2.size());
// calculate the actual colors
var cmap = new CMap();
while (pq2.size()) {cmap.push(pq2.pop());}
return cmap;
}
return {
quantize: quantize
};
})();
// Fantasy Map Generator main script
"use strict;"
fantasyMap();
function fantasyMap() {
// Declare variables
var svg = d3.select("svg"),
mapWidth = +svg.attr("width"),
mapHeight = +svg.attr("height"),
defs = svg.select("#deftemp"),
viewbox = svg.append("g").attr("id", "viewbox").on("touchmove mousemove", moved).on("click", clicked),
ocean = viewbox.append("g").attr("id", "ocean"),
oceanLayers = ocean.append("g").attr("id", "oceanLayers"),
oceanPattern = ocean.append("g").attr("id", "oceanPattern"),
landmass = viewbox.append("g").attr("id", "landmass"),
terrs = viewbox.append("g").attr("id", "terrs"),
cults = viewbox.append("g").attr("id", "cults"),
routes = viewbox.append("g").attr("id", "routes"),
roads = routes.append("g").attr("id", "roads"),
trails = routes.append("g").attr("id", "trails"),
rivers = viewbox.append("g").attr("id", "rivers"),
terrain = viewbox.append("g").attr("id", "terrain"),
regions = viewbox.append("g").attr("id", "regions"),
borders = viewbox.append("g").attr("id", "borders"),
stateBorders = borders.append("g").attr("id", "stateBorders"),
neutralBorders = borders.append("g").attr("id", "neutralBorders"),
coastline = viewbox.append("g").attr("id", "coastline"),
lakes = viewbox.append("g").attr("id", "lakes"),
grid = viewbox.append("g").attr("id", "grid"),
searoutes = routes.append("g").attr("id", "searoutes"),
labels = viewbox.append("g").attr("id", "labels"),
icons = viewbox.append("g").attr("id", "icons"),
burgs = icons.append("g").attr("id", "burgs"),
debug = viewbox.append("g").attr("id", "debug");
// Declare styles
landmass.attr("fill", "#eef6fb");
coastline.attr("opacity", .5).attr("stroke", "#1f3846").attr("stroke-width", .7).attr("filter", "url(#dropShadow)");
regions.attr("opacity", .55);
stateBorders.attr("opacity", .8).attr("stroke", "#56566d").attr("stroke-width", .5).attr("stroke-dasharray", "1.2 1.5").attr("stroke-linecap", "butt");
neutralBorders.attr("opacity", .8).attr("stroke", "#56566d").attr("stroke-width", .3).attr("stroke-dasharray", "1 1.5").attr("stroke-linecap", "butt");
cults.attr("opacity", .6);
rivers.attr("fill", "#5d97bb");
lakes.attr("fill", "#a6c1fd").attr("stroke", "#477794").attr("stroke-width", .3);
burgs.attr("fill", "#ffffff").attr("stroke", "#3e3e4b");
roads.attr("opacity", .8).attr("stroke", "#d06324").attr("stroke-width", .4).attr("stroke-dasharray", "1 2").attr("stroke-linecap", "round");
trails.attr("opacity", .8).attr("stroke", "#d06324").attr("stroke-width", .1).attr("stroke-dasharray", ".5 1").attr("stroke-linecap", "round");
searoutes.attr("opacity", .8).attr("stroke", "#ffffff").attr("stroke-width", .2).attr("stroke-dasharray", "1 2").attr("stroke-linecap", "round");
grid.attr("stroke", "#808080").attr("stroke-width", .1);
// canvas
var canvas = document.getElementById("canvas"),
ctx = canvas.getContext("2d");
// Color schemes
var color = d3.scaleSequential(d3.interpolateSpectral),
colors8 = d3.scaleOrdinal(d3.schemeSet2),
colors20 = d3.scaleOrdinal(d3.schemeCategory20);
// Version control
var version = "0.53b";
document.title = document.title + " v. " + version;
// Common variables
var customization, history = [], historyStage = -1, elSelected,
cells = [], land = [], riversData = [], manors = [],
queue = [], chain = {}, island = 0, cultureTree, manorTree;
// randomize options
var graphSize = +sizeInput.value,
manorsCount = manorsOutput.innerHTML = +manorsInput.value,
capitalsCount = regionsOutput.innerHTML = +regionsInput.value,
power = +powerInput.value,
neutral = +neutralInput.value,
swampiness = +swampinessInput.value,
sharpness = +sharpnessInput.value,
precipitation = +precInput.value;
// Groups for labels
var fonts = ["Amatic+SC:700", "Georgia", "Times New Roman", "Arial", "Comic Sans MS", "Lucida Sans Unicode", "Verdana", "Courier New"],
size = Math.round(10 - capitalsCount / 20),
capitals = labels.append("g").attr("id", "capitals").attr("fill", "#3e3e4b").attr("opacity", 1).attr("font-family", "Amatic SC").attr("data-font", "Amatic+SC:700").attr("font-size", size).attr("data-size", size),
towns = labels.append("g").attr("id", "towns").attr("fill", "#3e3e4b").attr("opacity", 1).attr("font-family", "Amatic SC").attr("data-font", "Amatic+SC:700").attr("font-size", 4).attr("data-size", 4).style("display", "none"),
size = Math.round(18 - capitalsCount / 6),
countries = labels.append("g").attr("id", "countries").attr("fill", "#3e3e4b").attr("opacity", 1).attr("font-family", "Amatic SC").attr("data-font", "Amatic+SC:700").attr("font-size", size).attr("data-size", size);
// append ocean pattern
oceanPattern.append("rect").attr("x", 0).attr("y", 0)
.attr("width", mapWidth).attr("height", mapHeight).attr("class", "pattern")
.attr("stroke", "none").attr("fill", "url(#oceanPattern)");
oceanLayers.append("rect").attr("x", 0).attr("y", 0)
.attr("width", mapWidth).attr("height", mapHeight).attr("id", "oceanBase").attr("fill", "#5167a9");
// D3 Line generator
var scX = d3.scaleLinear().domain([0, mapWidth]).range([0, mapWidth]),
scY = d3.scaleLinear().domain([0, mapHeight]).range([0, mapHeight]),
lineGen = d3.line().x(function(d) {return scX(d.scX);}).y(function(d) {return scY(d.scY);});
// main data variables
var voronoi = d3.voronoi().extent([[0, 0], [mapWidth, mapHeight]]);
var diagram, polygons, points = [], sample;
// D3 drag and zoom behavior
var scale = 1, viewX = 0, viewY = 0;
var zoom = d3.zoom().scaleExtent([1, 40]) // 40x is default max zoom
.translateExtent([[0, 0], [mapWidth, mapHeight]]) // 0,0 as default extent
.on("zoom", zoomed);
svg.call(zoom);
$("#optionsContainer").draggable({handle: ".drag-trigger", snap: "svg", snapMode: "both"});
$("#mapLayers").sortable({items: "li:not(.solid)", cancel: ".solid", update: moveLayer});
$("#templateBody").sortable({items: "div:not(div[data-type='Mountain'])"});
$("#mapLayers, #templateBody").disableSelection();
var drag = d3.drag()
.container(function() {return this;})
.subject(function() {var p=[d3.event.x, d3.event.y]; return [p, p];})
.on("start", dragstarted);
function zoomed() {
scale = d3.event.transform.k;
viewX = d3.event.transform.x;
viewY = d3.event.transform.y;
viewbox.attr("transform", d3.event.transform);
// rescale lables on zoom
labels.selectAll("g").each(function(d) {
var el = d3.select(this);
var desired = +el.attr("data-size");
var relative = roundN((desired + (desired / scale)) / 2, 2);
el.attr("font-size", relative);
var size = +el.attr("font-size");
size * scale < 6 ? el.style("display", "none") : el.style("display", "block");
});
}
// Manually update viewbox
function zoomUpdate() {
var transform = d3.zoomIdentity.translate(viewX, viewY).scale(scale);
svg.call(zoom.transform, transform);
}
// Reset zoom to initial with some duration
function resetZoom(duration) {
svg.transition().duration(duration).call(zoom.transform, d3.zoomIdentity);
}
var storedVersion = localStorage.getItem("version"); // show message on load
if (storedVersion != version) {
var message = "2018-03-07: The <i>Fantasy Map Generator</i> Demo is updated up to version " + version + ". <br><br>Main changes:";
message += "<br>* 'Graph size' option is usable now";
message += "<br>* 'Add River' button (click to auto-add a new River)";
message += "<br>* Re-worked 'Paint Brushes' (Undo-Redo etc)";
message += "<br>* 'Perspective' preview for Heightmap";
message += "<br>* Active zooming (rescale labels on zooming)";
message += "<br>* Save/Load buttons moved to Options footer";
message += "<br>* 'Print' button is added";
message += "<br><br>See ";
message += "<a href='https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Changelog' target='_blank'>changelog</a>";
message += " for additional details. Please report bugs and suggest enhancements ";
message += "<a href='https://github.com/Azgaar/Fantasy-Map-Generator/issues' target='_blank'>here</a>";
alertMessage.innerHTML = message;
$(function() {$("#alert").dialog({resizable: false, title: "Fantasy Map Generator v. " + version,
buttons: {
"Don't show again": function() {
localStorage.setItem("version", version);
$(this).dialog("close");
},
Close: function() {$(this).dialog("close");}
},
position: {my: "center", at: "center", of: "svg"}})
});
}
generate(); // genarate map on load
function generate() {
console.group("Random map");
console.time("TOTAL");
if (randomizeInput.value === "1") {randomizeOptions();}
placePoints();
calculateVoronoi(points);
detectNeighbors();
defineHeightmap();
markFeatures();
drawOcean();
reGraph();
resolveDepressions();
flux();
drawRelief();
drawCoastline();
manorsAndRegions();
if (!$("#toggleHeight").hasClass("buttonoff") && terrs.selectAll("path").size() === 0) {toggleHeight();}
console.timeEnd("TOTAL");
console.groupEnd("Random map");
}
// randomize options if randomization is allowed in option
function randomizeOptions() {
regionsInput.value = 7 + Math.floor(Math.random() * 10);
manorsInput.value = regionsInput.value * 5 + Math.floor(Math.random() * 500);
manorsCount = manorsOutput.innerHTML = manorsInput.value;
capitalsCount = regionsOutput.innerHTML = regionsInput.value;
precInput.value = 15 + Math.floor(Math.random() * 15);
precipitation = precOutput.value = +precInput.value;
}
// Locate points to calculate Voronoi diagram
function placePoints() {
console.time("placePoints");
points = [];
var radius = 5.9 / graphSize; // 5.9 is a radius to get 8k cells
var sampler = poissonDiscSampler(mapWidth, mapHeight, radius);
while (sample = sampler()) {
var x = roundN(sample[0], 2);
var y = roundN(sample[1], 2);
points.push([x, y]);
}
console.timeEnd("placePoints");
}
// Calculate Voronoi Diagram
function calculateVoronoi(points) {
console.time("calculateVoronoi");
diagram = voronoi(points),
polygons = diagram.polygons();
console.log(" cells: " + points.length);
console.timeEnd("calculateVoronoi");
}
// Get cell info on mouse move (useful for debugging)
function moved() {
var point = d3.mouse(this);
var i = diagram.find(point[0], point[1]).index;
if (i) {
var p = cells[i]; // get cell
$("#lx").text(Math.round(point[0]));
$("#ly").text(Math.round(point[1]));
$("#cell").text(i);
$("#height").text(ifDefined(p.height, 2));
$("#flux").text(ifDefined(p.flux, 3));
$("#river").text(ifDefined(p.river));
$("#region").text(ifDefined(p.region));
$("#feature").text(ifDefined(p.feature) + "" + ifDefined(p.featureNumber));
$("#score").text(ifDefined(p.score));
$("#path").text(ifDefined(p.path));
$("#culture").text(ifDefined(cultures[p.culture]));
d3.select("body").on("keydown", function() {
if (d3.event.keyCode == 32) {console.table(p);} // hotkey -space for cell
if (d3.event.keyCode == 67) {console.log(cells);} // hotkey -c for cells
if (d3.event.keyCode == 77) {console.table(manors);} // hotkey -m for manors
});
}
// draw line for Customization range placing
icons.selectAll(".line").remove();
if (customization === 1 && icons.selectAll(".tag").size() === 1) {
var x = +icons.select(".tag").attr("cx");
var y = +icons.select(".tag").attr("cy");
icons.insert("line", ":first-child").attr("class", "line").attr("x1", x).attr("y1", y).attr("x2", point[0]).attr("y2", point[1]);
}
// draw circle to show brush radius for Customization
var circle = icons.selectAll(".circle");
var brush = $("#brushesButtons .pressed");
if (customization === 1) {
if (brush.length === 0 || brush.hasClass("feature")) {circle.remove(); return;}
var r = roundN(6 / graphSize * brushRadius.value, 1);
if (circle.size() > 0) {circle.attr("r", r).attr("cx", point[0]).attr("cy", point[1]);}
else {icons.insert("circle", ":first-child").attr("class", "circle").attr("r", r).attr("cx", point[0]).attr("cy", point[1]);}
} else {circle.remove();}
}
// return value (e) if defined with specified number of decimals (f)
function ifDefined(e, f) {
if (e == undefined) {return "no";}
if (f) {return e.toFixed(f);}
return e;
}
// Drag actions
function dragstarted() {
var x0 = d3.event.x,
y0 = d3.event.y,
c0 = diagram.find(x0, y0).index,
c1 = c0;
d3.event.on("drag", function() {
if (customization === 1) {
var x1 = d3.event.x,
y1 = d3.event.y,
c2 = diagram.find(x1, y1).index;
if (c2 !== c1) {
c1 = c2;
var brush = $("#brushesButtons .pressed").attr("id");
var power = +brushPower.value;
if (brush === "brushHill") {add(c2, "hill", power);}
if (brush === "brushPit") {addPit(1, power, c2);}
if (!$("#brushesButtons .pressed").hasClass("feature")) {
// move a circle to show actual change radius
var radius = +brushRadius.value;
var r = roundN(6 / graphSize * radius, 1);
var circle = icons.selectAll(".circle");
if (circle.size() > 0) {circle.attr("r", r).attr("cx", x1).attr("cy", y1);}
else {icons.insert("circle", ":first-child").attr("class", "circle").attr("r", r).attr("cx", x1).attr("cy", y1);}
updateCellsInRadius(c2, c0);
}
}
mockHeightmap();
} else {
viewbox.on(".drag", null);
}
});
}
// update cells in radius if non-feature brush selected on both single click and drag
function updateCellsInRadius(cell, source) {
var power = +brushPower.value;
var radius = +brushRadius.value;
var brush = $("#brushesButtons .pressed").attr("id");
if ($("#brushesButtons .pressed").hasClass("feature")) {return;}
// define selection besed on radius
var selection = [cell];
while (radius > 1) {
var frontier = selection.slice();
frontier.map(function(s) {
cells[s].neighbors.forEach(function(e) {
if (selection.indexOf(e) === -1) {selection.push(e);}
});
});
radius--;
}
// change each cell in the selection
var sourceHeight = cells[source].height;
selection.map(function(s) {
if (brush === "brushElevate") {
if (cells[s].height < 0.2) {cells[s].height = 0.2}
else {cells[s].height += power;}
}
if (brush === "brushDepress") {cells[s].height -= power;}
if (brush === "brushAlign") {cells[s].height = sourceHeight;}
if (brush === "brushSmooth") {
var heights = [cells[s].height];
cells[s].neighbors.forEach(function(e) {heights.push(cells[e].height);});
cells[s].height = (cells[s].height + d3.mean(heights)) / 2;
}
});
}
// turn D3 polygons array into cell array, define neighbors for each cell
function detectNeighbors(withGrid) {
console.time("detectNeighbors");
var gridPath = ""; // store grid as huge single path string
polygons.map(function(i, d) {
var neighbors = [];
var type; // define type, -99 for map borders
if (withGrid) {gridPath += "M" + i.join("L") + "Z";} // grid path
diagram.cells[d].halfedges.forEach(function(e) {
var edge = diagram.edges[e], ea;
if (edge.left && edge.right) {
ea = edge.left.index;
if (ea === d) {ea = edge.right.index;}
neighbors.push(ea);
} else {
if (edge.left) {ea = edge.left.index;} else {ea = edge.right.index;}
type = -99; // polygon is on border if it has edge without opposite side polygon
}
})
cells.push({index: d, data: i.data, height: 0, type, neighbors});
});
if (withGrid) {grid.append("path").attr("d", round(gridPath, 1));}
console.timeEnd("detectNeighbors");
}
// Generate Heigtmap routine
function defineHeightmap() {
console.time('defineHeightmap');
var mapTemplate = templateInput.value;
if (mapTemplate === "Random") {
var rnd = Math.random();
if (rnd > 0.9) {mapTemplate = "Volcano";}
if (rnd > 0.8 && rnd <= 0.9) {mapTemplate = "High Island";}
if (rnd > 0.6 && rnd <= 0.8) {mapTemplate = "Low Island";}
if (rnd > 0.35 && rnd <= 0.6) {mapTemplate = "Continents";}
if (rnd > 0.01 && rnd <= 0.35) {mapTemplate = "Archipelago";}
if (rnd <= 0.01) {mapTemplate = "Atoll";}
}
addMountain();
if (mapTemplate === "Volcano") {templateVolcano();}
if (mapTemplate === "High Island") {templateHighIsland();}
if (mapTemplate === "Low Island") {templateLowIsland();}
if (mapTemplate === "Continents") {templateContinents();}
if (mapTemplate === "Archipelago") {templateArchipelago();}
if (mapTemplate === "Atoll") {templateAtoll();}
console.log(mapTemplate + " template is applied");
console.timeEnd('defineHeightmap');
}
// Heighmap Template: Volcano
function templateVolcano() {
modifyHeights("all", 0.05, 1.1);
addHill(5, 0.4);
addHill(2, 0.15);
addRange(3);
addRange(-3);
}
// Heighmap Template: High Island
function templateHighIsland() {
modifyHeights("all", 0.05, 0.9);
addRange(4);
addHill(12, 0.25);
addRange(-3);
modifyHeights("land", 0, 0.75);
addHill(3, 0.15);
}
// Heighmap Template: Low Island
function templateLowIsland() {
smoothHeights(2);
addRange(1);
addHill(4, 0.4);
addHill(12, 0.2);
addRange(-8);
modifyHeights("land", 0, 0.35);
}
// Heighmap Template: Continents
function templateContinents() {
addHill(24, 0.25);
addRange(4);
addHill(3, 0.18);
modifyHeights("land", 0, 0.7);
var count = Math.ceil(Math.random() * 6 + 2);
addStrait(count);
smoothHeights(2);
addPit(7);
addRange(-8);
modifyHeights("land", 0, 0.8);
modifyHeights("all", 0.02, 1);
}
// Heighmap Template: Archipelago
function templateArchipelago() {
modifyHeights("land", -0.2, 1);
addHill(14, 0.17);
addRange(5);
var count = Math.ceil(Math.random() * 2 + 2);
addStrait(count);
addRange(-12);
addPit(8);
modifyHeights("land", -0.05, 0.7);
smoothHeights(4);
}
// Heighmap Template: Atoll
function templateAtoll() {
addHill(2, 0.35);
addRange(2);
modifyHeights("all", 0.07, 1);
smoothHeights(1);
modifyHeights("0.27-10", 0, 0.1);
}
function addMountain() {
var x = Math.floor(Math.random() * mapWidth / 3 + mapWidth / 3);
var y = Math.floor(Math.random() * mapHeight * 0.2 + mapHeight * 0.4);
var rnd = diagram.find(x, y).index;
var height = Math.random() * 0.1 + 0.9;
add(rnd, "mountain", height);
}
function addHill(count, shift) {
// shift from 0 to 0.5
for (c = 0; c < count; c++) {
var limit = 0;
do {
var height = Math.random() * 0.4 + 0.1;
var x = Math.floor(Math.random() * mapWidth * (1-shift*2) + mapWidth * shift);
var y = Math.floor(Math.random() * mapHeight * (1-shift*2) + mapHeight * shift);
var rnd = diagram.find(x, y).index;
limit ++;
} while (cells[rnd].height + height > 0.9 && limit < 100)
add(rnd, "hill", height);
}
}
function add(start, type, height) {
var sharpness = 0.2;
var radius, hRadius, mRadius;
switch (+graphSize) {
case 1: hRadius = 0.991; mRadius = 0.91; break;
case 2: hRadius = 0.9967; mRadius = 0.951; break;
case 3: hRadius = 0.999; mRadius = 0.975; break;
case 4: hRadius = 0.9994; mRadius = 0.98; break;
}
radius = type === "mountain" ? mRadius : hRadius;
var queue = [], used = [];
cells[start].height += height;
cells[start].feature = undefined;
queue.push(start);
used.push(start);
for (i = 0; i < queue.length && height >= 0.01; i++) {
if (type == "mountain") {
height = +cells[queue[i]].height * radius - height / 100;
} else {
height *= radius;
}
cells[queue[i]].neighbors.forEach(function(e) {
if (used.indexOf(e) < 0) {
var mod = Math.random() * sharpness + 1.1 - sharpness;
if (sharpness == 0) {mod = 1;}
cells[e].height += height * mod;
if (cells[e].height > 1) {cells[e].height = 1;}
cells[e].feature = undefined;
queue.push(e);
used.push(e);
}
});
}
}
function addRange(mod, height, from, to) {
var count = Math.abs(mod);
for (c = 0; c < count; c++) {
var diff = 0, start = from, end = to;
if (!start || !end) {
do {
var xf = Math.floor(Math.random() * (mapWidth*0.7)) + mapWidth*0.15;
var yf = Math.floor(Math.random() * (mapHeight*0.6)) + mapHeight*0.2;
start = diagram.find(xf, yf).index;
var xt = Math.floor(Math.random() * (mapWidth*0.7)) + mapWidth*0.15;
var yt = Math.floor(Math.random() * (mapHeight*0.6)) + mapHeight*0.2;
end = diagram.find(xt, yt).index;
diff = Math.hypot(xt - xf, yt - yf);
} while (diff < 150 / graphSize || diff > 300 / graphSize)
}
var range = [], used = [];
if (start && end) {
for (var l = 0; start != end && l < 10000; l++) {
var min = 10000;
cells[start].neighbors.forEach(function(e) {
diff = Math.hypot(cells[end].data[0] - cells[e].data[0], cells[end].data[1] - cells[e].data[1]);
if (Math.random() > 0.8) {diff = diff / 2}
if (diff < min) {min = diff, start = e;}
});
range.push(start);
}
}
var change = height ? height : Math.random() * 0.1 + 0.1;
range.map(function(r) {
var rnd = Math.random() * 0.4 + 0.8;
if (mod > 0) {cells[r].height += change * rnd;}
else if (cells[r].height >= 0.1) {cells[r].height -= change * rnd;}
used.push(r);
cells[r].neighbors.forEach(function(e) {
if (used.indexOf(e) == -1) {
used.push(e);
rnd = Math.random() * 0.4 + 0.8;
if (mod > 0) {
cells[e].height += change / 2 * rnd;
} else if (cells[e].height >= 0.1) {
cells[e].height -= change / 2 * rnd;
}
}
});
});
}
}
function addStrait(width) {
var top = Math.floor(Math.random() * mapWidth * 0.35 + mapWidth * 0.3);
var bottom = Math.floor((mapWidth - top) - (mapWidth * 0.1) + (Math.random() * mapWidth * 0.2));
var start = diagram.find(top, mapHeight * 0.2).index;
var end = diagram.find(bottom, mapHeight * 0.8).index;
var range = [];
for (var l = 0; start !== end && l < 1000; l++) {
var min = 10000; // dummy value
cells[start].neighbors.forEach(function(e) {
diff = Math.hypot(cells[end].data[0] - cells[e].data[0], cells[end].data[1] - cells[e].data[1]);
if (Math.random() > 0.8) {diff = diff / 2}
if (diff < min) {min = diff; start = e;}
});
range.push(start);
}
var query = [], used = [];
for (; width > 0; width--) {
range.map(function(r) {
cells[r].neighbors.forEach(function(e) {
if (used.indexOf(e) == -1) {
query.push(e), used.push(e);
var height = cells[e].height * 0.23;
cells[e].height = roundN(height, 2);
}
});
range = query.slice();
});
}
}
function addPit(count, height, cell) {
for (c = 0; c < count; c++) {
var change = height ? height + 0.2 : Math.random() * 0.3 + 0.2;
var start = cell, used = [];
if (!start) {
var lowlands = $.grep(cells, function(e) {return (e.height >= 0.2);});
if (lowlands.length == 0) {return;}
var rnd = Math.floor(Math.random() * lowlands.length);
start = lowlands[rnd].index;
}
var query = [start];
for (var i = 1; change >= 0.01; i++) {
var rnd = Math.random() * 0.4 + 0.8;
change -= i / 60 * rnd;
query.map(function(p) {
cells[p].neighbors.forEach(function(e) {
if (used.indexOf(e) == -1 && change > 0) {
query.push(e);
used.push(e);
cells[e].height -= change;
if (cells[e].height < 0.1) {
cells[e].height = 0.1 + i / 100;
if (cells[e].height >= 0.2) {cells[e].height = 0.19;}
}
}
});
});
}
}
}
// Modify heights multiplying/adding by value
function modifyHeights(type, add, mult) {
cells.map(function(i) {
if (type === "land") {
if (i.height >= 0.2) {
i.height += add;
var dif = i.height - 0.2;
var factor = mult;
if (mult == "^2") {factor = dif}
if (mult == "^3") {factor = dif * dif;}
i.height = 0.2 + dif * factor;
}
} else if (type === "all") {
if (i.height > 0) {
i.height += add;
i.height *= mult;
}
} else {
var interval = type.split("-");
if (i.height >= +interval[0] && i.height <= +interval[1]) {
i.height += add;
if ($.isNumeric(mult)) {i.height *= mult; return;}
if (mult.slice(0,1) === "^") {
pow = mult.slice(1);
i.height = Math.pow(i.height, pow);
}
}
}
});
}
// Smooth heights using mean of neighbors
function smoothHeights(fraction) {
var fraction = fraction || 2;
cells.map(function(i) {
var heights = [i.height];
i.neighbors.forEach(function(e) {heights.push(cells[e].height);});
i.height = (i.height * (fraction - 1) + d3.mean(heights)) / fraction;
});
}
// Randomize heights a bit
function disruptHeights() {
cells.map(function(i) {
if (i.height < 0.18) {return;}
if (Math.random() > 0.5) {return;}
var rnd = Math.round(2 - Math.random() * 4) / 100;
i.height = roundN(i.height + rnd, 2);
});
}
// Get polygone neighbors and update their height with small optional modifier
function neighbors(i, height) {
cells[i].neighbors.forEach(function(e) {
if (!cells[e].used) {
var mod = Math.random() * sharpness + 1.1 - sharpness;
if (sharpness == 0.1) {mod = 1;}
cells[e].height += height * mod;
cells[e].used = 1;
queue.push(e);
}
});
}
// Mark features (ocean, lakes, islands)
function markFeatures() {
console.time("markFeatures");
var queue = [], lake = 0, number = 0, type, greater = 0, less = 0;
// ensure all near border cells are ocean
cells.map(function(l) {
l.height = roundN(l.height, 2);
if (l.type === -99) {
l.height = 0;
l.neighbors.forEach(function(e) {cells[e].height = 0;});
}
});
// start with top left corner to define Ocean first
var start = diagram.find(0, 0).index;
var unmarked = [cells[start]];
while (unmarked.length > 0) {
if (unmarked[0].height >= 0.2) {
type = "Island";
number = island;
island += 1;
greater = 0.2;
less = 100; // just to omit exclusion
} else {
type = "Lake";
number = lake;
lake += 1;
greater = -100; // just to omit exclusion
less = 0.2;
}
if (type == "Lake" && number == 0) {type = "Ocean";}
start = unmarked[0].index;
queue.push(start);
cells[start].feature = type;
cells[start].featureNumber = number;
while (queue.length > 0) {
var i = queue[0];
queue.shift();
cells[i].neighbors.forEach(function(e) {
if (!cells[e].feature && cells[e].height >= greater && cells[e].height < less) {
cells[e].feature = type;
cells[e].featureNumber = number;
queue.push(e);
}
if (type == "Island" && cells[e].height < 0.2) {
cells[i].type = 2;
cells[e].type = -1;
if (cells[e].feature === "Ocean") {
if (cells[i].harbor) {
cells[i].harbor += 1;
} else {
cells[i].harbor = 1;
}
}
}
});
}
unmarked = $.grep(cells, function(e) {return (!e.feature);});
}
console.log(" islands: " + island);
console.timeEnd("markFeatures");
}
function drawOcean() {
console.time("drawOcean");
var limits = [], odd = 0.8; // initial odd for ocean layer is 80%
// Define type of ocean cells based on cell distance form land
var frontier = $.grep(cells, function(e) {return (e.type === -1);});
if (Math.random() < odd) {limits.push(-1); odd = 0.3;}
for (var c = -2; frontier.length > 0 && c > -10; c--) {
if (Math.random() < odd) {limits.unshift(c); odd = 0.3;} else {odd += 0.2;}
frontier.map(function(i) {
i.neighbors.forEach(function(e) {
if (!cells[e].type) {cells[e].type = c;}
});
});
frontier = $.grep(cells, function(e) {return (e.type === c);});
}
if (outlineLayers.value !== "random") {limits = outlineLayers.value.split(",");}
// Define area edges
for (var c = 0; c < limits.length; c++) {
var edges = [];
for (var i = 0; i < cells.length; i++) {
if (cells[i].feature === "Ocean" && cells[i].type >= limits[c]) {
var cell = diagram.cells[i];
cell.halfedges.forEach(function(e) {
var edge = diagram.edges[e];
if (edge.left && edge.right) {
var ea = edge.left.index;
if (ea === i) {ea = edge.right.index;}
var type = cells[ea].type;
if (type < limits[c] || type == undefined) {
var start = edge[0].join(" ");
var end = edge[1].join(" ");
edges.push({start, end});
}
} else {
var start = edge[0].join(" ");
var end = edge[1].join(" ");
edges.push({start, end});
}
})
}
}
lineGen.curve(d3.curveBasisClosed);
var relax = 0.8 - c / 10;
if (relax < 0.2) {relax = 0.2};
var line = getContinuousLine(edges, 0, relax);
oceanLayers.append("path").attr("d", line).attr("fill", "#ecf2f9").style("opacity", 0.4 / limits.length);
}
console.timeEnd("drawOcean");
}
// recalculate Voronoi Graph to pack cells
function reGraph() {
console.time("reGraph");
var tempCells = [], newPoints = []; // to store new data
land = [], polygons= []; // clear old data
// get average precipitation based on graph size
var avPrec = roundN(precipitation / Math.sqrt(cells.length), 2);
cells.map(function(i) {
var height = Math.trunc(i.height * 100) / 100;
var type = i.type || undefined;
if (type !== -1 && type !== -2 && height < 0.2) {return;}
var x = roundN(i.data[0], 1);
var y = roundN(i.data[1], 1);
var feature = i.feature;
var featureNumber = i.featureNumber;
var harbor = i.harbor || 0;
var copy = $.grep(newPoints, function(e) {return (e[0] == x && e[1] == y);});
if (!copy.length) {
newPoints.push([x, y]);
tempCells.push({index:tempCells.length, data:[x, y], height, type, feature, featureNumber, harbor});
}
if (type === 2 || type === -1) { // add additional points
i.neighbors.forEach(function(e) {
if (cells[e].type == type) {
var x1 = (x * 2 + cells[e].data[0]) / 3;
var y1 = (y * 2 + cells[e].data[1]) / 3;
x1 = roundN(x1, 1);
y1 = roundN(y1, 1);
copy = $.grep(newPoints, function(e) {return (e[0] == x1 && e[1] == y1);});
if (!copy.length) {
newPoints.push([x1, y1]);
tempCells.push({index:tempCells.length, data:[x1, y1], height, type, feature, featureNumber, harbor});
}
};
});
}
});
cells = tempCells; // use tempCells as the only cells array
calculateVoronoi(newPoints); // recalculate Voronoi diagram using new points
var gridPath = ""; // store grid as huge single path string
cells.map(function(i, d) {
gridPath += "M" + polygons[d].join("L") + "Z";
var neighbors = []; // re-detect neighbors
diagram.cells[d].halfedges.forEach(function(e) {
var edge = diagram.edges[e], ea;
if (edge.left && edge.right) {
ea = edge.left.index;
if (ea === d) {ea = edge.right.index;}
neighbors.push(ea);
if (cells[ea].height < 0.2) {
if (i.type !== 1) {
i.type = 1; // mark coastal cells
// move cell point closer to coast
var x = (i.data[0] + cells[ea].data[0]) / 2;
var y = (i.data[1] + cells[ea].data[1]) / 2;
if (cells[ea].feature === "Lake") {
i.data[0] = x + (i.data[0] - x) * 0.4;
i.data[1] = y + (i.data[1] - y) * 0.4;
} else {
i.haven = ea; // harbor haven (oposite ocean cell)
var coastX = x + (i.data[0] - x) * 0.12;
var coastY = y + (i.data[1] - y) * 0.12;
var pointX = x + (i.data[0] - x) * 0.4;
var pointY = y + (i.data[1] - y) * 0.4;
i.coastX = roundN(coastX, 2);
i.coastY = roundN(coastY, 2);
i.data[0] = roundN(pointX, 2);
i.data[1] = roundN(pointY, 2);
}
}
}
}
})
i.neighbors = neighbors;
if (i.haven === undefined) {i.harbor = undefined;}
i.flux = avPrec;
});
grid.append("path").attr("d", round(gridPath, 1));
land = $.grep(cells, function(e) {return (e.height >= 0.2);});
land.sort(function(a, b) {return b.height - a.height;});
console.timeEnd("reGraph");
}
// Draw temp Heightmap for Customization
function mockHeightmap(log) {
$("#landmass").empty();
var heights = [];
var landCells = 0;
cells.map(function(i) {
if (i.height > 1) {i.height = 1;}
if (i.height < 0) {i.height = 0;}
if (i.height >= 0.2) {
landCells++;
landmass.append("path")
.attr("d", "M" + polygons[i.index].join("L") + "Z")
.attr("fill", color(1 - i.height))
.attr("stroke", color(1 - i.height));
}
heights.push(i.height);
});
// update history
if (log !== "nolog") {
history = history.slice(0, historyStage);
history[historyStage] = heights;
historyStage += 1;
}
redo.disabled = true;
undo.disabled = true;
if (historyStage < history.length - 1) {redo.disabled = false;}
if (historyStage > 0) {undo.disabled = false;}
var elevationAverage = roundN(d3.mean(heights), 2);
var landRatio = Math.round(landCells / cells.length * 100);
landmassCounter.innerHTML = landCells + " (" + landRatio + "%); Average Elevation: " + elevationAverage;
if (landCells > 100) {
$("#getMap").attr("disabled", false).removeClass("buttonoff");
} else {
$("#getMap").attr("disabled", true).addClass("buttonoff");
}
// if perspective is displayed, update it
if ($("#perspectivePanel").is(":visible")) {drawPerspective();}
}
// restoreHistory
function restoreHistory(step) {
historyStage = step;
var heights = history[historyStage];
if (heights === undefined) {return;}
cells.map(function(i, d) {
i.height = heights[d];
});
mockHeightmap("nolog");
}
// Detect and draw the coasline
function drawCoastline() {
console.time('drawCoastline');
getCurveType();
var oceanCoastline = "", lakeCoastline = "";
$("#landmass").empty();
for (var isle = 0; isle < island; isle++) {
var coastal = $.grep(land, function(e) {return (e.type === 1 && e.featureNumber === isle);});
if (!coastal.length) {continue;}
var oceanEdges = [], lakeEdges = [];
for (var i = 0; i < coastal.length; i++) {
var id = coastal[i].index, cell = diagram.cells[id];
cell.halfedges.forEach(function(e) {
var edge = diagram.edges[e];
if (edge.left && edge.right) {
var ea = edge.left.index;
if (ea === id) {ea = edge.right.index;}
if (cells[ea].height < 0.2) {
var start = edge[0].join(" ");
var end = edge[1].join(" ");
if (cells[ea].feature === "Lake") {
lakeEdges.push({start, end});
} else {
oceanEdges.push({start, end});
}
}
}
})
}
oceanCoastline += getContinuousLine(oceanEdges, 1.5, 0);
if (lakeEdges.length > 0) {lakeCoastline += getContinuousLine(lakeEdges, 1.5, 0);}
}
d3.select("#shape").append("path").attr("d", oceanCoastline).attr("fill", "white"); // draw the clippath
landmass.append("path").attr("d", oceanCoastline); // draw the landmass
coastline.append("path").attr("d", oceanCoastline); // draw the coastline
lakes.append("path").attr("d", lakeCoastline); // draw the lakes
console.timeEnd('drawCoastline');
}
function getContinuousLine(edges, indention, relax) {
var edgesOr = edges.slice();
var line = "";
while (edges.length > 2) {
var edgesOrdered = []; // to store points in a correct order
var start = edges[0].start;
var end = edges[0].end;
edges.shift();
var spl = start.split(" ");
edgesOrdered.push({scX: +spl[0], scY: +spl[1]});
spl = end.split(" ");
edgesOrdered.push({scX: +spl[0], scY: +spl[1]});
var x0 = +spl[0], y0 = +spl[1];
for (var i = 0; end !== start && i < 100000; i++) {
var next = null, index = null;
for (var e = 0; e < edges.length; e++) {
var edge = edges[e];
if (edge.start == end || edge.end == end) {
next = edge;
if (next.start == end) {end = next.end;} else {end = next.start;}
index = e;
break;
}
}
if (!next) {
console.error("Next edge is not found");
return "";
}
spl = end.split(" ");
if (indention || relax) {
var dist = Math.hypot(+spl[0] - x0, +spl[1] - y0);
if (dist >= indention && Math.random() > relax) {
edgesOrdered.push({scX: +spl[0], scY: +spl[1]});
x0 = +spl[0], y0 = +spl[1];
}
} else {
edgesOrdered.push({scX: +spl[0], scY: +spl[1]});
}
edges.splice(index, 1);
if (i === 100000-1) {
console.error("Line not ended, limit reached");
break;
}
}
line += lineGen(edgesOrdered) + "Z";
}
return round(line, 1);
}
// Resolve Heightmap Depressions (for a correct water flux modeling)
function resolveDepressions() {
console.time('resolveDepressions');
var depression = 1, limit = 100, minCell, minHigh;
for (var l = 0; depression > 0 && l < limit; l++) {
depression = 0;
for (var i = 0; i < land.length; i++) {
var heights = [];
land[i].neighbors.forEach(function(e) {heights.push(+cells[e].height);});
var minHigh = d3.min(heights);
if (land[i].height <= minHigh) {
depression += 1;
land[i].height = minHigh + 0.01;
}
}
if (l === limit - 1) {console.error("Error: resolveDepressions iteration limit");}
}
console.timeEnd('resolveDepressions');
}
function flux() {
console.time('flux');
riversData = [];
var riversOrder = [], riverNext = 0;
land.sort(function(a, b) {return b.height - a.height;});
for (var i = 0; i < land.length; i++) {
var id = land[i].index;
var heights = [];
land[i].neighbors.forEach(function(e) {heights.push(cells[e].height);});
var minId = heights.indexOf(d3.min(heights));
var min = land[i].neighbors[minId];
// Define river number
if (land[i].flux > 0.85) {
if (land[i].river == undefined) {
// State new River
land[i].river = riverNext;
riversData.push({river: riverNext, cell: id, x: land[i].data[0], y: land[i].data[1]});
riverNext += 1;
}
// Assing existing River to the downhill cell
if (cells[min].river == undefined) {
cells[min].river = land[i].river;
} else {
var riverTo = cells[min].river;
var iRiver = $.grep(riversData, function(e) {return (e.river == land[i].river);});
var minRiver = $.grep(riversData, function(e) {return (e.river == riverTo);});
var iRiverL = iRiver.length;
var minRiverL = minRiver.length;
// re-assing river nunber if new part is greater
if (iRiverL >= minRiverL) {
cells[min].river = land[i].river;
iRiverL += 1;
minRiverL -= 1;
}
// mark confluences
if (cells[min].height >= 0.2 && iRiverL > 1 && minRiverL > 1) {
if (!cells[min].confluence) {
cells[min].confluence = minRiverL-1;
} else {
cells[min].confluence += minRiverL-1;
}
}
}
}
cells[min].flux += land[i].flux;
if (land[i].river != undefined) {
var px = cells[min].data[0];
var py = cells[min].data[1];
if (cells[min].height < 0.2) {
// pour water to the Ocean
var sx = land[i].data[0];
var sy = land[i].data[1];
var x = (px + sx) / 2 + (px - sx) / 20;
var y = (py + sy) / 2 + (py - sy) / 20;
riversData.push({river: land[i].river, cell: id, x, y});
}
else {
// add next River segment
riversData.push({river: land[i].river, cell: min, x: px, y: py});
}
}
}
console.timeEnd('flux');
drawRiverLines(riverNext);
}
function drawRiverLines(riverNext) {
console.time('drawRiverLines');
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
for (var i = 0; i < riverNext; i++) {
var dataRiver = $.grep(riversData, function(e) {return e.river === i;});
if (dataRiver.length > 1) {
var riverAmended = amendRiver(dataRiver, 1);
var width = roundN(0.8 + Math.random() * 0.4, 1);
var increment = roundN(0.8 + Math.random() * 0.4, 1);
var d = drawRiver(riverAmended, width, increment);
rivers.append("path").attr("d", d).attr("id", "river"+i)
.attr("data-points", round(JSON.stringify(riverAmended), 1))
.attr("data-width", width).attr("data-increment", increment);
}
}
rivers.selectAll("path").on("click", editRiver);
console.timeEnd('drawRiverLines');
}
// add more river points on 1/3 and 2/3 of length
function amendRiver(dataRiver, rndFactor) {
var riverAmended = [], side = 1;
for (var r = 0; r < dataRiver.length; r++) {
var dX = dataRiver[r].x;
var dY = dataRiver[r].y;
riverAmended.push({scX:dX, scY:dY});
if (r+1 < dataRiver.length) {
var eX = dataRiver[r+1].x;
var eY = dataRiver[r+1].y;
var angle = Math.atan2(eY - dY, eX - dX);
var serpentine = 1 / (r+1);
var meandr = serpentine + 0.3 + Math.random() * 0.3 * rndFactor;
if (Math.random() > 0.5) {side *= -1};
var dist = Math.hypot(eX - dX, eY - dY);
// if dist is big or river is small add 2 extra points
if (dist > 8 || (dist > 4 && dataRiver.length < 6)) {
var stX = (dX * 2 + eX) / 3;
var stY = (dY * 2 + eY) / 3;
var enX = (dX + eX * 2) / 3;
var enY = (dY + eY * 2) / 3;
stX += -Math.sin(angle) * meandr * side;
stY += Math.cos(angle) * meandr * side;
if (Math.random() > 0.8) {side *= -1};
enX += Math.sin(angle) * meandr * side;
enY += -Math.cos(angle) * meandr * side;
riverAmended.push({scX:stX, scY:stY}, {scX:enX, scY:enY});
// if dist is medium or river is small add 1 extra point
} else if (dist > 4 || dataRiver.length < 6) {
var scX = (dX + eX) / 2;
var scY = (dY + eY) / 2;
scX += -Math.sin(angle) * meandr * side;
scY += Math.cos(angle) * meandr * side;
riverAmended.push({scX, scY});
}
}
}
return riverAmended;
}
function drawRiver(riverPoints, width, increment) {
var extraOffset = 0.02; // start offset to make river source visible
var width = width || 1; // river width modifier
var increment = increment || 1; // river bed widening modifier
var d = lineGen(riverPoints);
var river = defs.append("path").attr("d", d);
var riverLength = river.node().getTotalLength();
var widening = (4000 + (riverLength * 20)) * increment; // partially depends on riverLength
var riverPointsLeft = [], riverPointsRight = [];
for (var l=0; l < riverLength; l++) {
var point = river.node().getPointAtLength(l);
var cell = diagram.find(point.x, point.y, 1);
if (cell) {
var confluence = cells[cell.index].confluence;
if (confluence) {extraOffset += Math.atan(confluence / widening);}
}
var from = river.node().getPointAtLength(l - 0.1);
var to = river.node().getPointAtLength(l + 0.1);
var angle = Math.atan2(from.y - to.y, from.x - to.x);
// riverLength *
var offset = (Math.atan(Math.pow(l, 2) / widening) / 2 * width) + extraOffset;
var xLeft = point.x + -Math.sin(angle) * offset;
var yLeft = point.y + Math.cos(angle) * offset;
riverPointsLeft.push({scX:xLeft, scY:yLeft});
var xRight = point.x + Math.sin(angle) * offset;
var yRight = point.y + -Math.cos(angle) * offset;
riverPointsRight.unshift({scX:xRight, scY:yRight});
}
var point = river.node().getPointAtLength(riverLength);
var from = river.node().getPointAtLength(riverLength - 0.1);
var angle = Math.atan2(from.y - point.y, from.x - point.x);
var offset = (Math.atan(Math.pow(riverLength, 2) / widening) / 2 * width) + extraOffset;
var xLeft = point.x + -Math.sin(angle) * offset;
var yLeft = point.y + Math.cos(angle) * offset;
riverPointsLeft.push({scX:xLeft, scY:yLeft});
var xRight = point.x + Math.sin(angle) * offset;
var yRight = point.y + -Math.cos(angle) * offset;
riverPointsRight.unshift({scX:xRight, scY:yRight});
river.remove();
var right = lineGen(riverPointsRight);
var left = lineGen(riverPointsLeft);
left = left.substring(left.indexOf("C"));
var d = round(right + left + "Z", 2);
return d;
}
function editRiver() {
if (elSelected) {
if ($("#riverNew").hasClass('pressed')) {
var point = d3.mouse(this);
addRiverPoint({scX:point[0], scY:point[1]});
redrawRiver();
$("#riverNew").click();
return;
}
elSelected.call(d3.drag().on("drag", null)).classed("draggable", false);
rivers.select(".riverPoints").remove();
}
elSelected = d3.select(this);
elSelected.call(d3.drag().on("drag", riverDrag)).classed("draggable", true);
var points = JSON.parse(elSelected.attr("data-points"));
rivers.append("g").attr("class", "riverPoints").attr("transform", elSelected.attr("transform"));
points.map(function(p) {addRiverPoint(p)});
var tr = parseTransform(elSelected.attr("transform"));
riverAngle.value = tr[2];
riverAngleValue.innerHTML = Math.abs(+tr[2]) + "°";
riverScale.value = tr[5];
riverWidthInput.value = +elSelected.attr("data-width");
riverIncrement.value = +elSelected.attr("data-increment");
$("#riverEditor").dialog({
title: "Edit River",
minHeight: 30, width: "auto", maxWidth: 275, resizable: false,
position: {my: "center top", at: "top", of: this}
}).on("dialogclose", function(event) {
if (elSelected) {
elSelected.call(d3.drag().on("drag", null)).classed("draggable", false);
rivers.select(".riverPoints").remove();
$(".pressed").removeClass('pressed');
viewbox.style("cursor", "default");
}
});
}
function addRiverPoint(point) {
rivers.select(".riverPoints").append("circle")
.attr("cx", point.scX).attr("cy", point.scY).attr("r", 0.25)
.call(d3.drag().on("drag", riverPointDrag))
.on("click", function(d) {
if ($("#riverRemovePoint").hasClass('pressed')) {
$(this).remove(); redrawRiver();
}
if ($("#riverNew").hasClass('pressed')) {
$("#riverNew").click();
}
});
}
$("#riverEditor .editButton, #riverEditor .editButtonS").click(function() {
if (this.id == "riverRemove") {
alertMessage.innerHTML = `Are you sure you want to remove the river?`;
$(function() {$("#alert").dialog({resizable: false, title: "Remove river",
buttons: {
"Remove": function() {
$(this).dialog("close");
elSelected.remove();
rivers.select(".riverPoints").remove();
$("#riverEditor").dialog("close");
},
Cancel: function() {$(this).dialog("close");}
}})
});
return;
}
if (this.id == "riverCopy") {
var tr = parseTransform(elSelected.attr("transform"));
var d = elSelected.attr("d");
var points = elSelected.attr("data-points");
var width = elSelected.attr("data-width");
var increment = elSelected.attr("data-increment");
var x = 2, y = 2;
transform = `translate(${tr[0]-x},${tr[1]-y}) rotate(${tr[2]} ${tr[3]} ${tr[4]}) scale(${tr[5]})`;
while (rivers.selectAll("[transform='" + transform + "'][d='" + d + "']").size() > 0) {
x += 2; y += 2;
transform = `translate(${tr[0]-x},${tr[1]-y}) rotate(${tr[2]} ${tr[3]} ${tr[4]}) scale(${tr[5]})`;
}
var river = +$("#rivers > path").last().attr("id").slice(5) + 1;
rivers.append("path").attr("d", d).attr("data-points", points).attr("transform", transform)
.attr("id", "river"+river).on("click", editRiver)
.attr("data-width", width).attr("data-increment", increment);
return;
}
if (this.id == "riverRegenerate") {
// restore main points
var points = JSON.parse(elSelected.attr("data-points"));
var riverCells = [], dataRiver = [];
for (var p = 0; p < points.length; p++) {
var cell = diagram.find(points[p].scX, points[p].scY, 1);
if (cell !== null && cell !== riverCells[riverCells.length-1]) {riverCells.push(cell);}
}
for (var c = 0; c < riverCells.length; c++) {
dataRiver.push({x:riverCells[c][0], y:riverCells[c][1]});
}
// if last point not in cell center push it with one extra point
var last = points.pop();
if (dataRiver[dataRiver.length-1].x !== last.scX) {
dataRiver.push({x:last.scX, y:last.scY});
}
var rndFactor = 0.2 + Math.random() * 1.6; // random factor in range 0.2-1.8
var riverAmended = amendRiver(dataRiver, rndFactor);
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
var width = +elSelected.attr("data-width");
var increment = +elSelected.attr("data-increment");
var d = drawRiver(riverAmended, width, increment);
elSelected.attr("d", d).attr("data-points", round(JSON.stringify(riverAmended), 1));
rivers.select(".riverPoints").selectAll("*").remove();
riverAmended.map(function(p) {addRiverPoint(p);});
return;
}
if (this.id == "riverResize") {$("#riverAngle, #riverAngleValue, #riverScaleIcon, #riverScale, #riverReset").toggle();}
if (this.id == "riverWidth") {$("#riverWidthInput, #riverIncrementIcon, #riverIncrement").toggle();}
if (this.id == "riverAddPoint" || this.id == "riverRemovePoint" || this.id == "riverNew") {
if ($(this).hasClass('pressed')) {
$(".pressed").removeClass('pressed');
if (elSelected.attr("data-river") == "new") {
rivers.select(".riverPoints").selectAll("*").remove();
elSelected.attr("data-river", "");
elSelected.call(d3.drag().on("drag", riverDrag)).classed("draggable", true);
}
viewbox.style("cursor", "default");
} else {
$(".pressed").removeClass('pressed');
$(this).addClass('pressed');
if (this.id == "riverAddPoint" || this.id == "riverNew") {viewbox.style("cursor", "crosshair");}
if (this.id == "riverNew") {rivers.select(".riverPoints").selectAll("*").remove();}
}
return;
}
if (this.id == "riverReset") {
elSelected.attr("transform", "");
rivers.select(".riverPoints").attr("transform", "");
riverAngle.value = 0;
riverAngleValue.innerHTML = "0°";
riverScale.value = 1;
return;
}
$("#riverEditor .editButton").toggle();
$(this).show().next().toggle();
});
// on riverAngle change
$("#riverAngle").change(function() {
var tr = parseTransform(elSelected.attr("transform"));
riverAngleValue.innerHTML = Math.abs(+this.value) + "°";
$(this).attr("title", $(this).val());
var c = elSelected.node().getBBox();
var angle = this.value;
var scale = +tr[5];
transform = `translate(${tr[0]},${tr[1]}) rotate(${angle} ${(c.x+c.width/2)*scale} ${(c.y+c.height/2)*scale}) scale(${scale})`;
elSelected.attr("transform", transform);
rivers.select(".riverPoints").attr("transform", transform);
});
// on riverScale change
$("#riverScale").change(function() {
var tr = parseTransform(elSelected.attr("transform"));
$(this).attr("title", $(this).val());
var scaleOld = +tr[5];
var scale = +this.value;
var c = elSelected.node().getBBox();
var cx = c.x+c.width/2;
var cy = c.y+c.height/2;
var trX = +tr[0] + cx * (scaleOld - scale);
var trY = +tr[1] + cy * (scaleOld - scale);
var scX = +tr[3] * scale/scaleOld;
var scY = +tr[4] * scale/scaleOld;
transform = `translate(${trX},${trY}) rotate(${tr[2]} ${scX} ${scY}) scale(${scale})`;
elSelected.attr("transform", transform);
rivers.select(".riverPoints").attr("transform", transform);
});
// change river width
$("#riverWidthInput, #riverIncrement").change(function() {
var points = JSON.parse(elSelected.attr("data-points"));
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
var width = +$("#riverWidthInput").val();
var increment = +$("#riverIncrement").val();
var d = drawRiver(points, width, increment);
elSelected.attr("d", d)
.attr("data-points", JSON.stringify(points))
.attr("data-width", width).attr("data-increment", increment);
});
function riverDrag() {
var x = d3.event.x, y = d3.event.y;
var el = d3.select(this);
var tr = parseTransform(el.attr("transform"));
d3.event.on("drag", function() {
xc = d3.event.x, yc = d3.event.y;
var transform = `translate(${(+tr[0]+xc-x)},${(+tr[1]+yc-y)}) rotate(${tr[2]} ${tr[3]} ${tr[4]}) scale(${tr[5]})`;
el.attr("transform", transform);
rivers.select(".riverPoints").attr("transform", transform);
});
}
function parseTransform(string) {
// [translateX,translateY,rotateDeg,rotateX,rotateY,scale]
if (!string) {return [0,0,0,0,0,1];}
var a = string.replace(/[a-z()]/g,"").replace(/[ ]/g,",").split(",");
return [a[0] || 0, a[1] || 0, a[2] || 0, a[3] || 0, a[4] || 0, a[5] || 1];
}
function riverPointDrag() {
var x = d3.event.x, y = d3.event.y;
var el = d3.select(this);
d3.event
.on("drag", function() {el.attr("cx", d3.event.x).attr("cy", d3.event.y);})
.on("end", function() {
if (Math.abs(d3.event.x - x) + Math.abs(d3.event.y - y) > 0) {redrawRiver();}
});
}
function redrawRiver() {
var points = [];
rivers.select(".riverPoints").selectAll("circle").each(function() {
var el = d3.select(this);
points.push({scX: +el.attr("cx"), scY: +el.attr("cy")});
});
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
var d = drawRiver(points);
elSelected.attr("d", d).attr("data-points", round(JSON.stringify(points), 1));
}
function manorsAndRegions() {
console.group('manorsAndRegions');
calculateChains();
rankPlacesGeography();
getCurveType();
locateCultures();
locateCapitals();
generateMainRoads();
rankPlacesEconomy();
locateTowns();
// temporary off as now there are too many islands and searoutes produce mess
//checkAccessibility();
drawManors();
defineRegions();
drawRegions();
generatePortRoads();
generateSmallRoads();
generateOceanRoutes();
console.groupEnd('manorsAndRegions');
}
// Assess cells geographycal suitability for settlement
function rankPlacesGeography() {
console.time('rankPlacesGeography');
land.map(function(c) {
var score = (1 - c.height) * 5; // base score from height (will be biom)
if (c.type && Math.random() < 0.8 && !c.river) {
c.score = 0; // ignore 80% of extended cells
} else {
if (c.harbor) {
score += 2 - c.harbor; // good sea harbor is valued
if (c.river && c.type === 1) {score += 3;} // estuaries are valued
}
if (c.flux > 1) {score += c.flux / 2;} // riverbank is valued
if (c.confluence) {score += c.confluence;} // confluence is valued;
}
c.score = Math.round(score);
});
land.sort(compareScore);
console.timeEnd('rankPlacesGeography');
}
// Assess the cells economical suitability for settlement
function rankPlacesEconomy() {
console.time('rankPlacesEconomy');
land.map(function(c) {
var score = c.score;
if (c.path) {
var path = Math.ceil(c.path / 15);
if (path > 5) {path = 5;}
if (c.crossroad) {path *= 2;} // crossroads are valued
score += path; // roads are valued
}
c.score = Math.round(Math.random() * score + score); // 0.5 random factor
});
land.sort(compareScore);
console.timeEnd('rankPlacesEconomy');
}
function compareScore(a, b) {
if (a.score < b.score) return 1;
if (a.score > b.score) return -1;
return 0;
}
// Locate cultures
function locateCultures() {
var cultureCenters = d3.range(7).map(function(d) {return [Math.random() * mapWidth, Math.random() * mapHeight];});
cultureTree = d3.quadtree().extent([[0, 0], [mapHeight, mapWidth]]).addAll(cultureCenters);;
}
function locateCapitals() {
console.time('locateCapitals');
var spacing = mapWidth / capitalsCount;
manorTree = d3.quadtree().extent([[0, 0], [mapHeight, mapWidth]]);
if (power > 0) {spacing / power;}
console.log(" capitals: " + capitalsCount);
for (var l = 0; l < land.length && manors.length < capitalsCount; l++) {
var m = manors.length;
var dist = 10000;
if (l > 0) {
var closest = manorTree.find(land[l].data[0], land[l].data[1]);
dist = Math.hypot(land[l].data[0] - closest[0], land[l].data[1] - closest[1]);
}
if (dist >= spacing) {
var cell = land[l].index;
shiftSettlement(land[l], "capital");
queue.push(cell);
queue.push(...land[l].neighbors);
var closest = cultureTree.find(land[l].data[0], land[l].data[1]);
var culture = cultureTree.data().indexOf(closest);
var name = generateName(culture);
var capitalPower = roundN(Math.random() * power / 2 + 1, 1);
manors.push({i: m, cell, x: land[l].data[0], y: land[l].data[1], region: m, power: capitalPower, score: land[l].score, culture, name});
manorTree.add([land[l].data[0], land[l].data[1]]);
}
if (l === land.length - 1) {
console.error("Cannot place capitals with current spacing. Trying again with reduced spacing");
l = -1;
manors = [];
queue = [];
manorTree = d3.quadtree().extent([[0, 0], [mapHeight, mapWidth]]);
spacing /= 1.2;
}
}
manors.map(function(e, i) {
var p = cells[e.cell];
p.manor = i;
p.region = i;
p.culture = e.culture;
});
console.timeEnd('locateCapitals');
}
function locateTowns() {
console.time('locateTowns');
for (var l = 0; l < land.length && manors.length < manorsCount; l++) {
if (queue.indexOf(land[l].index) == -1) {
queue.push(land[l].index);
if (land[l].type || Math.random() > 0.6) {queue.push(...land[l].neighbors);}
shiftSettlement(land[l], "town");
var x = land[l].data[0];
var y = land[l].data[1];
var cell = land[l].index;
var region = "neutral", culture = -1, closest = neutral;
for (c = 0; c < capitalsCount; c++) {
var dist = Math.hypot(manors[c].x - x, manors[c].y - y) / manors[c].power;
var cap = manors[c].cell;
if (cells[cell].featureNumber !== cells[cap].featureNumber) {dist *= 3;}
if (dist < closest) {region = c; closest = dist;}
}
if (closest > neutral / 5 || region === "neutral") {
var closestCulture = cultureTree.find(x, y);
culture = cultureTree.data().indexOf(closestCulture);
} else {
culture = manors[region].culture;
}
var name = generateName(culture);
land[l].manor = manors.length;
land[l].culture = culture;
land[l].region = region;
manors.push({i: manors.length, cell, x, y, region, score: land[l].score, culture, name});
}
if (l === land.length - 1) {
console.error("Cannot place all towns. Towns requested: " + manorsCount + ". Towns placed: " + manors.length);
}
}
console.timeEnd('locateTowns');
}
function shiftSettlement(cell, type) {
if ((type === "capital" && cell.harbor > 0) || (type === "town" && cell.harbor === 1)) {
cell.port = true;
cell.data[0] = cell.coastX;
cell.data[1] = cell.coastY;
}
if (cell.river) {
var shift = 0.2 * cell.flux;
if (shift < 0.2) {shift = 0.2;}
if (shift > 1) {shift = 1;}
shift = Math.random() > .5 ? shift : shift * -1;
cell.data[0] += shift;
shift = Math.random() > .5 ? shift : shift * -1;
cell.data[1] += shift;
cell.data[0] = roundN(cell.data[0], 2);
cell.data[1] = roundN(cell.data[1], 2);
}
}
// Validate each island with manors has at least one port (so Island is accessible)
function checkAccessibility() {
console.time("checkAccessibility");
for (var i = 0; i < island; i++) {
var manorsOnIsland = $.grep(land, function(e) {return (typeof e.manor !== "undefined" && e.featureNumber === i);});
if (manorsOnIsland.length > 0) {
var ports = $.grep(manorsOnIsland, function(p) {return (p.port);});
if (ports.length === 0) {
var portCandidates = $.grep(manorsOnIsland, function(c) {return (c.harbor && c.type === 1);});
if (portCandidates.length > 0) {
console.error("No ports on Island " + manorsOnIsland[0].featureNumber + ". Upgrading first manor to port");
portCandidates[0].harbor = 1;
portCandidates[0].port = true;
portCandidates[0].data[0] = portCandidates[0].coastX;
portCandidates[0].data[1] = portCandidates[0].coastY;
manors[portCandidates[0].manor].x = portCandidates[0].coastX;
manors[portCandidates[0].manor].y = portCandidates[0].coastY;
} else {
console.error("Cannot generate ports on Island " + manorsOnIsland[0].featureNumber + ". Removing " + manorsOnIsland.length + " manors");
manorsOnIsland.map(function(e) {
manors.splice(e.manor, 1);
e.manor = undefined;
});
}
}
}
}
console.timeEnd("checkAccessibility");
}
function generateMainRoads() {
console.time("generateMainRoads");
for (var i = 0; i < island; i++) {
var manorsOnIsland = $.grep(land, function(e) {return (typeof e.manor !== "undefined" && e.featureNumber === i);});
if (manorsOnIsland.length > 1) {
for (var d = 1; d < manorsOnIsland.length; d++) {
for (var m = 0; m < d; m++) {
var path = findLandPath(manorsOnIsland[d].index, manorsOnIsland[m].index, "main");
restorePath(manorsOnIsland[m].index, manorsOnIsland[d].index, "main", path);
}
}
}
}
console.timeEnd("generateMainRoads");
}
function generatePortRoads() {
console.time("generatePortRoads");
var landCapitals = $.grep(land, function(e) {return (e.manor < capitalsCount && !e.port);});
landCapitals.map(function(e) {
var ports = $.grep(land, function(l) {return (l.port && l.region === e.manor);});
var minDist = 1000, end = -1;
ports.map(function(p) {
var dist = Math.hypot(e.data[0] - p.data[0], e.data[1] - p.data[1]);
if (dist < minDist) {minDist = dist; end = p.index;}
});
if (end !== -1) {
var start = e.index;
var path = findLandPath(start, end, "direct");
restorePath(end, start, "main", path);
}
});
console.timeEnd("generatePortRoads");
}
function generateSmallRoads() {
console.time("generateSmallRoads");
lineGen.curve(d3.curveBasis);
for (var i = 0; i < island; i++) {
var manorsOnIsland = $.grep(land, function(e) {return (typeof e.manor !== "undefined" && e.featureNumber === i);});
var l = manorsOnIsland.length;
if (l > 1) {
var secondary = Math.round((l + 8) / 10);
for (s = 0; s < secondary; s++) {
var start = manorsOnIsland[Math.floor(Math.random() * l)].index;
var end = manorsOnIsland[Math.floor(Math.random() * l)].index;
var dist = Math.hypot(cells[start].data[0] - cells[end].data[0], cells[start].data[1] - cells[end].data[1]);
if (dist > 10) {
var path = findLandPath(start, end, "direct");
restorePath(end, start, "small", path);
}
}
manorsOnIsland.map(function(e, d) {
if (!e.path && d > 0) {
var start = e.index, end = -1;
var road = $.grep(land, function(e) {return (e.path && e.featureNumber === i);});
if (road.length > 0) {
var minDist = 10000;
road.map(function(i) {
var dist = Math.hypot(e.data[0] - i.data[0], e.data[1] - i.data[1]);
if (dist < minDist) {minDist = dist; end = i.index;}
});
} else {
end = manorsOnIsland[0].index;
}
var path = findLandPath(start, end, "main");
restorePath(end, start, "small", path);
}
});
}
}
console.timeEnd("generateSmallRoads");
}
function generateOceanRoutes() {
console.time("generateOceanRoutes");
lineGen.curve(d3.curveBasis);
var ports = [];
for (var i = 0; i < island; i++) {
ports[i] = $.grep(land, function(e) {return (e.featureNumber === i && e.port);});
if (!ports[i]) {ports[i] = [];}
}
ports.sort(function(a, b) {return a.length < b.length;})
for (var i = 0; i < island; i++) {
if (ports[i].length === 0) {break;}
var length = ports[i].length;
ports[i].sort(function(a, b) {return a.score < b.score;})
var start = ports[i][0].index;
var paths = findOceanPaths(start, -1);
/* draw anchor icons
for (var p = 0; p < ports[i].length; p++) {
var x0 = ports[i][p].data[0];
var y0 = ports[i][p].data[1];
var x1 = cells[h.haven].data[0];
var y1 = cells[h.haven].data[1];
var x = x0 + (x1 - x0) * 0.8;
var y = y0 + (y1 - y0) * 0.8;
icons.append("use").attr("xlink:href", "#icon-anchor").attr("x", x).attr("y", y).attr("width", 1).attr("height", 1);
} */
for (var h = 1; h < length; h++) {
var end = ports[i][h].index;
restorePath(end, start, "ocean", paths);
}
for (var c = i + 1; c < island; c++) {
if (ports[c].length > 3 && length > 3) {
var end = ports[c][0].index;
restorePath(end, start, "ocean", paths);
}
}
if (length > 5) {
ports[i].sort(function(a, b) {return b.cost - a.cost;});
for (var a = 2; a < length && a < 10; a++) {
var dist = Math.hypot(ports[i][1].data[0] - ports[i][a].data[0], ports[i][1].data[1] - ports[i][a].data[1]);
var distPath = getPathDist(ports[i][1].index, ports[i][a].index);
if (distPath > dist * 4 + 10) {
var totalCost = ports[i][1].cost + ports[i][a].cost;
var paths = findOceanPaths(ports[i][1].index, ports[i][a].index);
if (ports[i][a].cost < totalCost) {
restorePath(ports[i][a].index, ports[i][1].index, "ocean", paths);
break;
}
}
}
}
}
console.timeEnd("generateOceanRoutes");
}
function findLandPath(start, end, type) {
// A* algorithm
var queue = new PriorityQueue({comparator: function(a, b) {return a.p - b.p}});
var cameFrom = [];
var costTotal = [];
costTotal[start] = 0;
queue.queue({e: start, p: 0});
while (queue.length > 0) {
var next = queue.dequeue().e;
if (next === end) {break;}
var pol = cells[next];
pol.neighbors.forEach(function(e) {
if (cells[e].height >= 0.2) {
var cost = cells[e].height * 2;
if (cells[e].path && type === "main") {
cost = 0.15;
} else {
if (typeof e.manor === "undefined") {cost += 0.1;}
if (typeof e.river !== "undefined") {cost -= 0.1;}
if (cells[e].harbor) {cost *= 0.3;}
if (cells[e].path) {cost *= 0.5;}
cost += Math.hypot(cells[e].data[0] - pol.data[0], cells[e].data[1] - pol.data[1]) / 30;
}
var costNew = costTotal[next] + cost;
if (!cameFrom[e] || costNew < costTotal[e]) { //
costTotal[e] = costNew;
cameFrom[e] = next;
var dist = Math.hypot(cells[e].data[0] - cells[end].data[0], cells[e].data[1] - cells[end].data[1]) / 15;
var priority = costNew + dist;
queue.queue({e, p: priority});
}
}
});
}
return cameFrom;
}
function findLandPaths(start, type) {
// Dijkstra algorithm
var queue = new PriorityQueue({comparator: function(a, b) {return a.p - b.p}});
var cameFrom = [];
var costTotal = [];
cameFrom[start] = "no";
costTotal[start] = 0;
queue.queue({e: start, p: 0});
while (queue.length > 0) {
var next = queue.dequeue().e;
var pol = cells[next];
pol.neighbors.forEach(function(e) {
var cost = cells[e].height;
if (cost >= 0.2) {
cost *= 2;
if (typeof e.river !== "undefined") {cost -= 0.2;}
if (pol.region !== cells[e].region) {cost += 1;}
if (cells[e].region === "neutral") {cost += 1;}
if (typeof e.manor !== "undefined") {cost = 0.1;}
var costNew = costTotal[next] + cost;
if (!cameFrom[e]) {
costTotal[e] = costNew;
cameFrom[e] = next;
queue.queue({e, p: costNew});
}
}
});
}
return cameFrom;
}
function findOceanPaths(start, end) {
var queue = new PriorityQueue({comparator: function(a, b) {return a.p - b.p}});
var next;
var cameFrom = [];
var costTotal = [];
cameFrom[start] = "no";
costTotal[start] = 0;
queue.queue({e: start, p: 0});
while (queue.length > 0 && next !== end) {
next = queue.dequeue().e;
var pol = cells[next];
pol.neighbors.forEach(function(e) {
if (cells[e].type < 0 || cells[e].haven === next) {
var cost = 1;
if (cells[e].type > 0) {cost += 100;}
if (cells[e].type < -1) {
var dist = Math.hypot(cells[e].data[0] - pol.data[0], cells[e].data[1] - pol.data[1]);
cost += 50 + dist * 2;
}
if (cells[e].path && cells[e].type < 0) {cost *= 0.8;}
var costNew = costTotal[next] + cost;
if (!cameFrom[e]) {
costTotal[e] = costNew;
cells[e].cost = costNew;
cameFrom[e] = next;
queue.queue({e, p: costNew});
}
}
});
}
return cameFrom;
}
function getPathDist(start, end) {
var queue = new PriorityQueue({comparator: function(a, b) {return a.p - b.p}});
var next, costNew;
var cameFrom = [];
var costTotal = [];
cameFrom[start] = "no";
costTotal[start] = 0;
queue.queue({e: start, p: 0});
while (queue.length > 0 && next !== end) {
next = queue.dequeue().e;
var pol = cells[next];
pol.neighbors.forEach(function(e) {
if (cells[e].path && (cells[e].type === -1 || cells[e].haven === next)) {
var dist = Math.hypot(cells[e].data[0] - pol.data[0], cells[e].data[1] - pol.data[1]);
costNew = costTotal[next] + dist;
if (!cameFrom[e]) {
costTotal[e] = costNew;
cameFrom[e] = next;
queue.queue({e, p: costNew});
}
}
});
}
return costNew;
}
function restorePath(end, start, type, from) {
var path = [], current = end, limit = 300;
var prev = cells[end];
if (type === "ocean" || !prev.path) {path.push({scX: prev.data[0], scY: prev.data[1]});}
if (!prev.path) {prev.path = 1;}
for (var i = 0; i < limit; i++) {
current = from[current];
var cur = cells[current];
if (!cur) {break;}
if (cur.path) {
cur.path += 1;
path.push({scX: cur.data[0], scY: cur.data[1]});
prev = cur;
drawPath();
} else {
cur.path = 1;
if (prev) {path.push({scX: prev.data[0], scY: prev.data[1]});}
prev = undefined;
path.push({scX: cur.data[0], scY: cur.data[1]});
}
if (current === start || !from[current]) {break;}
}
drawPath();
function drawPath() {
if (path.length > 1) {
var line = lineGen(path);
line = round(line, 1);
if (type === "main") {
roads.append("path").attr("d", line).attr("data-start", start).attr("data-end", end);
} else if (type === "small") {
trails.append("path").attr("d", line);
} else if (type === "ocean") {
searoutes.append("path").attr("d", line);
}
}
path = [];
}
}
// Append manors with random / generated names
// For each non-capital manor detect the closes capital (used for areas)
function drawManors() {
console.time('drawManors');
for (var i = 0; i < manors.length; i++) {
var x = manors[i].x;
var y = manors[i].y;
var cell = manors[i].cell;
var name = manors[i].name;
if (i < capitalsCount) {
burgs.append("circle").attr("r", 1).attr("stroke-width", .24).attr("class", "manor").attr("cx", x).attr("cy", y);
capitals.append("text").attr("x", x).attr("y", y).attr("dy", -1.3).text(name);
} else {
burgs.append("circle").attr("r", .5).attr("stroke-width", .12).attr("class", "manor").attr("cx", x).attr("cy", y);
towns.append("text").attr("x", x).attr("y", y).attr("dy", -.7).text(name);
}
}
labels.selectAll("text").on("click", editLabel);
burgs.selectAll("circle").call(d3.drag().on("drag", dragged).on("end", dragended)).on("click", changeBurg);
console.timeEnd('drawManors');
}
// calculate Markov's chain from real data
function calculateChains() {
var vowels = "aeiouy";
//var digraphs = ["ai","ay","ea","ee","ei","ey","ie","oa","oo","ow","ue","ch","ng","ph","sh","th","wh"];
for (var l = 0; l < cultures.length; l++) {
var probs = []; // Coleshill -> co les hil l-com
var inline = manorNames[l].join(" ").toLowerCase();
var syl = "";
for (var i = -1; i < inline.length - 2;) {
if (i < 0) {var f = " ";} else {var f = inline[i];}
var str = "", vowel = 0;
for (var c = i+1; str.length < 5; c++) {
if (inline[c] === undefined) {break;}
str += inline[c];
if (str === " ") {break;}
if (inline[c] !== "o" && inline[c] !== "e" && vowels.includes(inline[c]) && inline[c+1] === inline[c]) {break;}
if (inline[c+2] === " ") {str += inline[c+1]; break;}
if (vowels.includes(inline[c])) {vowel++;}
if (vowel && vowels.includes(inline[c+2])) {break;}
}
i += str.length;
probs[f] = probs[f] || [];
probs[f].push(str);
}
chain[l] = probs;
}
}
// generate random name using Markov's chain
function generateName(culture) {
var data = chain[culture], res = "", next = data[" "];
var cur = next[Math.floor(Math.random() * next.length)];
while (res.length < 7) {
var l = cur.charAt(cur.length - 1);
if (cur !== " ") {
res += cur;
next = data[l];
cur = next[Math.floor(Math.random() * next.length)];
} else if (res.length > 2 + Math.floor(Math.random() * 5)) {
break;
} else {
next = data[" "];
cur = next[Math.floor(Math.random() * next.length)];
}
}
var name = res.charAt(0).toUpperCase() + res.slice(1);
return name;
}
// Define areas based on the closest manor to a polygon
function defineRegions() {
console.time('defineRegions');
manorTree = d3.quadtree().extent([[0, 0], [mapHeight, mapWidth]]);
manors.map(function(m) {manorTree.add([m.x, m.y]);});
land.map(function(i) {
if (i.region === undefined) {
var closest = manorTree.find(i.data[0], i.data[1]);
var dist = Math.hypot(closest[0] - i.data[0], closest[1] - i.data[1]);
if (dist > neutral) {
i.region = "neutral";
var closestCulture = cultureTree.find(i.data[0], i.data[1]);
i.culture = cultureTree.data().indexOf(closestCulture);
} else {
var manor = $.grep(manors, function(e) {return (e.x === closest[0] && e.y === closest[1]);});
var cell = manor[0].cell;
if (cells[cell].featureNumber !== i.featureNumber) {
var minDist = dist * 3;
land.map(function(l) {
if (l.featureNumber === i.featureNumber && l.manor !== undefined) {
var distN = Math.hypot(l.data[0] - i.data[0], l.data[1] - i.data[1]);
if (distN < minDist) {minDist = distN; cell = l.index;}
}
});
}
i.region = cells[cell].region;
i.culture = cells[cell].culture;
}
}
});
console.timeEnd('defineRegions');
}
// Define areas cells
function drawRegions() {
console.time('drawRegions');
var edges = [], borderEdges = [], coastalEdges = [], neutralEdges = []; // arrays to store edges
for (var i = 0; i < capitalsCount; i++) {
edges[i] = [];
land.map(function(p) {
if (p.region === i) {
var cell = diagram.cells[p.index];
cell.halfedges.forEach(function(e) {
var edge = diagram.edges[e];
if (edge.left && edge.right) {
var ea = edge.left.index;
if (ea === p.index) {ea = edge.right.index;}
var opp = cells[ea];
if (opp.region !== i) {
var start = edge[0].join(" ");
var end = edge[1].join(" ");
edges[i].push({start, end});
if (opp.height >= 0.2 && opp.region > i) {borderEdges.push({start, end});}
if (opp.height >= 0.2 && opp.region === "neutral") {neutralEdges.push({start, end});}
if (opp.height < 0.2) {coastalEdges.push({start, end});}
}
}
})
}
});
drawRegion(edges[i], i);
drawRegionCoast(coastalEdges, i);
}
drawBorders(borderEdges, "state");
drawBorders(neutralEdges, "neutral");
console.timeEnd('drawRegions');
}
function drawRegion(edges, region) {
var path = "", array = [];
lineGen.curve(d3.curveLinear);
while (edges.length > 2) {
var edgesOrdered = []; // to store points in a correct order
var start = edges[0].start;
var end = edges[0].end;
edges.shift();
var spl = start.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
for (var i = 0; end !== start && i < 2000; i++) {
var next = $.grep(edges, function(e) {return (e.start == end || e.end == end);});
if (next.length > 0) {
if (next[0].start == end) {
end = next[0].end;
} else if (next[0].end == end) {
end = next[0].start;
}
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
}
var rem = edges.indexOf(next[0]);
edges.splice(rem, 1);
}
path += lineGen(edgesOrdered) + "Z ";
var edgesFormatted = [];
edgesOrdered.map(function(e) {edgesFormatted.push([+e.scX, +e.scY])});
array[array.length] = edgesFormatted;
}
if (capitalsCount <= 8) {
var scheme = colors8;
} else {
var scheme = colors20;
}
var color = scheme(region / capitalsCount);
regions.append("path").attr("d", round(path, 1)).attr("fill", color).attr("stroke", "none").attr("class", "region"+region);
array.sort(function(a, b){return b.length - a.length;});
generateRegionName(array, region);
}
function drawRegionCoast(edges, region) {
var path = "";
while (edges.length > 0) {
var edgesOrdered = []; // to store points in a correct order
var start = edges[0].start;
var end = edges[0].end;
edges.shift();
var spl = start.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
var next = $.grep(edges, function(e) {return (e.start == end || e.end == end);});
while (next.length > 0) {
if (next[0].start == end) {
end = next[0].end;
} else if (next[0].end == end) {
end = next[0].start;
}
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
var rem = edges.indexOf(next[0]);
edges.splice(rem, 1);
next = $.grep(edges, function(e) {return (e.start == end || e.end == end);});
}
path += lineGen(edgesOrdered);
}
if (capitalsCount <= 8) {
var scheme = colors8;
} else {
var scheme = colors20;
}
var color = scheme(region / capitalsCount);
regions.append("path").attr("d", round(path, 1)).attr("fill", "none").attr("stroke", color).attr("stroke-width", 1.5).attr("class", "region"+region);
}
function drawBorders(edges, type) {
var path = "";
while (edges.length > 0) {
var edgesOrdered = []; // to store points in a correct order
var start = edges[0].start;
var end = edges[0].end;
edges.shift();
var spl = start.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
var next = $.grep(edges, function(e) {return (e.start == end || e.end == end);});
while (next.length > 0) {
if (next[0].start == end) {
end = next[0].end;
} else if (next[0].end == end) {
end = next[0].start;
}
spl = end.split(" ");
edgesOrdered.push({scX: spl[0], scY: spl[1]});
var rem = edges.indexOf(next[0]);
edges.splice(rem, 1);
next = $.grep(edges, function(e) {return (e.start == end || e.end == end);});
}
path += lineGen(edgesOrdered);
}
if (type === "state") {stateBorders.append("path").attr("d", round(path, 1));}
if (type === "neutral") {neutralBorders.append("path").attr("d", round(path, 1));}
}
// generate region name and place label at pole of inaccessibility of the largest continuous element of the region
function generateRegionName(array, region) {
var name;
var culture = manors[region].culture;
var c = polylabel(array, 1.0); // pole of inaccessibility
// get source name (capital name = 20%; random name = 80%)
if (Math.random() < 0.8) {
name = generateName(culture);
} else {
name = manors[region].name;
}
name = addRegionSuffix(name, culture);
countries.append("text").attr("x", c[0].toFixed(2)).attr("y", c[1].toFixed(2)).text(name).on("click", editLabel);
}
function addRegionSuffix(name, culture) {
var suffix = "ia"; // common latin suffix
var vowels = "aeiouy";
if (Math.random() < 0.05 && (culture == 3 || culture == 4)) {suffix = "terra";} // 5% "terra" for Italian and Spanish
if (Math.random() < 0.05 && culture == 2) {suffix = "terre";} // 5% "terre" for French
if (Math.random() < 0.5 && culture == 0) {suffix = "land";} // 50% "land" for German
if (Math.random() < 0.33 && (culture == 1 || culture == 6)) {suffix = "land";} // 33% "land" for English and Scandinavian
if (culture == 5 && name.slice(-2) === "sk") {name.slice(0,-2);} // exclude -sk suffix for Slavic
if (name.indexOf(suffix) !== -1) {suffix = "";} // null suffix if name already contains it
var ending = name.slice(-1);
if (vowels.includes(ending) && name.length > 3) {
if (Math.random() > 0.2) {
ending = name.slice(-2,-1);
if (vowels.includes(ending)) {
name = name.slice(0,-2) + suffix; // 80% for vv
} else if (Math.random() > 0.2) {
name = name.slice(0,-1) + suffix; // 64% for cv
}
}
} else if (Math.random() > 0.5) {
name += suffix // 50% for cc
}
//if (name.slice(-2) !== "ia" && culture == 5 && Math.random() > 0.5) {name += "skaya Zemya";} // special case for Slavic
if (name.slice(-4) === "berg") {name += suffix;} // special case for -berg
return name;
}
// draw the Heightmap
function toggleHeight() {
var scheme = styleSchemeInput.value;
var hColor = color;
if (scheme === "light") {hColor = d3.scaleSequential(d3.interpolateRdYlGn);}
if (scheme === "green") {hColor = d3.scaleSequential(d3.interpolateGreens);}
if (scheme === "monochrome") {hColor = d3.scaleSequential(d3.interpolateGreys);}
if (terrs.selectAll("path").size() == 0) {
land.map(function(i) {
terrs.append("path")
.attr("d", "M" + polygons[i.index].join("L") + "Z")
.attr("fill", hColor(1 - i.height))
.attr("stroke", hColor(1 - i.height));
});
} else {
terrs.selectAll("path").remove();
}
}
// draw Cultures
function toggleCultures() {
if (cults.selectAll("path").size() == 0) {
land.map(function(i) {
cults.append("path")
.attr("d", "M" + polygons[i.index].join("L") + "Z")
.attr("fill", colors8(i.culture / cultures.length))
.attr("stroke", colors8(i.culture / cultures.length));
});
} else {
cults.selectAll("path").remove();
}
}
// Draw the water flux system (for dubugging)
function toggleFlux() {
var colorFlux = d3.scaleSequential(d3.interpolateBlues);
if (terrs.selectAll("path").size() == 0) {
land.map(function(i) {
terrs.append("path")
.attr("d", "M" + polygons[i.index].join("L") + "Z")
.attr("fill", colorFlux(0.1 + i.flux))
.attr("stroke", colorFlux(0.1 + i.flux));
});
} else {
terrs.selectAll("path").remove();
}
}
// Draw the Relief (need to create more beautiness)
function drawRelief() {
console.time('drawRelief');
var ea, edge, id, cell, x, y, height, path, dash = "", rnd;
var hill = [], hShade = [], swamp = "", swampCount = 0, forest = "", fShade = "", fLight = "", swamp = "";
hill[0] = "", hill[1] = "", hShade[0] = "", hShade[1] = "";
var strokes = terrain.append("g").attr("id", "strokes"),
hills = terrain.append("g").attr("id", "hills"),
mounts = terrain.append("g").attr("id", "mounts"),
swamps = terrain.append("g").attr("id", "swamps"),
forests = terrain.append("g").attr("id", "forests");
// sort the land to Draw the top element first (reduce the elements overlapping)
land.sort(compareY);
for (i = 0; i < land.length; i++) {
x = land[i].data[0];
y = land[i].data[1];
height = land[i].height;
if (height >= 0.7 && !land[i].river) {
h = (height - 0.55) * 12;
if (height < 0.8) {
count = 2;
} else {
count = 1;
}
rnd = Math.random() * 0.8 + 0.2;
for (c = 0; c < count; c++) {
cx = x - h * 0.9 - c;
cy = y + h / 4 + c / 2;
path = "M " + cx + "," + cy + " L " + (cx + h / 3 + rnd) + "," + (cy - h / 4 - rnd * 1.2) + " L " + (cx + h / 1.1) + "," + (cy - h) + " L " + (cx + h + rnd) + "," + (cy - h / 1.2 + rnd) + " L " + (cx + h * 2) + "," + cy;
mounts.append("path").attr("d", path).attr("stroke", "#5c5c70");
path = "M " + cx + "," + cy + " L " + (cx + h / 3 + rnd) + "," + (cy - h / 4 - rnd * 1.2) + " L " + (cx + h / 1.1) + "," + (cy - h) + " L " + (cx + h / 1.5) + "," + cy;
mounts.append("path").attr("d", path).attr("fill", "#999999");
dash += "M" + (cx - 0.1) + "," + (cy + 0.3) + " L" + (cx + 2 * h + 0.1) + "," + (cy + 0.3);
}
dash += "M" + (cx + 0.4) + "," + (cy + 0.6) + " L" + (cx + 2 * h - 0.3) + "," + (cy + 0.6);
} else if (height > 0.5 && !land[i].river) {
h = (height - 0.4) * 10;
count = Math.floor(4 - h);
if (h > 1.8) {
h = 1.8
}
for (c = 0; c < count; c++) {
cx = x - h - c;
cy = y + h / 4 + c / 2;
hill[c] += "M" + cx + "," + cy + " Q" + (cx + h) + "," + (cy - h) + " " + (cx + 2 * h) + "," + cy;
hShade[c] += "M" + (cx + 0.6 * h) + "," + (cy + 0.1) + " Q" + (cx + h * 0.95) + "," + (cy - h * 0.91) + " " + (cx + 2 * h * 0.97) + "," + cy;
dash += "M" + (cx - 0.1) + "," + (cy + 0.2) + " L" + (cx + 2 * h + 0.1) + "," + (cy + 0.2);
}
dash += "M" + (cx + 0.4) + "," + (cy + 0.4) + " L" + (cx + 2 * h - 0.3) + "," + (cy + 0.4);
}
if (height >= 0.21 && height < 0.22 && !land[i].river && swampCount < swampiness && land[i].used != 1) {
swampCount++;
land[i].used = 1;
swamp += drawSwamp(x, y);
id = land[i].index;
cell = diagram.cells[id];
cell.halfedges.forEach(function(e) {
edge = diagram.edges[e];
ea = edge.left.index;
if (ea === id || !ea) {
ea = edge.right.index;
}
if (cells[ea].height >= 0.2 && cells[ea].height < 0.3 && !cells[ea].river && cells[ea].used != 1) {
cells[ea].used = 1;
swamp += drawSwamp(cells[ea].data[0], cells[ea].data[1]);
}
})
}
if (Math.random() < height && height >= 0.22 && height < 0.48 && !land[i].river) {
for (c = 0; c < Math.floor(height * 8); c++) {
h = 0.6;
if (c == 0) {
cx = x - h - Math.random();
cy = y - h - Math.random();
}
if (c == 1) {
cx = x + h + Math.random();
cy = y + h + Math.random();
}
if (c == 2) {
cx = x - h - Math.random();
cy = y + 2 * h + Math.random();
}
forest += "M " + cx + " " + cy + " q -1 0.8 -0.05 1.25 v 0.75 h 0.1 v -0.75 q 0.95 -0.47 -0.05 -1.25 z";
fLight += "M " + cx + " " + cy + " q -1 0.8 -0.05 1.25 h 0.1 q 0.95 -0.47 -0.05 -1.25 z";
fShade += "M " + cx + " " + cy + " q -1 0.8 -0.05 1.25 q -0.2 -0.55 0 -1.1 z";
}
}
}
// draw all these stuff
strokes.append("path").attr("d", round(dash, 1));
hills.append("path").attr("d", round(hill[0], 1)).attr("stroke", "#5c5c70");
hills.append("path").attr("d", round(hShade[0], 1)).attr("fill", "white");
hills.append("path").attr("d", round(hill[1], 1)).attr("stroke", "#5c5c70");
hills.append("path").attr("d", round(hShade[1], 1)).attr("fill", "white").attr("stroke", "white");
swamps.append("path").attr("d", round(swamp, 1));
forests.append("path").attr("d", forest);
forests.append("path").attr("d", fLight).attr("fill", "white").attr("stroke", "none");
forests.append("path").attr("d", fShade).attr("fill", "#999999").attr("stroke", "none");
console.timeEnd('drawRelief');
}
function compareY(a, b) {
if (a.data[1] > b.data[1]) return 1;
if (a.data[1] < b.data[1]) return -1;
return 0;
}
function drawSwamp(x, y) {
var h = 0.6, line = "";
for (c = 0; c < 3; c++) {
if (c == 0) {
cx = x;
cy = y - 0.5 - Math.random();
}
if (c == 1) {
cx = x + h + Math.random();
cy = y + h + Math.random();
}
if (c == 2) {
cx = x - h - Math.random();
cy = y + 2 * h + Math.random();
}
line += "M" + cx + "," + cy + " H" + (cx - h / 6) + " M" + cx + "," + cy + " H" + (cx + h / 6) + " M" + cx + "," + cy + " L" + (cx - h / 3) + "," + (cy - h / 2) + " M" + cx + "," + cy + " V" + (cy - h / 1.5) + " M" + cx + "," + cy + " L" + (cx + h / 3) + "," + (cy - h / 2);
line += "M" + (cx - h) + "," + cy + " H" + (cx - h / 2) + " M" + (cx + h / 2) + "," + cy + " H" + (cx + h);
}
return line;
}
function dragged(e) {
var el = d3.select(this);
var x = d3.event.x;
var y = d3.event.y;
el.raise().classed("drag", true);
if (el.attr("x")) {
el.attr("x", x).attr("y", y + 0.8);
var matrix = el.attr("transform");
if (matrix) {
var angle = matrix.split('(')[1].split(')')[0].split(' ')[0];
var bbox = el.node().getBBox();
var rotate = "rotate("+ angle + " " + (bbox.x + bbox.width/2) + " " + (bbox.y + bbox.height/2) + ")";
el.attr("transform", rotate);
}
} else {
el.attr("cx", x).attr("cy", y);
}
}
function dragended(d) {
d3.select(this).classed("drag", false);
}
// Complete the map for the "customize" mode
function getMap() {
exitCustomization();
console.time("TOTAL");
if (randomizeInput.value === "1") {randomizeOptions();}
markFeatures();
drawOcean();
reGraph();
resolveDepressions();
flux();
drawRelief();
drawCoastline();
manorsAndRegions();
if (!$("#toggleHeight").hasClass("buttonoff") && terrs.selectAll("path").size() === 0) {toggleHeight();}
console.timeEnd("TOTAL");
}
// Add label or burg on mouseclick
function clicked() {
var brush = $(".pressed").attr("id");
var point = d3.mouse(this);
if (brush === "addLabel" || brush === "addBurg") {
var rnd = Math.floor(Math.random() * cultures.length);
var name = generateName(rnd);
if (brush === "addLabel") {
countries.append("text").attr("x", point[0]).attr("y", point[1]).text(name).on("click", editLabel);
} else {
burgs.append("circle").attr("r", 1).attr("stroke-width", .24)
.attr("cx", point[0]).attr("cy", point[1])
.call(d3.drag().on("drag", dragged).on("end", dragended)).on("click", changeBurg);
capitals.append("text").attr("x", point[0]).attr("y", point[1]).attr("dy", -1.3).text(name).on("click", editLabel);
}
return;
}
if (brush === "addRiver") {
var index = diagram.find(point[0], point[1]).index;
var cell = cells[index];
if (cell.river || cell.height < 0.2) {return;}
var dataRiver = []; // to store river points
var river = +$("#rivers > path").last().attr("id").slice(5) + 1;
cell.flux = 0.85;
while (cell) {
cell.river = river;
var x = cell.data[0], y = cell.data[1];
dataRiver.push({x, y});
var heights = [];
cell.neighbors.forEach(function(e) {heights.push(cells[e].height);});
var minId = heights.indexOf(d3.min(heights));
var min = cell.neighbors[minId];
var tx = cells[min].data[0], ty = cells[min].data[1];
if (cells[min].height < 0.2) {
var px = (x + tx) / 2;
var py = (y + ty) / 2;
dataRiver.push({x: px, y: py});
cell = undefined;
} else {
if (!cells[min].river) {cells[min].flux += cell.flux; cell = cells[min];}
if (cells[min].river) {
var r = cells[min].river;
var riverEl = $("#river"+r);
var points = JSON.parse(riverEl.attr("data-points"));
var riverCells = [];
for (var p = 0; p < points.length; p++) {
var c = diagram.find(points[p].scX, points[p].scY, 5);
if (c === null) {continue;}
if (c.index !== riverCells[riverCells.length-1]) {riverCells.push(c.index);}
}
if (dataRiver.length > riverCells.indexOf(min)) {
cells[min].flux = cell.flux + cells[min].flux / 2;
cell = cells[min];
riverEl.remove();
cells.map(function(c) {if (c.river === r) {c.river = undefined;}})
} else {
cells[min].confluence += dataRiver.length;
cells[min].flux += cell.flux;
dataRiver.push({x: tx, y: ty});
cell = undefined;
}
}
}
}
var rndFactor = 0.2 + Math.random() * 1.6; // random factor in range 0.2-1.8
var riverAmended = amendRiver(dataRiver, rndFactor);
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
var d = drawRiver(riverAmended, 2, 1);
elSelected = rivers.append("path").attr("d", d).attr("id", "river"+river)
.attr("data-points", round(JSON.stringify(riverAmended), 1))
.attr("data-width", 2).attr("data-increment", 1).on("click", editRiver);
rivers.select(".riverPoints").remove();
rivers.append("g").attr("class", "riverPoints");
riverAmended.map(function(p) {addRiverPoint(p);});
return;
}
if (customization === 1 && brush) {
var cell = diagram.find(point[0], point[1]).index;
var power = +brushPower.value;
if (brush === "brushHill") {add(cell, "hill", power);}
if (brush === "brushPit") {addPit(1, power, cell);}
if (brush === "brushRange" || brush === "brushTrough") {
if (icons.selectAll(".tag").size() === 0) {
icons.append("circle").attr("r", 3).attr("class", "tag").attr("cx", point[0]).attr("cy", point[1]);
} else {
var x = +icons.select(".tag").attr("cx");
var y = +icons.select(".tag").attr("cy");
var from = diagram.find(x, y).index;
icons.selectAll(".tag, .line").remove();
addRange(brush === "brushRange" ? 1 : -1, power, from, cell);
}
}
updateCellsInRadius(cell, cell);
mockHeightmap();
}
// add new river point if elSelected is river (has data-points) and add button pressed
if (!elSelected) {return;}
if (elSelected.attr("data-points")) {
if ($("#riverAddPoint").hasClass('pressed')) {
var dists = [], points = [];
var tr = parseTransform(elSelected.attr("transform"));
if (tr[5] == "1") {
point[0] -= +tr[0];
point[1] -= +tr[1];
}
rivers.select(".riverPoints").selectAll("circle").each(function() {
var x = +d3.select(this).attr("cx");
var y = +d3.select(this).attr("cy");
dists.push(Math.hypot(point[0] - x, point[1] - y));
points.push({scX:x, scY:y});
}).remove();
var index = dists.length;
if (points.length > 1) {
var sorded = dists.slice(0).sort(function(a, b) {return a-b;});
var closest = dists.indexOf(sorded[0]);
var next = dists.indexOf(sorded[1]);
if (closest <= next) {index = closest+1;} else {index = next+1;}
}
points.splice(index, 0, {scX:point[0], scY:point[1]});
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
var d = drawRiver(points, 2, 1);
elSelected.attr("d", d).attr("data-points", round(JSON.stringify(points), 1));
points.map(function(p) {addRiverPoint(p)});
return;
}
if ($("#riverNew").hasClass('pressed')) {
if (elSelected.attr("data-river") !== "new") {
elSelected.call(d3.drag().on("drag", null)).classed("draggable", false);
var river = +$("#rivers > path").last().attr("id").slice(5) + 1;
elSelected = rivers.append("path").attr("data-river", "new").attr("id", "river"+river)
.attr("data-width", 2).attr("data-increment", 1).on("click", editRiver);
}
addRiverPoint({scX:point[0], scY:point[1]});
redrawRiver();
return;
}
}
}
// Change burg marker size on click
function changeBurg() {
var size = this.getAttribute("r");
size = +size + .25;
if (size > 1.5) {size = .5;}
var width = this.getAttribute("stroke-width");
width = +width + .06;
if (width > .36) {width = .12;}
var type = this.getAttribute("class");
if (type) {
d3.selectAll("."+type).attr("r", size).attr("stroke-width", width);
} else {
this.setAttribute("r", size);
this.setAttribute("stroke-width", width);
}
}
function editLabel() {
if (elSelected) {
elSelected.call(d3.drag().on("drag", null)).classed("draggable", false);
}
elSelected = d3.select(this);
elSelected.call(d3.drag().on("drag", dragged).on("end", dragended)).classed("draggable", true);
var group = d3.select(this.parentNode);
updateGroupOptions();
editGroupSelect.value = group.attr("id");
editFontSelect.value = fonts.indexOf(group.attr("data-font"));
editSize.value = group.attr("data-size");
editColor.value = toHEX(group.attr("fill"));
editOpacity.value = group.attr("opacity");
editText.value = elSelected.text();
var matrix = elSelected.attr("transform");
if (matrix) {
var rotation = matrix.split('(')[1].split(')')[0].split(' ')[0];
} else {
var rotation = 0;
}
editAngle.value = rotation;
editAngleValue.innerHTML = rotation + "°";
$("#labelEditor").dialog({
title: "Edit Label: " + editText.value,
minHeight: 30, width: "auto", maxWidth: 275, resizable: false,
position: {my: "center top", at: "bottom", of: this}
});
// fetch default fonts if not done before
if (fonts.indexOf("Bitter") === -1) {
$("head").append('<link rel="stylesheet" type="text/css" href="fonts.css">');
fonts = fonts.concat(["Amatic+SC:700", "IM+Fell+English", "Great+Vibes", "Bitter", "Yellowtail", "Montez", "Lobster", "Josefin+Sans", "Shadows+Into+Light", "Orbitron", "Dancing+Script:700", "Bangers", "Chewy", "Architects+Daughter", "Kaushan+Script", "Gloria+Hallelujah", "Satisfy", "Comfortaa:700", "Cinzel"]);
updateFontOptions();
}
}
$("#labelEditor .editButton, #labelEditor .editButtonS").click(function() {
var group = d3.select(elSelected.node().parentNode);
if (this.id == "editRemoveSingle") {
alertMessage.innerHTML = "Are you sure you want to remove the label?";
$(function() {$("#alert").dialog({resizable: false, title: "Remove label",
buttons: {
"Remove": function() {
$(this).dialog("close");
elSelected.remove();
$("#labelEditor").dialog("close");
},
Cancel: function() {$(this).dialog("close");}
}})
});
return;
}
if (this.id == "editGroupRemove") {
var count = group.selectAll("text").size()
if (count < 2) {
group.remove();
$("#labelEditor").dialog("close");
return;
}
var message = "Are you sure you want to remove all labels (" + count + ") of that group?";
alertMessage.innerHTML = message;
$(function() {$("#alert").dialog({resizable: false, title: "Remove labels",
buttons: {
"Remove": function() {
$(this).dialog("close");
group.remove();
$("#labelEditor").dialog("close");
},
Cancel: function() {$(this).dialog("close");}
}})
});
return;
}
if (this.id == "editCopy") {
var shift = +group.attr("font-size") + 1;
var xn = +elSelected.attr("x") - shift;
var yn = +elSelected.attr("y") - shift;
while (group.selectAll("text[x='" + xn + "']").size() > 0) {xn -= shift; yn -= shift;}
group.append("text").attr("x", xn).attr("y", yn).text(elSelected.text())
.attr("transform", elSelected.attr("transform")).on("click", editLabel);
return;
}
if (this.id == "editGroupNew") {
if ($("#editGroupInput").css("display") === "none") {
$("#editGroupInput").css("display", "inline-block");
$("#editGroupSelect").css("display", "none");
editGroupInput.focus();
} else {
$("#editGroupSelect").css("display", "inline-block");
$("#editGroupInput").css("display", "none");
}
return;
}
if (this.id == "editExternalFont") {
if ($("#editFontInput").css("display") === "none") {
$("#editFontInput").css("display", "inline-block");
$("#editFontSelect").css("display", "none");
editFontInput.focus();
} else {
$("#editFontSelect").css("display", "inline-block");
$("#editFontInput").css("display", "none");
}
return;
}
if (this.id == "editTextRandom") {
var culture, index;
// check if label is manor name to get culture
var manor = $.grep(manors, function(e) {return (e.name === editText.value);});
if (manor.length === 1) {
culture = manor[0].culture;
index = manor[0].i;
} else {
// if not get cell's culture at BBox centre
var c = elSelected.node().getBBox();
var x = c.x + c.width / 2;
var y = c.y + c.height / 2;
culture = diagram.find(x, y).culture;
if (!culture) {culture = 0;}
}
var name = generateName(culture);
if (group.attr("id") === "countries") {name = addRegionSuffix(name, culture);}
editText.value = name;
elSelected.text(name);
$("div[aria-describedby='labelEditor'] .ui-dialog-title").text("Edit Label: " + name);
if (manor.length === 1) {manors[index].name = name;}
return;
}
$("#labelEditor .editButton").toggle();
if (this.id == "editGroupButton") {
if ($("#editGroupInput").css("display") !== "none") {$("#editGroupSelect").css("display", "inline-block");}
if ($("#editGroupRemove").css("display") === "none") {
$("#editGroupRemove, #editGroupNew").css("display", "inline-block");
} else {
$("#editGroupInput, #editGroupRemove, #editGroupNew").css("display", "none");
}
}
if (this.id == "editFontButton") {$("#editSizeIcon, #editFontSelect, #editSize").toggle();}
if (this.id == "editStyleButton") {$("#editOpacityIcon, #editOpacity, #editShadowIcon, #editShadow").toggle();}
if (this.id == "editAngleButton") {$("#editAngleValue").toggle();}
if (this.id == "editTextButton") {$("#editTextRandom").toggle();}
$(this).show().next().toggle();
});
function updateGroupOptions() {
editGroupSelect.innerHTML = "";
labels.selectAll("g").each(function(d) {
var opt = document.createElement("option");
opt.value = opt.innerHTML = d3.select(this).attr("id");
editGroupSelect.add(opt);
});
}
// on editAngle change
$("#editAngle").change(function() {
var c = elSelected.node().getBBox();
var rotate = `rotate(${this.value} ${(c.x+c.width/2)} ${(c.y+c.height/2)})`;
elSelected.attr("transform", rotate);
});
// on editFontInput change. Use a direct link to any @font-face declaration or just font name to fetch from Google Fonts
$("#editFontInput").change(function() {
if (editFontInput.value !== "") {
var url = (editFontInput.value).replace(/ /g, "+");
if (url.indexOf("http") === -1) {url = "https://fonts.googleapis.com/css?family=" + url;}
addFonts(url);
editFontInput.value = "";
editExternalFont.click();
}
});
function addFonts(url) {
$('head').append('<link rel="stylesheet" type="text/css" href="' + url + '">');
return fetch(url)
.then(resp => resp.text())
.then(text => {
let s = document.createElement('style');
s.innerHTML = text;
document.head.appendChild(s);
let styleSheet = Array.prototype.filter.call(
document.styleSheets,
sS => sS.ownerNode === s)[0];
let FontRule = rule => {
let family = rule.style.getPropertyValue('font-family');
let weight = rule.style.getPropertyValue('font-weight');
let font = family.replace(/['"]+/g, '').replace(/ /g, "+") + ":" + weight;
if (fonts.indexOf(font) == -1) {fonts.push(font);}
};
for (var r of styleSheet.cssRules) {FontRule(r);}
document.head.removeChild(s);
updateFontOptions();
})
}
// on any Editor input change
$("#labelEditor .editTrigger").change(function() {
$(this).attr("title", $(this).val());
elSelected.text(editText.value);
// check if Group was changed
var group = d3.select(elSelected.node().parentNode);
var groupOld = group.attr("id");
var groupNew = editGroupSelect.value;
if (editGroupInput.value !== "") {
groupNew = editGroupInput.value.toLowerCase().replace(/ /g, "_").replace(/[^\w\s]/gi, "");
if (Number.isFinite(+groupNew.charAt(0))) {groupNew = "g" + groupNew;}
}
if (groupOld !== groupNew) {
var removed = elSelected.remove();
if (labels.select("#"+groupNew).size() > 0) {
group = labels.select("#"+groupNew);
editFontSelect.value = fonts.indexOf(group.attr("data-font"));
editSize.value = group.attr("data-size");
editColor.value = toHEX(group.attr("fill"));
editOpacity.value = group.attr("opacity");
} else {
if (group.selectAll("text").size() === 0) {group.remove();}
group = labels.append("g").attr("id", groupNew);
updateGroupOptions();
$("#editGroupSelect, #editGroupInput").toggle();
editGroupInput.value = "";
}
group.append(function() {return removed.node();});
editGroupSelect.value = group.attr("id");
}
// update Group attributes
var size = +editSize.value;
var font = fonts[editFontSelect.value].split(':')[0].replace(/\+/g, " ");
group.attr("data-size", size)
.attr("font-size", roundN((size + (size / scale)) / 2, 2))
.attr("font-family", font)
.attr("data-font", fonts[editFontSelect.value])
.attr("fill", editColor.title)
.attr("opacity", editOpacity.value);
});
// Update font list for Label Editor
function updateFontOptions() {
editFontSelect.innerHTML = "";
for (var i=0; i < fonts.length; i++) {
var opt = document.createElement('option');
opt.value = i;
var font = fonts[i].split(':')[0].replace(/\+/g, " ");
opt.style.fontFamily = opt.innerHTML = font;
editFontSelect.add(opt);
}
}
// convert RGB color string to HEX without #
function toHEX(rgb){
if (rgb.charAt(0) === "#") {return rgb;}
rgb = rgb.match(/^rgba?[\s+]?\([\s+]?(\d+)[\s+]?,[\s+]?(\d+)[\s+]?,[\s+]?(\d+)[\s+]?/i);
return (rgb && rgb.length === 4) ? "#" +
("0" + parseInt(rgb[1],10).toString(16)).slice(-2) +
("0" + parseInt(rgb[2],10).toString(16)).slice(-2) +
("0" + parseInt(rgb[3],10).toString(16)).slice(-2) : '';
}
// get Curve Type
function getCurveType() {
type = curveType.value;
if (type === "Catmull–Rom") {lineGen.curve(d3.curveCatmullRom);}
if (type === "Linear") {lineGen.curve(d3.curveLinear);}
if (type === "Basis") {lineGen.curve(d3.curveBasisClosed);}
if (type === "Cardinal") {lineGen.curve(d3.curveCardinal);}
if (type === "Step") {lineGen.curve(d3.curveStep);}
}
// round string to d decimals
function round(string, d) {
var d = d || 1;
var m = Math.pow(10, d);
return string.replace(/[\d\.-][\d\.e-]*/g, function(n) {return Math.round(n * m) / m;})
}
// round number to d decimals
function roundN(number, d) {
var d = d || 2;
var m = Math.pow(10, d);
return Math.round(number * m) / m
}
// downalod map as SVG or PNG file
function saveAsImage(type) {
console.time("saveAsImage");
// get all used fonts
if (type === "svg") {viewbox.attr("transform", null);}
var fontsInUse = []; // to store fonts currently in use
labels.selectAll("g").each(function(d) {
var font = d3.select(this).attr("data-font");
if (fontsInUse.indexOf(font) === -1) {fontsInUse.push(font);}
});
var fontsToLoad = "https://fonts.googleapis.com/css?family=" + fontsInUse.join("|");
// clone svg
var cloneEl = document.getElementsByTagName("svg")[0].cloneNode(true);
cloneEl.id = "clone";
document.getElementsByTagName("body")[0].appendChild(cloneEl);
var clone = d3.select("#clone");
// for each g element get inline style so it could be used in saved svg
var emptyG = clone.append("g").node();
var defaultStyles = window.getComputedStyle(emptyG);
clone.selectAll("g").each(function(d) {
var compStyle = window.getComputedStyle(this);
var style = "";
for (var i=0; i < compStyle.length; i++) {
var key = compStyle[i];
var value = compStyle.getPropertyValue(key);
if (key !== "cursor" && value != defaultStyles.getPropertyValue(key)) {
style += key + ':' + value + ';';
}
}
if (style != "") {this.setAttribute('style', style);}
});
emptyG.remove();
// load fonts as dataURI so they will be available in downloaded svg/png
GFontToDataURI(fontsToLoad).then(cssRules => {
clone.select("defs").append("style").text(cssRules.join('\n'));
var svg_xml = (new XMLSerializer()).serializeToString(clone.node());
var blob = new Blob([svg_xml], {type:'image/svg+xml;charset=utf-8'});
var url = window.URL.createObjectURL(blob);
var link = document.createElement("a");
if (type === "png") {
canvas.width = mapWidth * 2;
canvas.height = mapHeight * 2;
var img = new Image();
img.src = url;
img.onload = function(){
ctx.drawImage(img, 0, 0, mapWidth * 2, mapHeight * 2);
link.download = "fantasy_map_" + Date.now() + ".png";
link.href = canvas.toDataURL('image/png');
canvas.width = mapWidth;
canvas.height = mapHeight;
canvas.style.opacity = 0;
document.body.appendChild(link);
link.click();
}
} else {
link.download = "fantasy_map_" + Date.now() + ".svg";
link.href = url;
document.body.appendChild(link);
link.click();
}
clone.remove();
console.timeEnd("saveAsImage");
window.setTimeout(function() {window.URL.revokeObjectURL(url);}, 2000);
});
}
// Code from Kaiido's answer:
// https://stackoverflow.com/questions/42402584/how-to-use-google-fonts-in-canvas-when-drawing-dom-objects-in-svg
function GFontToDataURI(url) {
"use strict;"
return fetch(url) // first fecth the embed stylesheet page
.then(resp => resp.text()) // we only need the text of it
.then(text => {
let s = document.createElement('style');
s.innerHTML = text;
document.head.appendChild(s);
let styleSheet = Array.prototype.filter.call(
document.styleSheets,
sS => sS.ownerNode === s)[0];
let FontRule = rule => {
let src = rule.style.getPropertyValue('src');
let family = rule.style.getPropertyValue('font-family');
let url = src.split('url(')[1].split(')')[0];
return {
rule: rule,
src: src,
url: url.substring(url.length - 1, 1)
};
};
let fontRules = [], fontProms = [];
for (var r of styleSheet.cssRules) {
let fR = FontRule(r)
fontRules.push(fR);
fontProms.push(
fetch(fR.url) // fetch the actual font-file (.woff)
.then(resp => resp.blob())
.then(blob => {
return new Promise(resolve => {
let f = new FileReader();
f.onload = e => resolve(f.result);
f.readAsDataURL(blob);
})
})
.then(dataURL => {
return fR.rule.cssText.replace(fR.url, dataURL);
})
)
}
document.head.removeChild(s); // clean up
return Promise.all(fontProms); // wait for all this has been done
});
}
// print displayed map segment
function printMap() {
var popUpAndPrint = function() {
//var printWindow = window.open('', 'PrintMap', 'width=' + mapWidth + ', height=' + mapHeight);
//var div = document.createElement("div");
//div.appendChild(svg.node());
//document.body.appendChild(div);
//printWindow.document.writeln(div);
//printWindow.document.close();
window.print();
window.close();
};
setTimeout(popUpAndPrint, 500);
}
// Save in .map format, based on FileSystem API
function saveMap() {
console.time("saveMap");
// data convention: 0 - version; 1 - all points; 2 - cells; 3 - manors; 4 - svg;
var svg_xml = (new XMLSerializer()).serializeToString(svg.node());
var line = "\r\n";
var data = version + line + JSON.stringify(points) + line + JSON.stringify(cells) + line + JSON.stringify(manors) + line + svg_xml;
var dataBlob = new Blob([data], {type:"text/plain"});
var dataURL = window.URL.createObjectURL(dataBlob);
var link = document.createElement("a");
link.download = "fantasy_map_" + Date.now() + ".map";
link.href = dataURL;
document.body.appendChild(link);
link.click();
console.timeEnd("saveMap");
window.setTimeout(function() {window.URL.revokeObjectURL(dataURL);}, 2000);
}
// Map Loader based on FileSystem API
$("#fileToLoad").change(function() {
console.time("loadMap");
var fileToLoad = this.files[0];
this.value = "";
var fileReader = new FileReader();
fileReader.onload = function(fileLoadedEvent) {
newPoints = [], points = [], cells = [], land = [], riversData = [], island = 0, manors = [], queue = [];
var dataLoaded = fileLoadedEvent.target.result;
var data = dataLoaded.split("\r\n");
// data convention: 0 - version; 1 - all points; 2 - cells; 3 - manors; 4 - svg;
var mapVersion = data[0];
if (mapVersion !== version) {
var message = `The Map version `;
if (mapVersion.length <= 10) {message += ` (${mapVersion}) `;}
message += `does not match the Generator version (${version}). In case of issues please send the .map file `;
message += `<a href="mailto:maxganiev@yandex.ru?Subject=Map%20update%20request" target="_top">to me</a>`;
message += ` for update or just keep using `
message += `<a href="https://github.com/Azgaar/Fantasy-Map-Generator/wiki/Changelog" target="_blank">an appropriate version</a>`;
message += ` of the Generator`;
alertMessage.innerHTML = message;
$(function() {$("#alert").dialog({title: "Load map", buttons: {OK: function() {$(this).dialog("close");}}});});
}
if (mapVersion.length > 10) {console.error("Cannot load map"); return;}
svg.remove();
points = JSON.parse(data[1]);
cells = JSON.parse(data[2]);
land = $.grep(cells, function(e) {return (e.height >= 0.2);});
cells.map(function(e) {newPoints.push(e.data);});
calculateVoronoi(newPoints);
manors = JSON.parse(data[3]);
document.body.insertAdjacentHTML("afterbegin", data[4]);
// redefine variables
customization = 0, elSelected = "";
svg = d3.select("svg").call(zoom);
mapWidth = +svg.attr("width");
mapHeight = +svg.attr("height");
defs = svg.select("#deftemp");
viewbox = svg.select("#viewbox").on("touchmove mousemove", moved).on("click", clicked);
ocean = viewbox.select("#ocean");
oceanLayers = ocean.select("#oceanLayers");
oceanPattern = ocean.select("#oceanPattern");
landmass = viewbox.select("#landmass");
terrs = viewbox.select("#terrs");
cults = viewbox.select("#cults");
routes = viewbox.select("#routes");
roads = routes.select("#roads");
trails = routes.select("#trails");
rivers = viewbox.select("#rivers");
terrain = viewbox.select("#terrain");
regions = viewbox.select("#regions");
borders = viewbox.select("#borders");
stateBorders = borders.select("#stateBorders");
neutralBorders = borders.select("#neutralBorders");
coastline = viewbox.select("#coastline");
lakes = viewbox.select("#lakes");
grid = viewbox.select("#grid");
searoutes = routes.select("#searoutes");
labels = viewbox.select("#labels");
icons = viewbox.select("#icons");
burgs = icons.select("#burgs");
debug = viewbox.select("#debug");
capitals = labels.select("#capitals");
towns = labels.select("#towns");
countries = labels.select("#countries");
// restore events
labels.selectAll("text").on("click", editLabel);
burgs.selectAll("circle").call(d3.drag().on("drag", dragged).on("end", dragended)).on("click", changeBurg);
// restore layers state
if (cults.selectAll("path").size() == 0) {$("#toggleCultures").addClass("buttonoff");} else {$("#toggleCultures").removeClass("buttonoff");}
if (terrs.selectAll("path").size() == 0) {$("#toggleHeight").addClass("buttonoff");} else {$("#toggleHeight").removeClass("buttonoff");}
if (regions.attr("display") === "none") {$("#toggleCountries").addClass("buttonoff");} else {$("#toggleCountries").removeClass("buttonoff");}
if (rivers.attr("display") === "none") {$("#toggleRivers").addClass("buttonoff");} else {$("#toggleRivers").removeClass("buttonoff");}
if (oceanPattern.attr("display") === "none") {$("#toggleOcean").addClass("buttonoff");} else {$("#toggleOcean").removeClass("buttonoff");}
if (landmass.attr("display") === "none") {$("#landmass").addClass("buttonoff");} else {$("#landmass").removeClass("buttonoff");}
if (terrain.attr("display") === "none") {$("#toggleRelief").addClass("buttonoff");} else {$("#toggleRelief").removeClass("buttonoff");}
if (borders.attr("display") === "none") {$("#toggleBorders").addClass("buttonoff");} else {$("#toggleBorders").removeClass("buttonoff");}
if (burgs.attr("display") === "none") {$("#toggleIcons").addClass("buttonoff");} else {$("#toggleIcons").removeClass("buttonoff");}
if (labels.attr("display") === "none") {$("#toggleLabels").addClass("buttonoff");} else {$("#toggleLabels").removeClass("buttonoff");}
if (routes.attr("display") === "none") {$("#toggleRoutes").addClass("buttonoff");} else {$("#toggleRoutes").removeClass("buttonoff");}
if (grid.attr("display") === "none") {$("#toggleGrid").addClass("buttonoff");} else {$("#toggleGrid").removeClass("buttonoff");}
console.timeEnd("loadMap");
};
fileReader.readAsText(fileToLoad, "UTF-8");
});
// Poisson-disc sampling for a points
// Source: bl.ocks.org/mbostock/99049112373e12709381; Based on https://www.jasondavies.com/poisson-disc
function poissonDiscSampler(width, height, radius) {
var k = 5, // maximum number of points before rejection
radius2 = radius * radius,
R = 3 * radius2,
cellSize = radius * Math.SQRT1_2,
gridWidth = Math.ceil(width / cellSize),
gridHeight = Math.ceil(height / cellSize),
grid = new Array(gridWidth * gridHeight),
queue = [],
queueSize = 0,
sampleSize = 0;
return function() {
if (!sampleSize) return sample(Math.random() * width, Math.random() * height);
// Pick a random existing sample and remove it from the queue
while (queueSize) {
var i = Math.random() * queueSize | 0,
s = queue[i];
// Make a new candidate between [radius, 2 * radius] from the existing sample.
for (var j = 0; j < k; ++j) {
var a = 2 * Math.PI * Math.random(),
r = Math.sqrt(Math.random() * R + radius2),
x = s[0] + r * Math.cos(a),
y = s[1] + r * Math.sin(a);
// Reject candidates that are outside the allowed extent, or closer than 2 * radius to any existing sample
if (0 <= x && x < width && 0 <= y && y < height && far(x, y)) return sample(x, y);
}
queue[i] = queue[--queueSize];
queue.length = queueSize;
}
};
function far(x, y) {
var i = x / cellSize | 0,
j = y / cellSize | 0,
i0 = Math.max(i - 2, 0),
j0 = Math.max(j - 2, 0),
i1 = Math.min(i + 3, gridWidth),
j1 = Math.min(j + 3, gridHeight);
for (j = j0; j < j1; ++j) {
var o = j * gridWidth;
for (i = i0; i < i1; ++i) {
if (s = grid[o + i]) {
var s,
dx = s[0] - x,
dy = s[1] - y;
if (dx * dx + dy * dy < radius2) return false;
}
}
}
return true;
}
function sample(x, y) {
var s = [x, y];
queue.push(s);
grid[gridWidth * (y / cellSize | 0) + (x / cellSize | 0)] = s;
++sampleSize;
++queueSize;
return s;
}
}
// Hotkeys
d3.select("body").on("keydown", function() {
if (!$("#labelEditor").is(":visible")) {
switch(d3.event.keyCode) {
case 27: // Escape
break;
case 37: // Left
if (viewX + 10 <= 0) {
viewX += 10;
zoomUpdate();
}
break;
case 39: // Right
if (viewX - 10 >= (mapWidth * (scale-1) * -1)) {
viewX -= 10;
zoomUpdate();
}
break;
case 38: // Up
if (viewY + 10 <= 0) {
viewY += 10;
zoomUpdate();
}
break;
case 40: // Down
if (viewY - 10 >= (mapHeight * (scale-1) * -1)) {
viewY -= 10;
zoomUpdate();
}
break;
case 107: // Plus
if (scale < 40) {
var dx = mapWidth / 2 * (scale-1) + viewX;
var dy = mapHeight / 2 * (scale-1) + viewY;
viewX = dx - mapWidth / 2 * scale;
viewY = dy - mapHeight / 2 * scale;
scale += 1;
if (scale > 40) {scale = 40;}
zoomUpdate();
}
break;
case 109: // Minus
if (scale > 1) {
var dx = mapWidth / 2 * (scale-1) + viewX;
var dy = mapHeight / 2 * (scale-1) + viewY;
viewX += mapWidth / 2 - dx;
viewY += mapHeight / 2 - dy;
scale -= 1;
if (scale < 1) {
scale = 1;
viewX = 0;
viewY = 0;
}
zoomUpdate();
}
break;
}
}
});
// Toggle Options pane
$("#optionsTrigger").on("click", function() {
if ($("#options").css("display") === "none") {
$("#regenerate").hide();
$("#options").fadeIn();
$("#layoutTab").click();
this.innerHTML = "◀";
} else {
$("#options").fadeOut();
this.innerHTML = "▶";
}
});
$("#collapsible").hover(function() {
if ($("#options").css("display") === "none") {$("#regenerate").show();}
}, function() {
$("#regenerate").hide();
});
// move layers on mapLayers dragging (jquery sortable)
function moveLayer(event, ui) {
var el = getLayer(ui.item.attr("id"));
if (el) {
var prev = getLayer(ui.item.prev().attr("id"));
var next = getLayer(ui.item.next().attr("id"));
if (prev) {el.insertAfter(prev);} else if (next) {el.insertBefore(next);}
}
}
// define connection between option layer buttons and actual svg groups
function getLayer(id) {
if (id === "toggleHeight") {return $("#terrs");}
if (id === "toggleCultures") {return $("#cults");}
if (id === "toggleRivers") {return $("#rivers");}
if (id === "toggleRelief") {return $("#terrain");}
if (id === "toggleBorders") {return $("#borders");}
if (id === "toggleCountries") {return $("#regions");}
if (id === "toggleIcons") {return $("#icons");}
if (id === "toggleLabels") {return $("#labels");}
if (id === "toggleRoutes") {return $("#routes");}
if (id === "toggleGrid") {return $("#grid");}
}
// UI Button handlers
$("button, a, li").on("click", function() {
var id = this.id;
var parent = this.parentNode.id;
if (icons.selectAll(".tag").size() > 0) {icons.selectAll(".tag, .line").remove();}
if (id === "toggleHeight") {toggleHeight();}
if (id === "toggleCountries") {
var countries = !$("#toggleCountries").hasClass("buttonoff");
var cultures = !$("#toggleCultures").hasClass("buttonoff");
if (!countries && cultures) {
$("#toggleCultures").toggleClass("buttonoff");
toggleCultures();
}
$('#regions').fadeToggle();
}
if (id === "toggleCultures") {
var countries = !$("#toggleCountries").hasClass("buttonoff");
var cultures = !$("#toggleCultures").hasClass("buttonoff");
if (!cultures && countries) {
$("#toggleCountries").toggleClass("buttonoff");
$('#regions').fadeToggle();
}
toggleCultures();
}
if (id === "toggleFlux") {toggleFlux();}
if (parent === "mapLayers" || parent === "styleContent") {$(this).toggleClass("buttonoff");}
if (id === "randomMap" || id === "regenerate") {
exitCustomization();
undraw();
resetZoom(1000);
generate();
}
if (id === "fromScratch") {
undraw();
placePoints();
calculateVoronoi(points);
detectNeighbors("grid");
customizeHeightmap();
}
if (id === "fromHeightmap") {
var heights = [];
for (var i = 0; i < points.length; i++) {
var cell = diagram.find(points[i][0], points[i][1]).index;
heights.push(cells[cell].height);
}
undraw();
calculateVoronoi(points);
detectNeighbors("grid");
for (var i = 0; i < points.length; i++) {
cells[i].height = heights[i];
}
mockHeightmap();
customizeHeightmap();
}
// heightmap customization buttons
if (customization === 1) {
if (id === "paintBrushes") {
$("#brushesPanel").dialog({
title: "Paint Brushes",
minHeight: 40, width: "auto", maxWidth: 200, resizable: false,
position: {my: "right top", at: "right-10 top+10", of: "svg"}});
}
if (id === "rescaleExecute") {
var subject = rescaleLower.value + "-" + rescaleHigher.value;
var sign = conditionSign.value;
var modifier = rescaleModifier.value;
if (sign === "×") {modifyHeights(subject, 0, +modifier);}
if (sign === "÷") {modifyHeights(subject, 0, (1 / modifier));}
if (sign === "+") {modifyHeights(subject, +modifier, 1);}
if (sign === "-") {modifyHeights(subject, (-1 * modifier), 1);}
if (sign === "^") {modifyHeights(subject, 0, "^" + modifier);}
mockHeightmap();
}
if (id === "rescaleButton") {
$("#modifyButtons").children().not("#rescaleButton, .condition").toggle();
}
if (id === "rescaleCondButton") {$("#modifyButtons").children().not("#rescaleCondButton, #rescaler").toggle();}
if (id === "undo") {restoreHistory(historyStage - 1);}
if (id === "redo") {restoreHistory(historyStage + 1);}
if (id === "smoothHeights") {smoothHeights(4); mockHeightmap();}
if (id === "disruptHeights") {disruptHeights(); mockHeightmap();}
if (id === "getMap") {getMap();}
if (id === "applyTemplate") {
$("#templateEditor").dialog({
title: "Template Editor",
minHeight: "auto", width: "auto", resizable: false,
position: {my: "right top", at: "right-10 top+10", of: "svg"}
});
}
if (id === "convertImage") {convertImage();}
if (id === "convertImageGrid") {$("#grid").fadeToggle();}
if (id === "convertImageHeights") {$("#landmass").fadeToggle();}
if (id === "perspectiveView") {
// Inputs control
const line = +$("#lineHandle0").attr("data-value");
const grad = +$("#lineHandle1").attr("data-value");
$("#lineSlider").slider({
min: 10, max: 320, step: 1, values: [line, grad],
create: function() {
$("#lineHandle0").text("x:"+line);
$("#lineHandle1").text("y:"+grad);
},
slide: function(event, ui) {
$("#lineHandle0").text("x:"+ui.values[0]).attr("data-value", ui.values[0]);
$("#lineHandle1").text("y:"+ui.values[1]).attr("data-value", ui.values[1]);
drawPerspective();
}
});
$("#ySlider").slider({
min: 1, max: 5, step: 0.1, value: +$("#yHandle").attr("data-value"),
create: function() {$("#yHandle").text($("#yHandle").attr("data-value"));},
slide: function(event, ui) {
$("#yHandle").text(ui.value).attr("data-value", ui.value);
drawPerspective();
}
});
$("#scaleSlider").slider({
min: 0.5, max: 2, step: 0.1, value: +$("#scaleHandle").attr("data-value"),
create: function() {$("#scaleHandle").text($("#scaleHandle").attr("data-value"));},
slide: function(event, ui) {
$("#scaleHandle").text(ui.value).attr("data-value", ui.value);
drawPerspective();
}
});
$("#heightSlider").slider({
min: 1, max: 50, step: 1, value: +$("#heightHandle").attr("data-value"),
create: function() {$("#heightHandle").text($("#heightHandle").attr("data-value"));},
slide: function(event, ui) {
$("#heightHandle").text(ui.value).attr("data-value", ui.value);
drawPerspective();
}
});
$("#perspectivePanel").dialog({
title: "Perspective View",
width: 520, height: 360,
position: {my: "center center", at: "center center", of: "svg"}
});
drawPerspective();
}
}
if ($(this).hasClass('radio') && (parent === "addFeature" || parent === "brushesButtons")) {
if ($(this).hasClass('pressed')) {
$(".pressed").removeClass('pressed');
viewbox.style("cursor", "default").on(".drag", null);
$("#brushRadiusLabel, #brushRadius").attr("disabled", true).addClass("disabled");
} else {
$(".pressed").removeClass('pressed');
$(this).addClass('pressed');
viewbox.style("cursor", "crosshair");
if (id.slice(0,5) === "brush" && id !== "brushRange" && id !== "brushTrough") {
viewbox.call(drag);
} else {
viewbox.on(".drag", null);
}
if (parent === "addFeature" || $(this).hasClass("feature")) {
$("#brushRadiusLabel, #brushRadius").attr("disabled", true).addClass("disabled");
} else {
$("#brushRadiusLabel, #brushRadius").attr("disabled", false).removeClass("disabled");
}
}
}
if ($(this).hasClass('radio') && parent === "mapFilters") {
$("svg").removeClass();
if ($(this).hasClass('pressed')) {
$("#mapFilters .pressed").removeClass('pressed');
} else {
$("#mapFilters .pressed").removeClass('pressed');
$(this).addClass('pressed');
if (id === "grayscale") {$("svg").addClass("grayscale");}
if (id === "sepia") {$("svg").addClass("sepia");}
if (id === "tint") {$("svg").addClass("tint");}
if (id === "dingy") {$("svg").addClass("dingy");}
}
}
if (id === "saveButton") {$("#saveDropdown").slideToggle();}
if (id === "loadMap") {fileToLoad.click();}
if (id === "printMap") {printMap();}
if (id === "zoomReset") {resetZoom(1000);}
if (id === "zoomPlus") {
scale += 1;
if (scale > 40) {scale = 40;}
zoomUpdate();
}
if (id === "zoomMinus") {
scale -= 1;
if (scale <= 1) {scale = 1; viewX = 0; viewY = 0;}
zoomUpdate();
}
if (id === "styleFontPlus" || id === "styleFontMinus") {
var el = viewbox.select("#"+styleElementSelect.value);
var mod = id === "styleFontPlus" ? 1.1 : 0.9;
el.selectAll("g").each(function() {
var el = d3.select(this);
var size = roundN(el.attr("font-size") * mod, 2);
if (size < 0.2) {size = 0.2;}
el.attr("data-size", size).attr("font-size", roundN((size + (size / scale)) / 2, 2));
});
return;
}
if (id === "styleFillPlus" || id === "styleFillMinus") {
var el = viewbox.select("#"+styleElementSelect.value);
var mod = id === "styleFillPlus" ? 1.1 : 0.9;
el.selectAll("*").each(function() {
var el = d3.select(this);
var size = roundN(el.attr("r") * mod, 2);
if (size < 0.1) {size = 0.1;}
if (el.node().nodeName === "circle") {el.attr("r", size);}
});
return;
}
if (id === "styleStrokePlus" || id === "styleStrokeMinus") {
var el = viewbox.select("#"+styleElementSelect.value);
var mod = id === "styleStrokePlus" ? 1.1 : 0.9;
el.selectAll("*").each(function() {
var el = d3.select(this);
var size = roundN(el.attr("stroke-width") * mod, 2);
if (size < 0.1) {size = 0.1;}
if (el.node().nodeName === "circle") {el.attr("stroke-width", size);}
});
return;
}
if (id === "templateClear" || id === "brushClear") {
if (customization === 1) {
var message = "Are you sure you want to clear the map?";
alertMessage.innerHTML = message;
$(function() {$("#alert").dialog({resizable: false, title: "Clear map",
buttons: {
"Clear": function() {
$(this).dialog("close");
viewbox.style("cursor", "crosshair").call(drag);
landmassCounter.innerHTML = "0";
$("#landmass").empty();
cells.map(function(i) {i.height = 0;});
// clear history
history = [];
historyStage = -1;
redo.disabled = true;
undo.disabled = true;
},
Cancel: function() {$(this).dialog("close");}
}})
});
} else {
start.click();
}
}
if (id === "templateComplete") {
if (customization === 1 && !$("#getMap").attr("disabled")) {getMap();}
}
if (id === "convertColorsMinus") {
var current = +convertColors.value - 1;
if (current < 4) {current = 3;}
convertColors.value = current;
heightsFromImage(current);
}
if (id === "convertColorsPlus") {
var current = +convertColors.value + 1;
if (current > 255) {current = 256;}
convertColors.value = current;
heightsFromImage(current);
}
if (id === "convertOverlayButton") {
$("#convertImageButtons").children().not(this).not("#imageToLoad, #convertColors").toggle();
}
if (id === "convertAutoLum") {autoAssing("lum");}
if (id === "convertAutoHue") {autoAssing("hue");}
if (id === "convertComplete") {completeConvertion();}
});
// support save options
$("#saveDropdown > div").click(function() {
var id = this.id;
if (id === "saveMap") {saveMap();}
if (id === "saveSVG") {saveAsImage("svg");}
if (id === "savePNG") {saveAsImage("png");}
$("#saveDropdown").slideUp("fast");
});
function drawPerspective() {
console.time("drawPerspective");
const width = 320, height = 180;
const wRatio = mapWidth / width, hRatio = mapHeight / height;
const lineCount = +$("#lineHandle0").attr("data-value");
const lineGranularity = +$("#lineHandle1").attr("data-value");
const perspective = document.getElementById("perspective");
const pContext = perspective.getContext("2d");
const lines = [];
let i = Math.floor(lineCount);
while (i--) {
const x = i / lineCount * width | 0;
const canvasPoints = [];
lines.push(canvasPoints);
let j = Math.floor(lineGranularity);
while (j--) {
const y = j / lineGranularity * height | 0;
let h = getHeightInPoint(x * wRatio, y * hRatio) - 0.2;
if (h < 0) {h = 0;}
canvasPoints.push([x, y, h]);
}
}
pContext.clearRect(0, 0, perspective.width, perspective.height);
for (let canvasPoints of lines) {
for (let i = 0; i < canvasPoints.length - 1; i++) {
const pt1 = canvasPoints[i];
const pt2 = canvasPoints[i + 1];
const avHeight = (pt1[2] + pt2[2]) / 2;
pContext.beginPath();
// transformPt
pContext.moveTo(...transformPt(pt1));
pContext.lineTo(...transformPt(pt2));
//const water = "rgb(81, 103, 169)", land = "rgb(130, 160, 115)";
//const clr = avHeight > 0 ? land : water;
let clr = "rgb(81, 103, 169)"; // water
if (avHeight !== 0) {clr = color(1 - avHeight - 0.2);}
pContext.strokeStyle = clr;
pContext.stroke();
}
}
console.timeEnd("drawPerspective");
}
// get Height value in point for Perspective view
function getHeightInPoint(x, y) {
const index = diagram.find(x, y).index;
return cells[index].height;
}
function transformPt(pt) {
const width = 320;
const maxHeight = +$("#heightHandle").attr("data-value");
var [x, y] = projectIsometric(pt[0], pt[1]);
return [x + width / 2 + 10, y + 10 - pt[2] * maxHeight];
}
function projectIsometric(x, y) {
const scale = $("#scaleHandle").attr("data-value");
const yProj = $("#yHandle").attr("data-value");
return [(x - y) * scale, (x + y) / yProj * scale];
}
// templateEditor Button handlers
$("#templateTools > button").on("click", function() {
var id = this.id;
id = id.replace("template", "");
if (id === "Mountain") {
var steps = $("#templateBody > div").length;
if (steps > 0) {return;}
}
$("#templateBody").attr("data-changed", 1);
$("#templateBody").append('<div data-type="' + id + '">' + id + '</div>');
var el = $("#templateBody div:last-child");
if (id === "Hill" || id === "Pit" || id === "Range" || id === "Trough") {
var count = '<label>count:<input class="templateElCount" title="Blobs to add" type="number" value="1" min="1" max="99"></label>';
}
if (id === "Hill") {
var dist = '<label>distribution:<input class="templateElDist" title="Set blobs distribution. 0.5 - map center; 0.1 - any place" type="number" value="0.25" min="0.1" max="0.5" step="0.01"></label>';
}
if (id === "Add" || id === "Multiply") {
var dist = '<label>to:<select class="templateElDist" title="Change only land or all cells"><option value="all" selected>all cells</option><option value="land">land only</option><option value="interval">interval</option></select></label>';
}
if (id === "Add") {
var count = '<label>value:<input class="templateElCount" title="Add value to height of all cells (negative values are allowed)" type="number" value="-0.1" min="-1" max="1" step="0.01"></label>';
}
if (id === "Multiply") {
var count = '<label>by value:<input class="templateElCount" title="Multiply all cells Height by the value" type="number" value="1.1" min="0" max="10" step="0.1"></label>';
}
if (id === "Smooth") {
var count = '<label>fraction:<input class="templateElCount" title="Set smooth fraction. 1 - full smooth, 2 - half-smooth, etc." type="number" min="1" max="10" value="2"></label>';
}
if (id === "Strait") {
var count = '<label>width:<input class="templateElCount" title="Set strait width" value="1-7"></label>';
}
el.append('<span title="Remove step" class="icon-trash-empty"></span>');
$(".icon-trash-empty").on("click", function() {$(this).parent().remove();});
if (dist) {el.append(dist);}
if (count) {el.append(count);}
el.find("select.templateElDist").on("input", fireTemplateElDist);
$("#templateBody").attr("data-changed", 1);
});
// fireTemplateElDist selector handlers
function fireTemplateElDist() {
if (this.value === "interval") {
var interval = prompt("Populate a height interval (e.g. from 0.17 to 0.2), without space, but with hyphen", "0.17-0.2");
if (interval) {
var option = '<option value="' + interval + '">' + interval + '</option>';
$(this).append(option).val(interval);
}
}
}
// templateSelect on change listener
$("#templateSelect").on("input", function() {
var steps = $("#templateBody > div").length;
var changed = +$("#templateBody").attr("data-changed");
var template = this.value;
if (steps && changed === 1) {
alertMessage.innerHTML = "Are you sure you want to change the base template? All the changes will be lost.";
$(function() {$("#alert").dialog({resizable: false, title: "Change Template",
buttons: {
"Change": function() {
changeTemplate(template);
$(this).dialog("close");
},
Cancel: function() {
var prev = $("#templateSelect").attr("data-prev");
$("#templateSelect").val(prev);
$(this).dialog("close");
}
}})
});
}
if (steps === 0 || changed === 0) {changeTemplate(template);}
});
function changeTemplate(template) {
$("#templateBody").empty();
$("#templateSelect").attr("data-prev", template);
addStep("Mountain");
if (template === "templateVolcano") {
addStep("Add", 0.05);
addStep("Multiply", 1.1);
addStep("Hill", 5, 0.4);
addStep("Hill", 2, 0.15);
addStep("Range", 3);
addStep("Trough", 3);
}
if (template === "templateHighIsland") {
addStep("Add", 0.05);
addStep("Multiply", 0.9);
addStep("Range", 4);
addStep("Hill", 12, 0.25);
addStep("Trough", 3);
addStep("Multiply", 0.75, "land");
addStep("Hill", 3, 0.15);
}
if (template === "templateLowIsland") {
addStep("Smooth", 2);
addStep("Range", 1);
addStep("Hill", 4, 0.4);
addStep("Hill", 12, 0.2);
addStep("Trough", 8);
addStep("Multiply", 0.35, "land");
}
if (template === "templateContinents") {
addStep("Hill", 24, 0.25);
addStep("Range", 4);
addStep("Hill", 3, 0.18);
addStep("Multiply", 0.7, "land");
addStep("Strait", "2-7");
addStep("Smooth", 2);
addStep("Pit", 7);
addStep("Trough", 8);
addStep("Multiply", 0.8, "land");
addStep("Add", 0.02, "all");
}
if (template === "templateArchipelago") {
addStep("Add", -0.2, "land");
addStep("Hill", 14, 0.17);
addStep("Range", 5);
addStep("Strait", "2-4");
addStep("Trough", 12);
addStep("Pit", 8);
addStep("Add", -0.05, "land");
addStep("Multiply", 0.7, "land");
addStep("Smooth", 4);
}
if (template === "templateAtoll") {
addStep("Hill", 2, 0.35);
addStep("Range", 2);
addStep("Add", 0.07, "all");
addStep("Smooth", 1);
addStep("Multiply", 0.1, "0.27-10");
}
$("#templateBody").attr("data-changed", 0);
}
// interprete template function
function addStep(feature, count, dist) {
if (!feature) {return;}
if (feature === "Mountain") {templateMountain.click();}
if (feature === "Hill") {templateHill.click();}
if (feature === "Pit") {templatePit.click();}
if (feature === "Range") {templateRange.click();}
if (feature === "Trough") {templateTrough.click();}
if (feature === "Strait") {templateStrait.click();}
if (feature === "Add") {templateAdd.click();}
if (feature === "Multiply") {templateMultiply.click();}
if (feature === "Smooth") {templateSmooth.click();}
if (count) {$("#templateBody div:last-child .templateElCount").val(count);}
if (dist) {
if (dist !== "land") {
var option = '<option value="' + dist + '">' + dist + '</option>';
$("#templateBody div:last-child .templateElDist").append(option);
}
$("#templateBody div:last-child .templateElDist").val(dist);
}
}
// Execute custom template
$("#templateRun").on("click", function() {
if (customization !== 1) {return;}
var steps = $("#templateBody > div").length;
if (steps) {cells.map(function(i) {i.height = 0;});}
for (var step=1; step <= steps; step++) {
var element = $("#templateBody div:nth-child(" + step + ")");
var type = element.attr("data-type");
if (type === "Mountain") {addMountain(); continue;}
var count = $("#templateBody div:nth-child(" + step + ") .templateElCount").val();
var dist = $("#templateBody div:nth-child(" + step + ") .templateElDist").val();
if (count) {
if (count[0] !== "-" && count.includes("-")) {
var lim = count.split("-");
count = Math.floor(Math.random() * (+lim[1] - +lim[0] + 1) + +lim[0]);
} else {
count = +count; // parse string
}
}
if (type === "Hill") {addHill(count, +dist);}
if (type === "Pit") {addPit(count);}
if (type === "Range") {addRange(count);}
if (type === "Trough") {addRange(-1 * count);}
if (type === "Strait") {addStrait(count);}
if (type === "Add") {modifyHeights(dist, count, 1);}
if (type === "Multiply") {modifyHeights(dist, 0, count);}
if (type === "Smooth") {smoothHeights(count);}
}
if (steps) {mockHeightmap();}
});
// Save custom template as text file
$("#templateSave").on("click", function() {
var steps = $("#templateBody > div").length;
var stepsData = "";
for (var step=1; step <= steps; step++) {
var element = $("#templateBody div:nth-child(" + step + ")");
var type = element.attr("data-type");
var count = $("#templateBody div:nth-child(" + step + ") .templateElCount").val();
var dist = $("#templateBody div:nth-child(" + step + ") .templateElDist").val();
if (!count) {count = "0";}
if (!dist) {dist = "0";}
stepsData += type + " " + count + " " + dist + "\r\n";
}
var dataBlob = new Blob([stepsData], {type:"text/plain"});
var url = window.URL.createObjectURL(dataBlob);
var link = document.createElement("a");
link.download = "template_" + Date.now() + ".txt";
link.href = url;
link.click();
$("#templateBody").attr("data-changed", 0);
});
// Load custom template as text file
$("#templateLoad").on("click", function() {templateToLoad.click();});
$("#templateToLoad").change(function() {
var fileToLoad = this.files[0];
this.value = "";
var fileReader = new FileReader();
fileReader.onload = function(fileLoadedEvent) {
var dataLoaded = fileLoadedEvent.target.result;
var data = dataLoaded.split("\r\n");
$("#templateBody").empty();
if (data.length > 0) {
$("#templateBody").attr("data-changed", 1);
$("#templateSelect").attr("data-prev", "templateCustom").val("templateCustom");
}
for (var i=0; i < data.length; i++) {
var line = data[i].split(" ");
addStep(line[0], line[1], line[2]);
}
};
fileReader.readAsText(fileToLoad, "UTF-8");
});
// Image to Heightmap Converter dialog
function convertImage() {
$(".pressed").removeClass('pressed');
viewbox.style("cursor", "default").on(".drag", null);
var div = d3.select("#colorScheme");
if (div.selectAll("*").size() === 0) {
for (var i = 0; i <= 100; i++) {
var width = i < 20 || i > 70 ? "1px" : "3px";
if (i === 0) {width = "4px";}
var clr = color(1-i/100);
var style = "background-color: " + clr + "; width: " + width;
div.append("div").attr("data-color", i/100).attr("style", style);
}
div.selectAll("*").on("touchmove mousemove", showHeight).on("click", assignHeight);
}
$("#imageConverter").dialog({
title: "Image to Heightmap Converter",
minHeight: 30, width: 260, resizable: false,
position: {my: "right top", at: "right-10 top+10", of: "svg"}})
.on('dialogclose', function() {completeConvertion();});
}
// Load image to convert
$("#convertImageLoad").on("click", function() {imageToLoad.click();});
$("#imageToLoad").change(function() {
console.time("loadImage");
// reset style
viewbox.attr("transform", null);
grid.attr("stroke-width", .3);
// load image
var file = this.files[0];
this.value = ""; // reset input value to get triggered if the same file is uploaded
var reader = new FileReader();
var img = new Image;
// draw image
img.onload = function() {
ctx.drawImage(img, 0, 0, mapWidth, mapHeight);
heightsFromImage(+convertColors.value);
console.timeEnd("loadImage");
}
reader.onloadend = function() {img.src = reader.result;}
reader.readAsDataURL(file);
});
function heightsFromImage(count) {
var imageData = ctx.getImageData(0, 0, mapWidth, mapHeight);
var data = imageData.data;
$("#landmass > path, .color-div").remove();
$("#landmass, #colorsUnassigned").fadeIn();
$("#colorsAssigned").fadeOut();
var colors = [], palette = [];
points.map(function(i) {
var x = Math.round(i[0]), y = Math.round(i[1]);
if (y == mapHeight) {y--;}
if (x == mapWidth) {x--;}
var p = (x + y * mapWidth) * 4;
var r = data[p], g = data[p + 1], b = data[p + 2];
colors.push([r, g, b]);
});
var cmap = MMCQ.quantize(colors, count);
polygons.map(function(i, d) {
cells[d].height = undefined;
var nearest = cmap.nearest(colors[d]);
var rgb = "rgb(" + nearest[0] + ", " + nearest[1] + ", " + nearest[2] + ")";
var hex = toHEX(rgb);
if (palette.indexOf(hex) === -1) {palette.push(hex);}
landmass.append("path").attr("d", "M" + i.join("L") + "Z").attr("data-i", d).attr("fill", hex).attr("stroke", hex);
});
landmass.selectAll("path").on("click", landmassClicked);
palette.sort(function(a, b) {return d3.lab(a).b - d3.lab(b).b;}).map(function(i) {
$("#colorsUnassigned").append('<div class="color-div" id="' + i.substr(1) + '" style="background-color: ' + i + ';"/>');
});
$(".color-div").click(selectColor);
}
function landmassClicked() {
var color = d3.select(this).attr("fill");
$("#"+color.slice(1)).click();
}
function selectColor() {
landmass.selectAll(".selectedCell").classed("selectedCell", 0);
var el = d3.select(this);
if (el.classed("selectedColor")) {
el.classed("selectedColor", 0);
} else {
$(".selectedColor").removeClass("selectedColor");
el.classed("selectedColor", 1);
$("#colorScheme .hoveredColor").removeClass("hoveredColor");
$("#colorsSelectValue").text(0);
if (el.attr("data-height")) {
var height = el.attr("data-height");
$("#colorScheme div[data-color='" + height + "']").addClass("hoveredColor");
$("#colorsSelectValue").text(Math.round(height * 100));
}
var color = "#" + d3.select(this).attr("id");
landmass.selectAll("path").classed("selectedCell", 0);
landmass.selectAll("path[fill='" + color + "']").classed("selectedCell", 1);
}
}
function showHeight() {
var el = d3.select(this);
var height = Math.round(el.attr("data-color") * 100);
$("#colorsSelectValue").text(height);
$("#colorScheme .hoveredColor").removeClass("hoveredColor");
el.classed("hoveredColor", 1);
}
function assignHeight() {
var sel = $(".selectedColor")[0];
var height = +d3.select(this).attr("data-color");
var rgb = color(1-height);
var hex = toHEX(rgb);
sel.style.backgroundColor = rgb;
sel.setAttribute("data-height", height);
var cur = "#" + sel.id;
sel.id = hex.substr(1);
landmass.selectAll(".selectedCell").each(function() {
d3.select(this).attr("fill", hex).attr("stroke", hex);
var i = +d3.select(this).attr("data-i");
cells[i].height = height;
});
var parent = sel.parentNode;
if (parent.id === "colorsUnassigned") {
colorsAssigned.appendChild(sel);
$("#colorsAssigned").fadeIn();
if ($("#colorsUnassigned .color-div").length < 1) {$("#colorsUnassigned").fadeOut();}
}
if ($("#colorsAssigned .color-div").length > 1) {sortAssignedColors();}
}
// sort colors based on assigned height
function sortAssignedColors() {
var data = [];
var colors = d3.select("#colorsAssigned").selectAll(".color-div");
colors.each(function(d) {
var id = d3.select(this).attr("id");
var height = +d3.select(this).attr("data-height");
data.push({id, height});
});
data.sort(function(a, b) {return a.height - b.height}).map(function(i) {
$("#colorsAssigned").append($("#"+i.id));
});
}
// auto assign color based on luminosity or hue
function autoAssing(type) {
var imageData = ctx.getImageData(0, 0, mapWidth, mapHeight);
var data = imageData.data;
$("#landmass > path, .color-div").remove();
$("#colorsAssigned").fadeIn();
$("#colorsUnassigned").fadeOut();
var heights = [];
polygons.map(function(i, d) {
var x = Math.round(i.data[0]), y = Math.round(i.data[1]);
if (y == mapHeight) {y--;}
if (x == mapWidth) {x--;}
var p = (x + y * mapWidth) * 4;
var r = data[p], g = data[p + 1], b = data[p + 2];
var lab = d3.lab("rgb(" + r + ", " + g + ", " + b + ")");
if (type === "hue") {
var normalized = roundN(normalize(lab.b + lab.a / 2, -50, 200), 2);
} else {
var normalized = roundN(normalize(lab.l, 0, 100), 2);
}
heights.push(normalized);
var rgb = color(1 - normalized);
var hex = toHEX(rgb);
cells[d].height = normalized;
landmass.append("path").attr("d", "M" + i.join("L") + "Z").attr("data-i", d).attr("fill", hex).attr("stroke", hex);
});
heights.sort(function(a, b) {return a - b;});
var unique = [...new Set(heights)];
unique.map(function(i) {
var rgb = color(1 - i);
var hex = toHEX(rgb);
$("#colorsAssigned").append('<div class="color-div" id="' + hex.substr(1) + '" data-height="' + i + '" style="background-color: ' + hex + ';"/>');
});
$(".color-div").click(selectColor);
}
function normalize(val, min, max) {
var normalized = (val - min) / (max - min);
if (normalized < 0) {normalized = 0;}
if (normalized > 1) {normalized = 1;}
return normalized;
}
function completeConvertion() {
mockHeightmap();
canvas.style.opacity = convertOverlay.value = convertOverlayValue.innerHTML = 0;
$("#imageConverter").dialog('close');
}
// Clear the map
function undraw() {
svg.selectAll("path, circle, text").remove();
cells = [], land = [], riversData = [], island = 0, manors = [], queue = [];
history = [], historyStage = -1; redo.disabled = true; undo.disabled = true; // clear history
}
// Enter Heightmap Customization mode
function customizeHeightmap() {
customization = 1;
svg.transition().duration(1000).call(zoom.transform, d3.zoomIdentity);
$("#customizationMenu").fadeIn("slow");
viewbox.style("cursor", "crosshair").call(drag);
landmassCounter.innerHTML = "0";
$('#grid').fadeIn();
$('#toggleGrid').removeClass("buttonoff");
if ($("#labelEditor").is(":visible")) {$("#labelEditor").dialog('close');}
if ($("#riverEditor").is(":visible")) {$("#riverEditor").dialog('close');}
}
// Remove all customization related styles, reset values
function exitCustomization() {
customization = 0;
canvas.style.opacity = 0;
$("#customizationMenu").fadeOut("slow");
$("#getMap").attr("disabled", true).addClass("buttonoff");
$("#landmass").empty();
$('#grid').empty().fadeOut();
$('#toggleGrid').addClass("buttonoff");
viewbox.style("cursor", "default").on(".drag", null);
if (!$("#toggleHeight").hasClass("buttonoff")) {toggleHeight();}
if ($("#imageConverter").is(":visible")) {$("#imageConverter").dialog('close');}
if ($("#brushesPanel").is(":visible")) {$("#brushesPanel").dialog('close');}
if ($("#templateEditor").is(":visible")) {$("#templateEditor").dialog('close');}
history = [];
historyStage = -1;
}
// Options handlers
$("input, select").on("input change", function() {
var id = this.id;
if (id === "styleElementSelect") {
var sel = this.value;
var el = viewbox.select("#"+sel);
$("#styleInputs div").hide();
if (sel === "rivers" || sel === "oceanBase" || sel === "lakes" || sel === "landmass" || sel === "burgs") {
$("#styleFill").css("display", "inline-block");
styleFillInput.value = styleFillOutput.value = el.attr("fill");
}
if (sel === "roads" || sel === "trails" || sel === "searoutes" || sel === "lakes" || sel === "stateBorders" || sel === "neutralBorders" || sel === "grid" || sel === "coastline") {
$("#styleStroke").css("display", "inline-block");
styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke");
$("#styleStrokeWidth").css("display", "block");
var width = el.attr("stroke-width") || "";
styleStrokeWidthInput.value = styleStrokeWidthOutput.value = width;
}
if (sel === "roads" || sel === "trails" || sel === "searoutes" || sel === "stateBorders" || sel === "neutralBorders") {
$("#styleStrokeDasharray, #styleStrokeLinecap").css("display", "block");
styleStrokeDasharrayInput.value = el.attr("stroke-dasharray") || "";
styleStrokeLinecapInput.value = el.attr("stroke-linecap") || "inherit";
}
if (sel === "regions") {
$("#styleMultiple").css("display", "inline-block");
$("#styleMultiple input").remove();
for (var r = 0; r < capitalsCount; r++) {
var color = regions.select(".region"+r).attr("fill");
$("#styleMultiple").append('<input type="color" id="regionColor' + r + '" value="' + color + '"/>');
}
$("#styleMultiple input").on("input", function() {
var id = this.id;
var r = +id.replace("regionColor", "");
regions.selectAll(".region"+r).attr("fill", this.value);
});
}
if (sel === "terrs") {$("#styleScheme").css("display", "block");}
if (sel === "heightmap") {$("#styleScheme").css("display", "block");}
if (sel === "cults") {
$("#styleMultiple").css("display", "inline-block");
$("#styleMultiple input").remove();
var colors = [];
cults.selectAll("path").each(function(d) {
var fill = d3.select(this).attr("fill");
if (colors.indexOf(fill) === -1) {colors.push(fill);}
});
for (var c = 0; c < colors.length; c++) {
$("#styleMultiple").append('<input type="color" id="' + colors[c].substr(1) + '" value="' + colors[c] + '"/>');
}
$("#styleMultiple input").on("input", function() {
var oldColor = "#" + d3.select(this).attr("id");
var newColor = this.value;
cults.selectAll("path").each(function() {
var fill = d3.select(this).attr("fill");
if (oldColor === fill) {d3.select(this).attr("fill", newColor).attr("stroke", newColor);}
});
$(this).attr("id", newColor.substr(1));
});
}
if (sel === "labels") {
$("#styleFill, #styleFontSize").css("display", "inline-block");
styleFillInput.value = styleFillOutput.value = el.select("g").attr("fill");
}
if (sel === "burgs") {
$("#styleSize").css("display", "block");
$("#styleStroke").css("display", "inline-block");
styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke");
}
// opacity
$("#styleOpacity, #styleFilter").css("display", "block");
var opacity = el.attr("opacity") || 1;
styleOpacityInput.value = styleOpacityOutput.value = opacity;
// filter
if (sel == "oceanBase") {el = oceanLayers;}
styleFilterInput.value = el.attr("filter") || "";
return;
}
if (id === "styleFillInput") {
styleFillOutput.value = this.value;
var el = viewbox.select("#"+styleElementSelect.value);
if (styleElementSelect.value !== "labels") {
el.attr('fill', this.value);
} else {
el.selectAll("g").attr('fill', this.value);
}
return;
}
if (id === "styleStrokeInput") {
styleStrokeOutput.value = this.value;
var el = viewbox.select("#"+styleElementSelect.value);
el.attr('stroke', this.value);
return;
}
if (id === "styleStrokeWidthInput") {
styleStrokeWidthOutput.value = this.value;
var sel = styleElementSelect.value;
viewbox.select("#"+sel).attr('stroke-width', +this.value);
return;
}
if (id === "styleStrokeDasharrayInput") {
var sel = styleElementSelect.value;
viewbox.select("#"+sel).attr('stroke-dasharray', this.value);
return;
}
if (id === "styleStrokeLinecapInput") {
var sel = styleElementSelect.value;
viewbox.select("#"+sel).attr('stroke-linecap', this.value);
return;
}
if (id === "styleOpacityInput") {
styleOpacityOutput.value = this.value;
var sel = styleElementSelect.value;
viewbox.select("#"+sel).attr('opacity', this.value);
return;
}
if (id === "styleFilterInput") {
var sel = styleElementSelect.value;
if (sel == "oceanBase") {sel = "oceanLayers";}
var el = viewbox.select("#"+sel);
el.attr('filter', this.value);
return;
}
if (id === "styleSchemeInput") {
terrs.selectAll("path").remove();
toggleHeight();
return;
}
if (id === "sizeInput") {graphSize = sizeOutput.value = this.value;}
if (id === "randomizeInput") {randomizeOutput.innerHTML = +this.value ? "✓" : "✕";}
if (id === "manorsInput") {
if (randomizeInput.value === "1") {
randomizeInput.value = 0;
randomizeOutput.innerHTML = "✕";
}
manorsCount = manorsOutput.value = this.value;
}
if (id === "regionsInput") {
if (randomizeInput.value === "1") {
randomizeInput.value = 0;
randomizeOutput.innerHTML = "✕";
}
capitalsCount = regionsOutput.value = this.value;
var size = Math.round(6 - capitalsCount / 20);
if (size < 3) {size = 3;}
capitals.attr("data-size", size);
size = Math.round(18 - capitalsCount / 6);
if (size < 4) {size = 4;}
countries.attr("data-size", size);
}
if (id === "powerInput") {power = powerOutput.value = this.value;}
if (id === "neutralInput") {neutral = neutralOutput.value = this.value;}
if (id === "swampinessInput") {swampiness = swampinessOutput.value = this.value;}
if (id === "sharpnessInput") {sharpness = sharpnessOutput.value = this.value;}
if (id === "precInput") {
precipitation = precOutput.value = +precInput.value;
if (randomizeInput.value === "1") {
randomizeInput.value = 0;
randomizeOutput.innerHTML = "✕";
}
}
if (id === "convertOverlay") {canvas.style.opacity = convertOverlayValue.innerHTML = +this.value;}
});
$("#rescaler").change(function() {
var change = roundN((+this.value - 5) / 10, 2);
modifyHeights("all", change, 1);
mockHeightmap();
rescaler.value = 5;
});
$("#layoutPreset").on("change", function() {
var preset = this.value;
$("#mapLayers li").not("#toggleOcean, #toggleLandmass").addClass("buttonoff");
$("#toggleOcean, #toggleLandmass").removeClass("buttonoff");
$("#oceanPattern, #landmass").fadeIn();
$("#rivers, #terrain, #borders, #regions, #burgs, #labels, #routes, #grid").fadeOut();
cults.selectAll("path").remove();
terrs.selectAll("path").remove();
if (preset === "layoutPolitical") {
toggleRivers.click();
toggleRelief.click();
toggleBorders.click();
toggleCountries.click();
toggleIcons.click();
toggleLabels.click();
toggleRoutes.click();
}
if (preset === "layoutCultural") {
toggleRivers.click();
toggleRelief.click();
toggleBorders.click();
$("#toggleCultures").click();
toggleIcons.click();
toggleLabels.click();
}
if (preset === "layoutEconomical") {
toggleRivers.click();
toggleRelief.click();
toggleBorders.click();
toggleIcons.click();
toggleLabels.click();
toggleRoutes.click();
}
if (preset === "layoutHeightmap") {
$("#toggleHeight").click();
toggleRivers.click();
}
});
// UI Button handlers
$(".tab > button").on("click", function() {
$(".tabcontent").hide();
$(".tab > button").removeClass("active");
$(this).addClass("active");
var id = this.id;
if (id === "layoutTab") {$("#layoutContent").show();}
if (id === "styleTab") {$("#styleContent").show();}
if (id === "optionsTab") {$("#optionsContent").show();}
if (id === "customizeTab") {$("#customizeContent").show();}
});
}
@sbryfcz
Copy link

sbryfcz commented Mar 22, 2017

Since you asked for comments (as a first timer), I'd suggest looking into ways to format your JS for easier readability. Maybe look at jslint or jsbeautify. Sorry but I'd love to read the code but its difficult to read with the current spacing/formatting.

Very cool work! Nice job.

@Azgaar
Copy link
Author

Azgaar commented Mar 23, 2017

Hi @sbryfcz.
Thank you! Sure, I will do some formatting. I understand the code is unreadable and not neat.

P,S. Done, code a bit formatted now. I've also added comments on how it works. Please go ahead with such a good suggestions!

@ikarth
Copy link

ikarth commented Mar 23, 2017

Suggestion for feature expansion: since you have talked about wanting to expand this into a dynasty simulator, you might want to look at using something like Tracery for text generation (which I'm guessing you're planning to do quite a lot of). It's already a Javascript library, so you can just import it and it'll simplify name generation and things like that.

@Azgaar
Copy link
Author

Azgaar commented Mar 23, 2017

Hi @ikarth! Thank you, I will think regarding using this library.

@XCJT
Copy link

XCJT commented Jun 27, 2017

Awesome work!

To change it to d3 pan/zoom you just need to do the following:

  // Fantasy Map Generator main script
  var svg = d3.select("svg"),
    g = svg.append("g"),
    terrs = g.append("g").attr("class", "terrs").on("touchmove mousemove", moved),
    areas = g.append("g").attr("class", "areas"),
    borders = g.append("g").attr("class", "borders"),
    rivers = g.append("g").attr("class", "rivers"),
    coastline = g.append("g").attr("class", "coastline"),
    terrain = g.append("g").attr("class", "terrain"), 
    names = g.append("g").attr("class", "names"),
    burgs = g.append("g").attr("class", "burgs");
  
  svg.call(d3.zoom()
    .scaleExtent([1 / 2, 4])
    .on("zoom", zoomed));
    
  console.log(g.attr("width"))
    
  function zoomed() {
    g.attr("transform", d3.event.transform);
  }
  
  generate(); // genarate map on load

@Saint-Ajora
Copy link

Saint-Ajora commented Jul 4, 2017

Firstly let me say this is one of the greatest things I have ever seen; I have been looking for this for years now. Secondly, I have literally zero experience in coding of any kind but I would like to know if there is a way to either save the image generated by your generator or a way to get the generator to make the same map again and again. For now I have been taking screenshots and matching up the edges in Paint.net (with some success) . I have tried clicking on the "Get Map" button and that seems to just change the island either black or white and draw lines (or show lines) from each manor/town/city.

@Azgaar
Copy link
Author

Azgaar commented Jul 18, 2017

Hi @XCJT. Thank you, zooming is already on D3, just not added to this old version. There are some new cool features developed but not yet deployed, see my blog and jsfiddle. I want to create a stable version and update this page or move to another host as blocks/gist are not really good for that kind of demo. Could you suggest a good platform for a working generator?

@Azgaar
Copy link
Author

Azgaar commented Jul 18, 2017

Hi @Saint-Ajora. Thank you for the feedback! "Get Map" is to complete the custom map, not to download it, sorry for the misleading labeling. I have developed download function (in svg), just not deployed it to this demo version as working on rivers, biomes and UI. I will try to update the page by the end of this week.

@Azgaar
Copy link
Author

Azgaar commented Jul 22, 2017

Hi All,
I've added download button ("save in SVG") and changed zooming to D3 built-in function.
Hope you will enjoy the update.

@dhbahr
Copy link

dhbahr commented Jul 22, 2017

Hi there @Azgaar, let me congratulate you on a really nice job, I discovered this today and I'm already looking forward to the new features.
1 issue though: when I download the image as SVG the Toogle Highmap, Area and Flux stop working, so I can only download 1 version of the map..
Best

@Azgaar
Copy link
Author

Azgaar commented Jul 23, 2017

Hi @dhbahr. Thank you for the bug report! I'll try to fix it. Looks weird.

@Azgaar
Copy link
Author

Azgaar commented Jul 23, 2017

Hi @dhbahr. The issue is fixed now. The root cause is that d3-save-svg library adds computed style attribute to all the elements. So I have to removed style after downloading. Not ideal, but I don't want to re-write d3-save-svg library, so it's OK as still works pretty fast. Please re-test :)

@Azgaar
Copy link
Author

Azgaar commented Aug 24, 2017

New changes are deployed.

@Azgaar
Copy link
Author

Azgaar commented Oct 15, 2017

Hi Guys. Generator is updated and now is more usable, even the project is still in progress and this is a demo version. Please use the main project GitHub page for bug reports or suggestions: https://github.com/Azgaar/Fantasy-Map-Generator/issues

And please let me know if you need manual on how to use the Generator (there are quite a lot of unobvious options / possibilities).

@Azgaar
Copy link
Author

Azgaar commented Mar 29, 2018

The Generator demo is moved to https://azgaar.github.io/Fantasy-Map-Generator

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment