Skip to content

Commit 4d6dcd9

Browse files
committed
add options for special browser to open preview page
1 parent 7e8c7df commit 4d6dcd9

File tree

15 files changed

+148
-12
lines changed

15 files changed

+148
-12
lines changed

README.md

+4
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,10 @@ let g:mkdp_command_for_global = 0
6363
" default: 0
6464
let g:mkdp_open_to_the_world = 0
6565
66+
" switch browser to open preview page
67+
" default: ''
68+
let g:mkdp_browser = ''
69+
6670
" options for markdown render
6771
" mkit: markdown-it options for render
6872
" katex: katex options for math

app/lib/util/opener.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export {};

app/lib/util/opener.js

+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
"use strict";
2+
Object.defineProperty(exports, "__esModule", { value: true });
3+
const tslib_1 = require("tslib");
4+
/*
5+
* fork from https://github.com/domenic/opener
6+
*/
7+
const child_process_1 = tslib_1.__importDefault(require("child_process"));
8+
const os_1 = tslib_1.__importDefault(require("os"));
9+
module.exports = function opener(args, tool, callback) {
10+
let platform = process.platform;
11+
args = [].concat(args);
12+
if (tool) {
13+
args.unshift(tool);
14+
}
15+
// Attempt to detect Windows Subystem for Linux (WSL).
16+
// WSL itself as Linux (which works in most cases), but in
17+
// this specific case we need to treat it as actually being Windows.
18+
// The "Windows-way" of opening things through cmd.exe works just fine here,
19+
// whereas using xdg-open does not, since there is no X Windows in WSL.
20+
if (platform === 'linux' && os_1.default.release().indexOf('Microsoft') !== -1) {
21+
platform = 'win32';
22+
}
23+
// http://stackoverflow.com/q/1480971/3191, but see below for Windows.
24+
let command;
25+
switch (platform) {
26+
case 'win32': {
27+
command = 'cmd.exe';
28+
break;
29+
}
30+
case 'darwin': {
31+
command = 'open';
32+
if (tool) {
33+
args.unshift('-a');
34+
}
35+
break;
36+
}
37+
default: {
38+
command = 'xdg-open';
39+
break;
40+
}
41+
}
42+
if (platform === 'win32') {
43+
// On Windows, we really want to use the "start" command.
44+
// But, the rules regarding arguments with spaces, and escaping them with quotes,
45+
// can get really arcane. So the easiest way to deal with this is to pass off the
46+
// responsibility to "cmd /c", which has that logic built in.
47+
//
48+
// Furthermore, if "cmd /c" double-quoted the first parameter,
49+
// then "start" will interpret it as a window title,
50+
// so we need to add a dummy empty-string window title: http://stackoverflow.com/a/154090/3191
51+
//
52+
// Additionally, on Windows ampersand needs to be escaped when passed to "start"
53+
args = args.map(value => {
54+
return value.replace(/&/g, '^&');
55+
});
56+
args = ['/c', 'start', '""'].concat(args);
57+
}
58+
return child_process_1.default.execFile(command, args, null, callback);
59+
};

