样式生成器预览放到iframe,防止影响到外面网页

This commit is contained in:
John Smith 2023-10-09 22:43:57 +08:00
parent 0f17dfdec7
commit fd1518db1d
3 changed files with 66 additions and 35 deletions

View File

@ -262,6 +262,9 @@ export default class ChatClientTest {
} else {
sleepTime = randInt(0, 400)
}
if (this.timerId) {
window.clearTimeout(this.timerId)
}
this.timerId = window.setTimeout(this.onTimeout.bind(this), sleepTime)
}

View File

@ -32,11 +32,16 @@ export default {
}
},
data() {
let customStyleElement = document.createElement('style')
document.head.appendChild(customStyleElement)
return {
config: chatConfig.deepCloneDefaultConfig(),
chatClient: null,
textEmoticons: [], //
pronunciationConverter: null,
textEmoticons: [], //
customStyleElement, //
}
},
computed: {
@ -79,12 +84,18 @@ export default {
// OBSOBS
document.addEventListener('visibilitychange', this.onVisibilityChange)
}
window.addEventListener('message', this.onWindowMessage)
},
beforeDestroy() {
window.removeEventListener('message', this.onWindowMessage)
document.removeEventListener('visibilitychange', this.onVisibilityChange)
if (this.chatClient) {
this.chatClient.stop()
}
document.head.removeChild(this.customStyleElement)
},
methods: {
onVisibilityChange() {
@ -105,6 +116,7 @@ export default {
}
try {
//
await initChatClientPromise
} catch (e) {
this.$message.error({
@ -199,11 +211,29 @@ export default {
this.textEmoticons = await chat.getTextEmoticons()
},
start() {
this.chatClient.start()
},
stop() {
this.chatClient.stop()
//
onWindowMessage(event) {
if (event.origin !== window.location.origin) {
console.warn(`消息origin错误${event.origin} != ${window.location.origin}`)
return
}
let { type, data } = event.data
switch (type) {
case 'roomSetCustomStyle':
this.customStyleElement.innerText = data.css
break
case 'roomStartClient':
if (this.chatClient) {
this.chatClient.start()
}
break
case 'roomStopClient':
if (this.chatClient) {
this.chatClient.stop()
}
break
}
},
onAddText(data) {

View File

@ -28,16 +28,16 @@
<div :style="{ position: 'relative', top: `${exampleTop}px` }">
<el-form inline style="line-height: 40px">
<el-form-item :label="$t('stylegen.playAnimation')" style="margin: 0">
<el-switch v-model="playAnimation" @change="onPlayAnimationChange"></el-switch>
<el-switch v-model="playAnimation" @change="setExampleRoomClientStart"></el-switch>
</el-form-item>
<el-form-item :label="$t('stylegen.backgrounds')" style="margin: 0 0 0 30px">
<el-switch v-model="exampleBgLight" :active-text="$t('stylegen.light')" :inactive-text="$t('stylegen.dark')"></el-switch>
</el-form-item>
</el-form>
<div id="example-container" :class="{ light: exampleBgLight }">
<div id="fakebody">
<room ref="room"></room>
</div>
<iframe id="example-room-iframe" ref="exampleRoomIframe"
:src="exampleRoomUrl" frameborder="0" @load="onExampleRoomLoad"
></iframe>
</div>
</div>
</el-col>
@ -52,17 +52,11 @@ import LineLike from './LineLike'
export default {
name: 'StyleGenerator',
components: {
Legacy,
LineLike,
Room: () => import('@/views/Room'),
},
components: { Legacy, LineLike },
data() {
let styleElement = document.createElement('style')
document.head.appendChild(styleElement)
//
// --\
// -> subComponentResults -> subComponentResult -> inputResult -> 0.5s -> debounceResult -> exampleCss
// -> subComponentResults -> subComponentResult -> inputResult -> 0.5s -> debounceResult
return {
//
subComponentResults: {
@ -75,20 +69,18 @@ export default {
//
debounceResult: '',
styleElement,
exampleTop: 0,
playAnimation: true,
exampleBgLight: false
}
},
computed: {
exampleRoomUrl() {
return this.$router.resolve({ name: 'test_room', query: { lang: this.$i18n.locale } }).href
},
//
subComponentResult() {
return this.subComponentResults[this.activeTab]
},
// CSS
exampleCss() {
return this.debounceResult.replace(/^body\b/gm, '#fakebody')
}
},
watch: {
@ -98,9 +90,7 @@ export default {
inputResult: _.debounce(function(val) {
this.debounceResult = val
}, 500),
exampleCss(val) {
this.styleElement.innerText = val
}
debounceResult: 'setExampleRoomCustomCss'
},
mounted() {
this.debounceResult = this.inputResult = this.subComponentResult
@ -109,8 +99,6 @@ export default {
},
beforeDestroy() {
this.$parent.$el.removeEventListener('scroll', this.onParentScroll)
document.head.removeChild(this.styleElement)
},
methods: {
onParentScroll(event) {
@ -120,13 +108,22 @@ export default {
this.exampleTop = event.target.scrollTop
}
},
onPlayAnimationChange(value) {
if (value) {
this.$refs.room.start()
} else {
this.$refs.room.stop()
}
onExampleRoomLoad() {
this.setExampleRoomCustomCss(this.debounceResult)
this.setExampleRoomClientStart(this.playAnimation)
},
setExampleRoomCustomCss(css) {
this.sendMessageToExampleRoom('roomSetCustomStyle', { css })
},
setExampleRoomClientStart(isStart) {
this.sendMessageToExampleRoom(isStart ? 'roomStartClient' : 'roomStopClient')
},
sendMessageToExampleRoom(type, data = null) {
let msg = { type, data }
this.$refs.exampleRoomIframe.contentWindow.postMessage(msg, window.location.origin)
},
copyResult() {
this.$refs.result.select()
document.execCommand('Copy')
@ -181,8 +178,9 @@ export default {
-webkit-gradient(linear, 0 0, 100% 100%, color-stop(.75, transparent), color-stop(.75, #eee));
}
#fakebody {
#example-room-iframe {
outline: 1px #999 dashed;
width: 100%;
height: 100%;
}
</style>