1
1
import colors from 'picocolors'
2
2
import type { ViteDevServer } from './server'
3
3
import { isDefined } from './utils'
4
+ import type { PreviewServer } from './preview'
5
+ import { openBrowser } from './server/openBrowser'
4
6
5
- export type BindShortcutsOptions = {
7
+ export type BindShortcutsOptions < Server = ViteDevServer | PreviewServer > = {
6
8
/**
7
9
* Print a one line hint to the terminal.
8
10
*/
9
11
print ?: boolean
10
- customShortcuts ?: ( CLIShortcut | undefined | null ) [ ]
12
+ customShortcuts ?: ( CLIShortcut < Server > | undefined | null ) [ ]
11
13
}
12
14
13
- export type CLIShortcut = {
15
+ export type CLIShortcut < Server = ViteDevServer | PreviewServer > = {
14
16
key : string
15
17
description : string
16
- action ( server : ViteDevServer ) : void | Promise < void >
18
+ action ( server : Server ) : void | Promise < void >
17
19
}
18
20
19
- export function bindShortcuts (
20
- server : ViteDevServer ,
21
- opts : BindShortcutsOptions ,
21
+ export function bindShortcuts < Server extends ViteDevServer | PreviewServer > (
22
+ server : Server ,
23
+ opts ? : BindShortcutsOptions < Server > ,
22
24
) : void {
23
25
if ( ! server . httpServer || ! process . stdin . isTTY || process . env . CI ) {
24
26
return
25
27
}
26
- server . _shortcutsOptions = opts
27
28
28
- if ( opts . print ) {
29
+ const isDev = isDevServer ( server )
30
+
31
+ if ( isDev ) {
32
+ server . _shortcutsOptions = opts
33
+ }
34
+
35
+ if ( opts ?. print ) {
29
36
server . config . logger . info (
30
37
colors . dim ( colors . green ( ' ➜' ) ) +
31
38
colors . dim ( ' press ' ) +
@@ -34,16 +41,25 @@ export function bindShortcuts(
34
41
)
35
42
}
36
43
37
- const shortcuts = ( opts . customShortcuts ?? [ ] )
44
+ const shortcuts = ( opts ? .customShortcuts ?? [ ] )
38
45
. filter ( isDefined )
39
- . concat ( BASE_SHORTCUTS )
46
+ // @ts -expect-error passing the right types, but typescript can't detect it
47
+ . concat ( isDev ? BASE_DEV_SHORTCUTS : BASE_PREVIEW_SHORTCUTS )
40
48
41
49
let actionRunning = false
42
50
43
51
const onInput = async ( input : string ) => {
44
52
// ctrl+c or ctrl+d
45
53
if ( input === '\x03' || input === '\x04' ) {
46
- await server . close ( ) . finally ( ( ) => process . exit ( 1 ) )
54
+ try {
55
+ if ( isDev ) {
56
+ await server . close ( )
57
+ } else {
58
+ server . httpServer . close ( )
59
+ }
60
+ } finally {
61
+ process . exit ( 1 )
62
+ }
47
63
return
48
64
}
49
65
@@ -81,7 +97,13 @@ export function bindShortcuts(
81
97
} )
82
98
}
83
99
84
- const BASE_SHORTCUTS : CLIShortcut [ ] = [
100
+ function isDevServer (
101
+ server : ViteDevServer | PreviewServer ,
102
+ ) : server is ViteDevServer {
103
+ return 'pluginContainer' in server
104
+ }
105
+
106
+ const BASE_DEV_SHORTCUTS : CLIShortcut < ViteDevServer > [ ] = [
85
107
{
86
108
key : 'r' ,
87
109
description : 'restart the server' ,
@@ -119,3 +141,25 @@ const BASE_SHORTCUTS: CLIShortcut[] = [
119
141
} ,
120
142
} ,
121
143
]
144
+
145
+ const BASE_PREVIEW_SHORTCUTS : CLIShortcut < PreviewServer > [ ] = [
146
+ {
147
+ key : 'o' ,
148
+ description : 'open in browser' ,
149
+ action ( server ) {
150
+ const url = server . resolvedUrls . local [ 0 ]
151
+ openBrowser ( url , true , server . config . logger )
152
+ } ,
153
+ } ,
154
+ {
155
+ key : 'q' ,
156
+ description : 'quit' ,
157
+ action ( server ) {
158
+ try {
159
+ server . httpServer . close ( )
160
+ } finally {
161
+ process . exit ( )
162
+ }
163
+ } ,
164
+ } ,
165
+ ]
0 commit comments