必须禁用双击缩放并适配安全区域:在HTML中添加viewport禁用缩放,Native端关闭scrollView回弹或侧滑返回;H5与Native通信需用WKScriptMessageHandler或JSBridge加密传输密码;触摸坐标须用getBoundingClientRect计算相对位置,并在capture阶段绑定touchmove事件。
iOS WebView 默认启用双击缩放,而手势密码依赖精确的 touchstart/touchmove 坐标判断,一旦触发缩放,e.touches[0].clientX 会因视口缩放失真,导致点选错位甚至无法识别路径。这不是 bug,是 Safari 渲染层行为。
中加入:
webView.scrollView.bounces = NO(UIWebView)或 webView.allowsBackForwardNavigationGestures = false(WKWebView 防止侧滑返回干扰)user-scalable=no —— iOS 10+ 对缺失此项非常敏感HTML5 手势密码完成绘制后,需把密码数组(如 [2,5,8,6])安全传回 iOS,不能靠 window.location.hash 或轮询 localStorage,既不安全也不可靠。
WKScriptMessageHandler:H5 调用 window.webkit.messageHandlers.gestureResult.postMessage(pwdArray),Native 在 userContentController:didReceive: 中接收stringByEvaluatingJavaScriptFromString: 注册全局回调函数,H5 执行 window.nativeCallback(pwdArray)
iOS 设备(尤其 iPhone X 及以后)存在刘海、Home Indicator 区域,document.documentElement.clientHeight 不等于可视区域真实高度,直接用 clientX/clientY 计算九宫格坐标会导致点位整体下移或错位。
const safeArea = window
.getComputedStyle(document.documentElement).getPropertyValue('--safe-area-inset-bottom')(需 CSS 注入支持),或用 window.innerHeight 替代 screen.height
getBoundingClientRect() 获取 canvas 实际渲染位置,再结合 e.touches[0].clientX - rect.left 算相对坐标,彻底避开 viewport 缩放和安全区干扰为防止手势过程中页面意外滚动,H5 必须调用 e.preventDefault(),但在 WKWebView 中,如果绑定在 bubbling 阶段(默认),部分 touchmove 事件仍会透出并触发滚动。
立即学习“前端免费学习笔记(深入)”;
canvas.addEventListener('touchmove', handler, { passive: false, capture: true })
passive: false 是关键——iOS Safari 默认给 touch 事件加了 passive=true,此时再调 preventDefault() 会被忽略且抛 warninginputClass: Hammer.SafariMobileInput,否则多点触控识别率骤降原生调用 HTML5 手势密码最易被忽略的不是加密逻辑,而是坐标系对齐和事件拦截时机——这两个点一旦出问题,用户“明明画对了却提示错误”,排查起来极难定位到 WebView 层。
来电咨询