2023-10-20 13:31:23 +02:00
"use strict" ;
2023-10-20 14:03:45 +02:00
2023-10-21 14:19:42 +02:00
// Default values for the min. and max. Y of biomes
2023-10-20 14:39:49 +02:00
const MIN _Y _DEFAULT = - 31000
const MAX _Y _DEFAULT = 31000
2023-10-21 14:19:42 +02:00
// Draw a grid line every GRID_STEP units
const GRID _STEP = 10
2023-10-21 19:26:24 +02:00
// Distance from the point's center in which a point can
// be selected by clicking on it
const POINT _SELECT _DISTANCE = 25
2023-10-23 18:09:23 +02:00
// Distance a point has to be dragged after a single click
// before drag-n-drop starts.
// Normally, the user can drag-n-drop a point
// when clickong and dragging it *after* it has already been
// selected; there's no distance requirement.
// But it is also possible to select the point (by clicking
// on it) and dragging it with the same mouse click. But for
// this method, the mouse had to move the distance below
// from the point coordinates (in pixels) before drag-n-drop
// activates.
const ONE _CLICK _DRAG _DROP _DISTANCE = 30
2023-10-23 08:48:00 +02:00
// Name to display if empty
2023-10-23 03:10:59 +02:00
const FALLBACK _NAME = "(no name)"
2023-10-21 02:18:29 +02:00
// Symbol for storing the biome ID in site objects
// for the Voronoi script
const biomeIDSymbol = Symbol ( "Biome ID" ) ;
2023-10-23 08:48:00 +02:00
// Colors (mostly self-explanatory)
const POINT _COLOR = "#913636" ;
const POINT _COLOR _SELECTED = "#e19696" ;
const EDGE _COLOR = "#0f2c2e" ;
const GRID _COLOR = "#00000040" ;
const AXIS _COLOR = "#000000" ;
// Color to be used when the diagram is cleared
const CLEAR _COLOR = "#ecddba" ;
// list of possible cell colors
const CELL _COLORS = [
2023-10-20 13:31:23 +02:00
"#64988e" ,
"#3d7085" ,
"#345644" ,
"#6b7f5c" ,
2023-10-21 12:08:24 +02:00
"#868750" ,
"#a7822c" ,
"#a06e38" ,
2023-10-20 13:31:23 +02:00
"#ad5f52" ,
"#692f11" ,
"#89542f" ,
"#796e63" ,
"#a17d5e" ,
2023-10-21 12:08:24 +02:00
"#5a3f20" ,
"#836299" ,
2023-10-20 13:31:23 +02:00
] ;
2023-10-23 08:48:00 +02:00
const CELL _COLOR _NEUTRAL = "#888888" ;
/* Status variables for the diagram calculations */
2023-10-20 13:31:23 +02:00
2023-10-21 17:12:43 +02:00
// Min. and max. mathematically possible values for heat and humidity
2023-10-21 17:55:32 +02:00
let limit _heat _min , limit _heat _max ;
let limit _humidity _min , limit _humidity _max ;
2023-10-21 17:12:43 +02:00
// Draw area. Slightly larger than the value area to avoid
// ugly edge rendering problems
const DRAW _OFFSET = 10
2023-10-21 17:55:32 +02:00
let draw _heat _min , draw _heat _max ;
let draw _humidity _min , draw _humidity _max ;
2023-10-21 17:12:43 +02:00
// The point that is considered the middle of heat/humidity;
// mathematically this value is the most probable.
2023-10-21 17:55:32 +02:00
let midpoint _heat ;
let midpoint _humidity ;
2023-10-21 17:12:43 +02:00
// Biome noise settings
const NOISE _OFFSET _DEFAULT = 50 ;
const NOISE _SCALE _DEFAULT = 50 ;
const NOISE _PERSISTENCE _DEFAULT = 0.5 ;
const NOISE _OCTAVES _DEFAULT = 3 ;
const NOISE _ABSVALUE _DEFAULT = false ;
2023-10-23 08:48:00 +02:00
// Current noise values
2023-10-21 17:55:32 +02:00
let noises = {
heat : {
offset : NOISE _OFFSET _DEFAULT ,
scale : NOISE _SCALE _DEFAULT ,
octaves : NOISE _OCTAVES _DEFAULT ,
persistence : NOISE _PERSISTENCE _DEFAULT ,
absvalue : NOISE _ABSVALUE _DEFAULT ,
} ,
humidity : {
offset : NOISE _OFFSET _DEFAULT ,
scale : NOISE _SCALE _DEFAULT ,
octaves : NOISE _OCTAVES _DEFAULT ,
persistence : NOISE _PERSISTENCE _DEFAULT ,
absvalue : NOISE _ABSVALUE _DEFAULT ,
} ,
} ;
function updateAreaVarsFor ( noiseType ) {
let noise = noises [ noiseType ] ;
let is _absolute = noise . absvalue === true
2023-10-21 17:12:43 +02:00
// Calculate min. and max. possible values
// Octaves
let [ o _min , o _max ] = [ 0 , 0 ]
2023-10-21 17:55:32 +02:00
for ( let o = 1 ; o <= noise . octaves ; o ++ ) {
2023-10-21 17:12:43 +02:00
let exp = o - 1
// Calculate the two possible extreme values
// with the octave value being either at 1 or -1.
2023-10-21 17:55:32 +02:00
let limit1 = ( 1 * noise . persistence * * exp )
2023-10-21 17:12:43 +02:00
let limit2
if ( ! is _absolute ) {
2023-10-21 17:55:32 +02:00
limit2 = ( - 1 * noise . persistence * * exp )
2023-10-21 17:12:43 +02:00
} else {
// If absvalue is set, one of the
// limits is always 0 because we
// can't get lower.
limit2 = 0
}
// To add to the maximum, pick the higher value
if ( limit1 > limit2 ) {
o _max = o _max + limit1
} else {
o _max = o _max + limit2
}
// To add to the minimum, pick the LOWER value
if ( limit1 > limit2 ) {
o _min = o _min + limit2
} else {
o _min = o _min + limit1
}
}
// Add offset and scale to min/max value (final step)
2023-10-21 17:55:32 +02:00
let min _value = noise . offset + noise . scale * o _min
let max _value = noise . offset + noise . scale * o _max
2023-10-21 17:12:43 +02:00
// Bring the 2 values in the correct order
// (min_value might be bigger for negative scale)
if ( min _value > max _value ) {
[ min _value , max _value ] = [ max _value , min _value ]
}
// Update globals
2023-10-21 17:55:32 +02:00
let limit _min = min _value ;
let limit _max = max _value ;
let draw _min = limit _min - DRAW _OFFSET
let draw _max = limit _max + DRAW _OFFSET
let midpoint = min _value + ( max _value - min _value ) / 2
if ( noiseType === "heat" ) {
limit _heat _min = limit _min ;
limit _heat _max = limit _max ;
draw _heat _min = draw _min ;
draw _heat _max = draw _max ;
midpoint _heat = midpoint ;
} else if ( noiseType === "humidity" ) {
limit _humidity _min = limit _min ;
limit _humidity _max = limit _max ;
draw _humidity _min = draw _min ;
draw _humidity _max = draw _max ;
midpoint _humidity = midpoint ;
} else {
console . log ( "ERROR! updateAreaVars() called with wrong noise_type!" )
}
}
function updateAreaVars ( ) {
updateAreaVarsFor ( "heat" ) ;
updateAreaVarsFor ( "humidity" ) ;
2023-10-21 17:12:43 +02:00
// Update element
2023-10-21 23:04:38 +02:00
rangeDisplay . innerHTML = "heat range: <span class='statHeat'>[" + ( + limit _heat _min ) + ", " + ( + limit _heat _max ) + "]</span>; " +
"humidity range: <span class='statHumidity'>[" + ( + limit _humidity _min ) + ", " + ( + limit _humidity _max ) + "]</span>" ;
2023-10-21 17:12:43 +02:00
}
updateAreaVars ( ) ;
2023-10-21 13:59:19 +02:00
// If true, point names are shown in diagram
let showNames = true ;
// If true, points are shown in diagram
let showPoints = true ;
// If true, cells are colorized in diagram
let showCellColors = true ;
// If true, show the grid in the diagram
let showGrid = true ;
2023-10-23 02:47:00 +02:00
// If true, show the heat/humidity axes
let showAxes = false ;
2023-10-21 13:59:19 +02:00
2023-10-21 17:12:43 +02:00
// Set to true if the draw canvas currently shows an error message
let drawError = false ;
2023-10-23 08:48:00 +02:00
// The last ID assigned to a biome (0 = none assigned yet).
2023-10-21 13:59:19 +02:00
let lastBiomeID = 0 ;
let biomePoints = [ ] ;
2023-10-23 08:48:00 +02:00
// Add a biome to the biome list
2023-10-21 13:59:19 +02:00
function addBiome ( biomeDef ) {
biomeDef . id = lastBiomeID ;
biomePoints . push ( biomeDef ) ;
2023-10-23 08:48:00 +02:00
// The biome ID is just a simple ascending number
2023-10-21 13:59:19 +02:00
lastBiomeID ++ ;
}
// Add a default biome at the midpoint
2023-10-23 03:10:59 +02:00
addBiome ( { name : "default" , heat : midpoint _heat , humidity : midpoint _humidity , min _y : MIN _Y _DEFAULT , max _y : MAX _Y _DEFAULT } )
2023-10-21 13:59:19 +02:00
2023-10-23 08:48:00 +02:00
// Get the Y vale of the widget to set the Y altitude
2023-10-21 13:59:19 +02:00
function getViewY ( ) {
if ( ! inputViewY ) {
return 0 ;
}
return inputViewY . value ;
}
2023-10-23 08:48:00 +02:00
// Returns the biome point by its given ID
2023-10-20 16:31:11 +02:00
// or null if it couldn't be found
function getBiomeByID ( id ) {
for ( let b = 0 ; b < biomePoints . length ; b ++ ) {
let biome = biomePoints [ b ] ;
if ( biome . id === id ) {
return biome ;
}
2023-10-20 13:31:23 +02:00
}
2023-10-20 16:31:11 +02:00
return null ;
}
2023-10-23 03:10:59 +02:00
// Returns a new biome name for displaying
function generateBiomeName ( id ) {
return + id ;
2023-10-20 13:31:23 +02:00
}
2023-10-23 08:48:00 +02:00
// Converts a biome point to a point object to be passed to the Voronoi API
2023-10-20 13:31:23 +02:00
function biomePointToVoronoiPoint ( point ) {
2023-10-21 02:18:29 +02:00
// Apart from x and y, we also add the biome ID to the Voronoi point
// so we can re-identify to which biome it belongs to when the
// object is returned from the Voronoi script output.
let newPoint = { x : point . heat , y : point . humidity , [ biomeIDSymbol ] : point . id }
2023-10-20 13:31:23 +02:00
return newPoint ;
}
2023-10-23 08:48:00 +02:00
// Converts a point object for the Voronoi API to a biome point
2023-10-20 13:31:23 +02:00
function voronoiPointToBiomePoint ( point ) {
2023-10-21 02:18:29 +02:00
let newPoint = { heat : point . x , humidity : point . y , id : point [ biomeIDSymbol ] }
2023-10-20 13:31:23 +02:00
return newPoint ;
}
2023-10-23 08:48:00 +02:00
/***** Draw functions *****/
/* Note: All of these need a draw context. */
/* Put the name of the given point on the draw context */
2023-10-21 11:35:22 +02:00
function putPointName ( context , point ) {
2023-10-22 12:40:01 +02:00
let he = point . heat
let hu = point . humidity
if ( he < limit _heat _min || he > limit _heat _max || hu < limit _humidity _min || hu > limit _humidity _max ) {
2023-10-21 11:35:22 +02:00
return ;
}
2023-10-22 12:40:01 +02:00
let [ x , y ] = biomeCoordsToCanvasPixelCoords ( he , hu ) ;
let w = voronoiCanvas . width ;
let h = voronoiCanvas . height ;
if ( x > w / 2 ) {
2023-10-21 11:35:22 +02:00
context . textAlign = "right" ;
2023-10-22 12:40:01 +02:00
x = x - 5 ;
2023-10-21 11:35:22 +02:00
} else {
context . textAlign = "left" ;
2023-10-22 12:40:01 +02:00
x = x + 5 ;
2023-10-21 11:35:22 +02:00
}
2023-10-22 12:40:01 +02:00
if ( y < h / 2 ) {
2023-10-21 12:08:24 +02:00
context . textBaseline = "top" ;
} else {
context . textBaseline = "alphabetic" ;
}
2023-10-22 12:40:01 +02:00
context . font = "120% sans-serif"
2023-10-23 03:10:59 +02:00
let displayName = point . name ;
if ( displayName === "" ) {
displayName = FALLBACK _NAME ;
}
context . fillText ( displayName , x , y ) ;
2023-10-21 11:35:22 +02:00
}
2023-10-23 08:48:00 +02:00
/* Put the given point on the draw context */
2023-10-20 13:31:23 +02:00
function putPoint ( context , point ) {
2023-10-22 12:40:01 +02:00
const ARROW _SIZE _SIDE = 7 ;
const ARROW _SIZE _CORNER = 9 ;
let he = point . heat
let hu = point . humidity
let [ x , y ] = biomeCoordsToCanvasPixelCoords ( he , hu ) ;
let w = voronoiCanvas . width ;
let h = voronoiCanvas . height ;
let [ limit _x _min , limit _y _min ] = biomeCoordsToCanvasPixelCoords ( limit _heat _min , limit _humidity _min ) ;
let [ limit _x _max , limit _y _max ] = biomeCoordsToCanvasPixelCoords ( limit _heat _max , limit _humidity _max ) ;
2023-10-20 15:23:44 +02:00
// Point is out of bounds: Draw an arrow at the border
2023-10-22 12:40:01 +02:00
if ( he < limit _heat _min || he > limit _heat _max || hu < limit _humidity _min || hu > limit _humidity _max ) {
2023-10-20 15:23:44 +02:00
context . beginPath ( ) ;
2023-10-23 02:54:45 +02:00
// bottom left corner
2023-10-22 12:40:01 +02:00
if ( he < limit _heat _min && hu < limit _humidity _min ) {
2023-10-23 02:54:45 +02:00
context . moveTo ( limit _x _min , limit _y _min ) ;
2023-10-22 12:40:01 +02:00
context . lineTo ( limit _x _min + ARROW _SIZE _CORNER , limit _y _min ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( limit _x _min , limit _y _min - ARROW _SIZE _CORNER ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
2023-10-23 02:54:45 +02:00
// bottom right corner
2023-10-22 12:40:01 +02:00
} else if ( he > limit _heat _max && hu < limit _humidity _min ) {
context . moveTo ( limit _x _max , limit _y _min ) ;
context . lineTo ( limit _x _max - ARROW _SIZE _CORNER , limit _y _min ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( limit _x _max , limit _y _min - ARROW _SIZE _CORNER ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
2023-10-23 02:54:45 +02:00
// top left corner
2023-10-22 12:40:01 +02:00
} else if ( he < limit _heat _min && hu > limit _humidity _max ) {
context . moveTo ( limit _x _min , limit _y _max ) ;
context . lineTo ( limit _x _min + ARROW _SIZE _CORNER , limit _y _max ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( limit _x _min , limit _y _max + ARROW _SIZE _CORNER ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
// top right corner
2023-10-22 12:40:01 +02:00
} else if ( he > limit _heat _max && hu > limit _humidity _max ) {
context . moveTo ( limit _x _max , limit _y _max ) ;
context . lineTo ( limit _x _max - ARROW _SIZE _CORNER , limit _y _max ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( limit _x _max , limit _y _max + ARROW _SIZE _CORNER ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
// left side
2023-10-22 12:40:01 +02:00
} else if ( he < limit _heat _min ) {
context . moveTo ( limit _x _min , y ) ;
context . lineTo ( limit _x _min + ARROW _SIZE _SIDE , y + ARROW _SIZE _SIDE ) ;
context . lineTo ( limit _x _min + ARROW _SIZE _SIDE , y - ARROW _SIZE _SIDE ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
// right side
2023-10-22 12:40:01 +02:00
} else if ( he > limit _heat _max ) {
context . moveTo ( limit _x _max , y ) ;
context . lineTo ( limit _x _max - ARROW _SIZE _SIDE , y + ARROW _SIZE _SIDE ) ;
context . lineTo ( limit _x _max - ARROW _SIZE _SIDE , y - ARROW _SIZE _SIDE ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
2023-10-23 02:54:45 +02:00
// bottom side
2023-10-22 12:40:01 +02:00
} else if ( hu < limit _humidity _min ) {
context . moveTo ( x , limit _y _min ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( x - ARROW _SIZE _SIDE , limit _y _min - ARROW _SIZE _SIDE ) ;
context . lineTo ( x + ARROW _SIZE _SIDE , limit _y _min - ARROW _SIZE _SIDE ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
2023-10-23 02:54:45 +02:00
// top side
2023-10-22 12:40:01 +02:00
} else if ( hu > limit _humidity _max ) {
context . moveTo ( x , limit _y _max ) ;
2023-10-23 02:54:45 +02:00
context . lineTo ( x - ARROW _SIZE _SIDE , limit _y _max + ARROW _SIZE _SIDE ) ;
context . lineTo ( x + ARROW _SIZE _SIDE , limit _y _max + ARROW _SIZE _SIDE ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
}
// Point is in bounds: Draw a dot
} else {
context . beginPath ( ) ;
context . moveTo ( 0 , 0 ) ;
2023-10-22 12:40:01 +02:00
context . arc ( x , y , 5 , 0 , Math . PI * 2 ) ;
2023-10-20 15:23:44 +02:00
context . closePath ( ) ;
context . fill ( ) ;
}
2023-10-20 13:31:23 +02:00
} ;
2023-10-23 08:48:00 +02:00
/* Put the grid on the draw context */
2023-10-20 13:31:23 +02:00
function putGrid ( context ) {
2023-10-22 12:40:01 +02:00
let [ limit _x _min , limit _y _min ] = biomeCoordsToCanvasPixelCoords ( limit _heat _min , limit _humidity _min ) ;
let [ limit _x _max , limit _y _max ] = biomeCoordsToCanvasPixelCoords ( limit _heat _max , limit _humidity _max ) ;
context . lineWidth = 2 ;
2023-10-23 08:48:00 +02:00
context . strokeStyle = GRID _COLOR ;
2023-10-22 12:40:01 +02:00
for ( let he = 0 ; he <= limit _heat _max ; he += GRID _STEP ) {
let [ x , _ ] = biomeCoordsToCanvasPixelCoords ( he , 0 ) ;
2023-10-20 13:31:23 +02:00
context . beginPath ( ) ;
2023-10-22 12:40:01 +02:00
context . moveTo ( x , limit _y _min ) ;
context . lineTo ( x , limit _y _max ) ;
2023-10-20 13:31:23 +02:00
context . stroke ( ) ;
}
2023-10-22 12:40:01 +02:00
for ( let he = - GRID _STEP ; he >= limit _heat _min ; he -= GRID _STEP ) {
let [ x , _ ] = biomeCoordsToCanvasPixelCoords ( he , 0 ) ;
2023-10-20 13:31:23 +02:00
context . beginPath ( ) ;
2023-10-22 12:40:01 +02:00
context . moveTo ( x , limit _y _min ) ;
context . lineTo ( x , limit _y _max ) ;
2023-10-20 13:31:23 +02:00
context . stroke ( ) ;
}
2023-10-22 12:40:01 +02:00
for ( let hu = 0 ; hu <= limit _humidity _max ; hu += GRID _STEP ) {
let [ _ , y ] = biomeCoordsToCanvasPixelCoords ( 0 , hu ) ;
2023-10-20 13:31:23 +02:00
context . beginPath ( ) ;
2023-10-22 12:40:01 +02:00
context . moveTo ( limit _x _min , y ) ;
context . lineTo ( limit _x _max , y ) ;
2023-10-20 13:31:23 +02:00
context . stroke ( ) ;
}
2023-10-22 12:40:01 +02:00
for ( let hu = - GRID _STEP ; hu >= limit _humidity _min ; hu -= GRID _STEP ) {
let [ _ , y ] = biomeCoordsToCanvasPixelCoords ( 0 , hu ) ;
2023-10-20 13:31:23 +02:00
context . beginPath ( ) ;
2023-10-22 12:40:01 +02:00
context . moveTo ( limit _x _min , y ) ;
context . lineTo ( limit _x _max , y ) ;
2023-10-20 13:31:23 +02:00
context . stroke ( ) ;
}
}
2023-10-23 08:48:00 +02:00
/* Put the labelled heat/humidity axes on the draw context */
2023-10-23 02:47:00 +02:00
function putAxes ( context ) {
const AXIS _ARROW _SIZE = 8 ;
context . lineWidth = 2 ;
2023-10-23 08:48:00 +02:00
context . strokeStyle = AXIS _COLOR ;
2023-10-23 02:47:00 +02:00
let [ x0 , y0 ] = biomeCoordsToCanvasPixelCoords ( 0 , 0 ) ;
let tick _heat = ( limit _heat _max - limit _heat _min ) * ( 100 / 175 ) ;
let tick _humidity = ( limit _humidity _max - limit _humidity _min ) * ( 100 / 175 ) ;
let [ tx , ty ] = biomeCoordsToCanvasPixelCoords ( tick _heat , tick _humidity ) ;
let w = voronoiCanvas . width ;
let h = voronoiCanvas . height ;
// horizontal axis
context . beginPath ( ) ;
context . moveTo ( 0 , y0 ) ;
context . lineTo ( w , y0 ) ;
// midpoint tick
context . moveTo ( tx , y0 ) ;
context . lineTo ( tx , y0 + AXIS _ARROW _SIZE ) ;
// arrow
context . moveTo ( w , y0 ) ;
context . lineTo ( w - AXIS _ARROW _SIZE , y0 - AXIS _ARROW _SIZE ) ;
context . moveTo ( w , y0 ) ;
context . lineTo ( w - AXIS _ARROW _SIZE , y0 + AXIS _ARROW _SIZE ) ;
context . stroke ( ) ;
context . closePath ( ) ;
// vertical axis
context . beginPath ( ) ;
context . moveTo ( x0 , 0 ) ;
context . lineTo ( x0 , h ) ;
// midpoint tick
context . moveTo ( x0 , ty ) ;
context . lineTo ( x0 - AXIS _ARROW _SIZE , ty ) ;
// arrow
context . moveTo ( x0 , 0 ) ;
context . lineTo ( x0 - AXIS _ARROW _SIZE , AXIS _ARROW _SIZE ) ;
context . moveTo ( x0 , 0 ) ;
context . lineTo ( x0 + AXIS _ARROW _SIZE , AXIS _ARROW _SIZE ) ;
context . stroke ( ) ;
context . closePath ( ) ;
// axis+midpoint labels
context . fillStyle = "black" ;
// heat label
context . textAlign = "right" ;
context . textBaseline = "top" ;
context . font = "100% sans-serif" ;
context . fillText ( "heat" , w - AXIS _ARROW _SIZE * 2 , y0 + 4 ) ;
context . textAlign = "center" ;
context . fillText ( Math . round ( tick _heat ) , tx , y0 + AXIS _ARROW _SIZE + 2 ) ;
// humidity label
context . textAlign = "right" ;
context . textBaseline = "bottom" ;
context . font = "100% sans-serif" ;
context . save ( ) ;
context . rotate ( - Math . PI / 2 ) ;
context . fillText ( "humidity" , - AXIS _ARROW _SIZE * 2 , x0 - 4 ) ;
context . restore ( ) ;
context . textAlign = "right" ;
context . textBaseline = "middle" ;
context . fillText ( Math . round ( tick _humidity ) , x0 - AXIS _ARROW _SIZE - 2 , ty ) ;
}
2023-10-20 13:31:23 +02:00
// Cache diagram object for performance boost
let cachedVoronoiDiagram = null ;
2023-10-23 08:48:00 +02:00
/ * G i v e n t h e l i s t o f b i o m e p o i n t s , r e t u r n s a V o r o n o i d i a g r a m o b j e c t
( which may have been cached ) . If recalculate is true , a recalculation is forced . * /
2023-10-20 17:22:48 +02:00
function getVoronoiDiagram ( points , recalculate ) {
2023-10-23 08:07:36 +02:00
// Calculate bounding box, defaults to heat/humidity limits ...
let vbbox = { xl : limit _heat _min , xr : limit _heat _max , yb : limit _humidity _max , yt : limit _humidity _min } ;
const BUFFER _ZONE = 1 ;
// ... unless a point is out of bounds,
// then we increase the bounding box size
for ( let p of points ) {
if ( p . heat < vbbox . xl ) {
vbbox . xl = p . heat - BUFFER _ZONE ;
} else if ( p . heat > vbbox . xr ) {
vbbox . xr = p . heat + BUFFER _ZONE ;
}
if ( p . humidity < vbbox . yt ) {
vbbox . yt = p . humidity - BUFFER _ZONE ;
} else if ( p . humidity > vbbox . yb ) {
vbbox . yb = p . humidity + BUFFER _ZONE ;
}
}
2023-10-20 13:31:23 +02:00
if ( ( cachedVoronoiDiagram === null ) || recalculate ) {
let sites = [ ]
2023-10-20 17:22:48 +02:00
for ( let p of points ) {
2023-10-20 13:31:23 +02:00
sites . push ( biomePointToVoronoiPoint ( p ) ) ;
}
let voronoi = new Voronoi ( ) ;
let diagram = null ;
if ( cachedVoronoiDiagram && recalculate ) {
diagram = cachedVoronoiDiagram ;
// This should improve performance
voronoi . recycle ( diagram ) ;
}
diagram = voronoi . compute ( sites , vbbox ) ;
cachedVoronoiDiagram = diagram ;
return diagram ;
} else {
return cachedVoronoiDiagram ;
}
}
2023-10-23 08:48:00 +02:00
/* Returns the context object required to draw on the canvas */
2023-10-20 13:31:23 +02:00
function getDrawContext ( ) {
let canvas = document . getElementById ( "voronoiCanvas" ) ;
2023-10-21 21:24:58 +02:00
if ( canvas . getContext ) {
return canvas . getContext ( "2d" ) ;
} else {
return null ;
}
2023-10-20 13:31:23 +02:00
}
2023-10-21 17:12:43 +02:00
// Clear draw area
function clear ( context ) {
if ( ! context ) {
context = getDrawContext ( ) ;
2023-10-21 21:24:58 +02:00
if ( ! context ) {
2023-10-23 18:09:23 +02:00
return false ;
2023-10-21 21:24:58 +02:00
}
2023-10-21 17:12:43 +02:00
}
2023-10-23 08:48:00 +02:00
context . fillStyle = CLEAR _COLOR ;
2023-10-22 12:40:01 +02:00
context . fillRect ( - DRAW _OFFSET , - DRAW _OFFSET , voronoiCanvas . width + DRAW _OFFSET , voronoiCanvas . height + DRAW _OFFSET ) ;
2023-10-21 21:24:58 +02:00
return true ;
2023-10-21 17:12:43 +02:00
}
2023-10-23 08:48:00 +02:00
/ * R e t u r n s a l l b i o m e p o i n t s e x c e p t t h o s e w h o s e Y l i m i t s f a l l o u t o f t h e
given y value * /
2023-10-20 22:03:15 +02:00
function getRenderedPoints ( y ) {
2023-10-20 17:22:48 +02:00
let points = [ ] ;
for ( let p = 0 ; p < biomePoints . length ; p ++ ) {
let point = biomePoints [ p ] ;
if ( y >= point . min _y && y <= point . max _y ) {
points . push ( point ) ;
}
}
2023-10-20 22:03:15 +02:00
return points ;
}
2023-10-21 17:12:43 +02:00
2023-10-23 08:48:00 +02:00
/ * G i v e n a b i o m e I D , r e t u r n s t h e m a t c h i n g H T M L e l e m e n t f r o m t h e
biome list widget or null if none * /
2023-10-21 19:26:24 +02:00
function getBiomeIDFromHTMLElement ( elem ) {
let strID = elem . id ;
if ( strID && strID . startsWith ( "biome_list_element_" ) ) {
let slice = strID . slice ( 19 ) ;
if ( slice ) {
return + slice ;
}
}
return null ;
}
2023-10-23 08:48:00 +02:00
/ * R e t u r n s b o t h t h e I D o f s e l e c t e d b i o m e a n d t h e a s s o c i a t e d H T M L e l e m e n t f r o m
the biome list widget , or null if nothing is selected .
Result is returned in the form [ id , htmlElement ] or [ null , null ] * /
2023-10-21 19:26:24 +02:00
function getSelectedBiomeIDAndElement ( ) {
if ( biomeSelector . selectedIndex === - 1 ) {
return [ null , null ] ;
}
let elem = biomeSelector . options [ biomeSelector . selectedIndex ] ;
let biomeID = getBiomeIDFromHTMLElement ( elem ) ;
if ( biomeID !== null ) {
return [ biomeID , elem ] ;
}
return [ null , null ] ;
}
2023-10-23 08:48:00 +02:00
/ * D r a w s t h e d i a g r a m o n t h e v o r o n o i C a n v a s .
Will ( re - ) calculate the Voronoi diagram if recalculate is true ;
otherwise it may re - use a previous diagram for performance reasons . * /
function draw ( recalculate ) {
2023-10-20 22:03:15 +02:00
let context = getDrawContext ( ) ;
2023-10-22 12:40:01 +02:00
let w = voronoiCanvas . width ;
let h = voronoiCanvas . height ;
2023-10-23 08:48:00 +02:00
let y = getViewY ( ) ;
2023-10-22 12:40:01 +02:00
// shorter function name (for "convert")
let conv = biomeCoordsToCanvasPixelCoords
2023-10-21 21:24:58 +02:00
if ( ! context ) {
if ( ! voronoiCanvas . hidden ) {
voronoiCanvas . hidden = true ;
2023-10-21 21:35:20 +02:00
coordinateDisplay . hidden = true ;
altitudeDisplay . hidden = true ;
2023-10-21 23:04:38 +02:00
rangeDisplay . hidden = true ;
2023-10-21 21:24:58 +02:00
configDiv . hidden = true ;
errorMessage . innerText = "ERROR: Could not get the canvas context which means this tool won't work for you. Maybe your browser does not support the HTML canvas element properly." ;
}
return false ;
}
2023-10-21 17:12:43 +02:00
clear ( context ) ;
// Render a special message if the value range is tiny
2023-10-21 17:55:32 +02:00
if ( ( limit _heat _max - limit _heat _min < 0.01 ) || ( limit _humidity _max - limit _humidity _min < 0.01 ) ) {
2023-10-21 17:12:43 +02:00
context . textAlign = "center" ;
context . fillStyle = "black" ;
context . textBaseline = "middle" ;
context . font = "200% sans-serif" ;
let msg = "Value range is too small." ;
2023-10-22 12:40:01 +02:00
context . fillText ( msg , w / 2 , h / 2 ) ;
2023-10-21 17:12:43 +02:00
drawError = true ;
2023-10-21 21:35:20 +02:00
updateAltitudeText ( ) ;
2023-10-21 21:24:58 +02:00
return true ;
2023-10-21 17:12:43 +02:00
}
2023-10-20 22:03:15 +02:00
let points = getRenderedPoints ( y ) ;
2023-10-21 17:12:43 +02:00
// Render a special message if there are no biomes
if ( points . length === 0 ) {
context . textAlign = "center" ;
context . fillStyle = "black" ;
context . textBaseline = "middle" ;
context . font = "200% sans-serif" ;
let msg ;
if ( biomePoints . length === 0 ) {
msg = "No biomes." ;
} else {
2023-10-21 23:04:38 +02:00
msg = "No biomes in this Y altitude." ;
2023-10-21 17:12:43 +02:00
}
2023-10-22 12:40:01 +02:00
context . fillText ( msg , w / 2 , h / 2 ) ;
2023-10-21 17:12:43 +02:00
drawError = true ;
2023-10-21 21:35:20 +02:00
updateAltitudeText ( ) ;
2023-10-21 21:24:58 +02:00
return true ;
2023-10-21 17:12:43 +02:00
}
drawError = false ;
2023-10-21 21:35:20 +02:00
updateAltitudeText ( ) ;
2023-10-21 17:12:43 +02:00
2023-10-20 17:22:48 +02:00
let diagram = getVoronoiDiagram ( points , recalculate ) ;
2023-10-20 16:31:11 +02:00
2023-10-23 09:30:55 +02:00
let createHalfedgesPath = function ( context , cell ) {
context . beginPath ( ) ;
for ( let h = 0 ; h < cell . halfedges . length ; h ++ ) {
let halfedge = cell . halfedges [ h ]
let start = halfedge . getStartpoint ( )
let end = halfedge . getEndpoint ( )
let [ cx1 , cy1 ] = conv ( start . x , start . y ) ;
let [ cx2 , cy2 ] = conv ( end . x , end . y ) ;
if ( h === 0 ) {
context . moveTo ( cx1 , cy1 ) ;
} else {
context . lineTo ( cx1 , cy1 ) ;
}
context . lineTo ( cx2 , cy2 ) ;
}
context . closePath ( ) ;
}
2023-10-20 16:31:11 +02:00
// Render cell colors
2023-10-21 13:59:19 +02:00
if ( showCellColors ) {
2023-10-23 08:48:00 +02:00
let colors = CELL _COLORS ;
2023-10-23 09:30:55 +02:00
/ * B e f o r e w e p a i n t t h e c e l l s ,
we paint the cell ( half ) edges in their
cell color . Painting the cells alone
leaves a tiny little space between the
cells unpainted , so there is an ugly boundary
between space . Painting the edges before
painting the cells should ensure these ugly
boundaries are gone * /
for ( let c = 0 ; c < diagram . cells . length ; c ++ ) {
let cell = diagram . cells [ c ] ;
// We use the biomeID to select a color
let biomeID = cell . site [ biomeIDSymbol ] ;
// This works because the biome ID is a number
2023-10-24 14:39:34 +02:00
let biome = getBiomeByID ( biomeID ) ;
if ( biome . color ) {
context . strokeStyle = biome . color ;
} else {
let ccol = biomeID % CELL _COLORS . length ;
context . strokeStyle = CELL _COLORS [ ccol ] ;
}
2023-10-23 09:30:55 +02:00
/ * T h e l i n e c a n b e q u i t e t h i c k b e c a u s e
most of it will be painted over by the
actual cell themselves anyway * /
context . lineWidth = 6 ;
createHalfedgesPath ( context , cell ) ;
context . stroke ( ) ;
}
// Paint the cells
2023-10-21 13:59:19 +02:00
for ( let c = 0 ; c < diagram . cells . length ; c ++ ) {
let cell = diagram . cells [ c ] ;
// We use the biomeID to select a color
let biomeID = cell . site [ biomeIDSymbol ] ;
// This works because the biome ID is a number
2023-10-24 14:39:34 +02:00
let biome = getBiomeByID ( biomeID ) ;
if ( biome . color ) {
context . fillStyle = biome . color ;
} else {
let ccol = biomeID % CELL _COLORS . length ;
context . fillStyle = CELL _COLORS [ ccol ] ;
}
2023-10-21 13:59:19 +02:00
2023-10-23 09:30:55 +02:00
createHalfedgesPath ( context , cell ) ;
2023-10-21 13:59:19 +02:00
context . fill ( ) ;
2023-10-20 13:31:23 +02:00
}
2023-10-21 13:59:19 +02:00
// If there's only 1 cell, we have to manually colorize it because
// the Voronoi script doesn't return that area in this special case.
if ( points . length === 1 && diagram . cells . length === 1 ) {
// 1 cell means the whole area is filled
context . fillStyle = colors [ points [ 0 ] . id % colors . length ] ;
2023-10-22 12:40:01 +02:00
context . fillRect ( - DRAW _OFFSET , - DRAW _OFFSET , w + DRAW _OFFSET , h + DRAW _OFFSET ) ;
2023-10-21 13:59:19 +02:00
}
} else {
// Use a "neutral" background color for the whole area if cell colors are disabled
2023-10-23 08:48:00 +02:00
context . fillStyle = CELL _COLOR _NEUTRAL ;
2023-10-22 12:40:01 +02:00
context . fillRect ( - DRAW _OFFSET , - DRAW _OFFSET , w + DRAW _OFFSET , h + DRAW _OFFSET ) ;
2023-10-20 13:31:23 +02:00
}
2023-10-20 17:24:06 +02:00
if ( points . length > 0 ) {
2023-10-21 14:19:42 +02:00
if ( showGrid ) {
2023-10-21 13:59:19 +02:00
putGrid ( context ) ;
}
2023-10-23 02:47:00 +02:00
if ( showAxes ) {
putAxes ( context ) ;
}
2023-10-20 17:24:06 +02:00
}
2023-10-20 13:31:23 +02:00
2023-10-20 16:31:11 +02:00
// Render Voronoi cell edges
2023-10-22 12:40:01 +02:00
context . lineWidth = 2.5 ;
2023-10-20 13:31:23 +02:00
for ( let e = 0 ; e < diagram . edges . length ; e ++ ) {
let edge = diagram . edges [ e ] ;
2023-10-20 15:32:14 +02:00
if ( edge . rSite === null ) {
context . strokeStyle = "transparent" ;
} else {
2023-10-23 08:48:00 +02:00
context . strokeStyle = EDGE _COLOR ;
2023-10-20 15:32:14 +02:00
}
2023-10-22 12:40:01 +02:00
let [ eax , eay ] = conv ( edge . va . x , edge . va . y ) ;
let [ ebx , eby ] = conv ( edge . vb . x , edge . vb . y ) ;
2023-10-20 13:31:23 +02:00
context . beginPath ( ) ;
2023-10-22 12:40:01 +02:00
context . moveTo ( eax , eay ) ;
context . lineTo ( ebx , eby ) ;
2023-10-20 13:31:23 +02:00
context . closePath ( ) ;
context . stroke ( ) ;
}
2023-10-21 19:26:24 +02:00
let [ selElemID , _ ] = getSelectedBiomeIDAndElement ( ) ;
2023-10-20 20:11:00 +02:00
2023-10-20 16:31:11 +02:00
// Render biome points
2023-10-21 13:59:19 +02:00
if ( showPoints ) {
for ( let point of points ) {
let pointID = point . id ;
// Highlight selected point
if ( selElemID !== null && pointID === selElemID ) {
2023-10-23 08:48:00 +02:00
context . fillStyle = POINT _COLOR _SELECTED ;
2023-10-21 13:59:19 +02:00
} else {
2023-10-23 08:48:00 +02:00
context . fillStyle = POINT _COLOR ;
2023-10-21 13:59:19 +02:00
}
putPoint ( context , point ) ;
2023-10-20 13:31:23 +02:00
}
2023-10-21 13:59:19 +02:00
}
if ( showNames ) {
// Render biome point names
for ( let point of points ) {
let pointID = point . id ;
// Highlight selected point
if ( selElemID !== null && pointID === selElemID ) {
context . fillStyle = "#FF8888FF" ;
} else {
context . fillStyle = "#FFFFFFAA" ;
}
putPointName ( context , point ) ;
2023-10-21 12:08:24 +02:00
}
2023-10-20 13:31:23 +02:00
}
2023-10-21 21:24:58 +02:00
return true ;
2023-10-20 13:31:23 +02:00
}
2023-10-23 08:48:00 +02:00
/ * C l e a r s t h e b i o m e l i s t w i d g e t a n d ( r e - a d d s ) t h e l i s t e l e m e n t s
for the biomes from scratch . * /
function repopulateBiomeSelector ( ) {
2023-10-20 13:31:23 +02:00
biomeSelector . innerHTML = "" ;
for ( let b = 0 ; b < biomePoints . length ; b ++ ) {
let num = b + 1 ;
let newElem = document . createElement ( "option" ) ;
newElem . value = num ;
2023-10-20 20:11:00 +02:00
newElem . id = "biome_list_element_" + biomePoints [ b ] . id ;
2023-10-23 03:10:59 +02:00
let displayName = biomePoints [ b ] . name ;
if ( displayName === "" ) {
displayName = FALLBACK _NAME ;
}
let newElemText = document . createTextNode ( displayName ) ;
2023-10-20 13:31:23 +02:00
newElem . append ( newElemText ) ;
biomeSelector . append ( newElem ) ;
}
}
2023-10-23 08:48:00 +02:00
/ * U p d a t e t h e s t a t u s ( l i k e d i s a b l e d o r v a l u e ) o f a l l w i d g e t s t h a t
affect the diagram , based on the internal data * /
2023-10-20 14:03:45 +02:00
function updateWidgetStates ( ) {
if ( biomePoints . length === 0 || biomeSelector . selectedIndex === - 1 ) {
removeBiomeButton . disabled = "disabled" ;
inputHeat . disabled = "disabled" ;
inputHumidity . disabled = "disabled" ;
2023-10-20 16:12:20 +02:00
inputMinY . disabled = "disabled" ;
inputMaxY . disabled = "disabled" ;
2023-10-23 03:10:59 +02:00
inputBiomeName . disabled = "disabled" ;
2023-10-24 14:39:34 +02:00
inputBiomeColor . disabled = "disabled" ;
2023-10-20 14:03:45 +02:00
} else {
removeBiomeButton . disabled = "" ;
inputHeat . disabled = "" ;
inputHumidity . disabled = "" ;
2023-10-20 16:12:20 +02:00
inputMinY . disabled = "" ;
inputMaxY . disabled = "" ;
2023-10-23 03:10:59 +02:00
inputBiomeName . disabled = "" ;
2023-10-24 14:39:34 +02:00
inputBiomeColor . disabled = "" ;
2023-10-20 14:18:22 +02:00
if ( biomeSelector . selectedIndex !== - 1 ) {
let selected = biomeSelector . options [ biomeSelector . selectedIndex ] ;
let point = biomePoints [ biomeSelector . selectedIndex ] ;
inputHeat . value = point . heat ;
inputHumidity . value = point . humidity ;
inputMinY . value = point . min _y ;
inputMaxY . value = point . max _y ;
2023-10-23 03:10:59 +02:00
inputBiomeName . value = point . name ;
2023-10-24 14:39:34 +02:00
inputBiomeColor . value = point . color ;
2023-10-20 14:18:22 +02:00
}
2023-10-20 14:03:45 +02:00
}
2023-10-21 17:55:32 +02:00
inputNoiseHeatOffset . value = noises . heat . offset ;
inputNoiseHeatScale . value = noises . heat . scale ;
inputNoiseHeatOctaves . value = noises . heat . octaves ;
inputNoiseHeatPersistence . value = noises . heat . persistence ;
inputNoiseHumidityOffset . value = noises . humidity . offset ;
inputNoiseHumidityScale . value = noises . humidity . scale ;
inputNoiseHumidityOctaves . value = noises . humidity . octaves ;
inputNoiseHumidityPersistence . value = noises . humidity . persistence ;
2023-10-20 14:03:45 +02:00
}
2023-10-23 08:48:00 +02:00
/ * T o b e c a l l e d w h e n a b i o m e v a l u e l i k e h e a t w a s c h a n g e d .
Will update internal data , biome list and the diagram .
* pointField : Name of the field it affects ( same as in biomePoints )
* value : The value to set * /
2023-10-20 14:18:22 +02:00
function onChangeBiomeValueWidget ( pointField , value ) {
if ( value === null ) {
return ;
}
if ( biomeSelector . selectedIndex === - 1 ) {
return ;
}
let selected = biomeSelector . options [ biomeSelector . selectedIndex ] ;
if ( selected === null ) {
return ;
}
let point = biomePoints [ biomeSelector . selectedIndex ] ;
2023-10-21 17:12:43 +02:00
point [ pointField ] = value ;
2023-10-23 03:10:59 +02:00
let displayName = point . name ;
if ( displayName === "" ) {
displayName = FALLBACK _NAME ;
}
selected . innerText = displayName ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-20 13:31:23 +02:00
}
2023-10-20 16:31:11 +02:00
2023-10-23 08:48:00 +02:00
// Select the given point.
2023-10-24 14:39:34 +02:00
// point is a point from biomePoints.
// returns [selected, alreadySelected]
// where seleted is true if the point has been selected
// and alreadySelected is true if the point was selected before
2023-10-20 22:03:15 +02:00
function selectPoint ( point ) {
for ( let elem of biomeSelector . options ) {
let strID = elem . id ;
let elemID = null ;
2023-10-21 19:26:24 +02:00
let biomeID = getBiomeIDFromHTMLElement ( elem ) ;
2023-10-24 14:39:34 +02:00
if ( ( biomeID !== null ) && ( point . id === biomeID ) ) {
if ( elem . selected === true ) {
return [ true , true ] ;
2023-10-20 22:03:15 +02:00
}
2023-10-24 14:39:34 +02:00
elem . selected = "selected" ;
draw ( true ) ;
updateWidgetStates ( ) ;
return [ true , false ] ;
2023-10-20 22:03:15 +02:00
}
}
2023-10-21 00:57:25 +02:00
return [ false , false ] ;
}
2023-10-23 08:48:00 +02:00
// Returns the distance between the two 2D points (x1, y1) and (x2, y2)
2023-10-21 00:57:25 +02:00
function getDistance ( x1 , y1 , x2 , y2 ) {
return Math . sqrt ( ( x2 - x1 ) * * 2 + ( y2 - y1 ) * * 2 ) ;
2023-10-20 22:03:15 +02:00
}
2023-10-23 08:48:00 +02:00
// Converts (x, y) canvas pixel coordinates to biome coordinates;
// returns [heat, humidity]
2023-10-21 01:55:49 +02:00
function canvasPixelCoordsToBiomeCoords ( x , y ) {
2023-10-21 17:55:32 +02:00
let w = ( voronoiCanvas . width / ( limit _heat _max - limit _heat _min ) ) ;
let h = ( voronoiCanvas . height / ( limit _humidity _max - limit _humidity _min ) ) ;
let heat = Math . round ( ( x + limit _heat _min * w ) / w ) ;
2023-10-22 12:53:47 +02:00
// This also flips the Y axis
let humidity = limit _humidity _min + ( limit _humidity _max - ( Math . round ( ( y + limit _humidity _min * h ) / h ) ) ) ;
2023-10-21 01:55:49 +02:00
return [ heat , humidity ] ;
}
2023-10-23 08:48:00 +02:00
// Converts heat and humidity coordinates to canvas pixel coordinates;
// returns [x, y]
2023-10-21 01:55:49 +02:00
function biomeCoordsToCanvasPixelCoords ( heat , humidity ) {
2023-10-21 17:55:32 +02:00
let w = ( voronoiCanvas . width / ( limit _heat _max - limit _heat _min ) ) ;
let h = ( voronoiCanvas . height / ( limit _humidity _max - limit _humidity _min ) ) ;
let pixelX = heat * w - limit _heat _min * w ;
2023-10-22 12:53:47 +02:00
// This also flips the Y axis
let pixelY = voronoiCanvas . height - ( humidity * h - limit _humidity _min * h ) ;
2023-10-21 01:55:49 +02:00
return [ pixelX , pixelY ] ;
}
2023-10-23 08:48:00 +02:00
// Given a (x, y) pixel positition on the canvas, returns
// the point that is nearest to it or null if none.
// maxDist is the maximum allowed point distance,
// if it is exceeded, null is returned.
2023-10-20 22:03:15 +02:00
function getNearestPointFromCanvasPos ( x , y , maxDist ) {
let nearestPoint = null ;
let nearestDist = null ;
let points = getRenderedPoints ( getViewY ( ) ) ;
for ( let i = 0 ; i < points . length ; i ++ ) {
let point = points [ i ] ;
2023-10-21 01:55:49 +02:00
let [ pixelX , pixelY ] = biomeCoordsToCanvasPixelCoords ( point . heat , point . humidity ) ;
let dist = getDistance ( x , y , pixelX , pixelY ) ;
2023-10-20 22:03:15 +02:00
if ( nearestPoint === null ) {
nearestPoint = point ;
nearestDist = dist ;
} else if ( dist < nearestDist ) {
nearestPoint = point ;
nearestDist = dist ;
}
}
if ( nearestDist < maxDist ) {
return nearestPoint ;
} else {
return null ;
}
}
2023-10-23 08:48:00 +02:00
// Whether the mouse is currently pressed
2023-10-20 22:03:15 +02:00
let mouseIsDown = false ;
2023-10-24 14:39:34 +02:00
// Coordinates of where the drag-n-drop started
// or is about to start
let dragDropStartPos = null ;
2023-10-23 08:48:00 +02:00
// ID of point being currently dragged by the user
// or null if none.
2023-10-21 00:57:25 +02:00
let dragDropPointID = null ;
2023-10-21 01:08:35 +02:00
2023-10-24 14:39:34 +02:00
// Current drag-n-drop state:
// 0 = no drag-n-drop
// 1 = preparing drag-n-drop right after selection
// 2 = drag-n-drop active
let dragDropState = 0 ;
2023-10-23 18:09:23 +02:00
2023-10-23 08:48:00 +02:00
/ * M o v e a n d u p d a t e t h e b i o m e p o i n t w h i l e t h e u s e r
is dragging it * /
2023-10-21 01:08:35 +02:00
function updatePointWhenDragged ( pointID ) {
2023-10-21 17:12:43 +02:00
if ( pointID !== null && ! drawError ) {
2023-10-21 00:57:25 +02:00
let selectedPoint = null ;
let points = getRenderedPoints ( getViewY ( ) ) ;
for ( let i = 0 ; i < points . length ; i ++ ) {
if ( points [ i ] . id === dragDropPointID ) {
selectedPoint = points [ i ] ;
2023-10-21 01:55:49 +02:00
let [ newHeat , newHumidity ] = canvasPixelCoordsToBiomeCoords ( event . offsetX , event . offsetY ) ;
2023-10-21 00:57:25 +02:00
selectedPoint . heat = newHeat ;
selectedPoint . humidity = newHumidity ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 00:57:25 +02:00
updateWidgetStates ( ) ;
2023-10-21 19:26:24 +02:00
let [ elemID , elem ] = getSelectedBiomeIDAndElement ( ) ;
if ( elemID !== null && points [ i ] . id === elemID ) {
2023-10-23 03:10:59 +02:00
let displayName = selectedPoint . name ;
if ( displayName === "" ) {
displayName = FALLBACK _NAME ;
}
elem . innerText = displayName ;
2023-10-21 19:26:24 +02:00
break ;
2023-10-21 01:08:35 +02:00
}
return ;
2023-10-21 00:57:25 +02:00
}
}
}
2023-10-21 01:08:35 +02:00
}
2023-10-23 08:48:00 +02:00
/ * U p d a t e s t h e t e x t s h o w i n g t h e c u r r e n t a l t i t u d e ( Y h e i g h t )
the diagram currently applies * /
function updateAltitudeText ( ) {
altitudeDisplay . innerHTML = "showing diagram for altitude Y=<span class='statAltitude'>" + inputViewY . value + "</span>" ;
}
/ * U p d a t e t h e t e x t t h a t s h o w s t h e b i o m e c o o r d i n a t e s
of the cursor when it ' s on the diagram * /
2023-10-21 17:12:43 +02:00
function updateCoordinateDisplay ( pixelX , pixelY ) {
// show coordinates
let [ heat , humidity ] = canvasPixelCoordsToBiomeCoords ( pixelX , pixelY ) ;
if ( ! drawError ) {
2023-10-21 23:04:38 +02:00
let html = "cursor coordinates: heat=<span class='statHeat'>" + heat + "</span>; humidity=<span class='statHumidity'>" + humidity + "</span>" ;
2023-10-21 17:12:43 +02:00
coordinateDisplay . innerHTML = html ;
} else {
coordinateDisplay . innerHTML = " " ;
}
}
2023-10-23 08:48:00 +02:00
/ * U p d a t e s a n d c h a n g e s t h e c u r s o r t y p e o n t h e d i a g r a m
canvas depending on whether we can select , drag or do nothing
at the pointed position * /
2023-10-21 19:26:24 +02:00
function updateDragDropCursorStatus ( ) {
if ( drawError || ! showPoints ) {
2023-10-23 08:48:00 +02:00
// a special message is shown; use auto cursor
2023-10-21 19:26:24 +02:00
voronoiCanvas . style . cursor = "auto" ;
return
}
let nearest = getNearestPointFromCanvasPos ( event . offsetX , event . offsetY , POINT _SELECT _DISTANCE ) ;
if ( nearest !== null ) {
let [ id , elem ] = getSelectedBiomeIDAndElement ( ) ;
if ( id !== null && nearest . id === id ) {
2023-10-23 08:48:00 +02:00
// This cursor indicates we can grab the point
2023-10-21 19:26:24 +02:00
voronoiCanvas . style . cursor = "grab" ;
} else {
2023-10-23 08:48:00 +02:00
// This cursor indicates we can grab the point
2023-10-21 19:26:24 +02:00
voronoiCanvas . style . cursor = "crosshair" ;
}
} else {
2023-10-23 08:48:00 +02:00
// Default cursor when a click doesn't to anything
2023-10-21 19:26:24 +02:00
voronoiCanvas . style . cursor = "auto" ;
}
}
2023-10-23 08:48:00 +02:00
/* Initializes checkbox variables of the view settings */
function checkboxVarsInit ( ) {
showNames = inputCheckboxNames . checked ;
showPoints = inputCheckboxPoints . checked ;
showCellColors = inputCheckboxCellColors . checked ;
showGrid = inputCheckboxGrid . checked ;
showAxes = inputCheckboxAxes . checked ;
}
/* Collapses/Expands a config section */
function toggleConfigSectionDisplay ( headerLink , container ) {
if ( container . style . display !== "none" ) {
headerLink . innerText = "▶" ;
container . style . display = "none" ;
} else {
headerLink . innerText = "▼" ;
container . style . display = "block" ;
}
}
/ * U n h i d e t h e m a i n c o n t e n t . U s e d t o d i s a b l e t h e n o s c r i p t
handling because the website starts with the main content
container hidden so the noscript version of the page
isn ' t cluttered . * /
function unhideContent ( ) {
mainContentContainer . hidden = false ;
/ * A l s o h i d e t h e c o n t a i n e r h o l d i n g t h e n o s c r i p t e r r o r
message to avoid spacing issues * /
noscriptContainer . hidden = true ;
}
/***** EVENTS *****/
/* Canvas events */
2023-10-21 01:08:35 +02:00
voronoiCanvas . onmousemove = function ( event ) {
2023-10-24 14:39:34 +02:00
// update drag-n-drop state
if ( dragDropState !== 2 && dragDropPointID !== null && mouseIsDown && dragDropStartPos !== null ) {
let dist = getDistance ( dragDropStartPos . x , dragDropStartPos . y , event . offsetX , event . offsetY )
if ( dragDropState === 1 && dist >= 1 ) {
dragDropState = 2 ;
} else if ( ( dragDropState === 0 || dragDropState === 1 ) && dist > ONE _CLICK _DRAG _DROP _DISTANCE ) {
dragDropState = 2 ;
2023-10-23 18:09:23 +02:00
}
}
2023-10-21 01:48:45 +02:00
// drag-n-drop
2023-10-24 14:39:34 +02:00
if ( dragDropState === 2 ) {
2023-10-21 01:08:35 +02:00
updatePointWhenDragged ( dragDropPointID ) ;
}
2023-10-21 17:12:43 +02:00
updateCoordinateDisplay ( event . offsetX , event . offsetY ) ;
2023-10-21 19:26:24 +02:00
updateDragDropCursorStatus ( ) ;
2023-10-21 01:55:49 +02:00
}
voronoiCanvas . onmouseenter = function ( event ) {
2023-10-21 17:12:43 +02:00
updateCoordinateDisplay ( event . offsetX , event . offsetY ) ;
2023-10-21 19:26:24 +02:00
updateDragDropCursorStatus ( ) ;
2023-10-21 01:08:35 +02:00
}
2023-10-21 01:55:49 +02:00
2023-10-21 01:08:35 +02:00
voronoiCanvas . onmousedown = function ( event ) {
2023-10-21 01:48:45 +02:00
// select point by clicking.
// initiate drag-n-drop if already selected.
2023-10-21 01:08:35 +02:00
mouseIsDown = true ;
2023-10-21 13:59:19 +02:00
if ( ! showPoints ) {
// Points need to be shown for drag-n-drop to work
return ;
}
2023-10-21 19:26:24 +02:00
let nearest = getNearestPointFromCanvasPos ( event . offsetX , event . offsetY , POINT _SELECT _DISTANCE ) ;
2023-10-21 01:08:35 +02:00
if ( nearest !== null ) {
let success , alreadySelected
[ success , alreadySelected ] = selectPoint ( nearest ) ;
2023-10-23 18:09:23 +02:00
dragDropPointID = nearest . id ;
2023-10-24 14:39:34 +02:00
if ( success ) {
let [ x , y ] = biomeCoordsToCanvasPixelCoords ( nearest . heat , nearest . humidity ) ;
dragDropStartPos = { x : x , y : y } ;
}
2023-10-21 01:08:35 +02:00
if ( alreadySelected ) {
2023-10-24 14:39:34 +02:00
dragDropState = 1 ;
2023-10-21 01:08:35 +02:00
}
2023-10-21 19:26:24 +02:00
updateDragDropCursorStatus ( ) ;
2023-10-21 01:08:35 +02:00
}
}
voronoiCanvas . onmouseup = function ( event ) {
2023-10-21 01:48:45 +02:00
// end drag-n-drop
2023-10-24 14:39:34 +02:00
if ( dragDropState === 2 ) {
updatePointWhenDragged ( dragDropPointID ) ;
}
2023-10-21 01:08:35 +02:00
mouseIsDown = false ;
2023-10-24 14:39:34 +02:00
dragDropStartPos = null ;
2023-10-21 00:57:25 +02:00
dragDropPointID = null ;
2023-10-24 14:39:34 +02:00
dragDropState = 0 ;
2023-10-20 22:03:15 +02:00
}
voronoiCanvas . onmouseleave = function ( ) {
2023-10-21 01:48:45 +02:00
// end drag-n-drop
2023-10-20 22:03:15 +02:00
mouseIsDown = false ;
2023-10-24 14:39:34 +02:00
dragDropStartPos = null ;
2023-10-21 00:57:25 +02:00
dragDropPointID = null ;
2023-10-24 14:39:34 +02:00
dragDropState = 0 ;
2023-10-21 01:48:45 +02:00
coordinateDisplay . innerHTML = " " ;
2023-10-20 22:03:15 +02:00
}
2023-10-23 08:48:00 +02:00
/* Biome list events */
biomeSelector . onchange = function ( ) {
draw ( false ) ;
if ( biomeSelector . selectedIndex !== - 1 ) {
let selected = biomeSelector . options [ biomeSelector . selectedIndex ] ;
let point = biomePoints [ biomeSelector . selectedIndex ] ;
inputHeat . value = point . heat ;
inputHumidity . value = point . humidity ;
inputMinY . value = point . min _y ;
inputMaxY . value = point . max _y ;
}
updateWidgetStates ( ) ;
}
addBiomeButton . onclick = function ( ) {
let he = Math . round ( limit _heat _min + Math . random ( ) * ( limit _heat _max - limit _heat _min ) ) ;
let hu = Math . round ( limit _humidity _min + Math . random ( ) * ( limit _humidity _max - limit _humidity _min ) ) ;
let newPoint = {
id : lastBiomeID ,
name : generateBiomeName ( lastBiomeID ) ,
heat : he ,
humidity : hu ,
min _y : MIN _Y _DEFAULT ,
max _y : MAX _Y _DEFAULT ,
2023-10-24 14:39:34 +02:00
color : null ,
2023-10-23 08:48:00 +02:00
} ;
biomePoints . push ( newPoint ) ;
let num = biomePoints . length
let newElem = document . createElement ( "option" ) ;
newElem . id = "biome_list_element_" + lastBiomeID ;
newElem . value = "" + num ;
let newElemText = document . createTextNode ( newPoint . name ) ;
newElem . append ( newElemText ) ;
biomeSelector . append ( newElem ) ;
newElem . selected = "selected" ;
draw ( true ) ;
updateWidgetStates ( ) ;
lastBiomeID ++ ;
}
removeBiomeButton . onclick = function ( ) {
if ( biomeSelector . selectedOptions . length === 0 ) {
return ;
}
let firstIndex = null ;
for ( let o = 0 ; o < biomeSelector . selectedOptions . length ; o ++ ) {
let opt = biomeSelector . selectedOptions [ o ]
let index = opt . index
if ( firstIndex === null ) {
firstIndex = index ;
}
biomePoints . splice ( index , 1 ) ;
opt . remove ( ) ;
}
if ( firstIndex !== null && biomePoints . length > 0 ) {
let newIndex = firstIndex - 1 ;
if ( newIndex < 0 ) {
newIndex = 0 ;
}
biomeSelector . options [ newIndex ] . selected = "selected" ;
}
draw ( true ) ;
updateWidgetStates ( ) ;
}
/* Biome editing widgets events */
inputHeat . oninput = function ( ) {
onChangeBiomeValueWidget ( "heat" , + this . value ) ;
}
inputHumidity . oninput = function ( ) {
onChangeBiomeValueWidget ( "humidity" , + this . value ) ;
}
inputMinY . oninput = function ( ) {
onChangeBiomeValueWidget ( "min_y" , + this . value ) ;
}
inputMaxY . oninput = function ( ) {
onChangeBiomeValueWidget ( "max_y" , + this . value ) ;
}
inputBiomeName . oninput = function ( ) {
onChangeBiomeValueWidget ( "name" , this . value ) ;
}
2023-10-24 14:39:34 +02:00
inputBiomeColor . oninput = function ( ) {
onChangeBiomeValueWidget ( "color" , this . value ) ;
}
2023-10-23 08:48:00 +02:00
/* Diagram view settings events */
inputViewY . oninput = function ( ) {
draw ( true ) ;
updateAltitudeText ( ) ;
}
2023-10-21 13:59:19 +02:00
inputCheckboxNames . onchange = function ( ) {
showNames = this . checked ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 13:59:19 +02:00
}
inputCheckboxPoints . onchange = function ( ) {
showPoints = this . checked ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 13:59:19 +02:00
}
inputCheckboxCellColors . onchange = function ( ) {
showCellColors = this . checked ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 13:59:19 +02:00
}
inputCheckboxGrid . onchange = function ( ) {
showGrid = this . checked ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 13:59:19 +02:00
}
2023-10-23 08:48:00 +02:00
/* Noise parameters events */
2023-10-23 02:47:00 +02:00
inputCheckboxAxes . onchange = function ( ) {
showAxes = this . checked ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-23 02:47:00 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHeatScale . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . heat . scale = + this . value ;
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:55:32 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHeatOffset . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . heat . offset = + this . value ;
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:55:32 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHeatPersistence . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . heat . persistence = + this . value ;
2023-10-21 17:12:43 +02:00
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:12:43 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHeatOctaves . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . heat . octaves = + this . value ;
2023-10-21 17:12:43 +02:00
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:12:43 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHumidityScale . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . humidity . scale = + this . value ;
2023-10-21 17:12:43 +02:00
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:12:43 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHumidityOffset . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . humidity . offset = + this . value ;
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:55:32 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHumidityPersistence . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . humidity . persistence = + this . value ;
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:55:32 +02:00
}
2023-10-21 21:00:51 +02:00
inputNoiseHumidityOctaves . oninput = function ( ) {
2023-10-21 17:55:32 +02:00
noises . humidity . octaves = + this . value ;
2023-10-21 17:12:43 +02:00
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:12:43 +02:00
}
inputNoiseReset . onclick = function ( ) {
2023-10-21 17:55:32 +02:00
noises . heat . offset = NOISE _OFFSET _DEFAULT ;
noises . heat . scale = NOISE _SCALE _DEFAULT ;
noises . heat . octaves = NOISE _OCTAVES _DEFAULT ;
noises . heat . persistence = NOISE _PERSISTENCE _DEFAULT ;
noises . heat . absvalue = NOISE _ABSVALUE _DEFAULT ;
inputNoiseHeatOffset . value = noises . heat . offset ;
inputNoiseHeatScale . value = noises . heat . scale ;
inputNoiseHeatOctaves . value = noises . heat . octaves ;
inputNoiseHeatPersistence . value = noises . heat . persistence ;
noises . humidity . offset = NOISE _OFFSET _DEFAULT ;
noises . humidity . scale = NOISE _SCALE _DEFAULT ;
noises . humidity . octaves = NOISE _OCTAVES _DEFAULT ;
noises . humidity . persistence = NOISE _PERSISTENCE _DEFAULT ;
noises . humidity . absvalue = NOISE _ABSVALUE _DEFAULT ;
inputNoiseHumidityOffset . value = noises . humidity . offset ;
inputNoiseHumidityScale . value = noises . humidity . scale ;
inputNoiseHumidityOctaves . value = noises . humidity . octaves ;
inputNoiseHumidityPersistence . value = noises . humidity . persistence ;
2023-10-21 17:12:43 +02:00
clear ( ) ;
updateAreaVars ( ) ;
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-21 17:12:43 +02:00
}
2023-10-23 08:48:00 +02:00
/* Events for collapsing/extending config section with the arrow thingie */
2023-10-20 16:31:11 +02:00
2023-10-22 02:11:36 +02:00
biomeConfigHeaderLink . onclick = function ( ) {
toggleConfigSectionDisplay ( this , biomeConfigContainer ) ;
}
viewConfigHeaderLink . onclick = function ( ) {
toggleConfigSectionDisplay ( this , viewConfigContainer ) ;
}
noiseConfigHeaderLink . onclick = function ( ) {
toggleConfigSectionDisplay ( this , noiseConfigContainer ) ;
}
2023-10-23 08:48:00 +02:00
/* Load events */
2023-10-23 01:46:24 +02:00
2023-10-21 13:59:19 +02:00
window . addEventListener ( "load" , checkboxVarsInit ) ;
2023-10-20 17:22:48 +02:00
window . addEventListener ( "load" , function ( ) {
2023-10-23 08:48:00 +02:00
draw ( true ) ;
2023-10-20 17:22:48 +02:00
} )
2023-10-23 08:48:00 +02:00
window . addEventListener ( "load" , repopulateBiomeSelector ) ;
2023-10-20 16:31:11 +02:00
window . addEventListener ( "load" , updateWidgetStates ) ;
2023-10-21 21:35:20 +02:00
window . addEventListener ( "load" , updateAltitudeText ) ;
2023-10-23 01:46:24 +02:00
window . addEventListener ( "load" , unhideContent ) ;