Fix touch events

This commit is contained in:
Eduard Kuzmenko 2020-12-12 18:30:01 +02:00
parent 84efdd6398
commit c36e5df589
10 changed files with 48 additions and 30 deletions

View File

@ -23,7 +23,7 @@ const ButtonMenuItem = (options: ButtonMenuItemOptions) => {
ripple(el); ripple(el);
// * cancel keyboard close // * cancel keyboard close
attachClickEvent(el, CLICK_EVENT_NAME == 'touchend' ? (e) => { attachClickEvent(el, CLICK_EVENT_NAME !== 'click' ? (e) => {
cancelEvent(e); cancelEvent(e);
onClick(e); onClick(e);
closeBtnMenu(); closeBtnMenu();

View File

@ -14,6 +14,7 @@ const ButtonMenuToggle = (options: Partial<{noRipple: true, onlyMobile: true, li
return button; return button;
}; };
// TODO: refactor for attachClickEvent, because if move finger after touchstart, it will start anyway
const ButtonMenuToggleHandler = (el: HTMLElement, onOpen?: () => void, options?: AttachClickOptions) => { const ButtonMenuToggleHandler = (el: HTMLElement, onOpen?: () => void, options?: AttachClickOptions) => {
const add = options?.listenerSetter ? options.listenerSetter.add.bind(options.listenerSetter, el) : el.addEventListener.bind(el); const add = options?.listenerSetter ? options.listenerSetter.add.bind(options.listenerSetter, el) : el.addEventListener.bind(el);

View File

@ -7,7 +7,7 @@ import type { AppInlineBotsManager } from "../../lib/appManagers/AppInlineBotsMa
import type { AppPhotosManager } from "../../lib/appManagers/appPhotosManager"; import type { AppPhotosManager } from "../../lib/appManagers/appPhotosManager";
import type { AppDocsManager } from "../../lib/appManagers/appDocsManager"; import type { AppDocsManager } from "../../lib/appManagers/appDocsManager";
import type { AppPeersManager } from "../../lib/appManagers/appPeersManager"; import type { AppPeersManager } from "../../lib/appManagers/appPeersManager";
import { findUpClassName, cancelEvent, findUpTag, CLICK_EVENT_NAME, whichChild, getElementByPoint } from "../../helpers/dom"; import { findUpClassName, cancelEvent, findUpTag, whichChild, getElementByPoint, attachClickEvent } from "../../helpers/dom";
import { getObjectKeysAndSort } from "../../helpers/object"; import { getObjectKeysAndSort } from "../../helpers/object";
import { isTouchSupported } from "../../helpers/touchSupport"; import { isTouchSupported } from "../../helpers/touchSupport";
import { logger } from "../../lib/logger"; import { logger } from "../../lib/logger";
@ -1480,7 +1480,7 @@ export default class ChatBubbles {
containerDiv.append(rowDiv); containerDiv.append(rowDiv);
}); });
containerDiv.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(containerDiv, (e) => {
cancelEvent(e); cancelEvent(e);
let target = e.target as HTMLElement; let target = e.target as HTMLElement;

View File

@ -11,7 +11,7 @@ import apiManager from "../../lib/mtproto/mtprotoworker";
//import Recorder from '../opus-recorder/dist/recorder.min'; //import Recorder from '../opus-recorder/dist/recorder.min';
import opusDecodeController from "../../lib/opusDecodeController"; import opusDecodeController from "../../lib/opusDecodeController";
import RichTextProcessor from "../../lib/richtextprocessor"; import RichTextProcessor from "../../lib/richtextprocessor";
import { blurActiveElement, cancelEvent, cancelSelection, CLICK_EVENT_NAME, findUpClassName, getRichValue, getSelectedNodes, isInputEmpty, markdownTags, MarkdownType, placeCaretAtEnd, serializeNodes } from "../../helpers/dom"; import { attachClickEvent, blurActiveElement, cancelEvent, cancelSelection, findUpClassName, getRichValue, getSelectedNodes, isInputEmpty, markdownTags, MarkdownType, placeCaretAtEnd, serializeNodes } from "../../helpers/dom";
import { ButtonMenuItemOptions } from '../buttonMenu'; import { ButtonMenuItemOptions } from '../buttonMenu';
import emoticonsDropdown from "../emoticonsDropdown"; import emoticonsDropdown from "../emoticonsDropdown";
import PopupCreatePoll from "../popupCreatePoll"; import PopupCreatePoll from "../popupCreatePoll";
@ -123,10 +123,10 @@ export default class ChatInput {
this.goDownBtn.append(this.goDownUnreadBadge); this.goDownBtn.append(this.goDownUnreadBadge);
this.chatInput.append(this.goDownBtn); this.chatInput.append(this.goDownBtn);
this.listenerSetter.add(this.goDownBtn, CLICK_EVENT_NAME, (e) => { attachClickEvent(this.goDownBtn, (e) => {
cancelEvent(e); cancelEvent(e);
this.chat.bubbles.onGoDownClick(); this.chat.bubbles.onGoDownClick();
}); }, {listenerSetter: this.listenerSetter});
// * constructor end // * constructor end
} }
@ -260,7 +260,23 @@ export default class ChatInput {
this.fileInput.value = ''; this.fileInput.value = '';
}, false); }, false);
this.listenerSetter.add(this.btnSend, CLICK_EVENT_NAME, this.onBtnSendClick); /* let time = Date.now();
this.btnSend.addEventListener('touchstart', (e) => {
time = Date.now();
});
let eventName1 = 'touchend';
this.btnSend.addEventListener(eventName1, (e: Event) => {
//cancelEvent(e);
console.log(eventName1 + ', time: ' + (Date.now() - time));
});
let eventName = 'mousedown';
this.btnSend.addEventListener(eventName, (e: Event) => {
cancelEvent(e);
console.log(eventName + ', time: ' + (Date.now() - time));
}); */
attachClickEvent(this.btnSend, this.onBtnSendClick, {listenerSetter: this.listenerSetter});
if(this.recorder) { if(this.recorder) {
const onCancelRecordClick = (e: Event) => { const onCancelRecordClick = (e: Event) => {
@ -269,7 +285,7 @@ export default class ChatInput {
this.recorder.stop(); this.recorder.stop();
opusDecodeController.setKeepAlive(false); opusDecodeController.setKeepAlive(false);
}; };
this.listenerSetter.add(this.btnCancelRecord, CLICK_EVENT_NAME, onCancelRecordClick); attachClickEvent(this.btnCancelRecord, onCancelRecordClick, {listenerSetter: this.listenerSetter});
this.recorder.onstop = () => { this.recorder.onstop = () => {
this.recording = false; this.recording = false;
@ -308,8 +324,8 @@ export default class ChatInput {
}; };
} }
this.listenerSetter.add(this.replyElements.cancelBtn, CLICK_EVENT_NAME, this.onHelperCancel); attachClickEvent(this.replyElements.cancelBtn, this.onHelperCancel, {listenerSetter: this.listenerSetter});
this.listenerSetter.add(this.replyElements.container, CLICK_EVENT_NAME, this.onHelperClick); attachClickEvent(this.replyElements.container, this.onHelperClick, {listenerSetter: this.listenerSetter});
} }
public constructPinnedHelpers() { public constructPinnedHelpers() {

View File

@ -5,7 +5,7 @@ import appPollsManager, { Poll, PollResults } from "../lib/appManagers/appPollsM
import serverTimeManager from "../lib/mtproto/serverTimeManager"; import serverTimeManager from "../lib/mtproto/serverTimeManager";
import { RichTextProcessor } from "../lib/richtextprocessor"; import { RichTextProcessor } from "../lib/richtextprocessor";
import rootScope from "../lib/rootScope"; import rootScope from "../lib/rootScope";
import { cancelEvent, CLICK_EVENT_NAME, findUpClassName } from "../helpers/dom"; import { attachClickEvent, cancelEvent, detachClickEvent, findUpClassName } from "../helpers/dom";
import { ripple } from "./ripple"; import { ripple } from "./ripple";
import appSidebarRight from "./sidebarRight"; import appSidebarRight from "./sidebarRight";
@ -338,7 +338,7 @@ export default class PollElement extends HTMLElement {
this.votersCountDiv.classList.add('hide'); this.votersCountDiv.classList.add('hide');
} }
this.sendVoteBtn.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.sendVoteBtn, (e) => {
cancelEvent(e); cancelEvent(e);
/* const indexes = this.answerDivs.filter(el => el.classList.contains('is-chosing')).map(el => +el.dataset.index); /* const indexes = this.answerDivs.filter(el => el.classList.contains('is-chosing')).map(el => +el.dataset.index);
if(indexes.length) { if(indexes.length) {
@ -364,7 +364,7 @@ export default class PollElement extends HTMLElement {
this.performResults(results, poll.chosenIndexes); this.performResults(results, poll.chosenIndexes);
} else if(!this.isClosed) { } else if(!this.isClosed) {
this.setVotersCount(results); this.setVotersCount(results);
this.addEventListener(CLICK_EVENT_NAME, this.clickHandler); attachClickEvent(this, this.clickHandler);
} }
} }
@ -406,7 +406,7 @@ export default class PollElement extends HTMLElement {
this.descDiv.append(toggleHint); this.descDiv.append(toggleHint);
//let active = false; //let active = false;
toggleHint.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(toggleHint, (e) => {
cancelEvent(e); cancelEvent(e);
//active = true; //active = true;
@ -514,9 +514,9 @@ export default class PollElement extends HTMLElement {
this.chosenIndexes = chosenIndexes.slice(); this.chosenIndexes = chosenIndexes.slice();
if(this.isRetracted) { if(this.isRetracted) {
this.addEventListener(CLICK_EVENT_NAME, this.clickHandler); attachClickEvent(this, this.clickHandler);
} else { } else {
this.removeEventListener(CLICK_EVENT_NAME, this.clickHandler); detachClickEvent(this, this.clickHandler);
} }
} }

View File

@ -8,7 +8,7 @@ import appStateManager from "../../lib/appManagers/appStateManager";
import appUsersManager from "../../lib/appManagers/appUsersManager"; import appUsersManager from "../../lib/appManagers/appUsersManager";
import { MOUNT_CLASS_TO } from "../../lib/mtproto/mtproto_config"; import { MOUNT_CLASS_TO } from "../../lib/mtproto/mtproto_config";
import rootScope from "../../lib/rootScope"; import rootScope from "../../lib/rootScope";
import { CLICK_EVENT_NAME, findUpClassName, findUpTag } from "../../helpers/dom"; import { attachClickEvent, findUpClassName, findUpTag } from "../../helpers/dom";
import AppSearch, { SearchGroup } from "../appSearch"; import AppSearch, { SearchGroup } from "../appSearch";
import "../avatar"; import "../avatar";
import { parseMenuButtonsTo } from "../misc"; import { parseMenuButtonsTo } from "../misc";
@ -292,32 +292,32 @@ export class AppSidebarLeft extends SidebarSlider {
this.archivedCount = this.buttons.archived.querySelector('.archived-count') as HTMLSpanElement; this.archivedCount = this.buttons.archived.querySelector('.archived-count') as HTMLSpanElement;
this.buttons.saved.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.buttons.saved, (e) => {
///////this.log('savedbtn click'); ///////this.log('savedbtn click');
setTimeout(() => { // menu doesn't close if no timeout (lol) setTimeout(() => { // menu doesn't close if no timeout (lol)
appImManager.setPeer(appImManager.myId); appImManager.setPeer(appImManager.myId);
}, 0); }, 0);
}); });
this.buttons.archived.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.buttons.archived, (e) => {
this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.archived); this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.archived);
}); });
this.buttons.contacts.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.buttons.contacts, (e) => {
this.contactsTab.openContacts(); this.contactsTab.openContacts();
}); });
this.buttons.settings.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.buttons.settings, (e) => {
this.settingsTab.fillElements(); this.settingsTab.fillElements();
this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.settings); this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.settings);
}); });
this.newButtons.channel.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.newButtons.channel, (e) => {
this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.newChannel); this.selectTab(AppSidebarLeft.SLIDERITEMSIDS.newChannel);
}); });
[this.newButtons.group, this.buttons.newGroup].forEach(btn => { [this.newButtons.group, this.buttons.newGroup].forEach(btn => {
btn.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(btn, (e) => {
this.addMembersTab.init(0, 'chat', false, (peerIds) => { this.addMembersTab.init(0, 'chat', false, (peerIds) => {
this.newGroupTab.init(peerIds); this.newGroupTab.init(peerIds);
}); });

View File

@ -8,7 +8,7 @@ import { ripple } from "../../ripple";
import { SliderTab } from "../../slider"; import { SliderTab } from "../../slider";
import { toast } from "../../toast"; import { toast } from "../../toast";
import appMessagesManager from "../../../lib/appManagers/appMessagesManager"; import appMessagesManager from "../../../lib/appManagers/appMessagesManager";
import { CLICK_EVENT_NAME } from "../../../helpers/dom"; import { attachClickEvent } from "../../../helpers/dom";
const MAX_FOLDER_NAME_LENGTH = 12; const MAX_FOLDER_NAME_LENGTH = 12;
@ -72,7 +72,7 @@ export default class AppEditFolderTab implements SliderTab {
this.animation = player; this.animation = player;
}); });
this.deleteFolderBtn.addEventListener(CLICK_EVENT_NAME, () => { attachClickEvent(this.deleteFolderBtn, () => {
this.deleteFolderBtn.setAttribute('disabled', 'true'); this.deleteFolderBtn.setAttribute('disabled', 'true');
appMessagesManager.filtersStorage.updateDialogFilter(this.filter, true).then(bool => { appMessagesManager.filtersStorage.updateDialogFilter(this.filter, true).then(bool => {
if(bool) { if(bool) {

View File

@ -5,7 +5,7 @@ import { parseMenuButtonsTo } from "../../misc";
import apiManager from "../../../lib/mtproto/mtprotoworker"; import apiManager from "../../../lib/mtproto/mtprotoworker";
import appSidebarLeft, { AppSidebarLeft } from ".."; import appSidebarLeft, { AppSidebarLeft } from "..";
import appUsersManager from "../../../lib/appManagers/appUsersManager"; import appUsersManager from "../../../lib/appManagers/appUsersManager";
import { CLICK_EVENT_NAME } from "../../../helpers/dom"; import { attachClickEvent } from "../../../helpers/dom";
export default class AppSettingsTab implements SliderTab { export default class AppSettingsTab implements SliderTab {
private container = document.querySelector('.settings-container') as HTMLDivElement; private container = document.querySelector('.settings-container') as HTMLDivElement;
@ -31,7 +31,7 @@ export default class AppSettingsTab implements SliderTab {
this.fillElements(); this.fillElements();
}); */ }); */
this.logOutBtn.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(this.logOutBtn, (e) => {
apiManager.logOut(); apiManager.logOut();
}); });