app/out/404.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
<!DOCTYPE html><html><head><meta charSet="utf-8" class="next-head"/><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/404.js" as="script"/><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_app.js" as="script"/><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_error.js" as="script"/><link rel="preload" href="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" as="script"/><link rel="preload" href="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" as="script"/><link rel="preload" href="/_next/static/runtime/main-94dad8183cac85e3af97.js" as="script"/></head><body><div id="__next"><div data-reactroot="">404</div></div><script>__NEXT_DATA__ = {"props":{"pageProps":{}},"page":"/404","query":{},"buildId":"Bl3Xk6SKdeG3dxMfwaFwz","nextExport":true};__NEXT_LOADED_PAGES__=[];__NEXT_REGISTER_PAGE=function(r,f){__NEXT_LOADED_PAGES__.push([r, f])}</script><script async="" id="__NEXT_PAGE__/404" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/404.js"></script><script async="" id="__NEXT_PAGE__/_app" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_app.js"></script><script async="" id="__NEXT_PAGE__/_error" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_error.js"></script><script src="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" async=""></script><script src="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" async=""></script><script src="/_next/static/runtime/main-94dad8183cac85e3af97.js" async=""></script></body></html>
1+
<!DOCTYPE html><html><head><meta charSet="utf-8" class="next-head"/><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/404.js" as="script"/><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_app.js" as="script"/><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_error.js" as="script"/><link rel="preload" href="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" as="script"/><link rel="preload" href="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" as="script"/><link rel="preload" href="/_next/static/runtime/main-94dad8183cac85e3af97.js" as="script"/></head><body><div id="__next"><div data-reactroot="">404</div></div><script>__NEXT_DATA__ = {"props":{"pageProps":{}},"page":"/404","query":{},"buildId":"262vxEr5Glefa1A~zX7ZC","nextExport":true};__NEXT_LOADED_PAGES__=[];__NEXT_REGISTER_PAGE=function(r,f){__NEXT_LOADED_PAGES__.push([r, f])}</script><script async="" id="__NEXT_PAGE__/404" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/404.js"></script><script async="" id="__NEXT_PAGE__/_app" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_app.js"></script><script async="" id="__NEXT_PAGE__/_error" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_error.js"></script><script src="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" async=""></script><script src="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" async=""></script><script src="/_next/static/runtime/main-94dad8183cac85e3af97.js" async=""></script></body></html>

