Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/nextcloud/server.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
path: root/core/src
diff options
context:
space:
mode:
authorVincent Petry <vincent@nextcloud.com>2020-10-30 02:03:24 +0300
committerVincent Petry <vincent@nextcloud.com>2020-11-02 19:31:18 +0300
commitf2de58db8cac3c0f10aa4e8e001b0e310be4a73e (patch)
tree02ac45c2762f1f35ce73a6dc902b5a4d87d5226b /core/src
parentff08b10a897f059adbd30b61e6f4ba1e9af7b719 (diff)
Work around snap.js state mess
Snap.js is not robust enough to prevent multiple calls to open() or close(), so we added more checks to prevent these happening even when an animation is in progress. Because if we let this through, snap.js will not notice that the animation is already done (or a duplicate animation was started), so the "transitionend" event will not fire a second time. During an animation, snap.js sets up a setInterval() that hogs the CPU during the animation. Since a transition doesn't always end due to the above conditions, that CPU hogging would stay forever. This is the best workaround so far, it seems it's not possible to cancel that interval from the outside nor to trigger events to make it do so. Signed-off-by: Vincent Petry <vincent@nextcloud.com>
Diffstat (limited to 'core/src')
-rw-r--r--core/src/init.js72
1 files changed, 64 insertions, 8 deletions
diff --git a/core/src/init.js b/core/src/init.js
index d2282beb023..b6bb49346bd 100644
--- a/core/src/init.js
+++ b/core/src/init.js
@@ -217,18 +217,74 @@ export const initCore = () => {
$('#app-content').prepend('<div id="app-navigation-toggle" class="icon-menu" style="display:none" tabindex="0"></div>')
- const toggleSnapperOnButton = () => {
- if (snapper.state().state === 'left') {
- snapper.close()
- } else {
- snapper.open('left')
+ // keep track whether snapper is currently animating, and
+ // prevent to call open or close while that is the case
+ // to avoid duplicating events (snap.js doesn't check this)
+ let animating = false
+ snapper.on('animating', () => {
+ // we need this because the trigger button
+ // is also implicitly wired to close by snapper
+ animating = true
+ })
+ snapper.on('animated', () => {
+ animating = false
+ })
+ snapper.on('start', () => {
+ // we need this because dragging triggers that
+ animating = true
+ })
+ snapper.on('end', () => {
+ // we need this because dragging stop triggers that
+ animating = false
+ })
+
+ // These are necessary because calling open or close
+ // on snapper during an animation makes it trigger an
+ // unfinishable animation, which itself will continue
+ // triggering animating events and cause high CPU load,
+ //
+ // Ref https://github.com/jakiestfu/Snap.js/issues/216
+ const oldSnapperOpen = snapper.open
+ const oldSnapperClose = snapper.close
+ const _snapperOpen = () => {
+ if (animating || snapper.state().state !== 'closed') {
+ return
+ }
+ oldSnapperOpen('left')
+ }
+
+ const _snapperClose = () => {
+ if (animating || snapper.state().state === 'closed') {
+ return
+ }
+ oldSnapperClose()
+ }
+
+ // Needs to be deferred to properly catch in-between
+ // events that snap.js is triggering after dragging.
+ //
+ // Skipped when running unit tests as we are not testing
+ // the snap.js workarounds...
+ if (!window.TESTING) {
+ snapper.open = () => {
+ _.defer(_snapperOpen)
+ }
+ snapper.close = () => {
+ _.defer(_snapperClose)
}
}
- $('#app-navigation-toggle').click(toggleSnapperOnButton)
+ $('#app-navigation-toggle').click((e) => {
+ // close is implicit in the button by snap.js
+ if (snapper.state().state !== 'left') {
+ snapper.open()
+ }
+ })
$('#app-navigation-toggle').keypress(e => {
- if (e.which === 13) {
- toggleSnapperOnButton()
+ if (snapper.state().state === 'left') {
+ snapper.close()
+ } else {
+ snapper.open()
}
})