mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
(core) Fixing scrolly bug on mobile
Summary: On mobile view not all rows are rendered when a section is expanded. Scrolly component calculates height of the GridView too soon (before animation is completed). With this change on mobile view we always take the screen height for calculation. A similar bug was on Card List, where cards were squeezed and their height was calculated to soon. Test Plan: Added test Reviewers: georgegevoian Reviewed By: georgegevoian Differential Revision: https://phab.getgrist.com/D3748
This commit is contained in:
parent
cd72a54bbb
commit
d107810127
@ -1,4 +1,4 @@
|
|||||||
/* globals $ */
|
/* globals $, window */
|
||||||
|
|
||||||
const _ = require('underscore');
|
const _ = require('underscore');
|
||||||
const ko = require('knockout');
|
const ko = require('knockout');
|
||||||
@ -40,7 +40,7 @@ const {RowContextMenu} = require('../ui/RowContextMenu');
|
|||||||
|
|
||||||
const {setPopupToCreateDom} = require('popweasel');
|
const {setPopupToCreateDom} = require('popweasel');
|
||||||
const {CellContextMenu} = require('app/client/ui/CellContextMenu');
|
const {CellContextMenu} = require('app/client/ui/CellContextMenu');
|
||||||
const {testId} = require('app/client/ui2018/cssVars');
|
const {testId, isNarrowScreen} = require('app/client/ui2018/cssVars');
|
||||||
const {contextMenu} = require('app/client/ui/contextMenu');
|
const {contextMenu} = require('app/client/ui/contextMenu');
|
||||||
const {mouseDragMatchElem} = require('app/client/ui/mouseDrag');
|
const {mouseDragMatchElem} = require('app/client/ui/mouseDrag');
|
||||||
const {menuToggle} = require('app/client/ui/MenuToggle');
|
const {menuToggle} = require('app/client/ui/MenuToggle');
|
||||||
@ -1306,14 +1306,18 @@ GridView.prototype.buildDom = function() {
|
|||||||
/** @inheritdoc */
|
/** @inheritdoc */
|
||||||
GridView.prototype.onResize = function() {
|
GridView.prototype.onResize = function() {
|
||||||
const activeFieldBuilder = this.activeFieldBuilder();
|
const activeFieldBuilder = this.activeFieldBuilder();
|
||||||
|
let height = null;
|
||||||
|
if (isNarrowScreen()) {
|
||||||
|
height = window.outerHeight;
|
||||||
|
}
|
||||||
if (activeFieldBuilder && activeFieldBuilder.isEditorActive()) {
|
if (activeFieldBuilder && activeFieldBuilder.isEditorActive()) {
|
||||||
// When the editor is active, the common case for a resize is if the virtual keyboard is being
|
// When the editor is active, the common case for a resize is if the virtual keyboard is being
|
||||||
// shown on mobile device. In that case, we need to scroll active cell into view, and need to
|
// shown on mobile device. In that case, we need to scroll active cell into view, and need to
|
||||||
// do it synchronously, to allow repositioning the editor to it in response to the same event.
|
// do it synchronously, to allow repositioning the editor to it in response to the same event.
|
||||||
this.scrolly.updateSize();
|
this.scrolly.updateSize(height);
|
||||||
this.scrolly.scrollRowIntoView(this.cursor.rowIndex.peek());
|
this.scrolly.scrollRowIntoView(this.cursor.rowIndex.peek());
|
||||||
} else {
|
} else {
|
||||||
this.scrolly.scheduleUpdateSize();
|
this.scrolly.scheduleUpdateSize(height);
|
||||||
}
|
}
|
||||||
this.width(this.scrollPane.clientWidth)
|
this.width(this.scrollPane.clientWidth)
|
||||||
};
|
};
|
||||||
|
@ -15,13 +15,14 @@ import {reportError} from 'app/client/models/errors';
|
|||||||
import {filterBar} from 'app/client/ui/FilterBar';
|
import {filterBar} from 'app/client/ui/FilterBar';
|
||||||
import {viewSectionMenu} from 'app/client/ui/ViewSectionMenu';
|
import {viewSectionMenu} from 'app/client/ui/ViewSectionMenu';
|
||||||
import {buildWidgetTitle} from 'app/client/ui/WidgetTitle';
|
import {buildWidgetTitle} from 'app/client/ui/WidgetTitle';
|
||||||
import {colors, isNarrowScreenObs, mediaSmall, testId, theme} from 'app/client/ui2018/cssVars';
|
import {colors, isNarrowScreen, isNarrowScreenObs, mediaSmall, testId, theme} from 'app/client/ui2018/cssVars';
|
||||||
import {icon} from 'app/client/ui2018/icons';
|
import {icon} from 'app/client/ui2018/icons';
|
||||||
import {DisposableWithEvents} from 'app/common/DisposableWithEvents';
|
import {DisposableWithEvents} from 'app/common/DisposableWithEvents';
|
||||||
import {mod} from 'app/common/gutil';
|
import {mod} from 'app/common/gutil';
|
||||||
import {Observable} from 'grainjs';
|
import {Observable} from 'grainjs';
|
||||||
import * as ko from 'knockout';
|
import * as ko from 'knockout';
|
||||||
import * as _ from 'underscore';
|
import * as _ from 'underscore';
|
||||||
|
import debounce from 'lodash/debounce';
|
||||||
import {computedArray, Disposable, dom, fromKo, Holder, IDomComponent, styled, subscribe} from 'grainjs';
|
import {computedArray, Disposable, dom, fromKo, Holder, IDomComponent, styled, subscribe} from 'grainjs';
|
||||||
|
|
||||||
// tslint:disable:no-console
|
// tslint:disable:no-console
|
||||||
@ -133,6 +134,23 @@ export class ViewLayout extends DisposableWithEvents implements IDomComponent {
|
|||||||
// (See https://stackoverflow.com/questions/2381336/detect-click-into-iframe-using-javascript).
|
// (See https://stackoverflow.com/questions/2381336/detect-click-into-iframe-using-javascript).
|
||||||
this.listenTo(this.gristDoc.app, 'clipboard_blur', this._maybeFocusInSection);
|
this.listenTo(this.gristDoc.app, 'clipboard_blur', this._maybeFocusInSection);
|
||||||
|
|
||||||
|
// On narrow screens (e.g. mobile), we need to resize the section after a transition.
|
||||||
|
// There will two transition events (one from section one from row), so we debounce them after a tick.
|
||||||
|
const handler = debounce((e: TransitionEvent) => {
|
||||||
|
// We work only on the transition of the flex-grow property, and only on narrow screens.
|
||||||
|
if (e.propertyName !== 'flex-grow' || !isNarrowScreen()) { return; }
|
||||||
|
// Make sure the view is still active.
|
||||||
|
if (this.viewModel.isDisposed() || !this.viewModel.activeSection) { return; }
|
||||||
|
const section = this.viewModel.activeSection.peek();
|
||||||
|
if (!section || section.isDisposed()) { return; }
|
||||||
|
const view = section.viewInstance.peek();
|
||||||
|
if (!view || view.isDisposed()) { return; }
|
||||||
|
// Make resize.
|
||||||
|
view.onResize();
|
||||||
|
}, 0);
|
||||||
|
this._layout.rootElem.addEventListener('transitionend', handler);
|
||||||
|
// Don't need to dispose the listener, as the rootElem is disposed with the layout.
|
||||||
|
|
||||||
const classActive = cssLayoutBox.className + '-active';
|
const classActive = cssLayoutBox.className + '-active';
|
||||||
const classInactive = cssLayoutBox.className + '-inactive';
|
const classInactive = cssLayoutBox.className + '-inactive';
|
||||||
this.autoDispose(subscribe(fromKo(this.viewModel.activeSection), (use, section) => {
|
this.autoDispose(subscribe(fromKo(this.viewModel.activeSection), (use, section) => {
|
||||||
@ -140,6 +158,7 @@ export class ViewLayout extends DisposableWithEvents implements IDomComponent {
|
|||||||
this._layout.forEachBox((box: {dom: Element}) => {
|
this._layout.forEachBox((box: {dom: Element}) => {
|
||||||
box.dom.classList.add(classInactive);
|
box.dom.classList.add(classInactive);
|
||||||
box.dom.classList.remove(classActive);
|
box.dom.classList.remove(classActive);
|
||||||
|
box.dom.classList.remove("transition");
|
||||||
});
|
});
|
||||||
let elem: Element|null = this._layout.getLeafBox(id)?.dom;
|
let elem: Element|null = this._layout.getLeafBox(id)?.dom;
|
||||||
while (elem?.matches('.layout_box')) {
|
while (elem?.matches('.layout_box')) {
|
||||||
@ -147,7 +166,9 @@ export class ViewLayout extends DisposableWithEvents implements IDomComponent {
|
|||||||
elem.classList.add(classActive);
|
elem.classList.add(classActive);
|
||||||
elem = elem.parentElement;
|
elem = elem.parentElement;
|
||||||
}
|
}
|
||||||
section.viewInstance.peek()?.onResize();
|
if (!isNarrowScreen()) {
|
||||||
|
section.viewInstance.peek()?.onResize();
|
||||||
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const commandGroup = {
|
const commandGroup = {
|
||||||
@ -406,7 +427,7 @@ const cssViewLeafInactive = styled('div', `
|
|||||||
const cssLayoutBox = styled('div', `
|
const cssLayoutBox = styled('div', `
|
||||||
@media screen and ${mediaSmall} {
|
@media screen and ${mediaSmall} {
|
||||||
&-active, &-inactive {
|
&-active, &-inactive {
|
||||||
transition: flex-grow 0.4s;
|
transition: flex-grow var(--grist-layout-animation-duration, 0.4s); // Exposed for tests
|
||||||
}
|
}
|
||||||
&-active > &-inactive,
|
&-active > &-inactive,
|
||||||
&-active > &-inactive.layout_hbox .layout_hbox,
|
&-active > &-inactive.layout_hbox .layout_hbox,
|
||||||
|
@ -274,9 +274,9 @@ Scrolly.prototype.addPane = function(containerElem, options, itemCreateFunc) {
|
|||||||
/**
|
/**
|
||||||
* Tells Scrolly to call updateSize after things have had a chance to render.
|
* Tells Scrolly to call updateSize after things have had a chance to render.
|
||||||
*/
|
*/
|
||||||
Scrolly.prototype.scheduleUpdateSize = function() {
|
Scrolly.prototype.scheduleUpdateSize = function(overrideHeight) {
|
||||||
if (!this.isDisposed() && !this.delayedUpdateSize.isPending()) {
|
if (!this.isDisposed() && !this.delayedUpdateSize.isPending()) {
|
||||||
this.delayedUpdateSize.schedule(0, this.updateSize, this);
|
this.delayedUpdateSize.schedule(0, this.updateSize.bind(this, overrideHeight), this);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -284,15 +284,16 @@ Scrolly.prototype.scheduleUpdateSize = function() {
|
|||||||
* Measures the size of the panes and adjusts Scrolly parameters for how many rows to render.
|
* Measures the size of the panes and adjusts Scrolly parameters for how many rows to render.
|
||||||
* This should be called as soon as all Scrolly panes have been attached to the Document, and any
|
* This should be called as soon as all Scrolly panes have been attached to the Document, and any
|
||||||
* time their outer size changes.
|
* time their outer size changes.
|
||||||
|
* Pass in an overrideHeight to use instead of the current height of the panes.
|
||||||
*/
|
*/
|
||||||
Scrolly.prototype.updateSize = function() {
|
Scrolly.prototype.updateSize = function(overrideHeight) {
|
||||||
this.resetHeights();
|
this.resetHeights();
|
||||||
this.shownHeight = Math.max(0, Math.max.apply(null, this.panes.map(function(pane) {
|
this.shownHeight = Math.max(0, Math.max.apply(null, this.panes.map(function(pane) {
|
||||||
return pane.container.clientHeight;
|
return pane.container.clientHeight;
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Update counts of rows that are shown.
|
// Update counts of rows that are shown.
|
||||||
var numVisible = Math.max(1, Math.ceil(this.shownHeight / this.minRowHeight));
|
var numVisible = Math.max(1, Math.ceil((overrideHeight ?? this.shownHeight) / this.minRowHeight));
|
||||||
this.numBuffered = 5;
|
this.numBuffered = 5;
|
||||||
this.numRendered = numVisible + 2 * this.numBuffered;
|
this.numRendered = numVisible + 2 * this.numBuffered;
|
||||||
|
|
||||||
|
@ -227,8 +227,13 @@ export function getSection(sectionOrTitle: string|WebElement): WebElement|WebEle
|
|||||||
* Click into a section without disrupting cursor positions.
|
* Click into a section without disrupting cursor positions.
|
||||||
*/
|
*/
|
||||||
export async function selectSectionByTitle(title: string) {
|
export async function selectSectionByTitle(title: string) {
|
||||||
// .test-viewsection is a special 1px width element added for tests only.
|
try {
|
||||||
await driver.findContent(`.test-viewsection-title`, title).find(".test-viewsection-blank").click();
|
// .test-viewsection is a special 1px width element added for tests only.
|
||||||
|
await driver.findContent(`.test-viewsection-title`, title).find(".test-viewsection-blank").click();
|
||||||
|
} catch (e) {
|
||||||
|
// We might be in mobile view.
|
||||||
|
await driver.findContent(`.test-viewsection-title`, title).findClosest(".view_leaf").click();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -2319,9 +2324,8 @@ export function bigScreen() {
|
|||||||
/**
|
/**
|
||||||
* Shrinks browser window dimensions to trigger mobile mode for a test suite.
|
* Shrinks browser window dimensions to trigger mobile mode for a test suite.
|
||||||
*/
|
*/
|
||||||
export function narrowScreen() {
|
export function narrowScreen() {
|
||||||
resizeWindowForSuite(400, 750);
|
resizeWindowForSuite(400, 750);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function addSupportUserIfPossible() {
|
export async function addSupportUserIfPossible() {
|
||||||
|
Loading…
Reference in New Issue
Block a user