整理文档并接入 H5 体验测试链路

This commit is contained in:
2026-03-27 15:36:27 +08:00
parent 0e025c3426
commit 0e0a724025
55 changed files with 4177 additions and 55 deletions

View File

@@ -0,0 +1,127 @@
let currentRequest = null
let currentEventChannel = null
let pageResolved = false
function appendQueryParam(url, key, value) {
const separator = url.indexOf('?') >= 0 ? '&' : '?'
return `${url}${separator}${key}=${encodeURIComponent(value)}`
}
function buildWebViewSrc(request) {
let nextUrl = request.url
nextUrl = appendQueryParam(nextUrl, 'cmrBridge', request.bridgeVersion)
nextUrl = appendQueryParam(nextUrl, 'cmrKind', request.kind)
return nextUrl
}
function emitFallbackAndClose() {
if (!currentRequest || !currentEventChannel) {
return
}
if (!pageResolved) {
pageResolved = true
currentEventChannel.emit('fallback', currentRequest.fallback)
}
wx.navigateBack({
fail() {},
})
}
function emitCloseAndBack(payload) {
if (currentEventChannel && !pageResolved) {
pageResolved = true
currentEventChannel.emit('close', payload || {})
}
wx.navigateBack({
fail() {},
})
}
Page({
data: {
webViewSrc: '',
webViewReady: false,
loadErrorText: '',
},
onLoad() {
pageResolved = false
currentRequest = null
currentEventChannel = null
this.setData({
webViewSrc: '',
webViewReady: false,
loadErrorText: '',
})
try {
currentEventChannel = this.getOpenerEventChannel()
} catch (error) {
currentEventChannel = null
}
if (!currentEventChannel) {
return
}
currentEventChannel.on('init', (request) => {
currentRequest = request
wx.setNavigationBarTitle({
title: request.title || '内容体验',
fail() {},
})
this.setData({
webViewSrc: buildWebViewSrc(request),
webViewReady: true,
loadErrorText: '',
})
})
},
onUnload() {
if (currentEventChannel && !pageResolved) {
currentEventChannel.emit('close', {})
}
pageResolved = false
currentRequest = null
currentEventChannel = null
},
handleWebViewMessage(event) {
const dataList = event.detail && Array.isArray(event.detail.data)
? event.detail.data
: []
const rawMessage = dataList.length ? dataList[dataList.length - 1] : null
if (!rawMessage || typeof rawMessage !== 'object') {
return
}
const action = rawMessage.action || rawMessage.type || ''
if (!action) {
return
}
if (action === 'close') {
emitCloseAndBack(rawMessage.payload)
return
}
if (action === 'submitResult') {
if (currentEventChannel) {
currentEventChannel.emit('submitResult', rawMessage.payload || {})
}
return
}
if (action === 'fallback') {
emitFallbackAndClose()
}
},
handleWebViewError() {
this.setData({
loadErrorText: '页面打开失败,已回退原生内容',
})
emitFallbackAndClose()
},
})

View File

@@ -0,0 +1,3 @@
{
"navigationBarTitleText": "内容体验"
}

View File

@@ -0,0 +1,136 @@
import { type H5BridgeMessage, type H5ExperienceRequest } from '../../game/experience/h5Experience'
type ExperienceWebViewPageData = {
webViewSrc: string
webViewReady: boolean
loadErrorText: string
}
let currentRequest: H5ExperienceRequest | null = null
let currentEventChannel: WechatMiniprogram.EventChannel | null = null
let pageResolved = false
function appendQueryParam(url: string, key: string, value: string): string {
const separator = url.indexOf('?') >= 0 ? '&' : '?'
return `${url}${separator}${key}=${encodeURIComponent(value)}`
}
function buildWebViewSrc(request: H5ExperienceRequest): string {
let nextUrl = request.url
nextUrl = appendQueryParam(nextUrl, 'cmrBridge', request.bridgeVersion)
nextUrl = appendQueryParam(nextUrl, 'cmrKind', request.kind)
return nextUrl
}
function emitFallbackAndClose() {
if (!currentRequest || !currentEventChannel) {
return
}
if (!pageResolved) {
pageResolved = true
currentEventChannel.emit('fallback', currentRequest.fallback)
}
wx.navigateBack({
fail: () => {},
})
}
function emitCloseAndBack(payload?: Record<string, unknown>) {
if (currentEventChannel && !pageResolved) {
pageResolved = true
currentEventChannel.emit('close', payload || {})
}
wx.navigateBack({
fail: () => {},
})
}
Page<ExperienceWebViewPageData, WechatMiniprogram.IAnyObject>({
data: {
webViewSrc: '',
webViewReady: false,
loadErrorText: '',
},
onLoad() {
pageResolved = false
currentRequest = null
currentEventChannel = null
this.setData({
webViewSrc: '',
webViewReady: false,
loadErrorText: '',
})
try {
currentEventChannel = this.getOpenerEventChannel()
} catch {
currentEventChannel = null
}
if (!currentEventChannel) {
return
}
currentEventChannel.on('init', (request: H5ExperienceRequest) => {
currentRequest = request
wx.setNavigationBarTitle({
title: request.title || '内容体验',
fail: () => {},
})
this.setData({
webViewSrc: buildWebViewSrc(request),
webViewReady: true,
loadErrorText: '',
})
})
},
onUnload() {
if (currentEventChannel && !pageResolved) {
currentEventChannel.emit('close', {})
}
pageResolved = false
currentRequest = null
currentEventChannel = null
},
handleWebViewMessage(event: WechatMiniprogram.CustomEvent) {
const dataList = event.detail && Array.isArray(event.detail.data)
? event.detail.data
: []
const rawMessage = dataList.length ? dataList[dataList.length - 1] : null
if (!rawMessage || typeof rawMessage !== 'object') {
return
}
const message = rawMessage as H5BridgeMessage
const action = message.action || message.type || ''
if (!action) {
return
}
if (action === 'close') {
emitCloseAndBack(message.payload)
return
}
if (action === 'submitResult') {
if (currentEventChannel) {
currentEventChannel.emit('submitResult', message.payload || {})
}
return
}
if (action === 'fallback') {
emitFallbackAndClose()
}
},
handleWebViewError() {
this.setData({
loadErrorText: '页面打开失败,已回退原生内容',
})
emitFallbackAndClose()
},
})

View File

@@ -0,0 +1,11 @@
<view wx:if="{{!webViewReady}}" class="experience-webview__loading">
<view class="experience-webview__loading-title">内容页加载中</view>
<view wx:if="{{loadErrorText}}" class="experience-webview__loading-error">{{loadErrorText}}</view>
</view>
<web-view
wx:if="{{webViewReady && webViewSrc}}"
src="{{webViewSrc}}"
bindmessage="handleWebViewMessage"
binderror="handleWebViewError"
></web-view>

View File

@@ -0,0 +1,27 @@
.page {
height: 100%;
}
page {
background: #f5f7f6;
}
.experience-webview__loading {
min-height: 100vh;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 48rpx;
color: #1f2f26;
}
.experience-webview__loading-title {
font-size: 32rpx;
font-weight: 600;
}
.experience-webview__loading-error {
margin-top: 20rpx;
font-size: 26rpx;
color: #a0523d;
}