View File

@ -12,7 +12,7 @@ import appMessagesManager from '../lib/appManagers/appMessagesManager';
import appPhotosManager, { MyPhoto } from '../lib/appManagers/appPhotosManager'; import appPhotosManager, { MyPhoto } from '../lib/appManagers/appPhotosManager';
import LottieLoader from '../lib/lottieLoader'; import LottieLoader from '../lib/lottieLoader';
import VideoPlayer from '../lib/mediaPlayer'; import VideoPlayer from '../lib/mediaPlayer';
import { cancelEvent, CLICK_EVENT_NAME, isInDOM } from "../helpers/dom"; import { attachClickEvent, cancelEvent, isInDOM } from "../helpers/dom";
import webpWorkerController from '../lib/webp/webpWorkerController'; import webpWorkerController from '../lib/webp/webpWorkerController';
import animationIntersector from './animationIntersector'; import animationIntersector from './animationIntersector';
import appMediaPlaybackController from './appMediaPlaybackController'; import appMediaPlaybackController from './appMediaPlaybackController';
@ -402,7 +402,7 @@ export function wrapDocument(doc: MyDocument, withTime = false, uploading = fals
let preloader: ProgressivePreloader; let preloader: ProgressivePreloader;
let download: DownloadBlob; let download: DownloadBlob;
docDiv.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(docDiv, (e) => {
cancelEvent(e); cancelEvent(e);
if(!download) { if(!download) {
if(downloadDiv.classList.contains('downloading')) { if(downloadDiv.classList.contains('downloading')) {
@ -757,7 +757,7 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
}, true); }, true);
if(emoji) { if(emoji) {
div.addEventListener(CLICK_EVENT_NAME, (e) => { attachClickEvent(div, (e) => {
cancelEvent(e); cancelEvent(e);
let animation = LottieLoader.getAnimation(div); let animation = LottieLoader.getAnimation(div);

View File

@ -477,6 +477,7 @@ export const attachClickEvent = (elem: HTMLElement, callback: (e: TouchEvent | M
const onTouchStart = (e: TouchEvent) => { const onTouchStart = (e: TouchEvent) => {
const onTouchMove = (e: TouchEvent) => { const onTouchMove = (e: TouchEvent) => {
remove('touchmove', onTouchMove, o);
remove('touchend', onTouchEnd, o); remove('touchend', onTouchEnd, o);
}; };