app/out/index.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
<!DOCTYPE html><html><head><meta charSet="utf-8" class="next-head"/><title class="next-head">「」</title><link rel="shortcut icon" type="image/ico" href="/_static/favicon.ico" class="next-head"/><link rel="stylesheet" href="/_static/page.css" class="next-head"/><link rel="stylesheet" href="/_static/markdown.css" class="next-head"/><link rel="stylesheet" href="/_static/highlight.css" class="next-head"/><link rel="stylesheet" href="/_static/[email protected]" class="next-head"/><script type="text/javascript" src="/_static/tweenlite.min.js" class="next-head"></script><script type="text/javascript" src="/_static/mermaid.min.js" class="next-head"></script><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/index.js" as="script"/><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_app.js" as="script"/><link rel="preload" href="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_error.js" as="script"/><link rel="preload" href="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" as="script"/><link rel="preload" href="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" as="script"/><link rel="preload" href="/_next/static/runtime/main-94dad8183cac85e3af97.js" as="script"/></head><body><div id="__next"><div id="page-ctn"><header id="page-header"><h3><svg viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path fill-rule="evenodd" d="M3 5h4v1H3V5zm0 3h4V7H3v1zm0 2h4V9H3v1zm11-5h-4v1h4V5zm0 2h-4v1h4V7zm0 2h-4v1h4V9zm2-6v9c0 .55-.45 1-1 1H9.5l-1 1-1-1H2c-.55 0-1-.45-1-1V3c0-.55.45-1 1-1h5.5l1 1 1-1H15c.55 0 1 .45 1 1zm-8 .5L7.5 3H2v9h6V3.5zm7-.5H9.5l-.5.5V12h6V3z"></path></svg></h3></header><section class="markdown-body"></section></div></div><script>__NEXT_DATA__ = {"props":{"pageProps":{}},"page":"/","query":{},"buildId":"Bl3Xk6SKdeG3dxMfwaFwz","nextExport":true};__NEXT_LOADED_PAGES__=[];__NEXT_REGISTER_PAGE=function(r,f){__NEXT_LOADED_PAGES__.push([r, f])}</script><script async="" id="__NEXT_PAGE__/" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/index.js"></script><script async="" id="__NEXT_PAGE__/_app" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_app.js"></script><script async="" id="__NEXT_PAGE__/_error" src="/_next/static/Bl3Xk6SKdeG3dxMfwaFwz/pages/_error.js"></script><script src="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" async=""></script><script src="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" async=""></script><script src="/_next/static/runtime/main-94dad8183cac85e3af97.js" async=""></script></body></html>
1+
<!DOCTYPE html><html><head><meta charSet="utf-8" class="next-head"/><title class="next-head">「」</title><link rel="shortcut icon" type="image/ico" href="/_static/favicon.ico" class="next-head"/><link rel="stylesheet" href="/_static/page.css" class="next-head"/><link rel="stylesheet" href="/_static/markdown.css" class="next-head"/><link rel="stylesheet" href="/_static/highlight.css" class="next-head"/><link rel="stylesheet" href="/_static/[email protected]" class="next-head"/><script type="text/javascript" src="/_static/tweenlite.min.js" class="next-head"></script><script type="text/javascript" src="/_static/mermaid.min.js" class="next-head"></script><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/index.js" as="script"/><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_app.js" as="script"/><link rel="preload" href="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_error.js" as="script"/><link rel="preload" href="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" as="script"/><link rel="preload" href="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" as="script"/><link rel="preload" href="/_next/static/runtime/main-94dad8183cac85e3af97.js" as="script"/></head><body><div id="__next"><div id="page-ctn"><header id="page-header"><h3><svg viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path fill-rule="evenodd" d="M3 5h4v1H3V5zm0 3h4V7H3v1zm0 2h4V9H3v1zm11-5h-4v1h4V5zm0 2h-4v1h4V7zm0 2h-4v1h4V9zm2-6v9c0 .55-.45 1-1 1H9.5l-1 1-1-1H2c-.55 0-1-.45-1-1V3c0-.55.45-1 1-1h5.5l1 1 1-1H15c.55 0 1 .45 1 1zm-8 .5L7.5 3H2v9h6V3.5zm7-.5H9.5l-.5.5V12h6V3z"></path></svg></h3></header><section class="markdown-body"></section></div></div><script>__NEXT_DATA__ = {"props":{"pageProps":{}},"page":"/","query":{},"buildId":"262vxEr5Glefa1A~zX7ZC","nextExport":true};__NEXT_LOADED_PAGES__=[];__NEXT_REGISTER_PAGE=function(r,f){__NEXT_LOADED_PAGES__.push([r, f])}</script><script async="" id="__NEXT_PAGE__/" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/index.js"></script><script async="" id="__NEXT_PAGE__/_app" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_app.js"></script><script async="" id="__NEXT_PAGE__/_error" src="/_next/static/262vxEr5Glefa1A~zX7ZC/pages/_error.js"></script><script src="/_next/static/runtime/webpack-42652fa8b82c329c0559.js" async=""></script><script src="/_next/static/chunks/commons.1c1b5ebd55cbcad2529a.js" async=""></script><script src="/_next/static/runtime/main-94dad8183cac85e3af97.js" async=""></script></body></html>

app/package.json

-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@
1010
"dependencies": {
1111
"log4js": "^3.0.6",
1212
"neovim": "^4.2.1",
13-
"opener": "^1.5.1",
1413
"socket.io": "^2.1.1",
1514
"tslib": "^1.9.3"
1615
}

app/server.js

