Skip to content

Instantly share code, notes, and snippets.

@monjudoh
Forked from hitode909/newmouse.user.js
Created October 5, 2010 13:05
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save monjudoh/611517 to your computer and use it in GitHub Desktop.
Save monjudoh/611517 to your computer and use it in GitHub Desktop.
// ==UserScript==
// @name newmouse
// @namespace http://www.hatena.ne.jp/hitode909
// @include *
// ==UserScript==
var Nears = function(x, y) {
this.centerX = x;
this.centerY = y;
this.i = 0;
}
Nears.prototype = {
getNext: function() {
this.i++;
var r = this.i / 3;
var t = this.i / 10;
var x = this.centerX + r * Math.cos(t);
var y = this.centerY + r * Math.sin(t);
return { x: x, y: y};
}
};
var Pointer = function() {
}
Pointer.prototype = {
setPoint: function(x, y, sx, sy) {
if (this._isSamePoint(x, y, sx, sy)) return;
if (this._isWorking()) this._clearPlot();
this.x = x;
this.y = y;
this.sx = sx;
this.sy = sy;
this.startPlot();
},
startPlot: function() {
var self = this;
this.nears = new Nears(this.x, this.y);
this._setPointerNormal();
this.timer = setInterval(function() {
for(var i = 0; i < 10; i++) {
var pos = self.nears.getNext();
var element = document.elementFromPoint(pos.x, pos.y);
self._plotPoint(self.sx + pos.x, self.sy + pos.y);
for(var j = 0; element && j < 3; j++) {
var tagName = element.tagName.toLowerCase();
if (["input", "a", "area", "iframe", "textarea", "button", "select"].indexOf(tagName) >= 0) {
self._focusElement(element);
self._setPointerMatched();
self._clearPlot();
return;
}
element = element.parentNode;
}
if (i == 0) self._unfocusElement(); // clear previous here to avoid blink
}
}, 0);
},
trigger: function(move, newTab) {
var target = this.focusedElement;
if (!target) return;
click(target);
if (document.querySelector(':focus') != target) target.focus();
// sendKey(target, 13);
if (move && target.href) {
if (newTab) {
window.open(target.href);
} else {
location.href = target.href;
}
}
},
// private
_focusElement: function(element) {
this._unfocusElement();
element.setAttribute('data-clicknear-focused', '1');
this.focusedElement = element;
},
_unfocusElement: function() {
if (!this.focusedElement) return;
this.focusedElement.removeAttribute('data-clicknear-focused');
this.focusedElement = null;
},
_clearPlot: function() {
clearInterval(this.timer);
this.timer = null;
},
_isWorking: function() {
return !!this.timer;
},
_isSamePoint: function(x, y, sx, sy) {
var diff = 0;
diff += Math.abs(this.x - x);
diff += Math.abs(this.y - y);
diff += Math.abs(this.sx - sx);
diff += Math.abs(this.sy - sy);
return diff < 4;
},
_plotPoint: function(x, y) {
if (!this.pointElement) return;
this.pointElement.setAttribute('style','z-index: 2147483647; position: absolute; left: ' + x + 'px; top: ' + y + 'px;');
},
_setPointerNormal: function() {
this.pointElement.firstChild.style.display = 'block';
this.pointElement.lastChild.style.display = 'none';
},
_setPointerMatched: function() {
this.pointElement.firstChild.style.display = 'none';
this.pointElement.lastChild.style.display = 'block';
}
};
// util
function click(target) {
["mouseup", "mousedown", "click"].forEach(function (type) {
var event = document.createEvent("MouseEvents");
event.initMouseEvent(type, true, true, window,
0, 0, 0, 0, 0, false, false, false, false, 0, null);
target.dispatchEvent(event);
});
}
function sendKey(target, code) {
["keydown", "keypress", "keyup"].forEach(function(type){
var event = document.createEvent("KeyboardEvent");
event.initKeyEvent(type, true, true, window,
false,false,false,false,code,1);
target.dispatchEvent(event);
});
}
// content script
(function() {
var pointer = new Pointer();
var step = function(event) {
pointer.setPoint(event.clientX, event.clientY, window.scrollX, window.scrollY);
};
document.body.addEventListener('mousemove',step, false);
document.body.addEventListener('scroll', step, false);
document.addEventListener('mouseup', function(event) {
if (["input", "a", "area", "iframe", "textarea", "button", "select"].indexOf(event.target.tagName.toLowerCase()) >= 0) return;
pointer.trigger(false);
}, false);
document.addEventListener('dblclick', function(event) {
if (["input", "a", "area", "iframe", "textarea", "button", "select"].indexOf(event.target.tagName.toLowerCase()) >= 0) return;
pointer.trigger(true, event.metaKey);
}, false);
var style = document.createElement('style');
document.body.appendChild(style);
style.textContent = [
"[data-clicknear-focused] {",
"background: yellow;",
"}"
].join("\n");
})();
// image
(function(){
var div = document.createElement('div');
with(div.style) {
zIndex = '2147483647';
position = 'absolute';
};
document.body.appendChild(div);
Pointer.prototype.pointElement = div;
})();
(function(){
var img = document.createElement('img');
img.src = '';
// with(img.style) {
// zIndex = '2147483647';
// position = 'absolute';
// };
Pointer.prototype.pointElement.appendChild(img);
})();
(function(){
var img = document.createElement('img');
img.src = '';
// with(img.style) {
// zIndex = '2147483647';
// position = 'absolute';
// };
Pointer.prototype.pointElement.appendChild(img);
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment