diff --git a/sim/visuals/board.ts b/sim/visuals/board.ts index 64bb6ead..69d3e952 100644 --- a/sim/visuals/board.ts +++ b/sim/visuals/board.ts @@ -239,6 +239,8 @@ namespace pxsim.visuals { const state = ev3board().getInputNodes()[port] as ColorSensorNode; if (state.getMode() == ColorSensorMode.Colors) { view = new ColorGridControl(this.element, this.defs, state, port); + } else if (state.getMode() == ColorSensorMode.RgbRaw) { + view = new ColorRGBWheelControl(this.element, this.defs, state, port); } else if (state.getMode() == ColorSensorMode.Reflected) { view = new ColorWheelControl(this.element, this.defs, state, port); } else if (state.getMode() == ColorSensorMode.RefRaw) { diff --git a/sim/visuals/controls/colorRGBWheel.ts b/sim/visuals/controls/colorRGBWheel.ts new file mode 100644 index 00000000..0f8783f3 --- /dev/null +++ b/sim/visuals/controls/colorRGBWheel.ts @@ -0,0 +1,165 @@ +namespace pxsim.visuals { + + export class ColorRGBWheelControl extends ControlView { + private group: SVGGElement; + private colorGradient: SVGLinearGradientElement; + private reporter: SVGTextElement[] = []; + private rect: SVGElement[] = []; + + private printOffsetH = 18; + + getInnerWidth() { + return 120; + } + + getInnerHeight() { + return 192; + } + + private getReporterHeight() { + return 70; + } + + private getSliderWidth() { + return 24; + } + + private getSliderHeight() { + return 100; + } + + private getMaxValue() { + return 1023; + } + + private mapValue(x: number, inMin: number, inMax: number, outMin: number, outMax: number) { + return (x - inMin) * (outMax - outMin) / (inMax - inMin) + outMin; + } + + updateState() { + if (!this.visible) return; + const node = this.state; + const value = node.getValue(); + let inverseValue = this.getMaxValue() - value; + inverseValue = this.mapValue(inverseValue, 0, 1023, 0, 100); + svg.setGradientValue(this.colorGradient, inverseValue + "%"); + this.reporter[0].textContent = "R: " + `${parseFloat((value).toString()).toFixed(0)}`; + this.reporter[1].textContent = "G: " + `${parseFloat((value).toString()).toFixed(0)}`; + this.reporter[2].textContent = "B: " + `${parseFloat((value).toString()).toFixed(0)}`; + } + + updateColorLevel(pt: SVGPoint, parent: SVGSVGElement, ev: MouseEvent) { + let cur = svg.cursorPoint(pt, parent, ev); + const bBox = this.rect[0].getBoundingClientRect(); + const height = bBox.height; + let t = Math.max(0, Math.min(1, (height + bBox.top / this.scaleFactor - cur.y / this.scaleFactor) / height)); + const state = this.state; + state.setColor(t * this.getMaxValue()); + } + + getInnerView(parent: SVGSVGElement, globalDefs: SVGDefsElement) { + this.group = svg.elt("g") as SVGGElement; + + let gc = "gradient-color-" + this.getPort(); + const prevColorGradient = globalDefs.querySelector(`#${gc}`) as SVGLinearGradientElement; + this.colorGradient = prevColorGradient ? prevColorGradient : svg.linearGradient(globalDefs, gc, false); + svg.setGradientValue(this.colorGradient, "50%"); + svg.setGradientColors(this.colorGradient, "black", "yellow"); + + let reporterGroup: SVGElement[] = []; + for (let i = 0; i < 3; i++) { + reporterGroup[i] = pxsim.svg.child(this.group, "g"); + + console.log(`reporterGroup[${i}]:`); + console.log(reporterGroup[i]); + + reporterGroup[i].setAttribute("transform", `translate(${this.getWidth() / 2}, ${18 + this.printOffsetH * i})`); + this.reporter[i] = pxsim.svg.child(reporterGroup[i], "text", { 'text-anchor': 'middle', 'class': 'sim-text number large inverted', 'style': 'font-size: 18px;' }) as SVGTextElement; + + console.log(`this.reporter[${i}]:`); + console.log(this.reporter[0]); + } + + let sliderGroup: SVGElement[] = []; + for (let i = 0; i < 3; i++) { + sliderGroup[i] = pxsim.svg.child(this.group, "g"); + const translateX = (this.getWidth() / 2 - this.getSliderWidth() / 2 - 36) + 36 * i; + sliderGroup[i].setAttribute("transform", `translate(${translateX}, ${this.getReporterHeight()})`); + + console.log(`sliderGroup[${i}]:`); + console.log(sliderGroup[i]); + + this.rect[0] = pxsim.svg.child(sliderGroup[i], "rect", { + "width": this.getSliderWidth(), + "height": this.getSliderHeight(), + "style": `fill: url(#${gc})` + } + ); + + console.log(`this.rect[${i}]:`); + console.log(this.rect[0]); + } + /* + const sliderGroupR = pxsim.svg.child(this.group, "g"); + const sliderGroupG = pxsim.svg.child(this.group, "g"); + const sliderGroupB = pxsim.svg.child(this.group, "g"); + sliderGroupR.setAttribute("transform", `translate(${this.getWidth() / 2 - this.getSliderWidth() / 2 - 36}, ${this.getReporterHeight()})`); + sliderGroupG.setAttribute("transform", `translate(${this.getWidth() / 2 - this.getSliderWidth() / 2}, ${this.getReporterHeight()})`); + sliderGroupB.setAttribute("transform", `translate(${this.getWidth() / 2 - this.getSliderWidth() / 2 + 36}, ${this.getReporterHeight()})`); + console.log("sliderGroupR:"); + console.log(sliderGroupR); + console.log("sliderGroupG:"); + console.log(sliderGroupG); + console.log("sliderGroupB:"); + console.log(sliderGroupB); + + this.rect[0] = pxsim.svg.child(sliderGroupR, "rect", { + "width": this.getSliderWidth(), + "height": this.getSliderHeight(), + "style": `fill: url(#${gc})` + } + ); + this.rect[1] = pxsim.svg.child(sliderGroupG, "rect", { + "width": this.getSliderWidth(), + "height": this.getSliderHeight(), + "style": `fill: url(#${gc})` + } + ); + this.rect[2] = pxsim.svg.child(sliderGroupB, "rect", { + "width": this.getSliderWidth(), + "height": this.getSliderHeight(), + "style": `fill: url(#${gc})` + } + ); + + console.log("this.rectR:"); + console.log(this.rect[0]); + console.log("this.rectG:"); + console.log(this.rect[1]); + console.log("this.rectB:"); + console.log(this.rect[2]); + */ + + let pt = parent.createSVGPoint(); + let captured: boolean[] = [false, false, false]; + + touchEvents(this.rect[0], ev => { + if (captured[0] && (ev as MouseEvent).clientY) { + ev.preventDefault(); + this.updateColorLevel(pt, parent, ev as MouseEvent); + } + }, ev => { + captured[0] = true; + if ((ev as MouseEvent).clientY) { + this.rect[0].setAttribute('cursor', '-webkit-grabbing'); + this.updateColorLevel(pt, parent, ev as MouseEvent); + } + }, () => { + captured[0] = false; + this.rect[0].setAttribute('cursor', '-webkit-grab'); + }); + + return this.group; + } + } +} \ No newline at end of file