我正在开发一个webapp(不是一个有趣文本页面的网站),它有一个非常不同的触摸界面(当你点击时你的手指隐藏屏幕)和鼠标(严重依赖于悬停预览)。 如何检测到我的用户没有鼠标向他显示正确的界面?我计划为鼠标和触摸的人留下一个开关(就像一些笔记本电脑)。
浏览器中的触摸事件功能实际上并不意味着用户正在使用触摸设备(例如,Modernizr不会剪切它)。如果设备有鼠标,正确回答问题的代码应该返回false,否则返回true。对于具有鼠标和触摸功能的设备,它应该返回false(不是仅触摸)
作为旁注,我的触控界面可能也适用于仅限键盘的设备,所以更多的是缺少我想要检测的鼠标。
为了更清楚地说明需求,以下是我要实现的API:
// Level 1
// The current answers provide a way to do that.
hasTouch();
// Returns true if a mouse is expected.
// Note: as explained by the OP, this is not !hasTouch()
// I don't think we have this in the answers already, that why I offer a bounty
hasMouse();
// Level 2 (I don't think it's possible, but maybe I'm wrong, so why not asking)
// callback is called when the result of "hasTouch()" changes.
listenHasTouchChanges(callback);
// callback is called when the result of "hasMouse()" changes.
listenHasMouseChanges(callback);
答案 0 :(得分:57)
主要问题是您有以下不同类别的设备/用例:
更糟糕的是,人们可以从其中一些类转换到其他类(插入鼠标,连接到键盘),或者用户可能会在普通笔记本电脑上显示,直到它们伸出并触摸屏幕。 / p>
假设浏览器中存在事件构造函数不是向前推进的好方法(并且它有些不一致),这是正确的。此外,除非您正在跟踪特定事件或仅尝试排除上述几个类别,否则使用事件本身并非完整的证据。
例如,假设您发现用户已发出真正的鼠标移动(而不是触摸事件中的错误移动,请参阅http://www.html5rocks.com/en/mobile/touchandmouse/)。
然后是什么?
您启用悬停样式?你添加更多按钮?
无论哪种方式,你都要增加玻璃时间,因为你必须等待事件发生。
但是当你的高贵用户决定拔掉他的鼠标并完全触摸时会发生什么...你是否等他触摸你现在挤满的界面,然后在他努力确定你现在拥挤的UI之后立即改变它?
以子弹形式,在https://github.com/Modernizr/Modernizr/issues/869#issuecomment-15264101
引用stucox
- 我们想检测是否存在鼠标
- 在事件发生之前可能无法检测到
- 因此,我们检测的是,如果在此会话中使用了鼠标 - 它不会立即从页面加载
- 我们可能也检测不到没有鼠标 - 它是真的未定义(我认为这比设置更有意义) 它被证明是错误的)
- 我们可能无法检测鼠标是否在会话期间断开连接 - 这与用户只是放弃他们的能力无法区分 小鼠
暂且不说:浏览器知道用户何时插入鼠标/连接键盘,但不会将其暴露给JavaScript .. dang!
这会引导您进行以下操作:
跟踪给定用户的当前功能是复杂,不可靠和可疑的优点
渐进增强的想法在这里很适用。无论用户的上下文如何,都可以构建一种平稳运行的体验。然后根据浏览器功能/媒体查询进行假设,以添加在假定上下文中相对的功能。鼠标的存在只是不同设备上不同用户体验您的网站的众多方式之一。在内核中创建具有优点的东西,不要过多担心人们如何点击按钮。
答案 1 :(得分:56)
如何在文档上侦听mousemove事件。然后,直到您听到该事件,您才认为该设备仅为触摸或键盘。
var mouseDetected = false;
function onMouseMove(e) {
unlisten('mousemove', onMouseMove, false);
mouseDetected = true;
// initializeMouseBehavior();
}
listen('mousemove', onMouseMove, false);
(其中listen
和unlisten
视情况委托给addEventListener
或attachEvent
。)
希望这不会导致太多的视觉冲击,如果您需要基于模式的大量重新布局,它会很糟糕...
答案 2 :(得分:23)
@ Wyatt的回答很棒,让我们有很多想法。
在我的情况下,我选择听第一次互动,然后才设置行为。因此,即使用户有鼠标,如果第一次互动是触摸,我将视为触摸设备。
考虑given order in which events are processed:
我们可以假设如果在触摸之前触发鼠标事件,则它是真正的鼠标事件,而不是模拟事件。示例(使用jQuery):
$(document).ready(function() {
var $body = $('body');
var detectMouse = function(e){
if (e.type === 'mousedown') {
alert('Mouse interaction!');
}
else if (e.type === 'touchstart') {
alert('Touch interaction!');
}
// remove event bindings, so it only runs once
$body.off('mousedown touchstart', detectMouse);
}
// attach both events to body
$body.on('mousedown touchstart', detectMouse);
});
这对我有用
答案 3 :(得分:15)
从2018年开始,有一种很好且可靠的方法来检测浏览器是否具有鼠标(或类似的输入设备): CSS4媒体交互功能,现在几乎所有现代浏览器都支持该功能(除了IE 11和特殊的移动浏览器。
W3C:
指针媒体功能用于查询状态和准确性 指点设备(例如鼠标)。
请参阅以下选项:
/* The primary input mechanism of the device includes a
pointing device of limited accuracy. */
@media (pointer: coarse) { ... }
/* The primary input mechanism of the device
includes an accurate pointing device. */
@media (pointer: fine) { ... }
/* The primary input mechanism of the
device does not include a pointing device. */
@media (pointer: none) { ... }
/* Primary input mechanism system can
hover over elements with ease */
@media (hover: hover) { ... }
/* Primary input mechanism cannot hover
at all or cannot conveniently hover
(e.g., many mobile devices emulate hovering
when the user performs an inconvenient long tap),
or there is no primary pointing input mechanism */
@media (hover: none) { ... }
/* One or more available input mechanism(s)
can hover over elements with ease */
@media (any-hover: hover) { ... }
/* One or more available input mechanism(s) cannot
hover (or there are no pointing input mechanisms) */
@media (any-hover: none) { ... }
媒体查询也可以在JS中使用:
if(window.matchMedia("(any-hover: none)").matches) {
// do sth
}
相关:
正式工作草案:https://drafts.csswg.org/mediaqueries/#mf-interaction
浏览器支持:https://caniuse.com/#search=media%20features
类似的问题:Detect if a client device supports :hover and :focus states
答案 4 :(得分:9)
只能检测浏览器是否触摸能力。没有办法知道它是否真的有触摸屏或鼠标连接。
如果检测到触摸功能,可以通过听触摸事件而不是鼠标事件来优先使用。
跨浏览器检测触摸功能:
function hasTouch() {
return (('ontouchstart' in window) || // html5 browsers
(navigator.maxTouchPoints > 0) || // future IE
(navigator.msMaxTouchPoints > 0)); // current IE10
}
然后可以用它来检查:
if (!hasTouch()) alert('Sorry, need touch!);
或选择要收听的事件:
var eventName = hasTouch() ? 'touchend' : 'click';
someElement.addEventListener(eventName , handlerFunction, false);
或使用单独的触摸与非触摸方法:
if (hasTouch() === true) {
someElement.addEventListener('touchend' , touchHandler, false);
} else {
someElement.addEventListener('click' , mouseHandler, false);
}
function touchHandler(e) {
/// stop event somehow
e.stopPropagation();
e.preventDefault();
window.event.cancelBubble = true;
// ...
return false; // :-)
}
function mouseHandler(e) {
// sorry, touch only - or - do something useful and non-restrictive for user
}
对于鼠标,只能检测鼠标是否正在使用,而不是是否存在。可以设置一个全局标志来指示鼠标是否被使用检测到(类似于现有答案,但简化了一点):
var hasMouse = false;
window.onmousemove = function() {
hasMouse = true;
}
(一个不能包括mouseup
或mousedown
,因为这些事件也可以通过触摸触发)
浏览器限制对低级系统API的访问,这些API需要能够检测其正在使用的系统的硬件功能等功能。
有可能编写一个插件/扩展来访问这些,但是通过JavaScript和DOM这样的检测仅限于此目的,并且必须编写专用于各种OS平台的插件。
总而言之:这种检测只能通过“好猜”来估计。
答案 5 :(得分:7)
当Media Queries Level 4在浏览器中可用时,我们将能够使用“指针”和“悬停”查询来使用鼠标检测设备。
如果我们真的想要将这些信息传达给Javascript,我们可以使用CSS查询根据设备类型设置特定样式,然后在Javascript中使用public ActionResult EditCustomerRequest(int id, string name, string date)
{
CustomerFoodModel CRequest = Helper.getCustomerDetails(id);
CustomerHistoryModel CHModel = Helper. getCustomerHistoryDetails(id);
return PartialView("EditCustomerRequest",new CustomerFoodModel(){
CustomerFood = CRequest,
CustomerHistory = CHModel
});
}
来读取该样式,并派生原始设备从中输入。
但是鼠标可以随时连接或拔掉,并且用户可能想要在触摸和鼠标之间切换。因此,我们可能需要检测此更改,并提供更改界面或自动执行更改。
答案 6 :(得分:6)
由于您计划提供一种在接口之间切换的方法,是否可以简单地要求用户单击链接或按钮以“输入”正确版本的应用程序?然后你可以记住他们对未来访问的偏好。它不是高科技,但它100%可靠: - )
答案 7 :(得分:4)
@SamuelRossille遗憾的是,没有任何我知道的浏览器会暴露鼠标的存在(或缺少它)。
所以,话虽如此,我们只需要尝试使用我们现有的选项...事件做到最好。我知道这并不是你想要的......同意它目前还远非理想。
我们可以尽力确定用户是否在任何特定时刻使用鼠标或触摸。这是一个使用jQuery和amp;的快速而肮脏的例子。敲除:
//namespace
window.ns = {};
// for starters, we'll briefly assume if touch exists, they are using it - default behavior
ns.usingTouch = ko.observable(Modernizr.touch); //using Modernizr here for brevity. Substitute any touch detection method you desire
// now, let's sort out the mouse
ns.usingMouse = ko.computed(function () {
//touch
if (ns.usingTouch()) {
//first, kill the base mousemove event
//I really wish browsers would stop trying to handle this within touch events in the first place
window.document.body.addEventListener('mousemove', function (e) {
e.preventDefault();
e.stopImmediatePropagation();
}, true);
//remove mouse class from body
$('body').removeClass("mouse");
//switch off touch listener
$(document).off(".ns-touch");
// switch on a mouse listener
$(document).on('mousemove.ns-mouse', function (e) {
if (Math.abs(window.lastX - e.clientX) > 0 || window.lastY !== e.clientY) {
ns.usingTouch(false); //this will trigger re-evaluation of ns.usingMouse() and result in ns.usingMouse() === true
}
});
return false;
}
//mouse
else {
//add mouse class to body for styling
$('body').addClass("mouse");
//switch off mouse listener
$(document).off(".ns-mouse");
//switch on a touch listener
$(document).on('touchstart.ns-touch', function () { ns.usingTouch(true) });
return true;
}
});
//tests:
//ns.usingMouse()
//$('body').hasClass('mouse');
您现在可以绑定/订阅usingMouse()& usingTouch()和/或使用 body.mouse 类设置界面样式。一旦检测到鼠标光标并且在touchstart上,界面将立即来回切换。
希望很快我们会从浏览器供应商处获得一些更好的选择。
答案 8 :(得分:2)
Tera-WURFL可以通过将浏览器签名与其数据库进行比较,告诉您访问您网站的设备的功能。看看,它是免费的!
答案 9 :(得分:2)
为什么不检测它是否能够感知触摸和/或对鼠标移动作出反应?
// This will also return false on
// touch-enabled browsers like Chrome
function has_touch() {
return !!('ontouchstart' in window);
}
function has_mouse() {
return !!('onmousemove' in window);
}
答案 10 :(得分:2)
这在类似情况下对我有用。基本上,假设用户没有鼠标,直到您看到一系列连续的鼠标移动,而不会干预mousedowns或mouseups。不是很优雅,但它有效。
var mousedown = false;
var mousemovecount = 0;
function onMouseDown(e){
mousemovecount = 0;
mousedown = true;
}
function onMouseUp(e){
mousedown = false;
mousemovecount = 0;
}
function onMouseMove(e) {
if(!mousedown) {
mousemovecount++;
if(mousemovecount > 5){
window.removeEventListener('mousemove', onMouseMove, false);
console.log("mouse moved");
$('body').addClass('has-mouse');
}
} else {
mousemovecount = 0;
}
}
window.addEventListener('mousemove', onMouseMove, false);
window.addEventListener('mousedown', onMouseDown, false);
window.addEventListener('mouseup', onMouseUp, false);
答案 11 :(得分:0)
运行以下代码段进行尝试:
var msg = (window.matchMedia("(any-pointer: coarse)").matches ? "Touchscreen" : "Mouse");
var msg = (window.matchMedia("(any-pointer: coarse)").matches ? "Touchscreen" : "Mouse");
document.getElementById('info').innerHTML = msg;
body {
background: black;
color: cornflowerblue;
}
#info {
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
font-size: 30vmin;
font-family: verdana, arial, helvetica;
}
<div id='info'></div>
(有关window.matchMedia
的更多信息。)
答案 12 :(得分:0)
我花了数小时为我的Phonegap应用找出了这个问题,然后想到了这个问题。如果触发的事件是“被动”事件,它将生成一个控制台警告,这意味着它不会促动任何更改,但是可以!我将对任何改进或更好的方法感兴趣。但这有效地使我可以普遍使用$ .touch()。
$(document).ready(function(){
$("#aButton").touch(function(origElement, origEvent){
console.log('Original target ID: ' + $(origEvent.target).attr('id'));
});
});
$.fn.touch = function (callback) {
var touch = false;
$(this).on("click", function(e){
if (!touch)
{
console.log("I click!");
let callbackReal = callback.bind(this);
callbackReal(this, e);
}else{
touch = true;
}
touch = false;
});
$(this).on("touchstart", function(e){
if (typeof e.touches != typeof undefined)
{
e.preventDefault();
touch = true;
console.log("I touch!");
let callbackReal = callback.bind(this);
callbackReal(this, e);
}
});
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
<button id="aButton">A Button</button>
答案 13 :(得分:0)
我遇到了同样的问题,一次点击也被注册为一次点击。 在阅读完投票最多的答案的评论后,我想出了自己的解决方案:
var body = document.getElementsByTagName('body')[0];
var mouseCount = 0;
// start in an undefined state
// (i use this to blend in elements once we decide what input is used)
var interactionMode = 'undefined';
var registerMouse = function() {
// count up mouseCount every time, the mousemove event is triggered
mouseCount++;
// but dont set it instantly.
// instead wait 20 miliseconds (seems to be a good value for multiple move actions),
// if another mousemove event accoures switch to mouse as interaction
setTimeout(function() {
// a touch event triggers also exactly 1 mouse move event.
// So only if mouseCount is higher than 1 we are really moving the cursor by mouse.
if (mouseCount > 1) {
body.removeEventListener('mousemove', registerMouse);
body.removeEventListener('touchend', registerTouch);
interactionMode = 'mouse';
console.log('now mousing');
listenTouch();
}
// set the counter to zero again
mouseCount = 0;
}, 20);
};
var registerTouch = function() {
body.removeEventListener('mousemove', registerMouse);
body.removeEventListener('touchend', registerTouch);
interactionMode = 'touch';
console.log('now touching');
mouseCount = 0;
listenMouse();
};
var listenMouse = function() {
body.addEventListener("mousemove", registerMouse);
};
var listenTouch = function() {
body.addEventListener("touchend", registerTouch);
};
listenMouse();
listenTouch();
// after one second without input, assume, that we are touching
// could be adjusted to several seconds or deleted
// without this, the interactionMode would stay 'undefined' until first mouse or touch event
setTimeout(function() {
if (!body.classList.contains('mousing') || body.classList.contains('touching')) {
registerTouch();
}
}, 1000);
/* fix, so that scrolling is possible */
html,
body {
height: 110%;
}
Mouse or touch me
我发现的唯一问题是,您必须能够滚动才能正确检测到触摸事件。单个选项卡(触摸)可能会出现问题。
答案 14 :(得分:0)
刚刚找到了一个我认为非常优雅的解决方案。
// flag as mouse interaction by default
var isMouse = true;
// detect a touch event start and flag it
$(window).on('touchstart', function () {
// if triggers touchstart, toggle flag
// since touch events come before mouse event / click
// callback of mouse event will get in callback
// `isMouse === false`
isMouse = false;
});
// now the code that you want to write
// should also work with `mouse*` events
$('a.someClass').on('click', function () {
if (isMouse) {
// interaction with mouse event
// ...
} else {
// reset for the next event detection
// this is crucial for devices that have both
// touch screen and mouse
isMouse = true;
// interaction with touch event
// ...
}
});
答案 15 :(得分:0)
jQuery中用于检测鼠标使用情况的简单解决方案,解决了移动设备也触发“mousemove”事件的问题。只需添加一个touchstart监听器即可删除mousemove监听器,以便在触摸时不会触发它。
$('body').one('touchstart.test', function(e) {
// Remove the mousemove listener for touch devices
$('body').off('mousemove.test');
}).one('mousemove.test', function(e) {
// MOUSE!
});
当然,该设备仍然可以触摸和鼠标,但上面将保证使用真正的鼠标。
答案 16 :(得分:0)
在各种PC,Linux,iPhone,Android手机和标签上进行一些测试。很奇怪,没有简单的防弹解决方案。当一些具有Touch且没有鼠标的人仍然将Touch和Mouse事件呈现给应用程序时出现问题。由于确实要支持仅鼠标和仅触摸实例,因此希望同时处理这两个实例,但这会导致用户交互的两次出现。如果可以知道设备上没有鼠标,则可以知道忽略伪造/插入的鼠标事件。如果遇到MouseMove,尝试设置标志,但有些浏览器抛出假的MouseMove以及MouseUp和MouseDown。试过检查时间戳,但认为这太冒险了。结论:我发现创建假鼠标事件的浏览器总是在插入的MouseDown之前插入一个MouseMove。在99.99%的情况下,当在具有真实鼠标的系统上运行时,有多个连续的MouseMove事件 - 至少两个。因此,跟踪系统是否遇到两个连续的MouseMove事件,并声明如果从未满足此条件,则不存在鼠标。这可能太简单了,但是我的所有测试设置都在工作。我想我会坚持下去,直到找到更好的解决方案。 - Jim W
答案 17 :(得分:0)
我认为最好的想法是mousemove
听众(目前是最佳答案)。我相信这种方法需要稍微调整一下。确实,基于触摸的浏览器甚至可以模拟mousemove事件,正如您在this iOS discussion中看到的那样,所以我们应该小心一点。
有意义的是,基于触摸的浏览器只会在用户点击屏幕时模拟此事件(用户的手指已关闭)。这意味着我们应该在mousemove处理程序中添加一个测试,以查看事件期间哪个鼠标按钮关闭(如果有)。如果没有鼠标按钮关闭,我们可以安全地假设存在真正的鼠标。如果鼠标按钮关闭,则测试仍然无法确定。
那么如何实施呢?这个question表明在鼠标移动期间检查哪个鼠标按钮关闭的最可靠方法是实际监听文档级别中的3个事件:mousemove,mousedown和mouseup。向上和向下只会设置一个全局布尔标志。此举将进行测试。如果你有一个移动并且布尔值为假,我们可以假设存在一个鼠标。有关确切的代码示例,请参阅问题。
最后一条评论..此测试并不理想,因为它无法在加载时执行。因此,我会使用先前建议的渐进增强方法。默认情况下显示不支持鼠标特定悬停界面的版本。如果发现鼠标,请使用JS在运行时启用此模式。这应该尽可能与用户无缝连接。
为了支持用户配置的更改(即鼠标已断开连接),您可以定期重新测试。虽然我认为在这种情况下更好地通知用户有关这两种模式并让用户在它们之间手动切换(很像移动/桌面选择,总是可以颠倒)。
答案 18 :(得分:0)
我认为不可能识别仅限触控的设备(据我所知)。主要问题是触摸设备也触发了所有鼠标和键盘事件。 请参阅以下示例,两个警报都会为触摸设备返回true。
function is_touch_present() {
return ('ontouchstart' in window) || ('onmsgesturechange' in window);
}
function is_mouse_present() {
return (('onmousedown' in window) && ('onmouseup' in window) && ('onmousemove' in window) && ('onclick' in window) && ('ondblclick' in window) && ('onmousemove' in window) && ('onmouseover' in window) && ('onmouseout' in window) && ('oncontextmenu' in window));
}
alert("Touch Present: " + is_touch_present());
alert("Mouse Present: " + is_mouse_present());
答案 19 :(得分:0)
正如其他人所指出的,明确地检测他们是否有鼠标是不可靠的。这可以很容易地改变,具体取决于设备。对于布尔值为true或false,这绝对是你无法做到的事情,至少在文档规模上是这样。
触摸事件和鼠标事件是独占的。所以这有助于采取不同的行动。问题是触摸事件更接近鼠标向上/向下/移动事件,并且还触发点击事件。
从你的问题,你说你想要悬停预览。除此之外,我不知道有关您的界面的任何其他细节。我假设由于缺少鼠标而你想要点击预览,而点击会因为悬停预览而执行不同的动作。
如果是这种情况,您可以采取一些懒惰的检测方法:
onclick事件将始终带有带鼠标的onmouseover事件。因此请注意鼠标位于已单击的元素的顶部。
您可以使用文档范围的onmousemove事件执行此操作。您可以使用event.target
来记录鼠标所在的元素。然后在您的onclick事件中,您可以检查鼠标是否实际位于被单击的元素上(或元素的子元素)。
从那里你可以选择依赖于两者的click事件,并根据结果采取A或B动作。如果某些触摸设备没有发出点击事件(而您必须依赖ontouch *事件),则B动作可能无效。
答案 20 :(得分:0)
看看modenizr的一个功能是触摸
http://modernizr.com/docs/#features-misc
虽然我没有完全测试,但似乎效果很好
这也是来自modernizr页面的链接 http://www.html5rocks.com/en/mobile/touchandmouse/
答案 21 :(得分:0)
我在这里看到的主要问题是,大多数触控设备会触发鼠标事件以及相应的触控设备(touchstart - &gt; mousedown,touchmove - &gt; mousemove等)。 对于仅有键盘的键盘,最后用于现代键盘,它们具有通用浏览器,因此您甚至无法检测到MouseEvent类的存在。
在我看来,这里不那么痛苦的解决方案是在启动时显示一个菜单(仅对键盘用户进行'alt'管理),并且可以将选项存储在localStorage / cookies / serverside中,或者保留salme下次访客来时选择。
答案 22 :(得分:-4)
我已阅读并尝试了所有建议,但没有足够的结果。
然后我调查了一些,发现了这段代码 - device.js
我在客户的网站上使用它来检测鼠标存在:
(<html>
应该有desktop
类)并且似乎非常好,对于touch
支持,我只是进行常规检查'ontouchend' in document
并使用来自两个检测的信息来假设具体我需要的东西。
答案 23 :(得分:-4)
我强烈建议不要采用这种方法。考虑触摸屏,桌面大小的设备,您需要解决一组不同的问题。
请在没有鼠标的情况下使用您的应用程序(即无悬停预览)。
答案 24 :(得分:-7)
通常最好检测是否支持鼠标悬停功能,而不是检测操作系统/浏览器类型。您可以通过以下方式完成此操作:
if (element.mouseover) {
//Supports the mouseover event
}
请确保不执行以下操作:
if (element.mouseover()) {
// Supports the mouseover event
}
后者实际上会调用该方法,而不是检查它的存在。
您可以在此处阅读更多内容: http://www.quirksmode.org/js/support.html