2016-03-24 20:15:04 +00:00
|
|
|
'use strict'
|
2016-03-18 18:51:02 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
const { app, BrowserWindow, deprecate } = require('electron')
|
2016-03-24 20:15:04 +00:00
|
|
|
const binding = process.atomBinding('dialog')
|
|
|
|
const v8Util = process.atomBinding('v8_util')
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2016-07-11 04:30:18 +00:00
|
|
|
const fileDialogProperties = {
|
2016-01-12 02:40:23 +00:00
|
|
|
openFile: 1 << 0,
|
|
|
|
openDirectory: 1 << 1,
|
|
|
|
multiSelections: 1 << 2,
|
2016-07-11 04:30:18 +00:00
|
|
|
createDirectory: 1 << 3,
|
2017-02-02 16:30:02 +00:00
|
|
|
showHiddenFiles: 1 << 4,
|
2017-02-12 04:23:27 +00:00
|
|
|
promptToCreate: 1 << 5,
|
2017-07-18 01:40:57 +00:00
|
|
|
noResolveAliases: 1 << 6,
|
|
|
|
treatPackageAsDirectory: 1 << 7
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
const messageBoxTypes = ['none', 'info', 'warning', 'error', 'question']
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
const messageBoxOptions = {
|
2016-01-12 02:40:23 +00:00
|
|
|
noLink: 1 << 0
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
const parseArgs = function (window, options, callback, ...args) {
|
2016-12-13 00:25:56 +00:00
|
|
|
if (window != null && window.constructor !== BrowserWindow) {
|
2016-01-14 18:35:29 +00:00
|
|
|
// Shift.
|
2016-05-07 14:55:26 +00:00
|
|
|
[callback, options, window] = [options, window, null]
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-03-31 16:07:56 +00:00
|
|
|
|
2016-01-12 02:40:23 +00:00
|
|
|
if ((callback == null) && typeof options === 'function') {
|
2016-01-14 18:35:29 +00:00
|
|
|
// Shift.
|
2016-05-07 14:55:26 +00:00
|
|
|
[callback, options] = [options, null]
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-03-31 16:07:56 +00:00
|
|
|
|
|
|
|
// Fallback to using very last argument as the callback function
|
2016-12-01 23:00:12 +00:00
|
|
|
const lastArgument = args[args.length - 1]
|
2016-05-07 14:52:52 +00:00
|
|
|
if ((callback == null) && typeof lastArgument === 'function') {
|
|
|
|
callback = lastArgument
|
2016-03-31 16:07:56 +00:00
|
|
|
}
|
|
|
|
|
2016-03-24 20:15:04 +00:00
|
|
|
return [window, options, callback]
|
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2016-12-29 20:05:58 +00:00
|
|
|
const normalizeAccessKey = (text) => {
|
|
|
|
if (typeof text !== 'string') return text
|
|
|
|
|
|
|
|
// macOS does not have access keys so remove single ampersands
|
|
|
|
// and replace double ampersands with a single ampersand
|
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
return text.replace(/&(&?)/g, '$1')
|
|
|
|
}
|
|
|
|
|
|
|
|
// Linux uses a single underscore as an access key prefix so escape
|
|
|
|
// existing single underscores with a second underscore, replace double
|
|
|
|
// ampersands with a single ampersand, and replace a single ampersand with
|
|
|
|
// a single underscore
|
|
|
|
if (process.platform === 'linux') {
|
|
|
|
return text.replace(/_/g, '__').replace(/&(.?)/g, (match, after) => {
|
|
|
|
if (after === '&') return after
|
|
|
|
return `_${after}`
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return text
|
|
|
|
}
|
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
const checkAppInitialized = function () {
|
2016-01-12 02:40:23 +00:00
|
|
|
if (!app.isReady()) {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new Error('dialog module can only be used after app is ready')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
const openDialog = (sync, window, options) => {
|
|
|
|
checkAppInitialized()
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
if (window.constructor !== BrowserWindow) options = window
|
|
|
|
if (options == null) {
|
|
|
|
options = {
|
|
|
|
title: 'Open',
|
|
|
|
properties: ['openFile']
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2019-03-05 13:54:48 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
const {
|
|
|
|
buttonLabel = '',
|
|
|
|
defaultPath = '',
|
|
|
|
filters = [],
|
|
|
|
properties = ['openFile'],
|
|
|
|
title = '',
|
|
|
|
message = '',
|
|
|
|
securityScopedBookmarks = false
|
|
|
|
} = options
|
|
|
|
|
|
|
|
if (!Array.isArray(properties)) throw new TypeError('Properties must be an array')
|
|
|
|
|
|
|
|
let dialogProperties = 0
|
|
|
|
for (const prop in fileDialogProperties) {
|
|
|
|
if (properties.includes(prop)) {
|
|
|
|
dialogProperties |= fileDialogProperties[prop]
|
2016-05-06 18:10:31 +00:00
|
|
|
}
|
2019-03-05 13:54:48 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
if (typeof title !== 'string') throw new TypeError('Title must be a string')
|
|
|
|
if (typeof buttonLabel !== 'string') throw new TypeError('Button label must be a string')
|
|
|
|
if (typeof defaultPath !== 'string') throw new TypeError('Default path must be a string')
|
|
|
|
if (typeof message !== 'string') throw new TypeError('Message must be a string')
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
const settings = { title, buttonLabel, defaultPath, filters, message, securityScopedBookmarks, window }
|
|
|
|
settings.properties = dialogProperties
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
return (sync) ? binding.showOpenDialogSync(settings) : binding.showOpenDialog(settings)
|
|
|
|
}
|
2017-02-01 16:01:01 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
module.exports = {
|
|
|
|
showOpenDialog: function (window, options) {
|
|
|
|
return openDialog(false, window, options)
|
|
|
|
},
|
|
|
|
showOpenDialogSync: function (window, options) {
|
|
|
|
return openDialog(true, window, options)
|
2016-01-12 02:40:23 +00:00
|
|
|
},
|
2016-03-24 20:15:04 +00:00
|
|
|
showSaveDialog: function (...args) {
|
|
|
|
checkAppInitialized()
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-12-01 22:37:03 +00:00
|
|
|
let [window, options, callback] = parseArgs(...args)
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-01-12 02:40:23 +00:00
|
|
|
if (options == null) {
|
|
|
|
options = {
|
|
|
|
title: 'Save'
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2018-09-13 16:10:51 +00:00
|
|
|
let { buttonLabel, defaultPath, filters, title, message, securityScopedBookmarks = false, nameFieldLabel, showsTagField } = options
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (title == null) {
|
|
|
|
title = ''
|
|
|
|
} else if (typeof title !== 'string') {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Title must be a string')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (buttonLabel == null) {
|
|
|
|
buttonLabel = ''
|
|
|
|
} else if (typeof buttonLabel !== 'string') {
|
2016-12-01 22:16:33 +00:00
|
|
|
throw new TypeError('Button label must be a string')
|
2016-05-06 18:10:31 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (defaultPath == null) {
|
|
|
|
defaultPath = ''
|
|
|
|
} else if (typeof defaultPath !== 'string') {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Default path must be a string')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (filters == null) {
|
|
|
|
filters = []
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2017-02-01 15:34:21 +00:00
|
|
|
if (message == null) {
|
|
|
|
message = ''
|
|
|
|
} else if (typeof message !== 'string') {
|
|
|
|
throw new TypeError('Message must be a string')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nameFieldLabel == null) {
|
|
|
|
nameFieldLabel = ''
|
|
|
|
} else if (typeof nameFieldLabel !== 'string') {
|
|
|
|
throw new TypeError('Name field label must be a string')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (showsTagField == null) {
|
2017-02-09 13:01:40 +00:00
|
|
|
showsTagField = true
|
2017-02-01 15:34:21 +00:00
|
|
|
}
|
|
|
|
|
2018-02-12 18:25:06 +00:00
|
|
|
const wrappedCallback = typeof callback === 'function' ? function (success, result, bookmarkData) {
|
|
|
|
return success ? callback(result, bookmarkData) : callback()
|
2016-03-24 20:15:04 +00:00
|
|
|
} : null
|
2018-09-13 16:10:51 +00:00
|
|
|
const settings = { title, buttonLabel, defaultPath, filters, message, securityScopedBookmarks, nameFieldLabel, showsTagField, window }
|
2017-02-08 01:32:58 +00:00
|
|
|
return binding.showSaveDialog(settings, wrappedCallback)
|
2016-01-12 02:40:23 +00:00
|
|
|
},
|
2016-03-18 18:51:02 +00:00
|
|
|
|
2016-03-24 20:15:04 +00:00
|
|
|
showMessageBox: function (...args) {
|
|
|
|
checkAppInitialized()
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-12-01 22:37:03 +00:00
|
|
|
let [window, options, callback] = parseArgs(...args)
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-01-12 02:40:23 +00:00
|
|
|
if (options == null) {
|
|
|
|
options = {
|
|
|
|
type: 'none'
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2017-02-06 15:35:36 +00:00
|
|
|
let {
|
|
|
|
buttons, cancelId, checkboxLabel, checkboxChecked, defaultId, detail,
|
|
|
|
icon, message, title, type
|
|
|
|
} = options
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (type == null) {
|
|
|
|
type = 'none'
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
const messageBoxType = messageBoxTypes.indexOf(type)
|
|
|
|
if (messageBoxType === -1) {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Invalid message box type')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-12-12 18:05:01 +00:00
|
|
|
if (buttons == null) {
|
|
|
|
buttons = []
|
|
|
|
} else if (!Array.isArray(buttons)) {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Buttons must be an array')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2016-12-29 21:21:43 +00:00
|
|
|
if (options.normalizeAccessKeys) {
|
2016-12-29 20:05:58 +00:00
|
|
|
buttons = buttons.map(normalizeAccessKey)
|
|
|
|
}
|
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
if (title == null) {
|
|
|
|
title = ''
|
|
|
|
} else if (typeof title !== 'string') {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Title must be a string')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (message == null) {
|
|
|
|
message = ''
|
|
|
|
} else if (typeof message !== 'string') {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Message must be a string')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (detail == null) {
|
|
|
|
detail = ''
|
|
|
|
} else if (typeof detail !== 'string') {
|
2016-03-24 20:15:04 +00:00
|
|
|
throw new TypeError('Detail must be a string')
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2017-02-06 15:35:36 +00:00
|
|
|
checkboxChecked = !!checkboxChecked
|
|
|
|
|
|
|
|
if (checkboxLabel == null) {
|
|
|
|
checkboxLabel = ''
|
|
|
|
} else if (typeof checkboxLabel !== 'string') {
|
|
|
|
throw new TypeError('checkboxLabel must be a string')
|
|
|
|
}
|
|
|
|
|
2016-12-01 23:00:12 +00:00
|
|
|
if (icon == null) {
|
|
|
|
icon = null
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
if (defaultId == null) {
|
|
|
|
defaultId = -1
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
|
|
|
|
2016-01-14 18:35:29 +00:00
|
|
|
// Choose a default button to get selected when dialog is cancelled.
|
2016-12-01 23:00:12 +00:00
|
|
|
if (cancelId == null) {
|
2019-02-27 08:24:13 +00:00
|
|
|
// If the defaultId is set to 0, ensure the cancel button is a different index (1)
|
|
|
|
cancelId = (defaultId === 0 && buttons.length > 1) ? 1 : 0
|
2016-12-01 23:00:12 +00:00
|
|
|
for (let i = 0; i < buttons.length; i++) {
|
|
|
|
const text = buttons[i].toLowerCase()
|
|
|
|
if (text === 'cancel' || text === 'no') {
|
|
|
|
cancelId = i
|
2016-03-24 20:15:04 +00:00
|
|
|
break
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-01 23:00:12 +00:00
|
|
|
|
|
|
|
const flags = options.noLink ? messageBoxOptions.noLink : 0
|
|
|
|
return binding.showMessageBox(messageBoxType, buttons, defaultId, cancelId,
|
2018-09-13 16:10:51 +00:00
|
|
|
flags, title, message, detail, checkboxLabel,
|
|
|
|
checkboxChecked, icon, window, callback)
|
2016-01-12 02:40:23 +00:00
|
|
|
},
|
2016-03-18 18:51:02 +00:00
|
|
|
|
2016-03-24 20:15:04 +00:00
|
|
|
showErrorBox: function (...args) {
|
2016-12-01 22:37:03 +00:00
|
|
|
return binding.showErrorBox(...args)
|
2017-04-03 19:25:06 +00:00
|
|
|
},
|
|
|
|
|
2017-04-04 17:49:10 +00:00
|
|
|
showCertificateTrustDialog: function (...args) {
|
2018-10-02 01:56:31 +00:00
|
|
|
const [window, options, callback] = parseArgs(...args)
|
2017-04-04 17:49:10 +00:00
|
|
|
|
2017-04-04 01:33:21 +00:00
|
|
|
if (options == null || typeof options !== 'object') {
|
|
|
|
throw new TypeError('options must be an object')
|
|
|
|
}
|
|
|
|
|
2018-09-13 16:10:51 +00:00
|
|
|
let { certificate, message } = options
|
2017-04-04 15:45:27 +00:00
|
|
|
if (certificate == null || typeof certificate !== 'object') {
|
2017-04-04 01:33:21 +00:00
|
|
|
throw new TypeError('certificate must be an object')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message == null) {
|
|
|
|
message = ''
|
|
|
|
} else if (typeof message !== 'string') {
|
|
|
|
throw new TypeError('message must be a string')
|
|
|
|
}
|
|
|
|
|
|
|
|
return binding.showCertificateTrustDialog(window, certificate, message, callback)
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
module.exports.showOpenDialog = deprecate.promisify(module.exports.showOpenDialog)
|
|
|
|
|
2016-01-14 18:35:29 +00:00
|
|
|
// Mark standard asynchronous functions.
|
2016-12-01 23:00:12 +00:00
|
|
|
v8Util.setHiddenValue(module.exports.showMessageBox, 'asynchronous', true)
|
|
|
|
v8Util.setHiddenValue(module.exports.showOpenDialog, 'asynchronous', true)
|
|
|
|
v8Util.setHiddenValue(module.exports.showSaveDialog, 'asynchronous', true)
|