RxJS:锁定水平或垂直鼠标拖动
RxJS: Locked horizontal or vertical mouse dragging
我想构建一个界面,您可以在一定距离后选择模式向各个方向拖动。例如,如果您水平拖动 25px,它会锁定到该模式并保持在那里,直到您释放鼠标。如果你垂直拖动它也会做同样的事情。如果您长时间单击或按住,可能会发生其他操作。
下面是简化的 fiddle 说明我的目标:https://jsfiddle.net/ud37p0y2/2/
反应式编程似乎非常适合这个,但我似乎无法弄清楚如何启动这些模式,然后坚持使用它们,直到您松开鼠标。我的起点是许多拖放示例,但我似乎无法更进一步..
一些代码(TypeScript):
var mouseDown = Rx.Observable.fromEvent($element[0], 'mousedown').select((event: MouseEvent): IPoint => {
event.preventDefault();
return { x: event.clientX, y: event.clientY };
});
var mouseUp = Rx.Observable.fromEvent($element[0], 'mouseup');
var mouseMove = Rx.Observable.fromEvent($element[0], 'mousemove');
var mouseDrag = mouseDown.selectMany((mouseDownPos: IPoint) => {
return mouseMove.select((event: MouseEvent) => {
return {
x: event.clientX - mouseDownPos.x,
y: event.clientY - mouseDownPos.y
};
}).takeUntil(mouseUp);
});
var horizontalDrag = mouseDrag.filter((pos: IPoint) => {
return pos.x < -25 || pos.x > 25;
});
// How would i continue from here?
horizontalDrag.subscribe((pos: IPoint) => {
console.log('drag'); // This fires all the time, i'd like to do it once when the mode starts and then something else to be called every time the mouse has moved
});
从这里我想获得水平拖动、垂直拖动和保持事件的可观察值。在一个模式启动后,其他模式应该被禁用,例如拖动不会触发长按事件。
我会使用 amb
+ skipWhile
的组合。
amb
会给你锁定状态的行为,
skipWhile
将阻止事件触发,直到它们超过阈值
一段时间。
核心逻辑如下所示:
//Waits for either X or Y to emit then only propagates that one
return Rx.Observable.amb(
mouseMove
.pluck('clientX')
//Wait until the threshold is reached
.skipWhile(function (x) {
return Math.abs(startAt.clientX - x) < 25;
})
//Transform the outgoing event
.map(function (x) {
return {
prop: 'clientX',
delta: x - startAt.clientX
};
}),
mouseMove
.pluck('clientY')
.skipWhile(function (y) {
return Math.abs(startAt.clientY - y) < 25;
})
.map(function (y) {
return {
prop: 'clientY',
delta: y - startAt.clientY
};
}),
//If neither propagates for a second, then subscribe to this instead
mouseMove
.startWith(startAt)
.delaySubscription(1000)
.tap(function (e) {
box.className = 'press';
prop = 'timeStamp';
box.innerHTML = '';
})
.map(function (e) {
return {
prop: 'timeStamp',
delta: e.timeStamp - startAt.timeStamp
};
}))
.takeUntil(mouseUp);
编辑 1
通过将延续 Observable 移动到 amb
并使用 delaySubscription
代替 timeout
。
这里是您的代码的完全修改版本:
var box = document.getElementById('box');
var mouseDown = Rx.Observable.fromEvent(box, 'mousedown');
var mouseUp = Rx.Observable.fromEvent(document.body, 'mouseup');
var mouseMove = Rx.Observable.fromEvent(box, 'mousemove')
.tap(function(e) { e.preventDefault(); });
mouseDown.flatMapLatest(function (start) {
var startAt = start;
box.className = 'waiting';
box.innerHTML = 'waiting...';
return Rx.Observable.amb(
mouseMove
.pluck('clientX')
.skipWhile(function (x) {
return Math.abs(startAt.clientX - x) < 25;
})
.map(function (x) {
return {
prop: 'clientX',
delta: x - startAt.clientX
};
}),
mouseMove
.pluck('clientY')
.skipWhile(function (y) {
return Math.abs(startAt.clientY - y) < 25;
})
.map(function (y) {
return {
prop: 'clientY',
delta: y - startAt.clientY
};
}),
mouseMove
.startWith(startAt)
.delaySubscription(1000)
.tap(function (e) {
box.className = 'press';
prop = 'timeStamp';
box.innerHTML = '';
}).map(function (e) {
return {
prop: 'timeStamp',
delta: e.timeStamp - startAt.timeStamp
};
}))
.takeUntil(mouseUp);
})
.subscribe(function (x) {
box.innerHTML = x.prop + ': ' + x.delta;
});
mouseUp.subscribe(function() {
box.className = '';
box.innerHTML = '';
});
body {
font: 12px sans-serif;
}
#box {
width: 300px;
height: 300px;
border: 1px #000 solid;
text-align: center;
padding: 20px;
transition: 0.2s background-color;
cursor: pointer;
}
#box.waiting {
background-color: gray;
cursor: move;
}
#box.dragX {
background-color: red;
cursor: ew-resize;
}
#box.dragY {
background-color: green;
cursor: ns-resize;
}
#box.press {
background-color: yellow;
cursor: progress;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/rxjs/2.5.3/rx.all.js"></script>
<ol>
<li>Drag horizontally</li>
<li>Release</li>
<li>Drag vertically</li>
<li>Relase</li>
<li>Press and hold</li>
</ol>
<div id="box"></div>
我想构建一个界面,您可以在一定距离后选择模式向各个方向拖动。例如,如果您水平拖动 25px,它会锁定到该模式并保持在那里,直到您释放鼠标。如果你垂直拖动它也会做同样的事情。如果您长时间单击或按住,可能会发生其他操作。
下面是简化的 fiddle 说明我的目标:https://jsfiddle.net/ud37p0y2/2/
反应式编程似乎非常适合这个,但我似乎无法弄清楚如何启动这些模式,然后坚持使用它们,直到您松开鼠标。我的起点是许多拖放示例,但我似乎无法更进一步..
一些代码(TypeScript):
var mouseDown = Rx.Observable.fromEvent($element[0], 'mousedown').select((event: MouseEvent): IPoint => {
event.preventDefault();
return { x: event.clientX, y: event.clientY };
});
var mouseUp = Rx.Observable.fromEvent($element[0], 'mouseup');
var mouseMove = Rx.Observable.fromEvent($element[0], 'mousemove');
var mouseDrag = mouseDown.selectMany((mouseDownPos: IPoint) => {
return mouseMove.select((event: MouseEvent) => {
return {
x: event.clientX - mouseDownPos.x,
y: event.clientY - mouseDownPos.y
};
}).takeUntil(mouseUp);
});
var horizontalDrag = mouseDrag.filter((pos: IPoint) => {
return pos.x < -25 || pos.x > 25;
});
// How would i continue from here?
horizontalDrag.subscribe((pos: IPoint) => {
console.log('drag'); // This fires all the time, i'd like to do it once when the mode starts and then something else to be called every time the mouse has moved
});
从这里我想获得水平拖动、垂直拖动和保持事件的可观察值。在一个模式启动后,其他模式应该被禁用,例如拖动不会触发长按事件。
我会使用 amb
+ skipWhile
的组合。
amb
会给你锁定状态的行为,skipWhile
将阻止事件触发,直到它们超过阈值 一段时间。
核心逻辑如下所示:
//Waits for either X or Y to emit then only propagates that one
return Rx.Observable.amb(
mouseMove
.pluck('clientX')
//Wait until the threshold is reached
.skipWhile(function (x) {
return Math.abs(startAt.clientX - x) < 25;
})
//Transform the outgoing event
.map(function (x) {
return {
prop: 'clientX',
delta: x - startAt.clientX
};
}),
mouseMove
.pluck('clientY')
.skipWhile(function (y) {
return Math.abs(startAt.clientY - y) < 25;
})
.map(function (y) {
return {
prop: 'clientY',
delta: y - startAt.clientY
};
}),
//If neither propagates for a second, then subscribe to this instead
mouseMove
.startWith(startAt)
.delaySubscription(1000)
.tap(function (e) {
box.className = 'press';
prop = 'timeStamp';
box.innerHTML = '';
})
.map(function (e) {
return {
prop: 'timeStamp',
delta: e.timeStamp - startAt.timeStamp
};
}))
.takeUntil(mouseUp);
编辑 1
通过将延续 Observable 移动到 amb
并使用 delaySubscription
代替 timeout
。
这里是您的代码的完全修改版本:
var box = document.getElementById('box');
var mouseDown = Rx.Observable.fromEvent(box, 'mousedown');
var mouseUp = Rx.Observable.fromEvent(document.body, 'mouseup');
var mouseMove = Rx.Observable.fromEvent(box, 'mousemove')
.tap(function(e) { e.preventDefault(); });
mouseDown.flatMapLatest(function (start) {
var startAt = start;
box.className = 'waiting';
box.innerHTML = 'waiting...';
return Rx.Observable.amb(
mouseMove
.pluck('clientX')
.skipWhile(function (x) {
return Math.abs(startAt.clientX - x) < 25;
})
.map(function (x) {
return {
prop: 'clientX',
delta: x - startAt.clientX
};
}),
mouseMove
.pluck('clientY')
.skipWhile(function (y) {
return Math.abs(startAt.clientY - y) < 25;
})
.map(function (y) {
return {
prop: 'clientY',
delta: y - startAt.clientY
};
}),
mouseMove
.startWith(startAt)
.delaySubscription(1000)
.tap(function (e) {
box.className = 'press';
prop = 'timeStamp';
box.innerHTML = '';
}).map(function (e) {
return {
prop: 'timeStamp',
delta: e.timeStamp - startAt.timeStamp
};
}))
.takeUntil(mouseUp);
})
.subscribe(function (x) {
box.innerHTML = x.prop + ': ' + x.delta;
});
mouseUp.subscribe(function() {
box.className = '';
box.innerHTML = '';
});
body {
font: 12px sans-serif;
}
#box {
width: 300px;
height: 300px;
border: 1px #000 solid;
text-align: center;
padding: 20px;
transition: 0.2s background-color;
cursor: pointer;
}
#box.waiting {
background-color: gray;
cursor: move;
}
#box.dragX {
background-color: red;
cursor: ew-resize;
}
#box.dragY {
background-color: green;
cursor: ns-resize;
}
#box.press {
background-color: yellow;
cursor: progress;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/rxjs/2.5.3/rx.all.js"></script>
<ol>
<li>Drag horizontally</li>
<li>Release</li>
<li>Drag vertically</li>
<li>Relase</li>
<li>Press and hold</li>
</ol>
<div id="box"></div>