Welcome to my blog: BlueSun – wechat small program routing combat
Catalogue of 0.
- 1. Introduction
- 2. Intelligent route redirect – Navigator module
- 3. Virtual routing policy – Router module
- 4. Landing transfer policy – LandTransfer module
- 4.1. For the first problem to be solved: uniform landing page
- 4.2. For the second problem to be solved: short chain parameters
- 4.3. Design of LandTransfer module
- 5. Better development experience
- 5.1. The Typescript + Router
- 5.2. Intelligently Generate route configuration
- 5.3. Customize component hops
- 6. Overall architecture diagram
- 7. Last but not least
1. Introduction
In wechat applets consist of an App() instance, and numerous pages (). In the applet, the routing of all pages is managed by the framework, which maintains all pages in the form of a stack, and then provides the following API to jump between routes:
wx.navigateTo
wx.redirectTo
wx.navigateBack
wx.switchTab
wx.reLaunch
However, for an enterprise application, these questions are left to the developer:
- Native API used
Callback
Function implementation form, with our modern commonPromise
和async/await
There is gap. - Based on the design of small program routing, exposed to the external is the real routing (such as scanning code, public number link, etc.), the subsequent project reconstruction left a historical burden.
- Applets stack up to 10 levels, and after 10 or more
wx.navigateTo
Invalid, need developer judgement to usewx.redirectTo
Or other API - There is a special page in the applet page stack: the Tab page, which needs to be used
wx.switchTab
To jump. Developers need to take the initiative to judge, it is not convenient to change Tab page attributes. - Additionally, for small code, use the unlimited API wxacode.getunlimited, which has a parameter length limit of 32 bits. It needs to be solved by the developers themselves.
This paper is expected to provide solutions to these problems one by one.
2. Intelligent route redirect – Navigator module
Here we solve together:
- Native API is not Promsie
- Special handling when page stack breaks 10 layers
- Special page Tab jump processing
Our idea is to design a logic that can automatically determine which wechat routing API to use according to the scenario, and then provide only one function externally, for example:
gotoPage('/pages/goods/index')
Copy the code
The specific logic is as follows:
- This parameter is used when the route to the TAB page is displayed
wx.switchTab
. - When the page stack reaches 10 levels, if the page to jump to is in the page stack, use
wx.navigateBack({ delta: X })
Push out to the target page. - When the page stack reaches 10 layers, the target page does not exist in the page stack
wx.redirectTo
Replace the top of the stack page. - For other purposes
wx.navigateTo
Incidentally, we implement this function as a Promise and support parameters passed in as objects, such as:
gotoPage('/pages/goods/index', { name: 'jc'}).then(...) .catch(...) ;Copy the code
For most scenarios, just use gotoPage.
It must also have a specific case, need to explicitly specify use navigateTo/switchTab/redirectTo/navigateBack which one.
So let’s follow a similar implementation, an API that satisfies the same pattern
navigateTo('/pages/goods/index', { name: 'jc'}).then(...) .catch(...) ; switchTab('/pages/goods/index', { name: 'jc'}).then(...) .catch(...) ; redirectTo('/pages/goods/index', { name: 'jc'}).then(...) .catch(...) ; navigateBack('/pages/goods/index', { name: 'jc'}).then(...) .catch(...) ;Copy the code
These functions can all converge into the same module, which we call Navigator
const navigator = newNavigator(); navigator.gotoPage(...) ; navigator.navigateTo(...) ; navigator.switchTab(...) ; navigator.redirectTo(...) ; navigator.navigateBack(...) ;Copy the code
Module design:
3. Virtual routing policy – Router module
Here, we address:
- Real routes are exposed externally, causing heavy historical burden.
In many applications, we often need to map all the routes that a pattern matches to the same page. For example, we have a Goods page that will be used to host all the Goods with different ids.
At the code level, the call is expected to look like this:
// Create a routing instance
const router = new Router();
// Register the route
router.register({
path: '/goods/:id'.// Virtual route
route: '/pages/goods/index'.// Real route
});
// pages/goods/index: options = {id: '123'}
router.gotoPage('/goods/123');
// pages/goods/index: options = {id: '456'}
router.gotoPage('/goods/456');
Copy the code
The core logic of the Class Router is to do:
- You can register routes to store the relationship between virtual paths and real paths.
- It can convert virtual paths to real paths and recognize dynamic segments.
- Route redirection.
For “route registration”, we can do this by storing a map inside it.
For “path conversion”, vue-Router has a similar implementation, found through its source code, internal uses path-to-regexp as a path matching engine, we can use it.
Then for “route jump”, we can directly reuse the Navigator module mentioned above and complete the route jump by inputting the real path.
Module design:
Among them:
- RouteMatcher: Provides dynamic routing parameter matching and internally uses path-to-regexp as the path matching engine.
- Route: Creates a router for each path and stores the relationship between the virtual path and the real Route of each Route.
- Router: Integrates all internal modules and provides a unified and elegant call method externally.
4. Landing transfer policy – LandTransfer module
Here, we address:
- Small program scan code, public link and other scenarios under the landing page unification.
- Small program code, for unlimited API wxacode.getunlimited, breaks the parameter length limit of 32 bits.
4.1. For the first problem to be solved: uniform landing page
We put such as: scan small program code, public number menu, public number articles and other ways to open a small program a page path called “external route”.
According to the design of the small program, the connection exposed to the external is the real page path, such as: /pages/home/index. The disadvantages of this design in practice: each landing page is scattered, and it is difficult to modify the real file path in the later stage.
In “long life cycle” products, it is inevitable that we will encounter project refactoring as the product iterates. If all we distribute are unprocessed real paths, our refactoring is tied down and requires a lot of compatibility. Because you don’t know how many of the qr codes that are distributed are printed into physical materials.
Then, the “virtual route” + “landing transfer” strategy appears basic and important.
The function of “virtual route”, **Router ** module provides us with support, we also need to provide a unified landing page externally, let it complete the transfer of internal routes.
Basic logic:
- A real route to be distributed to a unique landing page, such as:
$LAND_PAGE: /pages/land-page/index
- From this landing page, internal routes are redirected and forwarded by receiving parameters such as:
path=/user&name=jc&age=18
At the code level, we want to implement something like this:
// /pages/land-page/index.ts
const landTransfer = new LandTransfer(landTransferOptions);
Page({
onLoad(options) {
landTransfer
.run(options)
.then(() = >{... }) .catch(() = > {...});
}
});
Copy the code
Then for TS, we can also use the decorator version, which is even easier:
import { landTransferDecorator } from 'wxapp-router';
Page({
@landTransferDecorator(landTransferOptions)
onLoad(options) {
// ...}});Copy the code
4.2. For the second problem to be solved: short chain parameters
Wechat applets mainly provide two interfaces to generate applets:
- Wxacod. get: obtains small program codes. This method is applicable to service scenarios where a small number of codes are required. The applets generated through this interface are valid forever and are limited to 100,000
- Wxacode. getUnlimited: obtains small program codes, which is applicable to service scenarios where a large number of codes are required. The small program code generated through this interface is valid forever and the number is temporarily unlimited.
The first way is to limit the number of wxacode.get to 10W, which is a lot and probably not used by most small programs.
But if we operate a small program of medium and large e-commerce, if: 1W kinds of goods x 10 kinds of commodity specifications, it will exceed this amount. It will be difficult to retrofit when the time comes.
So, if we’re running a “long life cycle” product, we use the second approach: wxacode.getunlimited
Unfortunately, although it has no number limit, it does have a limit of 32 characters for parameters, which is obviously not enough (a UUID is 32 characters).
In this case, we can use the form of “short chain parameter” to solve the problem. Since wxacode.getunLimited will be passed to the applet as the scene parameter as the query parameter, we can use the scene parameter to implement the short chain service, which requires the backend cooperation.
The front and back end interactions are as follows:
- When a small program needs to generate a small program code, request the interface provided by the back end, for example:
/api/encodeShortParams
- The back end converts the content to a 32-character string and stores it in the database.
- The back-end throughwxacode.getUnlimitedInterface, as a short-chained string
scene
Values to agree on a good unified landing page$LAND_PAGE
As apage
Value to generate small program code. - When the applet code is entered into the applet, the applet gets
scene
Parameter to request the interface provided by the back end, for example:/api/decodeShrotParams
- Applets understand the content and jump to the target page.
For the logic processing of the unified landing page, we only need to add a logic to convert the content of short chain parameters on the basis of the first question:
At the code level, we just need to define the way to convert short chain parameters: convertScenePrams
// in /pages/land-page/index.js
import { landTransferDecorator } from 'wxapp-router';
const landTransferOptions = {
// Receive options.scene in onLoad(options)
convertSceneParams: (sceneParams) = > {
return API.convertScene({ sceneParams }).then((content) = > {
// If the back-end store is JSON string, front-end decode
Content = {path: '/home', a: 1, b:2}
return JSON.parse(content); }); }}; Page({@landTransferDecorator(landTransferOptions)
onLoad(options) {
// ...}});Copy the code
The API. ConvertScene connects to the server to provide HTTP interface services.
4.3. Design of LandTransfer module
5. Better development experience
5.1. The Typescript + Router
For a route jump inside a small program, instead of specifying a string route, can we also make a chain call, like a function call, to jump to the page? Something like that;
routes.pages.user.go({ name: 'jc' });
Copy the code
The benefits of this are:
- A more natural way to call.
- Can combine TS, to do type hint and association.
Because the Wxapp-Router does not know what route the developer needs to register in advance, the route TS declaration file needs to be defined by the developer.
For example, we maintain a routing file in our project:
// config/routes.ts
// Create a routing instance
const router = new Router();
const routesConfig = [{
path: '/user'.route: '/pages/user/index'}, {path: '/goods'.route: '/pages/goods/index',}];type RoutesType {
paegs: {
user: Route<{name: string}>,
goods: Route,
}
}
// Register the route
router.batchRegister(routesConfig);
/ / access routes
const routes: RoutesType = router.getRoutes();
export default routes;
Copy the code
Then use it somewhere else:
import routes from './routes.ts';
routes.pages.user.go({ name: 'jc' });
Copy the code
5.2. Intelligently Generate route configuration
If we need to manually write RoutesType for each route when there are more routes, it’s a little uncomfortable.
In the applet, we configure the formal routes to app.json, so we can do most of the work by automating the build while following the established project structure, such as:
- Intelligent Registered Routing
- Intelligent identification page entry declaration
5.3. Customize component hops
All of these are used at the script level, and there is also WXML in the applet, which we want to use quickly in a component:
<Router path="/pageA" query="{{pageAQuery}}"></Router>
<Router path="/pageB" query="{{pageBQuery}}" type="redirectTo"></Router>
<Router path="/pageC/katy"></Router>
Copy the code
Then, implementing a custom component and wrapping the Router module around it should not be a problem.
Sample code:
// components/router.wxml
<view class="wxapp-router" bind:tap="gotoPage">
<slot />
</view>
Copy the code
// components/router.ts
Component({
properties: {
path: String.type: {
type: String.value: 'gotoPage'
},
route: String.query: Object.delta: Number.setData: Object,},methods: {
gotoPage(event) {
const router = getApp().router;
const { path, route, type, query} = this.data;
const toPath = route || path;
if (['gotoPage'.'navigateTo'.'switchTab'.'redirectTo'].includes(type)) {
(router as any) [type](toPath, query);
}
if (type= = ='navigateBack') {
const { delta, setData } = this.data;
router.navigateBack({ delta }, { setData })
}
}
}
})
Copy the code
6. Overall architecture diagram
Finally, we review the design of each module as a whole
- Navigator: Encapsulates wechat native routing API and provides intelligent jump policy.
- LandTransfer: Provides the landing page transfer policy.
- RouteMatcher: Provides dynamic routing parameter matching.
- Route: Creates a router for each path.
- Router: Integrates internal modules and provides elegant calls externally.
- Logger: internal Logger.
- Path-to-regexp: Routing matching engine for the open source community.
7. Last but not least
Since I have written a lot of practical articles, many students would like to learn the whole sample code, this time I simply wrote a tool, Enjoy It!
Wxapp-router: 🛵 The Router for Wechat Miniprogram