Skip to content

Commit 05833b6

Browse files
authoredAug 3, 2024··
feat(extension): support open-in-editor (#543)
1 parent 5e60d64 commit 05833b6

File tree

6 files changed

+102
-7
lines changed

6 files changed

+102
-7
lines changed
 

‎docs/.vitepress/config.ts

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ const GETTING_STARTED: DefaultTheme.NavItemWithLink[] = [
77
{ text: 'Introduction', link: '/getting-started/introduction' },
88
{ text: 'Installation', link: '/getting-started/installation' },
99
{ text: 'Features', link: '/getting-started/features' },
10+
{ text: 'Open in editor', link: '/getting-started/open-in-editor' },
1011
]
1112

1213
const GUIDES: DefaultTheme.NavItemWithLink[] = [
+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
# Open component in editor
2+
3+
When you select a component, you have the option to open the corresponding source file in your code editor.
4+
5+
## Used in devtools vite plugin
6+
7+
Vite plugin supports this feature out-of-the-box.
8+
9+
The feature is based on the [vite-plugin-vue-inspector](https://github.com/webfansplz/vite-plugin-vue-inspector) plugin and requires configuration, which you can do by looking at the [configuration documentation](https://github.com/webfansplz/vite-plugin-vue-inspector?tab=readme-ov-file#--configuration-ide--editor).
10+
11+
Starting from **v7.2.0**, you can specify the editor by `launchEditor` option:
12+
13+
This is a list of [supported editors](https://github.com/yyx990803/launch-editor?tab=readme-ov-file#supported-editors), please ensure that the editor's environment variables are correctly configured beforehand.
14+
15+
```ts
16+
import VueDevTools from 'vite-plugin-vue-devtools'
17+
export default defineConfig({
18+
plugins: [
19+
VueDevTools({
20+
launchEditor: 'webstorm',
21+
}),
22+
Unocss(),
23+
],
24+
})
25+
```
26+
27+
## Used in devtools browser extension
28+
29+
### Vite & Nuxt & Quasar CLI
30+
31+
Vite & Nuxt & Quasar CLI supports this feature out-of-the-box. Make sure to be in debug mode.
32+
33+
### Webpack
34+
35+
In your Vue project, install the [launch-editor-middleware](https://github.com/yyx990803/launch-editor#middleware) package and modify your webpack configuration:
36+
37+
1. Import the package:
38+
39+
```ts
40+
const openInEditor = require('launch-editor-middleware')
41+
```
42+
43+
2. In the `devServer` option, register the `/__open-in-editor` HTTP route:
44+
45+
```js
46+
devServer: {
47+
before: (app) => {
48+
app.use('/__open-in-editor', openInEditor())
49+
}
50+
}
51+
```
52+
53+
3. The editor to launch is guessed. You can also specify the editor app with the editor option. See the [supported editors list.](https://github.com/yyx990803/launch-editor?tab=readme-ov-file#supported-editors)
54+
55+
```js
56+
openInEditor('code')
57+
```
58+
59+
4. You can now click on the name of the component in the Component inspector pane (if the devtools knows about its file source, a tooltip will appear).
60+
61+
### Node.js
62+
63+
You can use the [launch-editor](https://github.com/yyx990803/launch-editor) package to setup an HTTP route with the `/__open-in-editor` path. It will receive file as an URL variable.
64+
65+
### Customize request
66+
67+
You can change the request host (default `/`) with the following code in your frontend app, e.g.
68+
69+
```ts
70+
if (process.env.NODE_ENV !== 'production') {
71+
window.VUE_DEVTOOLS_CONFIG = {
72+
openInEditorHost: 'http://localhost:3000/'
73+
}
74+
}
75+
```

‎packages/applet/src/modules/components/index.vue

+1-1
Original file line numberDiff line numberDiff line change
@@ -321,7 +321,7 @@ function closeComponentRenderCode() {
321321
<i v-tooltip.bottom="'Scroll to component'" class="i-material-symbols-light:eye-tracking-outline h-4 w-4 cursor-pointer hover:(op-70)" @click="scrollToComponent" />
322322
<i v-tooltip.bottom="'Show render code'" class="i-material-symbols-light:code h-5 w-5 cursor-pointer hover:(op-70)" @click="getComponentRenderCode" />
323323
<i v-if="isInChromePanel" v-tooltip.bottom="'Inspect DOM'" class="i-material-symbols-light:menu-open h-5 w-5 cursor-pointer hover:(op-70)" @click="inspectDOM" />
324-
<i v-if="activeTreeNodeFilePath && !isInChromePanel" v-tooltip.bottom="'Open in Editor'" class="i-carbon-launch h-4 w-4 cursor-pointer hover:(op-70)" @click="openInEditor" />
324+
<i v-if="activeTreeNodeFilePath" v-tooltip.bottom="'Open in Editor'" class="i-carbon-launch h-4 w-4 cursor-pointer hover:(op-70)" @click="openInEditor" />
325325
</div>
326326
</div>
327327
<RootStateViewer class="no-scrollbar flex-1 select-none overflow-scroll" :data="filteredState" :node-id="activeComponentId" :inspector-id="inspectorId" expanded-state-id="component-state" />
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,12 @@
11
import { rpc } from '@vue/devtools-core'
2+
import { isInChromePanel } from '@vue/devtools-shared'
23

34
export const vueInspectorDetected = ref(false)
45

56
export const openInEditor = async (file: string) => {
6-
return rpc.value.openInEditor({ file })
7+
const opts: { file: string, host?: string } = { file }
8+
if (isInChromePanel) {
9+
opts.host = 'chrome-extension'
10+
}
11+
return rpc.value.openInEditor(opts)
712
}

‎packages/devtools-kit/src/core/open-in-editor/index.ts

+14-5
Original file line numberDiff line numberDiff line change
@@ -6,21 +6,30 @@ export interface OpenInEditorOptions {
66
file?: string
77
line?: number
88
column?: number
9+
host?: string
910
}
1011

1112
export function setOpenInEditorBaseUrl(url: string) {
1213
target.__VUE_DEVTOOLS_OPEN_IN_EDITOR_BASE_URL__ = url
1314
}
1415

1516
export function openInEditor(options: OpenInEditorOptions = {}) {
16-
const { file, baseUrl = window.location.origin, line = 0, column = 0 } = options
17+
const { file, host, baseUrl = window.location.origin, line = 0, column = 0 } = options
1718
if (file) {
18-
if (devtoolsState.vitePluginDetected) {
19+
if (host === 'chrome-extension') {
20+
const fileName = file.replace(/\\/g, '\\\\')
21+
// @ts-expect-error skip type check
22+
const _baseUrl = window.VUE_DEVTOOLS_CONFIG?.openInEditorHost ?? '/'
23+
fetch(`${_baseUrl}__open-in-editor?file=${encodeURI(file)}`).then((response) => {
24+
if (!response.ok) {
25+
const msg = `Opening component ${fileName} failed`
26+
console.log(`%c${msg}`, 'color:red')
27+
}
28+
})
29+
}
30+
else if (devtoolsState.vitePluginDetected) {
1931
const _baseUrl = target.__VUE_DEVTOOLS_OPEN_IN_EDITOR_BASE_URL__ ?? baseUrl
2032
target.__VUE_INSPECTOR__.openInEditor(_baseUrl, file, line, column)
2133
}
22-
else {
23-
// @TODO: support other
24-
}
2534
}
2635
}

‎packages/playground/basic/src/main.ts

+5
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,11 @@ app.use(ElementPlus)
2222

2323
// devtools.connect()
2424

25+
// // @ts-expect-error skip type check
26+
// window.VUE_DEVTOOLS_CONFIG = {
27+
// openInEditorHost: 'http://localhost:3000',
28+
// }
29+
2530
const routes: RouteRecordRaw[] = [
2631
{
2732
path: '/',

0 commit comments

Comments
 (0)
Please sign in to comment.