-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.js
122 lines (89 loc) · 3 KB
/
main.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
//
// Audio
//
var data = [r=1];
function rng() { return .5+(r = (16807*r + 1)|0)/8e9; }
var audio_data;
var k = 100;
for (var i = 0; i < 960000; i++) {
if (!(i % 8000)) k *= Math.pow(1.06,rng()*8-4|0);
audio_data += String.fromCharCode(70 * (((i%k)>k/2) + (i%k/2)/k/2 * (1-(i % 8000)/8000)) | 0);
}
//
// Video
//
c.parentNode.innerHTML = '<canvas id=c style=width:100%;height:100%;cursor:none>'
c.parentNode.style = "margin:0"
new Audio("data:audio/wav;base64,"+btoa("RIFFdataWAVEfmt \x10\x00\x00\x00\x01\x00\x01\x00\x80>\x00\x00\x80>\x00\x00\x01\x00\x08\x00data\x00\xa6\x0e\x00"+audio_data)).play();
var viewport_g = c.getContext("2d");
var viewport_w = c.width = 960;
var viewport_h = c.height = 540;
//document.body.appendChild(c);
var heightmap_dim = 1024;
var viewport_imagedata = viewport_g.getImageData(0,0,viewport_w,viewport_h);
var buffer = viewport_imagedata.data;
var player_x = 200;
var player_y = 0
var far_distance = 400;
var water_level = 120;
var player_h = 120;
var h_scale = 2500;
function px(x,y) {
var j = (viewport_w*y + x)*4;
buffer[j++] = r;
buffer[j++] = g;
buffer[j++] = b;
buffer[j++] = 255;
}
function vertLine(x, y_start, y_buf){
for (var i = y_start|0; i < y_buf; i++)
px(x,i), px(x, viewport_h-i)
}
function getSample(x, y) {
return data[((x|0)+heightmap_dim)%heightmap_dim +
((y|0)+heightmap_dim)%heightmap_dim * heightmap_dim];
}
function drawViewport() {
for (var i = 0; i < viewport_w; i++) {
var y_buf = viewport_h;
for (var d = 0; d < far_distance; d++) {
var my = player_y + d/2;
var mx = (player_x + d/2 + i * -d / viewport_w)|0;
var map_h = getSample(mx*1.1,my);
var fog = 1 - d / far_distance;
var viewport_y = viewport_h/2 + (player_h - map_h) / d * h_scale;
var depth = Math.min(1,Math.pow(map_h / water_level, 8));
var wave = 1.1*getSample(mx*8+player_y*16, my*16+player_y*8)/2*fog*(1-depth)
r = wave + map_h*fog*depth*.8;
g = 1.6*wave + map_h*fog*depth*.5;
b = wave + map_h*fog*depth*.3;
vertLine(i, Math.max(viewport_y, viewport_h/2), y_buf);
y_buf = Math.min(viewport_y, y_buf);
}
// clear sky
vertLine(i, viewport_h/2, y_buf);
}
};
var skip = heightmap_dim/2;
var diamond = 1;
function color(avg) {
return avg/4 * (1-skip / heightmap_dim * 2) + rng()*255 * skip / heightmap_dim * 2;
}
while (skip|0) {
var odd = 0;
for (var y = 0; y < heightmap_dim; y += skip) {
if (diamond)
y+=skip;
for (var x = (diamond||!odd) ? skip : 0; x < heightmap_dim; x += skip*2)
data[x + heightmap_dim * y] = color(diamond ? getSample(x-skip,y-skip) + getSample(x+skip,y-skip) + getSample(x-skip,y+skip) + getSample(x+skip,y+skip) :
getSample(x-skip,y) + getSample(x+skip,y) + getSample(x,y-skip) + getSample(x,y+skip));
odd = !odd;
}
if (diamond = !diamond) skip /= 2;
}
(function tick() {
setTimeout(tick,16);
player_y += .3;
drawViewport();
viewport_g.putImageData(viewport_imagedata, 0, 0);
})();