class ColorViewer {
constructor({
colors = cvConfig.colors(),
gamut = cvConfig.gamut(),
surfaces = [],
inputs = cvConfig.inputs(),
colorspace = "munsell",
height = 600,
camera = cvConfig.camera()
} = {}) {
this.inputs = inputs;
this.colors = colors.srgb.map((x) => ({
linearRGB: colorUtils.linearRgbFromSRgb(x),
mesh: threeColor(colors.size)
}));
this.scene = new THREE.Scene();
this.colors.forEach((x) => this.scene.add(x.mesh));
const aspect = width / height;
this.camera = new THREE.PerspectiveCamera(
camera.fov,
aspect,
camera.near,
camera.far
);
this.camera.position.set(...camera.position);
this.camera.up = new THREE.Vector3(1, 0, 0);
this.renderer = new THREE.WebGLRenderer({ antialias: true });
this.renderer.setSize(width, height);
this.renderer.setPixelRatio(devicePixelRatio);
this.controls = new THREE.OrbitControls(
this.camera,
this.renderer.domElement
);
this.controls.target = new THREE.Vector3(50, 0, 0);
this.controls.addEventListener("change", () =>
this.renderer.render(this.scene, this.camera)
);
this.view = this.renderer.domElement;
this.animate = function* () {
// initialize
// `children[2]`: ColorSpace input
const transGenColorSpace = transitionGenerator(this.inputs.children[2], {
states: Array.from(colorUtils.mapColorSpaces().values())
});
// `children[3]`: CVD input
const transGenCvd = transitionGenerator(
this.inputs.children[3].children[1],
{
states: ["none", "protan", "deuteran", "tritan"]
}
);
let transitionColorSpaceOld = {};
let transitionMatCvdOld = [];
while (true) {
// color space
let transitionColorSpace = transGenColorSpace.next().value;
let changedColorSpace = !_.isEqual(
transitionColorSpaceOld,
transitionColorSpace
);
transitionColorSpaceOld = _.clone(transitionColorSpace);
// background color
// let background = fnBackgroundToSRgb(this.inputs.value.background);
let background = transitionInterpolate(
transitionColorSpace,
(state) => {
return _.flow([
(x) => [x, 0, 0],
colorSpaces[state].from100,
colorSpaces[state].toSRgb,
colorUtils.clampRgbGamut
])(this.inputs.value.background);
}
);
this.scene.background = new THREE.Color(...background);
// scale of color-spheres
let scale = transitionInterpolate(transitionColorSpace, (state) => {
// about 1 for Munsell, OkLab; about 2 for CIELuv
return (
(colorSpaces[state].scale.chroma /
colorSpaces[state].scale.luminance) *
2.5 +
0.75
);
});
// color-vision deficiency
let transitionCvd = transGenCvd.next().value;
let transitionMatCvd = transitionInterpolate(transitionCvd, (state) =>
matrixCvd(state, this.inputs.value.cvd.severity)
);
let changedCvd = !_.isEqual(transitionMatCvdOld, transitionMatCvd);
transitionMatCvdOld = _.cloneDeep(transitionMatCvd);
// Number[3] -> Number[3]
let linearRgbtoCvdSRgbClamp = _.flow([
(x) => math.multiply(transitionMatCvd, x),
colorUtils.linearRgbToSRgb,
colorUtils.clampRgbGamut
]);
if (changedCvd || changedColorSpace) {
this.colors.forEach((x) => {
// calculate new sRGB values, coordinates
const sRgbCvd = linearRgbtoCvdSRgbClamp(x.linearRGB);
const coordsCvd = transitionInterpolate(
transitionColorSpace,
(state) => {
return _.flow([
colorSpaces[state].fromSRgb,
colorSpaces[state].to100
])(sRgbCvd);
}
);
// implement
x.mesh.position.set(...coordsCvd);
x.mesh.material.color.setRGB(...sRgbCvd);
x.mesh.material.needsUpdate = true;
// size
x.mesh.scale.setScalar(scale);
});
}
this.controls.autoRotate = true;
this.controls.autoRotateSpeed = this.inputs.value.rotationSpeed;
this.controls.update();
this.renderer.render(this.scene, this.camera);
yield changedCvd || changedColorSpace;
}
};
invalidation.then(() => (this.controls.dispose(), this.renderer.dispose()));
}
}