pxt-ev3/sim/visuals/controls/motorSlider.ts

157 lines
5.3 KiB
TypeScript
Raw Permalink Normal View History

2017-12-28 22:23:30 +01:00
namespace pxsim.visuals {
export class MotorSliderControl extends ControlView<MotorNode> {
private group: SVGGElement;
private gradient: SVGLinearGradientElement;
private slider: SVGGElement;
private reporter: SVGTextElement;
private dial: SVGGElement;
private static SLIDER_RADIUS = 100;
private internalAngle: number = 0;
2017-12-28 22:23:30 +01:00
getInnerView(parent: SVGSVGElement, globalDefs: SVGDefsElement) {
this.group = svg.elt("g") as SVGGElement;
const slider = pxsim.svg.child(this.group, 'g', { 'transform': 'translate(25,25)' })
const outerCircle = pxsim.svg.child(slider, "circle", {
'stroke-dasharray': '565.48', 'stroke-dashoffset': '0',
'cx': 100, 'cy': 100, 'r': '90', 'style': `fill:transparent;`,
'stroke': '#a8aaa8', 'stroke-width': '1rem'
}) as SVGCircleElement;
this.reporter = pxsim.svg.child(this.group, "text", {
2018-01-10 22:47:39 +01:00
'x': this.getInnerWidth() / 2, 'y': this.getInnerHeight() / 2,
'text-anchor': 'middle', 'dominant-baseline': 'middle',
2017-12-28 22:23:30 +01:00
'style': 'font-size: 50px',
'class': 'sim-text inverted number'
}) as SVGTextElement;
this.dial = pxsim.svg.child(slider, "g", { 'cursor': '-webkit-grab' }) as SVGGElement;
const handleInner = pxsim.svg.child(this.dial, "g");
pxsim.svg.child(handleInner, "circle", { 'cx': 0, 'cy': 0, 'r': 30, 'style': 'fill: #f12a21;' });
pxsim.svg.child(handleInner, "circle", { 'cx': 0, 'cy': 0, 'r': 29.5, 'style': 'fill: none;stroke: #b32e29' });
2018-01-10 22:47:39 +01:00
this.updateDial();
2017-12-28 22:23:30 +01:00
let pt = parent.createSVGPoint();
let captured = false;
const dragSurface = svg.child(this.group, "rect", {
x: 0,
y: 0,
width: this.getInnerWidth(),
2018-01-10 22:47:39 +01:00
height: this.getInnerHeight(),
2017-12-28 22:23:30 +01:00
opacity: 0,
cursor: '-webkit-grab'
})
touchEvents(dragSurface, ev => {
if (captured && (ev as MouseEvent).clientY != undefined) {
ev.preventDefault();
this.updateSliderValue(pt, parent, ev as MouseEvent);
this.handleSliderMove();
2017-12-28 22:23:30 +01:00
}
}, ev => {
captured = true;
if ((ev as MouseEvent).clientY != undefined) {
this.updateSliderValue(pt, parent, ev as MouseEvent);
this.handleSliderDown();
2017-12-28 22:23:30 +01:00
}
}, () => {
captured = false;
2018-01-10 22:47:39 +01:00
this.handleSliderUp();
2017-12-28 22:23:30 +01:00
})
return this.group;
}
getInnerWidth() {
return 250;
}
getInnerHeight() {
2018-01-10 22:47:39 +01:00
return 250;
2017-12-28 22:23:30 +01:00
}
private lastPosition: number;
private prevVal: number;
private updateSliderValue(pt: SVGPoint, parent: SVGSVGElement, ev: MouseEvent) {
let cur = svg.cursorPoint(pt, parent, ev);
let bBox = this.content.getBoundingClientRect();
2017-12-28 22:23:30 +01:00
const coords = {
x: cur.x / this.scaleFactor - bBox.left / this.scaleFactor,
y: cur.y / this.scaleFactor - bBox.top / this.scaleFactor
2017-12-28 22:23:30 +01:00
};
const radius = MotorSliderControl.SLIDER_RADIUS / 2;
const dx = coords.x - radius;
const dy = coords.y - radius;
const atan = Math.atan(-dy / dx);
let deg = Math.ceil(atan * (180 / Math.PI));
if (dx < 0) {
deg -= 270;
} else if (dy > 0) {
deg -= 450;
} else if (dx >= 0 && dy <= 0) {
deg = 90 - deg;
}
const value = Math.abs(Math.ceil((deg % 360)));
2017-12-28 22:23:30 +01:00
this.internalAngle = value;
2018-01-10 22:47:39 +01:00
this.updateDial();
2017-12-28 22:23:30 +01:00
this.prevVal = deg;
this.lastPosition = cur.x;
}
2018-01-10 22:47:39 +01:00
private handleSliderDown() {
const state = this.state;
state.manualMotorDown();
}
2017-12-28 22:23:30 +01:00
2018-01-10 22:47:39 +01:00
private handleSliderMove() {
this.dial.setAttribute('cursor', '-webkit-grabbing');
const state = this.state;
state.manualMotorAngle(this.internalAngle);
2018-01-10 22:47:39 +01:00
}
2017-12-28 22:23:30 +01:00
2018-01-10 22:47:39 +01:00
private handleSliderUp() {
this.dial.setAttribute('cursor', '-webkit-grab');
const state = this.state;
state.manualMotorUp();
this.internalAngle = 0;
2018-01-10 22:47:39 +01:00
this.updateDial();
}
private updateDial() {
let angle = this.internalAngle;
2017-12-28 22:23:30 +01:00
// Update dial position
const radius = MotorSliderControl.SLIDER_RADIUS;
const dialRadius = 5;
const x = Math.ceil((radius - dialRadius) * Math.sin(angle * Math.PI / 180)) + radius;
const y = Math.ceil((radius - dialRadius) * -Math.cos(angle * Math.PI / 180)) + radius;
2017-12-28 22:23:30 +01:00
this.dial.setAttribute('transform', `translate(${x}, ${y})`);
}
2018-01-10 22:47:39 +01:00
updateState() {
if (!this.visible) {
return;
}
const node = this.state;
const angle = node.getAngle() % 360;
2018-01-10 22:47:39 +01:00
// Update reporter
this.reporter.textContent = `${angle}°`;
2017-12-28 22:23:30 +01:00
}
}
}