(core) Fixing anchor link navigation.

Summary: Last document position was overwritting anchor link navigation.

Test Plan: Browser tests

Reviewers: georgegevoian

Reviewed By: georgegevoian

Differential Revision: https://phab.getgrist.com/D2934
This commit is contained in:
Jarosław Sadziński 2021-07-22 11:14:35 +02:00
parent f8e4fe54ba
commit a07395855a
3 changed files with 38 additions and 9 deletions

View File

@ -1,8 +1,8 @@
import { CursorPos } from "app/client/components/Cursor"; import {CursorPos} from 'app/client/components/Cursor';
import { getStorage } from "app/client/lib/localStorageObs"; import {GristDoc} from 'app/client/components/GristDoc';
import { IDocPage } from "app/common/gristUrls"; import {getStorage} from 'app/client/lib/localStorageObs';
import { Disposable } from "grainjs"; import {IDocPage} from 'app/common/gristUrls';
import { GristDoc } from "app/client/components/GristDoc"; import {Disposable} from 'grainjs';
/** /**
* Enriched cursor position with a view id * Enriched cursor position with a view id
@ -54,6 +54,12 @@ export class CursorMonitor extends Disposable {
} }
private _whenDocumentLoadsRestorePosition(doc: GristDoc) { private _whenDocumentLoadsRestorePosition(doc: GristDoc) {
// if doc was opened with a hash link, don't restore last position
if (doc.hasCustomNav.get()) {
this._restored = true;
return;
}
// on view shown // on view shown
this.autoDispose(doc.currentView.addListener(async view => { this.autoDispose(doc.currentView.addListener(async view => {
// if the position was restored for this document do nothing // if the position was restored for this document do nothing

View File

@ -43,7 +43,7 @@ import {isSchemaAction} from 'app/common/DocActions';
import {OpenLocalDocResult} from 'app/common/DocListAPI'; import {OpenLocalDocResult} from 'app/common/DocListAPI';
import {HashLink, IDocPage} from 'app/common/gristUrls'; import {HashLink, IDocPage} from 'app/common/gristUrls';
import {RecalcWhen} from 'app/common/gristTypes'; import {RecalcWhen} from 'app/common/gristTypes';
import {encodeQueryParams, waitObs} from 'app/common/gutil'; import {encodeQueryParams, undef, waitObs} from 'app/common/gutil';
import {StringUnion} from 'app/common/StringUnion'; import {StringUnion} from 'app/common/StringUnion';
import {TableData} from 'app/common/TableData'; import {TableData} from 'app/common/TableData';
import {DocStateComparison} from 'app/common/UserAPI'; import {DocStateComparison} from 'app/common/UserAPI';
@ -106,7 +106,8 @@ export class GristDoc extends DisposableWithEvents {
public editorMonitor: EditorMonitor; public editorMonitor: EditorMonitor;
// component for keeping track of a cell that is being edited // component for keeping track of a cell that is being edited
public draftMonitor: Drafts; public draftMonitor: Drafts;
// will document perform its own navigation (from anchor link)
public hasCustomNav: Observable<boolean>;
// Emitter triggered when the main doc area is resized. // Emitter triggered when the main doc area is resized.
public readonly resizeEmitter = this.autoDispose(new Emitter()); public readonly resizeEmitter = this.autoDispose(new Emitter());
@ -297,6 +298,11 @@ export class GristDoc extends DisposableWithEvents {
return undefined; return undefined;
}); });
this.hasCustomNav = Computed.create(this, urlState().state, (_, state) => {
const hash = state.hash;
return !!(hash && (undef(hash.colRef, hash.rowId, hash.sectionId) !== undefined));
});
this.draftMonitor = Drafts.create(this, this); this.draftMonitor = Drafts.create(this, this);
this.cursorMonitor = CursorMonitor.create(this, this); this.cursorMonitor = CursorMonitor.create(this, this);
this.editorMonitor = EditorMonitor.create(this, this); this.editorMonitor = EditorMonitor.create(this, this);

View File

@ -105,8 +105,7 @@ export function undefDefault<T>(x: T|undefined, y: T): T {
// for typescript 4 // for typescript 4
// type Undef<T> = T extends [infer A, ...infer B] ? undefined extends A ? NonNullable<A> | Undef<B> : A : unknown; // type Undef<T> = T extends [infer A, ...infer B] ? undefined extends A ? NonNullable<A> | Undef<B> : A : unknown;
type Undef1<T> = T extends [infer A] ? type Undef1<T> = T extends [infer A] ? A : unknown;
undefined extends A ? NonNullable<A> : A : unknown;
type Undef2<T> = T extends [infer A, infer B] ? type Undef2<T> = T extends [infer A, infer B] ?
undefined extends A ? NonNullable<A> | Undef1<[B]> : A : Undef1<T>; undefined extends A ? NonNullable<A> | Undef1<[B]> : A : Undef1<T>;
@ -117,6 +116,23 @@ type Undef3<T> = T extends [infer A, infer B, infer C] ?
type Undef<T> = T extends [infer A, infer B, infer C, infer D] ? type Undef<T> = T extends [infer A, infer B, infer C, infer D] ?
undefined extends A ? NonNullable<A> | Undef3<[B, C, D]> : A : Undef3<T>; undefined extends A ? NonNullable<A> | Undef3<[B, C, D]> : A : Undef3<T>;
/*
Undef<T> can detect correct type that will be returned as a first defined value:
const t1: number = undef(1, 1 as number | undefined);
const t1: number | undefined = undef(2 as number | undefined, 3 as number | undefined);
const t3: number = undef(3 as number | undefined, undefined, 4);
const t4: number = undef(1, '');
const t5: number = undef(1 as number | undefined, 4);
const t6: string = undef('1', 2);
const t7: string | number = undef(undefined, 2 as number | undefined, '3');
const t8: string = undef(undefined, undefined, '3');
const t9: string = undef(undefined, '2' as string | undefined, '3');
const ta: string | number | undefined = undef(undefined, '2' as string | undefined, 3 as number | undefined);
const tb: string | number = undef(undefined, '2' as string | undefined, 3 as number | undefined, 5);
*/
/** /**
* Returns the first defined value from the list or unknown. * Returns the first defined value from the list or unknown.
* Use with typed result, so the typescript type checker can provide correct type. * Use with typed result, so the typescript type checker can provide correct type.
@ -128,6 +144,7 @@ export function undef<T extends Array<any>>(...list: T): Undef<T> {
return undefined as any; return undefined as any;
} }
/** /**
* Parses json and returns the result, or returns defaultVal if parsing fails. * Parses json and returns the result, or returns defaultVal if parsing fails.
*/ */