本帖最后由 riyad 于 2025-2-23 19:38 编辑
3 e+ ?- E2 Q, l" d0 w- z) N7 l* w9 b1 t( A: P
BrowserNavigatorBrowserNavigator 是一个使用浏览器历史 API 实现导航的类。 导航器对标准导航器进行了扩展,提供了基本的导航功能和专门用于 浏览器历史记录的逻辑。
% X% ^6 T7 B% u! [构造函数要创建该类的实例,开发人员可以使用该类的构造函数,该构造函数接受历史 项、指向历史项列表条目的光标(索引)以及其他选项: - import { BrowserNavigator } from '@telegram-apps/sdk';
8 i5 A( q3 Z9 z+ T. y7 t; e a - + [, C( v0 O1 L" ^+ B5 W
- new BrowserNavigator(['/index'], 0);" b; p7 m4 F# c% o7 Z& |
- new BrowserNavigator([{ pathname: '/a' }, { pathname: '/b' }], 1);
复制代码WARNING 如果传入的历史项目列表为空,或者游标指向一个不存在的项目, 构造函数将抛出一个适当的错误。 postEventpostEvent 函数允许替换代表 导航器调用 Mini Apps 方法的函数。 - const n = new BrowserNavigator(['/'], 0, {
' V, C* z, t- | - postEvent(method, payload) {
% v! t4 C- @0 m2 p; R3 i4 ?$ Z - console.log('postEvent:', method, payload);
+ n: v/ l; D! `$ R( L3 _. ?8 I& X0 I - },$ K" f; b( v- D8 O$ ^3 N" x
- });
复制代码 hashModehashMode 属性允许开发人员设置导航器使用的导航模式。 允许的值有:classic(#pathname)、slash(#/pathname)和null。 传递 null 值 会将导航器从 SPA 模式切换到 MPA 模式,使用整个 href 而不是位置的哈希部分 。 默认情况下,导航器使用 classic 模式。 - const n = new BrowserNavigator(['/'], 0, {
% a6 S: j# ~, W% [$ { w - hashMode: 'slash'
3 y* p8 R3 O9 W. j) R - });
复制代码 basebase 属性负责解析路由和渲染路径,假定它们必须 以指定的 base 值开始。 - const n = new BrowserNavigator(['/'], 0, {
3 }) }$ j- v0 |& a - hashMode: null,
8 l; N' T( S' j( {3 ^5 i! h) {( O - base: '/solidjs-template',
( V# v5 g& e. Y - });
7 q% K3 V! N' s - + `1 R' u9 j8 ]# h; C+ \
- n.renderPath('/home'); // -> /solidjs-template/home
复制代码该值只有在使用 MPA 模式时才有用。
. `+ \9 q& M/ [5 ~, B0 dcreateBrowserNavigatorFromLocation为了简化创建 BrowserNavigator 的过程,软件包提供了 createBrowserNavigatorFromLocation 方法。 根据所传递的选项(即前面描述的构造函数 选项),它会解析当前位置并创建 一个 BrowserNavigator 实例。 - import { createBrowserNavigatorFromLocation } from '@telegram-apps/sdk';/ E; a* ^* V! V1 F1 Y) f/ _" l
- & v; h% [) f$ |4 d$ }5 X9 w
- const navigator = createBrowserNavigatorFromLocation({4 Q6 F- M9 e6 Q
- hashMode: 'slash',
& L( w- Z- J0 l2 R - });
复制代码 initNavigatorinitNavigator 函数负责使用先前保存的导航器状态创建一个 BrowserNavigator 实例 。 它还会自动将其保存在会话存储 中,以便将来恢复状态。 该函数接受会话存储密钥名称(其中将包含导航器状态)和 可选对象(代表导航器构造器选项)。 - import { initNavigator } from '@telegram-apps/sdk';
2 x0 R1 E* a$ J* G, ^ - / s5 ~. ]7 ~) W2 T" V8 `$ i7 ]
- const n = initNavigator('app-navigator-state', {
! _3 s$ L/ f* D$ R+ W! c, X6 n' e - hashMode: 'slash',
# c# O7 W! k1 e0 w. ?" F - });
复制代码如果函数无法使用会话存储恢复导航器,它将 使用 createBrowserNavigatorFromLocation 函数。
8 p( f+ H, u0 d E2 }连接创建 BrowserNavigator 实例并使用其方法不会自动更新 浏览器历史记录。 要做到这一点,开发人员应将其附加到系统中。 在此之前, 导航器只会更新其内部状态,并通知所有订阅者有关更改。 手动 附件是必要的,以防止出现创建多个此类导航器的情况,即 每个导航器都试图用其内部状态控制浏览器历史记录。 为了让导航器控制浏览器的历史记录,需要通过 attach方法来附加历史记录: - import { BrowserNavigator } from '@telegram-apps/sdk';9 u6 V# E2 o8 u# g9 R
- ~. ]7 X% u% C! e: f- const navigator = new BrowserNavigator(...);
6 s! Q( e' s) M! P5 h# L6 o1 K
' ]; R) t4 w! O8 o) h- navigator.attach().then(() => {
0 ]8 `, T5 f9 Y0 j z - console.log('Attachment completed');% w6 r7 X. s7 m, {4 s- p
- });
复制代码该方法返回一个承诺,当附件完成时,该承诺将被解析。 要阻止导航器修改浏览器的历史记录,请使用 detach 方法: 导航导航器为开发人员提供了一系列操作导航历史记录的方法。 back历史倒退 1. forward历史向前推移 1. go按指定的 delta 值更改当前活动的历史项目索引。 如果更新后的索引指向一个不存在的历史项,该方法不会更改 索引。 在指定 fit 参数之前, 。 如果指定了它,该方法会调整传入的 delta,使 适合[0, history.length - 1]范围。 - // Goes back in history by 3 entries.& o# J: T. a/ s% ~! m1 e. k1 J7 O1 A
- navigator.go(-3);
8 D' }$ h4 ~- a0 K/ ` - ( l$ j% o5 j1 Q- K! d
- // Will surely do nothing in real-world applications.
2 o, G9 H3 ?% c( I1 O - navigator.go(-100000);
( @; Q! y6 b/ K+ h+ m
5 z$ e U4 s( x9 G ^- // Will go back to the oldest entry.
3 u0 `& [+ [. v$ E; Q' b7 g - navigator.go(-100000, true);2 Y+ N Q) s2 l1 l
4 h0 C8 s3 s: U8 p9 t) W- // Will go forward to the newest entry.3 B. M% I4 V2 J' ?
- navigator.go(100000, true);
复制代码 goTo转到指定的索引。 如果传递的索引超出范围,该方法将不起任何作用。 如果指定了 fit 选项,并且索引超出了界限,那么它将被调整为最接近 的界限。 - // Will do nothing.
+ _+ L0 W/ _' v6 ^3 G, ^ - navigator.goTo(-1);
; l+ ?5 f4 ^2 h$ H - + ], o9 f* b O4 D; d: L' f0 x- T
- // Will go to the oldest entry.% `- o, U5 S2 g# n% A7 c/ B
- navigator.goTo(0);3 U! P$ ] S8 w! Z6 f
) E, h7 H* f+ d$ P. s- // Will surely do nothing in real-world applications.: f( O5 s$ K4 n0 d( O& H
- navigator.goTo(100000);2 W3 i2 p* K, d% d
- 6 p; T0 Y0 S+ y/ [
- // Will go back to the oldest entry.
4 O% p4 L3 M$ s' F$ k" J- g - navigator.goTo(-100000, true);) D6 V! { v' _* u) H: [! Z
% X' e0 k) y2 ]: z$ k, N- // Will go forward to the newest entry.
. O+ F" z! m- j7 Y/ t - navigator.goTo(100000, true);
复制代码 push推送新的历史项目。 该方法用 推送的条目替换当前条目之后的所有条目。 请注意,传递的项目总是相对的。 如果要将其用作绝对值,请使用 / 前缀。 例如:/absolute, { pathname: '/absolute' }. 要创建最终路径,导航器会使用 URL 类构造函数中的一种方法,根据当前路径解析 路径。 在接下来的示例中,我们假设当前条目为 /home/blog。 绝对路径名指定绝对路径不会将其与当前路径合并,而是使用 的完整路径: - navigator.push('/database');2 K% ^% E) {6 e; a% w0 t: h
- // or
, B. M7 n, K3 x e! b" ^ - navigator.push({ pathname: '/database' });/ N# y9 x" N, J0 L: C: Y+ Q0 b
- // Navigator will add a new entry, and the current one becomes /database
复制代码 相对路径名指定相对路径名的行为与浏览器中的行为相同: - navigator.push('license'); O7 u7 ]$ W3 ^
- // or$ Q) o$ @8 I2 N5 P% |2 o$ Y
- navigator.push({ pathname: 'license' });
2 N7 k* \% N- _& k - // Navigator will add a new entry, and the current one becomes /home/license
复制代码 搜索要添加带查询参数的条目,请使用问号 (?) 或 search 条目属性: - navigator.push('?id=1');9 Y% L# r: P ^5 E
- // or
6 X) z0 S5 @4 u: y - navigator.push({ search: '?id=1' }); C8 n y& i2 L$ J5 ^8 D/ v
- // Navigator will add a new entry, and the current one becomes /home/blog?id=1
复制代码INFO 推送带有不同查询参数或缺少查询参数的新路径名将导致当前 查询参数丢失。 为防止出现这种情况,开发人员应再次通过它们。 Hash添加条目的散列部分的逻辑与 search相同,但使用的是 hashtag (#) 和 hash 条目属性。 - navigator.push('#introduction');" y$ ^4 q+ n- C0 P$ N
- // or
" R1 f3 l' x0 l% _8 v7 v; ] - navigator.push({ hash: '#introduction' });* ?: @, a3 M; @% J$ f
- // Navigator will add a new entry, and the current
7 g. M# m/ @% A. t, T5 t - // one becomes /home/blog#introduction
复制代码 replacereplace 方法的功能与 push 方法类似,但它不会创建新的 条目。 相反,它取代了当前的版本。 属性index当前历史光标。 - const navigator = new BrowserNavigator(['/'], 0);
/ n/ _" u% r0 ^ - navigator.index; // 0
复制代码 id当前历史项目标识符。 如果没有明确指定 id,导航器会自行生成。 - const navigator = new BrowserNavigator(['/'], 0);
; C( d t) q2 C! E6 D- W* @! @ F - navigator.id; // 'abb721'5 I4 j& l) P% ?. w6 r4 x! O4 g. Z
0 k1 n7 I" J1 ?- const navigator2 = new BrowserNavigator([{ id: 'a', pathname: '/' }], 0);+ L: L. Y8 \ O
- navigator2.id; // 'a'
复制代码 hash当前历史项目哈希值。 - const navigator = new BrowserNavigator(['/#jungle'], 0);
" s: g; h( k/ x+ |$ Z4 j - navigator.hash; // "#jungle"
复制代码 hasPrev如果导航器在当前项目之前有项目,则为 True。 - const navigator = new BrowserNavigator(['/'], 0);
# _6 v0 Q( |- h1 m5 d* u8 q - navigator.hasPrev; // false
5 P0 M) k/ t# e - |$ Q( p6 P0 _$ t! _3 `
- const navigator2 = new BrowserNavigator(['/a', '/b'], 1);
+ U W/ }- @) s/ {1 z - navigator2.hasPrev; // true
复制代码 hasNext如果导航器在当前项目之后还有项目,则为 True。 - const navigator = new BrowserNavigator(['/'], 0);& c# Q& I+ M4 n& g! G. Y
- navigator.hasNext; // false0 e7 ]% m: M& C/ Y5 e6 _0 c" n
- 2 \3 O3 E" { y! N9 N5 t
- const navigator2 = new BrowserNavigator(['/a', '/b'], 0);5 B4 M* L* s" F. L9 _7 u4 p; q/ e
- navigator2.hasNext; // true
复制代码 history安全修改导航历史记录。 - const navigator = new BrowserNavigator(['/a#a-hash', '/b?b-query=1'], 0);3 j; M9 i( i u& x o
- navigator.history;
: r; [0 e7 X5 o! e - // [
; ~ w3 d% N6 s# |! [7 }% _ - // { pathname: '/a', hash: '#a-hash', search: '', id: 'ahJJ123' },
' `. H8 T7 z8 k# { - // { pathname: '/b', hash: '', search: '?b-query=1', id: 'dd82' },
# k1 v' m) K) h0 V& s4 z; ^ - // ]
复制代码 path路径,包括路径名、搜索和哈希值。 - const navigator = new BrowserNavigator([{
" w) u W i6 v m - pathname: '/a',- f4 S9 b6 E d6 ^# g
- hash: '#mama',
$ N$ Q8 D, b6 e: d - search: '?joe',
# G5 P* l0 Q$ k( Y# w - }], 0);
0 Z' d0 z% A* w - navigator.path; // '/a?joe#mama'
复制代码 pathname当前路径名。 总是以斜线开头。 - const navigator = new BrowserNavigator([{" T7 P+ o* O2 V2 u! w$ X
- pathname: '/a',
F4 M" C8 a& y$ z$ a& q; w3 ] - hash: '#mama',; h& r- z$ v- I. l* e
- search: '?joe',7 D! G3 s0 F$ ~* W
- }], 0);3 A- t) K' ~; H2 ~6 }5 |4 {
- navigator.pathname; // '/a'
复制代码 search当前查询参数。 - const navigator = new BrowserNavigator([{
) o5 c' E$ K7 F - pathname: '/a',
4 ]! I8 I, U ` - hash: '#mama',( k7 T- F. b: ~6 z4 N1 T
- search: '?joe',
) Z+ o/ k ^: i* e+ {0 y - }], 0);
4 n. O3 Q" b7 Y) O0 y0 _ - navigator.search; // '?joe'
复制代码
) m" R& S. V6 I# y3 K4 N/ Cstate当前历史项目状态。 - const navigator = new BrowserNavigator([{ state: 'test' }], 0);1 P; n/ s( K% ]1 i. T3 `8 `% S7 S
- 7 ]8 H( E! u) b& K2 x
$ W, c& D) ^# M. N- navigator.state; // 'test'
复制代码 2 i! Y' _" l7 i/ o. j' B$ G2 v
事件BrowserNavigator 提供了 on 和 off 方法来管理事件监听器。 目前,可用于监听的 事件只有 change。 change 事件的有效载荷是一个对象,包含 以下属性: - navigator: BrowserNavigator: 相关导航器实例。
- delta: number: 当前活动项目光标的 delta。
- from: { pathname: string; hash: string; search: string; id: string; state?状态 }: 以前活动的历史项目。
- to:结构与 from 相同的对象,代表当前活动的历史项目。
& \6 \; _6 Q9 l" w& x6 m $ b4 r1 D. ~ i" K
添加事件监听器要为 change 事件添加事件监听器,请使用 on 方法: - const removeEventListener = navigator.on('change', (ev) => {! ?; @: D5 O: p# j6 G
- console.warn('Navigation state changed', ev);. N1 r) c \% W' m$ l9 y
- });
复制代码 删除事件监听器要移除事件监听器,请调用 on 方法返回的函数: 或者,您也可以使用导航器的 off 方法:
( O& _" P! R; r6 s- function listener(ev) {% c. y2 y% K! {% P! B) w. }
- console.warn('Navigation state changed', ev);8 D9 G- Y; r4 F) I$ V& [& O
- }( N0 m0 d* X' N+ H1 s! @; |' e9 V% h
- navigator.on('change', listener);
# J; ?1 @9 v) w+ a. V+ W- X6 \- T - navigator.off('change', listener);
复制代码 其他方法renderPathrenderPath 方法将导航器的 base 属性与给定的路径数据相结合,并应用 导航器的导航模式。 该方法以字符串形式返回完全渲染的路径。 - const n = new BrowserNavigator(['/'], 0, {- u5 j5 u8 G& b; D& K+ C) A
- hashMode: 'slash',# h3 v7 p2 B& `% S7 t2 ?! {
- });7 f9 P0 e9 V6 U5 s$ H+ K
- n.renderPath('/test'); // '#/test'3 |5 m- T6 i8 D' \3 Z. Y& I
' d2 x2 b3 O4 T3 m% u' D- const n2 = new BrowserNavigator(['/'], 0, {
$ e5 [( _/ c9 w5 N+ s* V( S5 [ - base: '/my-base',. r( Y3 I. A2 |* S" C! y# Y3 ]* z1 {
- hashMode: 'slash',5 f2 f5 q4 e* {2 t4 o7 H
- });+ X' ^' Z8 w3 K2 x
- n2.renderPath('/test'); // '#/my-base/test'
% ]7 Z! H' _+ F
. n5 N1 c9 H, {- l) w. I- const n3 = new BrowserNavigator(['/'], 0, {$ F+ H6 N5 r- |0 j
- base: '/my-base',/ u: L8 r( ?2 S4 r5 F$ e
- hashMode: null,# J6 l# L8 }( I* T2 M$ k4 K
- });
8 K6 j9 ? l" h+ b5 U& ]; i0 x! {. x - n3.renderPath('/test'); // '/my-base/test'
复制代码 parsePathparsePath 方法会根据当前导航类型解析所提供的路径,并以对象形式返回 。 这种方法有助于了解导航器如何解释给定路径。 " j: B5 j. b5 K* X; C
- const n = new BrowserNavigator(['/'], 0);
4 N' P' W5 \( {3 `) [ - n.parsePath('/test');/ N. {; L+ Y! H' x1 m
- // { pathname: '/', search: '', hash: '' }
$ ]* j1 d5 ]1 j' l9 G6 }
2 z, {$ }( e6 |6 c7 l5 {- n.parsePath('/test#abc');
8 Z8 @! s8 l. I( d - // { pathname: '/abc', search: '', hash: '' }
$ `( u2 j' P) g- B5 D
) c: K2 S& B- |) U- n.parsePath('/test#abc?query#some-hash');
1 _2 z5 |& v+ U7 s1 i - // { pathname: '/abc', search: '?query', hash: '#some-hash' }
7 v4 r& `! c% k2 i, K7 E Y) B - ( X$ |/ ?; ^$ f
- const n2 = new BrowserNavigator(['/'], 0, { hashMode: null });1 L3 {+ j) G$ L7 Y5 ?, e( [1 T5 D
- n2.parsePath('/test');
2 `4 ]9 _. i* v- ]" U1 m: Y - // { pathname: '/test', search: '', hash: '' } C& M K3 q: ?( ?* w/ w2 Y& ~( T
- 0 C+ D1 B6 E# a
- n2.parsePath('/test#abc');
2 @$ L! ^4 y1 \5 P' Y7 }( y. s) n - // { pathname: '/test', search: '', hash: '#abc' }
8 T! C; o9 I0 E* p- z
9 _% e+ a( x, U* |5 I- n2.parsePath('/test?query#abc');
2 b+ f2 b( W& Y+ X1 h2 Q5 } - // { pathname: '/test', search: '?query', hash: '#abc' }
复制代码 + N: H* f6 c# L, @6 \7 a6 ?
7 u! v/ d, f/ |) M5 w8 d' o
3 _, f7 A, N1 v% A3 W |