input-observer.ts 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959
  1. /**
  2. * Copyright (c) 2018-2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. * @author David Sehnal <david.sehnal@gmail.com>
  6. * @author Russell Parker <russell@benchling.com>
  7. */
  8. import { Subject, Observable } from 'rxjs';
  9. import { Viewport } from '../../mol-canvas3d/camera/util';
  10. import { Vec2, EPSILON } from '../../mol-math/linear-algebra';
  11. import { BitFlags, noop } from '../../mol-util';
  12. export function getButtons(event: MouseEvent | Touch) {
  13. if (typeof event === 'object') {
  14. if ('buttons' in event) {
  15. return event.buttons;
  16. } else if ('which' in event) {
  17. const b = (event as any).which; // 'any' to support older browsers
  18. if (b === 2) {
  19. return 4;
  20. } else if (b === 3) {
  21. return 2;
  22. } else if (b > 0) {
  23. return 1 << (b - 1);
  24. }
  25. }
  26. }
  27. return 0;
  28. }
  29. export function getButton(event: MouseEvent | Touch) {
  30. if (typeof event === 'object') {
  31. if ('button' in event) {
  32. const b = event.button;
  33. if (b === 1) {
  34. return 4;
  35. } else if (b === 2) {
  36. return 2;
  37. } else if (b >= 0) {
  38. return 1 << b;
  39. }
  40. }
  41. }
  42. return 0;
  43. }
  44. export function getModifiers(event: MouseEvent | Touch): ModifiersKeys {
  45. return {
  46. alt: 'altKey' in event ? event.altKey : false,
  47. shift: 'shiftKey' in event ? event.shiftKey : false,
  48. control: 'ctrlKey' in event ? event.ctrlKey : false,
  49. meta: 'metaKey' in event ? event.metaKey : false
  50. };
  51. }
  52. export const DefaultInputObserverProps = {
  53. noScroll: true,
  54. noMiddleClickScroll: true,
  55. noContextMenu: true,
  56. noPinchZoom: true,
  57. noTextSelect: true,
  58. preventGestures: false,
  59. mask: (x: number, y: number) => true,
  60. pixelScale: 1
  61. };
  62. export type InputObserverProps = Partial<typeof DefaultInputObserverProps>
  63. export type ModifiersKeys = {
  64. shift: boolean,
  65. alt: boolean,
  66. control: boolean,
  67. meta: boolean
  68. }
  69. export namespace ModifiersKeys {
  70. export const None = create();
  71. export function areEqual(a: ModifiersKeys, b: ModifiersKeys) {
  72. return a.shift === b.shift && a.alt === b.alt && a.control === b.control && a.meta === b.meta;
  73. }
  74. export function size(a?: ModifiersKeys) {
  75. if (!a) return 0;
  76. let ret = 0;
  77. if (!!a.shift) ret++;
  78. if (!!a.alt) ret++;
  79. if (!!a.control) ret++;
  80. if (!!a.meta) ret++;
  81. return ret;
  82. }
  83. export function create(modifierKeys: Partial<ModifiersKeys> = {}): ModifiersKeys {
  84. return {
  85. shift: !!modifierKeys.shift,
  86. alt: !!modifierKeys.alt,
  87. control: !!modifierKeys.control,
  88. meta: !!modifierKeys.meta
  89. };
  90. }
  91. }
  92. export type ButtonsType = BitFlags<ButtonsType.Flag>
  93. export namespace ButtonsType {
  94. export const has: (btn: ButtonsType, f: Flag) => boolean = BitFlags.has;
  95. export const create: (fs: Flag) => ButtonsType = BitFlags.create;
  96. export enum Flag {
  97. /** No button or un-initialized */
  98. None = 0x0,
  99. /** Primary button (usually left) */
  100. Primary = 0x1,
  101. /** Secondary button (usually right) */
  102. Secondary = 0x2,
  103. /** Auxilary button (usually middle or mouse wheel button) */
  104. Auxilary = 0x4,
  105. /** 4th button (typically the "Browser Back" button) */
  106. Forth = 0x8,
  107. /** 5th button (typically the "Browser Forward" button) */
  108. Five = 0x10,
  109. }
  110. }
  111. export type KeyCode = string
  112. type BaseInput = {
  113. buttons: ButtonsType
  114. button: ButtonsType.Flag
  115. modifiers: ModifiersKeys
  116. }
  117. export type DragInput = {
  118. x: number,
  119. y: number,
  120. dx: number,
  121. dy: number,
  122. pageX: number,
  123. pageY: number,
  124. isStart: boolean
  125. } & BaseInput
  126. export type WheelInput = {
  127. x: number,
  128. y: number,
  129. pageX: number,
  130. pageY: number,
  131. dx: number,
  132. dy: number,
  133. dz: number,
  134. spinX: number,
  135. spinY: number
  136. } & BaseInput
  137. export type ClickInput = {
  138. x: number,
  139. y: number,
  140. pageX: number,
  141. pageY: number,
  142. } & BaseInput
  143. export type MoveInput = {
  144. x: number,
  145. y: number,
  146. pageX: number,
  147. pageY: number,
  148. movementX?: number,
  149. movementY?: number,
  150. inside: boolean,
  151. // Move is subscribed to window element
  152. // This indicates that the event originated from the element the InputObserver was created on
  153. onElement: boolean
  154. } & BaseInput
  155. export type PinchInput = {
  156. delta: number,
  157. fraction: number,
  158. fractionDelta: number,
  159. distance: number,
  160. isStart: boolean
  161. } & BaseInput
  162. export type GestureInput = {
  163. scale: number,
  164. rotation: number,
  165. deltaScale: number,
  166. deltaRotation: number
  167. isStart?: boolean,
  168. isEnd?: boolean
  169. }
  170. export type KeyInput = {
  171. key: string,
  172. code: string,
  173. modifiers: ModifiersKeys
  174. x: number,
  175. y: number,
  176. pageX: number,
  177. pageY: number,
  178. /** for overwriting browser shortcuts like `ctrl+s` as needed */
  179. preventDefault: () => void
  180. }
  181. export const EmptyKeyInput: KeyInput = {
  182. key: '',
  183. code: '',
  184. modifiers: ModifiersKeys.None,
  185. x: -1,
  186. y: -1,
  187. pageX: -1,
  188. pageY: -1,
  189. preventDefault: noop,
  190. };
  191. export type ResizeInput = {
  192. }
  193. enum DraggingState {
  194. Stopped = 0,
  195. Started = 1,
  196. Moving = 2
  197. }
  198. type PointerEvent = {
  199. clientX: number
  200. clientY: number
  201. pageX: number
  202. pageY: number
  203. movementX?: number
  204. movementY?: number
  205. target: EventTarget | null
  206. preventDefault?: () => void
  207. }
  208. type GestureEvent = {
  209. scale: number,
  210. rotation: number,
  211. } & MouseEvent
  212. interface InputObserver {
  213. noScroll: boolean
  214. noContextMenu: boolean
  215. readonly width: number
  216. readonly height: number
  217. readonly pixelRatio: number
  218. readonly pointerLock: boolean
  219. readonly drag: Observable<DragInput>,
  220. // Equivalent to mouseUp and touchEnd
  221. readonly interactionEnd: Observable<undefined>,
  222. readonly wheel: Observable<WheelInput>,
  223. readonly pinch: Observable<PinchInput>,
  224. readonly gesture: Observable<GestureInput>,
  225. readonly click: Observable<ClickInput>,
  226. readonly move: Observable<MoveInput>,
  227. readonly leave: Observable<undefined>,
  228. readonly enter: Observable<undefined>,
  229. readonly resize: Observable<ResizeInput>,
  230. readonly modifiers: Observable<ModifiersKeys>
  231. readonly key: Observable<KeyInput>
  232. readonly keyUp: Observable<KeyInput>
  233. readonly keyDown: Observable<KeyInput>
  234. readonly lock: Observable<boolean>
  235. requestPointerLock: (viewport: Viewport) => void
  236. exitPointerLock: () => void
  237. dispose: () => void
  238. }
  239. function createEvents() {
  240. return {
  241. drag: new Subject<DragInput>(),
  242. interactionEnd: new Subject<undefined>(),
  243. click: new Subject<ClickInput>(),
  244. move: new Subject<MoveInput>(),
  245. wheel: new Subject<WheelInput>(),
  246. pinch: new Subject<PinchInput>(),
  247. gesture: new Subject<GestureInput>(),
  248. resize: new Subject<ResizeInput>(),
  249. leave: new Subject<undefined>(),
  250. enter: new Subject<undefined>(),
  251. modifiers: new Subject<ModifiersKeys>(),
  252. key: new Subject<KeyInput>(),
  253. keyUp: new Subject<KeyInput>(),
  254. keyDown: new Subject<KeyInput>(),
  255. lock: new Subject<boolean>(),
  256. };
  257. }
  258. const AllowedNonPrintableKeys = ['Backspace', 'Delete'];
  259. namespace InputObserver {
  260. export function create(props: InputObserverProps = {}): InputObserver {
  261. const { noScroll, noContextMenu } = { ...DefaultInputObserverProps, ...props };
  262. return {
  263. noScroll,
  264. noContextMenu,
  265. pointerLock: false,
  266. width: 0,
  267. height: 0,
  268. pixelRatio: 1,
  269. ...createEvents(),
  270. requestPointerLock: noop,
  271. exitPointerLock: noop,
  272. dispose: noop
  273. };
  274. }
  275. export function fromElement(element: Element, props: InputObserverProps = {}): InputObserver {
  276. let { noScroll, noMiddleClickScroll, noContextMenu, noPinchZoom, noTextSelect, mask, pixelScale, preventGestures } = { ...DefaultInputObserverProps, ...props };
  277. let width = element.clientWidth * pixelRatio();
  278. let height = element.clientHeight * pixelRatio();
  279. let isLocked = false;
  280. let lockedViewport = Viewport();
  281. let lastTouchDistance = 0, lastTouchFraction = 0;
  282. const pointerDown = Vec2();
  283. const pointerStart = Vec2();
  284. const pointerEnd = Vec2();
  285. const pointerDelta = Vec2();
  286. const rectSize = Vec2();
  287. const modifierKeys: ModifiersKeys = {
  288. shift: false,
  289. alt: false,
  290. control: false,
  291. meta: false
  292. };
  293. const position = {
  294. x: -1,
  295. y: -1,
  296. pageX: -1,
  297. pageY: -1,
  298. };
  299. function pixelRatio() {
  300. return window.devicePixelRatio * pixelScale;
  301. }
  302. function getModifierKeys(): ModifiersKeys {
  303. return { ...modifierKeys };
  304. }
  305. function getKeyOnElement(event: Event): boolean {
  306. return event.target === document.body || event.target === element;
  307. }
  308. let dragging: DraggingState = DraggingState.Stopped;
  309. let disposed = false;
  310. let buttons = ButtonsType.create(ButtonsType.Flag.None);
  311. let button = ButtonsType.Flag.None;
  312. let isInside = false;
  313. let hasMoved = false;
  314. let resizeObserver: ResizeObserver | undefined;
  315. if (typeof window.ResizeObserver !== 'undefined') {
  316. resizeObserver = new window.ResizeObserver(onResize);
  317. }
  318. const events = createEvents();
  319. const { drag, interactionEnd, wheel, pinch, gesture, click, move, leave, enter, resize, modifiers, key, keyUp, keyDown, lock } = events;
  320. attach();
  321. function attach() {
  322. element.addEventListener('contextmenu', onContextMenu as any, false);
  323. element.addEventListener('wheel', onMouseWheel as any, false);
  324. element.addEventListener('mousedown', onMouseDown as any, false);
  325. // for dragging to work outside canvas bounds,
  326. // mouse move/up events have to be added to a parent, i.e. window
  327. window.addEventListener('mousemove', onMouseMove as any, false);
  328. window.addEventListener('mouseup', onMouseUp as any, false);
  329. element.addEventListener('touchstart', onTouchStart as any, false);
  330. element.addEventListener('touchmove', onTouchMove as any, false);
  331. element.addEventListener('touchend', onTouchEnd as any, false);
  332. element.addEventListener('gesturechange', onGestureChange as any, false);
  333. element.addEventListener('gesturestart', onGestureStart as any, false);
  334. element.addEventListener('gestureend', onGestureEnd as any, false);
  335. // reset buttons and modifier keys state when browser window looses focus
  336. window.addEventListener('blur', handleBlur);
  337. window.addEventListener('keyup', handleKeyUp as EventListener, false);
  338. window.addEventListener('keydown', handleKeyDown as EventListener, false);
  339. window.addEventListener('keypress', handleKeyPress as EventListener, false);
  340. document.addEventListener('pointerlockchange', onPointerLockChange, false);
  341. document.addEventListener('pointerlockerror', onPointerLockError, false);
  342. if (resizeObserver != null) {
  343. resizeObserver.observe(element.parentElement!);
  344. } else {
  345. window.addEventListener('resize', onResize, false);
  346. }
  347. }
  348. function dispose() {
  349. if (disposed) return;
  350. disposed = true;
  351. element.removeEventListener('contextmenu', onContextMenu as any, false);
  352. element.removeEventListener('wheel', onMouseWheel as any, false);
  353. element.removeEventListener('mousedown', onMouseDown as any, false);
  354. window.removeEventListener('mousemove', onMouseMove as any, false);
  355. window.removeEventListener('mouseup', onMouseUp as any, false);
  356. element.removeEventListener('touchstart', onTouchStart as any, false);
  357. element.removeEventListener('touchmove', onTouchMove as any, false);
  358. element.removeEventListener('touchend', onTouchEnd as any, false);
  359. element.removeEventListener('gesturechange', onGestureChange as any, false);
  360. element.removeEventListener('gesturestart', onGestureStart as any, false);
  361. element.removeEventListener('gestureend', onGestureEnd as any, false);
  362. window.removeEventListener('blur', handleBlur);
  363. window.removeEventListener('keyup', handleKeyUp as EventListener, false);
  364. window.removeEventListener('keydown', handleKeyDown as EventListener, false);
  365. window.removeEventListener('keypress', handleKeyPress as EventListener, false);
  366. document.removeEventListener('pointerlockchange', onPointerLockChange, false);
  367. document.removeEventListener('pointerlockerror', onPointerLockError, false);
  368. cross.remove();
  369. if (resizeObserver != null) {
  370. resizeObserver.unobserve(element.parentElement!);
  371. resizeObserver.disconnect();
  372. } else {
  373. window.removeEventListener('resize', onResize, false);
  374. }
  375. }
  376. function onPointerLockChange() {
  377. if (element.ownerDocument.pointerLockElement === element) {
  378. isLocked = true;
  379. } else {
  380. isLocked = false;
  381. }
  382. toggleCross(isLocked);
  383. lock.next(isLocked);
  384. }
  385. function onPointerLockError() {
  386. console.error('Unable to use Pointer Lock API');
  387. isLocked = false;
  388. toggleCross(isLocked);
  389. lock.next(isLocked);
  390. }
  391. function onContextMenu(event: MouseEvent) {
  392. if (!mask(event.clientX, event.clientY)) return;
  393. if (noContextMenu) {
  394. event.preventDefault();
  395. }
  396. }
  397. function updateModifierKeys(event: MouseEvent | WheelEvent | TouchEvent) {
  398. modifierKeys.alt = event.altKey;
  399. modifierKeys.shift = event.shiftKey;
  400. modifierKeys.control = event.ctrlKey;
  401. modifierKeys.meta = event.metaKey;
  402. }
  403. function handleBlur() {
  404. if (buttons || modifierKeys.shift || modifierKeys.alt || modifierKeys.meta || modifierKeys.control) {
  405. buttons = 0 as ButtonsType;
  406. modifierKeys.shift = modifierKeys.alt = modifierKeys.control = modifierKeys.meta = false;
  407. }
  408. }
  409. function handleKeyDown(event: KeyboardEvent) {
  410. let changed = false;
  411. if (!modifierKeys.alt && event.altKey) { changed = true; modifierKeys.alt = true; }
  412. if (!modifierKeys.shift && event.shiftKey) { changed = true; modifierKeys.shift = true; }
  413. if (!modifierKeys.control && event.ctrlKey) { changed = true; modifierKeys.control = true; }
  414. if (!modifierKeys.meta && event.metaKey) { changed = true; modifierKeys.meta = true; }
  415. if (changed && isInside) modifiers.next(getModifierKeys());
  416. if (getKeyOnElement(event) && isInside) {
  417. keyDown.next({
  418. key: event.key,
  419. code: event.code,
  420. modifiers: getModifierKeys(),
  421. ...position,
  422. preventDefault: () => event.preventDefault(),
  423. });
  424. }
  425. }
  426. function handleKeyUp(event: KeyboardEvent) {
  427. let changed = false;
  428. if (modifierKeys.alt && !event.altKey) { changed = true; modifierKeys.alt = false; }
  429. if (modifierKeys.shift && !event.shiftKey) { changed = true; modifierKeys.shift = false; }
  430. if (modifierKeys.control && !event.ctrlKey) { changed = true; modifierKeys.control = false; }
  431. if (modifierKeys.meta && !event.metaKey) { changed = true; modifierKeys.meta = false; }
  432. if (changed && isInside) modifiers.next(getModifierKeys());
  433. if (AllowedNonPrintableKeys.includes(event.key)) handleKeyPress(event);
  434. if (getKeyOnElement(event) && isInside) {
  435. keyUp.next({
  436. key: event.key,
  437. code: event.code,
  438. modifiers: getModifierKeys(),
  439. ...position,
  440. preventDefault: () => event.preventDefault(),
  441. });
  442. }
  443. }
  444. function handleKeyPress(event: KeyboardEvent) {
  445. if (!getKeyOnElement(event) || !isInside) return;
  446. key.next({
  447. key: event.key,
  448. code: event.code,
  449. modifiers: getModifierKeys(),
  450. ...position,
  451. preventDefault: () => event.preventDefault(),
  452. });
  453. }
  454. function getCenterTouch(ev: TouchEvent): PointerEvent {
  455. const t0 = ev.touches[0];
  456. const t1 = ev.touches[1];
  457. return {
  458. clientX: (t0.clientX + t1.clientX) / 2,
  459. clientY: (t0.clientY + t1.clientY) / 2,
  460. pageX: (t0.pageX + t1.pageX) / 2,
  461. pageY: (t0.pageY + t1.pageY) / 2,
  462. target: ev.target
  463. };
  464. }
  465. function getTouchDistance(ev: TouchEvent) {
  466. const dx = ev.touches[0].pageX - ev.touches[1].pageX;
  467. const dy = ev.touches[0].pageY - ev.touches[1].pageY;
  468. return Math.sqrt(dx * dx + dy * dy);
  469. }
  470. function onTouchStart(ev: TouchEvent) {
  471. ev.preventDefault();
  472. if (ev.touches.length === 1) {
  473. buttons = button = ButtonsType.Flag.Primary;
  474. onPointerDown(ev.touches[0]);
  475. } else if (ev.touches.length === 2) {
  476. buttons = ButtonsType.Flag.Secondary & ButtonsType.Flag.Auxilary;
  477. button = ButtonsType.Flag.Secondary;
  478. onPointerDown(getCenterTouch(ev));
  479. const touchDistance = getTouchDistance(ev);
  480. lastTouchDistance = touchDistance;
  481. pinch.next({
  482. distance: touchDistance,
  483. fraction: 1,
  484. fractionDelta: 0,
  485. delta: 0,
  486. isStart: true,
  487. buttons,
  488. button,
  489. modifiers: getModifierKeys()
  490. });
  491. } else if (ev.touches.length === 3) {
  492. buttons = button = ButtonsType.Flag.Forth;
  493. onPointerDown(getCenterTouch(ev));
  494. }
  495. }
  496. function onTouchEnd(ev: TouchEvent) {
  497. endDrag();
  498. }
  499. function onTouchMove(ev: TouchEvent) {
  500. button = ButtonsType.Flag.None;
  501. if (noPinchZoom) {
  502. ev.preventDefault();
  503. ev.stopPropagation();
  504. if ((ev as any).originalEvent) {
  505. (ev as any).originalEvent.preventDefault();
  506. (ev as any).originalEvent.stopPropagation();
  507. }
  508. }
  509. if (ev.touches.length === 1) {
  510. buttons = ButtonsType.Flag.Primary;
  511. onPointerMove(ev.touches[0]);
  512. } else if (ev.touches.length === 2) {
  513. const touchDistance = getTouchDistance(ev);
  514. const touchDelta = lastTouchDistance - touchDistance;
  515. if (Math.abs(touchDelta) < 4) {
  516. buttons = ButtonsType.Flag.Secondary;
  517. onPointerMove(getCenterTouch(ev));
  518. } else {
  519. buttons = ButtonsType.Flag.Auxilary;
  520. updateModifierKeys(ev);
  521. const fraction = lastTouchDistance / touchDistance;
  522. pinch.next({
  523. delta: touchDelta,
  524. fraction,
  525. fractionDelta: lastTouchFraction - fraction,
  526. distance: touchDistance,
  527. isStart: false,
  528. buttons,
  529. button,
  530. modifiers: getModifierKeys()
  531. });
  532. lastTouchFraction = fraction;
  533. }
  534. lastTouchDistance = touchDistance;
  535. } else if (ev.touches.length === 3) {
  536. buttons = ButtonsType.Flag.Forth;
  537. onPointerMove(getCenterTouch(ev));
  538. }
  539. }
  540. function onMouseDown(ev: MouseEvent) {
  541. updateModifierKeys(ev);
  542. buttons = getButtons(ev);
  543. button = getButton(ev);
  544. if (noMiddleClickScroll && buttons === ButtonsType.Flag.Auxilary) {
  545. ev.preventDefault;
  546. }
  547. onPointerDown(ev);
  548. }
  549. function onMouseMove(ev: MouseEvent) {
  550. updateModifierKeys(ev);
  551. buttons = getButtons(ev);
  552. button = ButtonsType.Flag.None;
  553. onPointerMove(ev);
  554. }
  555. function onMouseUp(ev: MouseEvent) {
  556. updateModifierKeys(ev);
  557. buttons = getButtons(ev);
  558. button = getButton(ev);
  559. onPointerUp(ev);
  560. endDrag();
  561. }
  562. function endDrag() {
  563. interactionEnd.next(void 0);
  564. }
  565. function onPointerDown(ev: PointerEvent) {
  566. if (!mask(ev.clientX, ev.clientY)) return;
  567. eventOffset(pointerStart, ev);
  568. Vec2.copy(pointerDown, pointerStart);
  569. if (insideBounds(pointerStart)) {
  570. dragging = DraggingState.Started;
  571. }
  572. }
  573. function onPointerUp(ev: PointerEvent) {
  574. dragging = DraggingState.Stopped;
  575. if (!mask(ev.clientX, ev.clientY)) return;
  576. eventOffset(pointerEnd, ev);
  577. if (!hasMoved && Vec2.distance(pointerEnd, pointerDown) < 4) {
  578. const { pageX, pageY } = getPagePosition(ev);
  579. const [x, y] = pointerEnd;
  580. click.next({ x, y, pageX, pageY, buttons, button, modifiers: getModifierKeys() });
  581. }
  582. hasMoved = false;
  583. }
  584. function onPointerMove(ev: PointerEvent) {
  585. eventOffset(pointerEnd, ev);
  586. const { pageX, pageY } = getPagePosition(ev);
  587. const [x, y] = pointerEnd;
  588. const { movementX, movementY } = ev;
  589. const inside = insideBounds(pointerEnd) && mask(ev.clientX, ev.clientY);
  590. if (isInside && !inside) {
  591. leave.next(void 0);
  592. } else if (!isInside && inside) {
  593. enter.next(void 0);
  594. }
  595. isInside = inside;
  596. position.x = x;
  597. position.y = y;
  598. position.pageX = pageX;
  599. position.pageY = pageY;
  600. move.next({ x, y, pageX, pageY, movementX, movementY, buttons, button, modifiers: getModifierKeys(), inside, onElement: ev.target === element });
  601. if (dragging === DraggingState.Stopped) return;
  602. if (noTextSelect) {
  603. ev.preventDefault?.();
  604. }
  605. Vec2.div(pointerDelta, Vec2.sub(pointerDelta, pointerEnd, pointerStart), getClientSize(rectSize));
  606. if (Vec2.magnitude(pointerDelta) < EPSILON) return;
  607. const isStart = dragging === DraggingState.Started;
  608. if (isStart && !mask(ev.clientX, ev.clientY)) return;
  609. if (Vec2.distance(pointerEnd, pointerDown) >= 4) {
  610. hasMoved = true;
  611. }
  612. const [dx, dy] = pointerDelta;
  613. drag.next({ x, y, dx, dy, pageX, pageY, buttons, button, modifiers: getModifierKeys(), isStart });
  614. Vec2.copy(pointerStart, pointerEnd);
  615. dragging = DraggingState.Moving;
  616. }
  617. function onMouseWheel(ev: WheelEvent) {
  618. if (!mask(ev.clientX, ev.clientY)) return;
  619. eventOffset(pointerEnd, ev);
  620. const { pageX, pageY } = getPagePosition(ev);
  621. const [x, y] = pointerEnd;
  622. if (noScroll) {
  623. ev.preventDefault();
  624. }
  625. const normalized = normalizeWheel(ev);
  626. buttons = button = ButtonsType.Flag.Auxilary;
  627. if (normalized.dx || normalized.dy || normalized.dz) {
  628. wheel.next({ x, y, pageX, pageY, ...normalized, buttons, button, modifiers: getModifierKeys() });
  629. }
  630. }
  631. function tryPreventGesture(ev: GestureEvent) {
  632. // console.log(ev, preventGestures);
  633. if (!preventGestures) return;
  634. ev.preventDefault();
  635. ev.stopImmediatePropagation?.();
  636. ev.stopPropagation?.();
  637. }
  638. let prevGestureScale = 0, prevGestureRotation = 0;
  639. function onGestureStart(ev: GestureEvent) {
  640. tryPreventGesture(ev);
  641. prevGestureScale = ev.scale;
  642. prevGestureRotation = ev.rotation;
  643. gesture.next({ scale: ev.scale, rotation: ev.rotation, deltaRotation: 0, deltaScale: 0, isStart: true });
  644. }
  645. function gestureDelta(ev: GestureEvent, isEnd?: boolean) {
  646. gesture.next({
  647. scale: ev.scale,
  648. rotation: ev.rotation,
  649. deltaRotation: prevGestureRotation - ev.rotation,
  650. deltaScale: prevGestureScale - ev.scale,
  651. isEnd
  652. });
  653. prevGestureRotation = ev.rotation;
  654. prevGestureScale = ev.scale;
  655. }
  656. function onGestureChange(ev: GestureEvent) {
  657. tryPreventGesture(ev);
  658. gestureDelta(ev);
  659. }
  660. function onGestureEnd(ev: GestureEvent) {
  661. tryPreventGesture(ev);
  662. gestureDelta(ev, true);
  663. }
  664. function onResize() {
  665. resize.next({});
  666. }
  667. function insideBounds(pos: Vec2) {
  668. if (element instanceof Window || element instanceof Document || element === document.body) {
  669. return true;
  670. } else {
  671. const rect = element.getBoundingClientRect();
  672. return pos[0] >= 0 && pos[1] >= 0 && pos[0] < rect.width && pos[1] < rect.height;
  673. }
  674. }
  675. function getClientSize(out: Vec2) {
  676. out[0] = element.clientWidth;
  677. out[1] = element.clientHeight;
  678. return out;
  679. }
  680. function eventOffset(out: Vec2, ev: PointerEvent) {
  681. width = element.clientWidth * pixelRatio();
  682. height = element.clientHeight * pixelRatio();
  683. if (isLocked) {
  684. const pr = pixelRatio();
  685. out[0] = (lockedViewport.x + lockedViewport.width / 2) / pr;
  686. out[1] = (height - (lockedViewport.y + lockedViewport.height / 2)) / pr;
  687. } else {
  688. const rect = element.getBoundingClientRect();
  689. out[0] = (ev.clientX || 0) - rect.left;
  690. out[1] = (ev.clientY || 0) - rect.top;
  691. }
  692. return out;
  693. }
  694. function getPagePosition(ev: PointerEvent) {
  695. if (isLocked) {
  696. return {
  697. pageX: Math.round(window.innerWidth / 2) + lockedViewport.x,
  698. pageY: Math.round(window.innerHeight / 2) + lockedViewport.y
  699. };
  700. } else {
  701. return {
  702. pageX: ev.pageX,
  703. pageY: ev.pageY
  704. };
  705. }
  706. }
  707. const cross = addCross();
  708. const crossWidth = 30;
  709. function addCross() {
  710. const cross = document.createElement('div');
  711. const b = '30%';
  712. const t = '10%';
  713. const c = `#000 ${b}, #0000 0 calc(100% - ${b}), #000 0`;
  714. const vline = `linear-gradient(0deg, ${c}) 50%/${t} 100% no-repeat`;
  715. const hline = `linear-gradient(90deg, ${c}) 50%/100% ${t} no-repeat`;
  716. const cdot = 'radial-gradient(circle at 50%, #000 5%, #0000 5%)';
  717. Object.assign(cross.style, {
  718. width: `${crossWidth}px`,
  719. aspectRatio: 1,
  720. background: `${vline}, ${hline}, ${cdot}`,
  721. display: 'none',
  722. zIndex: 1000,
  723. position: 'absolute',
  724. mixBlendMode: 'difference',
  725. filter: 'invert(1)',
  726. });
  727. element.parentElement?.appendChild(cross);
  728. return cross;
  729. }
  730. function toggleCross(value: boolean) {
  731. cross.style.display = value ? 'block' : 'none';
  732. if (value) {
  733. const pr = pixelRatio();
  734. const offsetX = (lockedViewport.x + lockedViewport.width / 2) / pr;
  735. const offsetY = (lockedViewport.y + lockedViewport.height / 2) / pr;
  736. cross.style.width = `${crossWidth}px`;
  737. cross.style.left = `calc(${offsetX}px - ${crossWidth / 2}px)`;
  738. cross.style.bottom = `calc(${offsetY}px - ${crossWidth / 2}px)`;
  739. }
  740. }
  741. return {
  742. get noScroll() { return noScroll; },
  743. set noScroll(value: boolean) { noScroll = value; },
  744. get noContextMenu() { return noContextMenu; },
  745. set noContextMenu(value: boolean) { noContextMenu = value; },
  746. get width() { return width; },
  747. get height() { return height; },
  748. get pixelRatio() { return pixelRatio(); },
  749. get pointerLock() { return isLocked; },
  750. ...events,
  751. requestPointerLock: (viewport: Viewport) => {
  752. lockedViewport = viewport;
  753. if (!isLocked) {
  754. element.requestPointerLock();
  755. }
  756. },
  757. exitPointerLock: () => {
  758. if (isLocked) {
  759. element.ownerDocument.exitPointerLock();
  760. }
  761. },
  762. dispose
  763. };
  764. }
  765. }
  766. // Adapted from https://stackoverflow.com/a/30134826
  767. // License: https://creativecommons.org/licenses/by-sa/3.0/
  768. function normalizeWheel(event: any) {
  769. // Reasonable defaults
  770. const PIXEL_STEP = 10;
  771. const LINE_HEIGHT = 40;
  772. const PAGE_HEIGHT = 800;
  773. let spinX = 0, spinY = 0,
  774. dx = 0, dy = 0, dz = 0; // pixelX, pixelY, pixelZ
  775. // Legacy
  776. if ('detail' in event) { spinY = event.detail; }
  777. if ('wheelDelta' in event) { spinY = -event.wheelDelta / 120; }
  778. if ('wheelDeltaY' in event) { spinY = -event.wheelDeltaY / 120; }
  779. if ('wheelDeltaX' in event) { spinX = -event.wheelDeltaX / 120; }
  780. // side scrolling on FF with DOMMouseScroll
  781. if ('axis' in event && event.axis === event.HORIZONTAL_AXIS) {
  782. spinX = spinY;
  783. spinY = 0;
  784. }
  785. dx = spinX * PIXEL_STEP;
  786. dy = spinY * PIXEL_STEP;
  787. if ('deltaY' in event) { dy = event.deltaY; }
  788. if ('deltaX' in event) { dx = event.deltaX; }
  789. if ('deltaZ' in event) { dz = event.deltaZ; }
  790. if ((dx || dy || dz) && event.deltaMode) {
  791. if (event.deltaMode === 1) { // delta in LINE units
  792. dx *= LINE_HEIGHT;
  793. dy *= LINE_HEIGHT;
  794. dz *= LINE_HEIGHT;
  795. } else { // delta in PAGE units
  796. dx *= PAGE_HEIGHT;
  797. dy *= PAGE_HEIGHT;
  798. dz *= PAGE_HEIGHT;
  799. }
  800. }
  801. // Fall-back if spin cannot be determined
  802. if (dx && !spinX) { spinX = (dx < 1) ? -1 : 1; }
  803. if (dy && !spinY) { spinY = (dy < 1) ? -1 : 1; }
  804. return { spinX, spinY, dx, dy, dz };
  805. }
  806. export { InputObserver };