vue-kityminder-ggg
Version:
百度脑图
190 lines (170 loc) • 6.65 kB
JavaScript
/**
* @fileOverview
*
* 根据按键控制状态机的跳转
*
* @author: techird
* @copyright: Baidu FEX, 2014
*/
define(function(require, exports, module) {
var Hotbox = require('../hotbox');
console.log('Hotbox',Hotbox)
// Nice: http://unixpapa.com/js/key.html
function isIntendToInput(e) {
if (e.ctrlKey || e.metaKey || e.altKey) return false;
// a-zA-Z
if (e.keyCode >= 65 && e.keyCode <= 90) return true;
// 0-9 以及其上面的符号
if (e.keyCode >= 48 && e.keyCode <= 57) return true;
// 小键盘区域 (除回车外)
if (e.keyCode != 108 && e.keyCode >= 96 && e.keyCode <= 111) return true;
// 小键盘区域 (除回车外)
// @yinheli from pull request
if (e.keyCode != 108 && e.keyCode >= 96 && e.keyCode <= 111) return true;
// 输入法
if (e.keyCode == 229 || e.keyCode === 0) return true;
return false;
}
/**
* @Desc: 下方使用receiver.enable()和receiver.disable()通过
* 修改div contenteditable属性的hack来解决开启热核后依然无法屏蔽浏览器输入的bug;
* 特别: win下FF对于此种情况必须要先blur在focus才能解决,但是由于这样做会导致用户
* 输入法状态丢失,因此对FF暂不做处理
* @Editor: Naixor
* @Date: 2015.09.14
*/
function JumpingRuntime() {
var fsm = this.fsm;
var minder = this.minder;
var receiver = this.receiver;
var container = this.container;
var receiverElement = receiver.element;
var hotbox = this.hotbox;
var compositionLock = false;
// normal -> *
receiver.listen('normal', function(e) {
// 为了防止处理进入edit模式而丢失处理的首字母,此时receiver必须为enable
receiver.enable();
// normal -> hotbox
if (e.is('Space')) {
e.preventDefault();
// safari下Space触发hotbox,然而这时Space已在receiver上留下作案痕迹,因此抹掉
if (kity.Browser.safari) {
receiverElement.innerHTML = '';
}
return fsm.jump('hotbox', 'space-trigger');
}
/**
* check
* @editor Naixor
* @Date 2015-12-2
*/
switch (e.type) {
case 'keydown': {
// debugger
if (minder.getSelectedNode()) {
if (isIntendToInput(e)) {
return fsm.jump('input', 'user-input');
};
} else {
receiverElement.innerHTML = '';
}
// normal -> normal shortcut
fsm.jump('normal', 'shortcut-handle', e);
break;
}
case 'keyup': {
break;
}
default: {}
}
});
// hotbox -> normal
receiver.listen('hotbox', function(e) {
receiver.disable();
e.preventDefault();
var handleResult = hotbox.dispatch(e);
if (hotbox.state() == Hotbox.STATE_IDLE && fsm.state() == 'hotbox') {
return fsm.jump('normal', 'hotbox-idle');
}
});
// input => normal
receiver.listen('input', function(e) {
receiver.enable();
if (e.type == 'keydown') {
if (e.is('Enter')) {
e.preventDefault();
// debugger
return fsm.jump('normal', 'input-commit');
}
if (e.is('Esc')) {
e.preventDefault();
return fsm.jump('normal', 'input-cancel');
}
if (e.is('Tab') || e.is('Shift + Tab')) {
e.preventDefault();
}
} else if (e.type == 'keyup' && e.is('Esc')) {
e.preventDefault();
if (!compositionLock) {
return fsm.jump('normal', 'input-cancel');
}
}
else if (e.type == 'compositionstart') {
compositionLock = true;
}
else if (e.type == 'compositionend') {
setTimeout(function () {
compositionLock = false;
});
}
});
//////////////////////////////////////////////
/// 右键呼出热盒
/// 判断的标准是:按下的位置和结束的位置一致
//////////////////////////////////////////////
var downX, downY;
var MOUSE_RB = 2; // 右键
container.addEventListener('mousedown', function(e) {
// console.log(e.button)
// return // 禁用热盒模式
// debugger
if (e.button == MOUSE_RB) {
e.preventDefault();
}
if (fsm.state() == 'hotbox') {
hotbox.active(Hotbox.STATE_IDLE);
fsm.jump('normal', 'blur');
} else if (fsm.state() == 'normal' && e.button == MOUSE_RB) {
downX = e.clientX;
downY = e.clientY;
}
}, false);
container.addEventListener('mousewheel', function(e) {
if (fsm.state() == 'hotbox') {
hotbox.active(Hotbox.STATE_IDLE);
fsm.jump('normal', 'mousemove-blur');
}
}, false);
container.addEventListener('contextmenu', function(e) {
e.preventDefault();
});
container.addEventListener('mouseup', function(e) {
if (fsm.state() != 'normal') {
return;
}
if (e.button != MOUSE_RB || e.clientX != downX || e.clientY != downY) {
return;
}
if (!minder.getSelectedNode()) {
return;
}
fsm.jump('hotbox', 'content-menu');
}, false);
// 阻止热盒事件冒泡,在热盒正确执行前导致热盒关闭
hotbox.$element.addEventListener('mousedown', function(e) {
e.stopPropagation();
});
}
return module.exports = JumpingRuntime;
});