Skip to content

Commit 841fd76

Browse files
committedJul 2, 2024
fix: capture for circular reference in state editor
1 parent 4b3ba2e commit 841fd76

File tree

1 file changed

+40
-19
lines changed

1 file changed

+40
-19
lines changed
 

‎packages/devtools/client/components/StateEditor.vue

+40-19
Original file line numberDiff line numberDiff line change
@@ -14,24 +14,41 @@ const emit = defineEmits<{
1414
1515
const isOpen = useVModel(props, 'open', emit, { passive: true })
1616
const colorMode = useColorMode()
17-
const proxy = ref()
17+
const proxy = shallowRef()
18+
const error = shallowRef()
1819
1920
const state = useState(props.name)
20-
if (props.state)
21-
proxy.value = JSON.parse(JSON.stringify(props.state))
22-
else if (typeof props.state === 'number' || typeof props.state !== 'string')
23-
proxy.value = props.state
2421
25-
const watcher = watchPausable(
26-
proxy,
27-
(value) => {
28-
if (typeof value !== 'number' && typeof value !== 'string')
29-
deepSync(value, props.state)
30-
else
31-
state.value = value
32-
},
33-
{ deep: true },
34-
)
22+
function clone() {
23+
error.value = undefined
24+
try {
25+
if (props.state)
26+
proxy.value = JSON.parse(JSON.stringify(props.state))
27+
else if (typeof props.state === 'number' || typeof props.state !== 'string')
28+
proxy.value = props.state
29+
}
30+
catch (e) {
31+
console.error(e)
32+
error.value = e
33+
}
34+
}
35+
36+
let watcher: ReturnType<typeof watchPausable> | undefined
37+
38+
onMounted(() => {
39+
clone()
40+
41+
watcher = watchPausable(
42+
proxy,
43+
(value) => {
44+
if (typeof value !== 'number' && typeof value !== 'string')
45+
deepSync(value, props.state)
46+
else
47+
state.value = value
48+
},
49+
{ deep: true },
50+
)
51+
})
3552
3653
function deepSync(from: any, to: any) {
3754
for (const key in from) {
@@ -47,10 +64,10 @@ function deepSync(from: any, to: any) {
4764
}
4865
4966
async function refresh() {
50-
watcher.pause()
51-
proxy.value = JSON.parse(JSON.stringify(props.state))
67+
watcher?.pause()
68+
clone()
5269
await nextTick()
53-
watcher.resume()
70+
watcher?.resume()
5471
}
5572
</script>
5673

@@ -75,13 +92,17 @@ async function refresh() {
7592
<template v-if="isOpen">
7693
<NButton v-tooltip.bottom="'Refresh View'" title="Refresh View" icon="carbon-renew" :border="false" @click="refresh" />
7794
<DataSchemaButton
78-
v-if="proxy"
95+
v-if="proxy && !error"
7996
:getter="() => ({ name, input: JSON.stringify(proxy) })"
8097
/>
8198
</template>
8299
</div>
83100
<template v-if="isOpen || !name">
101+
<div v-if="error" class="bg-red:10 px5 py3 text-red">
102+
Error: {{ error }}
103+
</div>
84104
<JsonEditorVue
105+
v-else
85106
v-model="proxy"
86107
v-bind="$attrs"
87108
class="json-editor-vue"

0 commit comments

Comments
 (0)
Please sign in to comment.