+10-6
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,10 @@ if (!/^(\/|C:\\)snapshot/.test(__dirname)) {
88
const { plugin } = require('./nvim')
99
const http = require('http')
1010
const websocket = require('socket.io')
11-
const opener = require('opener')
11+
12+
const opener = require('./lib/util/opener')
1213
const logger = require('./lib/util/logger')('app/server')
1314
const { getIP } = require('./lib/util/getIP')
14-
1515
const routes = require('./routes')
1616

1717
let clients = {}
@@ -108,11 +108,16 @@ async function startServer () {
108108
})
109109
clients = {}
110110
}
111-
function openBrowser ({ bufnr }) {
111+
async function openBrowser ({ bufnr }) {
112112
const openHost = openToTheWord ? getIP() : '127.0.0.1'
113113
const url = `http://${openHost}:${port}/page/${bufnr}`
114-
logger.info('open page: ', url)
115-
opener(url)
114+
const browser = await plugin.nvim.getVar('mkdp_browser')
115+
logger.info(`open page [${browser || 'default'}]: `, url)
116+
if (browser !== '') {
117+
opener(url, browser)
118+
} else {
119+
opener(url)
120+
}
116121
}
117122
plugin.init({
118123
refreshPage,
@@ -126,4 +131,3 @@ async function startServer () {
126131
}
127132

128133
startServer()
129-

package.json

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "markdown-preview.vim",
3-
"version": "0.0.3",
3+
"version": "0.0.4",
44
"description": "markdown preview plugin for (neo)vim",
55
"bin": "./app/server.js",
66
"repository": "https://github.com/iamcco/markdown-preview.vim.git",
@@ -25,7 +25,6 @@
2525
"neovim": "^4.2.1",
2626
"next": "^7.0.2",
2727
"next-routes": "^1.4.2",
28-
"opener": "^1.5.1",
2928
"react": "^16.5.2",
3029
"react-dom": "^16.5.2",
3130
"socket.io": "^2.1.1",

plugin/mkdp.vim

+5
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,11 @@ if !exists('g:mkdp_open_to_the_world')
3333
let g:mkdp_open_to_the_world = 0
3434
endif
3535

36+
" switch browser to open preview page
37+
if !exists('g:mkdp_browser')
38+
let g:mkdp_browser = ''
39+
endif
40+
3641
if !exists('g:mkdp_preview_options')
3742
let g:mkdp_preview_options = {
3843
\ 'mkit': {},

src/util/logger.ts

-1
Original file line numberDiff line numberDiff line change
@@ -41,4 +41,3 @@ if (!isRoot) {
4141
module.exports = (name = 'mkdp'): log4js.Logger => {
4242
return log4js.getLogger(name)
4343
}
44-

src/util/opener.ts

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
/*
2+
* fork from https://github.com/domenic/opener
3+
*/
4+
import childProcess from 'child_process'
5+
import os from 'os'
6+
7+
module.exports = function opener(
8+
args: string | string[],
9+
tool: string | undefined,
10+
callback: (() => any) | undefined
11+
) {
12+
let platform = process.platform
13+
args = [].concat(args)
14+
15+
if (tool) {
16+
args.unshift(tool)
17+
}
18+
19+
// Attempt to detect Windows Subystem for Linux (WSL).
20+
// WSL itself as Linux (which works in most cases), but in
21+
// this specific case we need to treat it as actually being Windows.
22+
// The "Windows-way" of opening things through cmd.exe works just fine here,
23+
// whereas using xdg-open does not, since there is no X Windows in WSL.
24+
if (platform === 'linux' && os.release().indexOf('Microsoft') !== -1) {
25+
platform = 'win32'
26+
}
27+
28+
// http://stackoverflow.com/q/1480971/3191, but see below for Windows.
29+
let command
30+
switch (platform) {
31+
case 'win32': {
32+
command = 'cmd.exe'
33+
break
34+
}
35+
case 'darwin': {
36+
command = 'open'
37+
if (tool) {
38+
args.unshift('-a')
39+
}
40+
break
41+
}
42+
default: {
43+
command = 'xdg-open'
44+
break
45+
}
46+
}
47+
48+
if (platform === 'win32') {
49+
// On Windows, we really want to use the "start" command.
50+
// But, the rules regarding arguments with spaces, and escaping them with quotes,
51+
// can get really arcane. So the easiest way to deal with this is to pass off the
52+
// responsibility to "cmd /c", which has that logic built in.
53+
//
54+
// Furthermore, if "cmd /c" double-quoted the first parameter,
55+
// then "start" will interpret it as a window title,
56+
// so we need to add a dummy empty-string window title: http://stackoverflow.com/a/154090/3191
57+
//
58+
// Additionally, on Windows ampersand needs to be escaped when passed to "start"
59+
args = args.map(value => {
60+
return value.replace(/&/g, '^&')
61+
})
62+
args = ['/c', 'start', '""'].concat(args)
63+
}
64+
65+
return childProcess.execFile(command, args, null, callback)
66+
}

0 commit comments

Comments
 (0)