-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcone-mapper.js
153 lines (115 loc) · 4.17 KB
/
cone-mapper.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
'use strict';
import Canvas from 'canvas';
const Image = Canvas.Image;
import fs from 'fs-extra';
import path from 'path';
import colors from 'colors';
import GIFEncoder from 'gifencoder';
import cliProgress from 'cli-progress';
import * as color_convert from './color-convert.js';
import {writeImage, writeImageSilent, getOutputImage} from './image-loader.js';
export async function map(settings, pixels, sequences, sequence_keys, reverse_color_map, image_size, frames) {
const unduplicated_frames = frames / settings.input.duplicate_frames;
const height = settings.output.cylinder_image.height;
const width = height * settings.output.cylinder_image.diameter_scalar * Math.PI;
const output = getOutputImage(width, height, {r: 255, g: 255, b: 255, a: 0});
const context = output.getContext("2d");
const colors = sequence_keys.map(string => string.split('')
.map( color => reverse_color_map[color])
.map(color_convert.toARGBObject));
console.log(`colors: ${colors.length} colors[0]: ${colors[0].length} frames: ${frames} unduplicated_frames: ${unduplicated_frames}`);
console.log(colors)
const row_height = height / frames;
const column_width = width / colors.length;
for (let row = 0; row < frames; row++) {
for (let column = 0; column < colors.length; column++) {
const x = column * column_width;
const y = row * row_height;
const color = colors[column][row];
context.fillStyle = `rgba(${color.r}, ${color.g}, ${color.b}, 1.0})`;
context.strokeStyle = `rgba(${color.r}, ${color.g}, ${color.b}, 1.0})`;
context.beginPath();
context.rect(x, y, column_width, row_height);
context.fill();
context.stroke();
}
}
const color_mapping = findColorMapping(pixels, sequences);
// the width of the cylinder watched from the side
const visible_width = height * settings.output.cylinder_image.diameter_scalar;
const visible_columns = Math.max(...color_mapping.map( i => i.offset));
console.log(`${visible_width}, ${visible_columns}`.red);
const items = color_mapping.map( item => {
const column = sequence_keys.indexOf(item.offset_key);
const rows = colors[column].length;
const row = item.offset;
const color = colors[column][row];
//const x = (columns / 2 + visible_columns / 2 - column) * column_width;
//const y = row_height * row + row_height / 2; // middle of row
const x = column * column_width + column_width * 0.5;
const y = (row+1) * row_height;
const item_colors = Array.from({length:unduplicated_frames}, (_) => 0).map((_,i) => {
let r = (rows-row + i);
r = ((r%rows)+rows)%rows; // this handles negative modulo
let color = colors[column][r];
return color;
});
return {row, column, aim_position: {x,y}, string: item.string, color, colors: item_colors};
});
var mapping = pixels.map((pixel, i) => {
const item = items[i];
const aim_position = item.aim_position;
return {
mirror: {
x: pixel.x - 0.5, // offset to be between -0.5 and 0.5
y: pixel.y - 0.5,
},
palette: {
x: aim_position.x / width - 0.5, // offset to be between -0.5 and 0.5
y: aim_position.y / height - 0.5, // offset to be between -0.5 and 0.5
colors: item.colors,
},
row: item.row,
column: item.column,
string: item.string
};
})
const mapping_conf = {
mirror: {
width: image_size.width,
height: image_size.height,
},
palette: {
width,
height,
},
mapping,
colors,
frames,
}
if (settings.output.texture) {
await writeImage(path.join(settings.output.path,'texture.png'), output);
}
return mapping_conf;
}
function findColorMapping(pixels, sequences) {
let stats = {};
let result = pixels.map( pixel => {
const pixel_string = pixel.pixel_colors.join('');
let key = findKeyAndOffset(sequences, pixel_string);
stats[key.offset_key] = (stats[key.offset_key] || 0) + 1;
return key;
});
return result;
}
function findKeyAndOffset(sequences, pixel_string) {
for (var key in sequences) {
const offsets = sequences[key];
for (let offset of offsets) {
if (pixel_string === offset.string) {
return {offset_key: key, string: offset.string, offset: offset.offset};
}
}
}
throw new Error(`Could not find key offset for "${pixel_string}"`);
}