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')
|
2019-03-18 19:37:06 +00:00
|
|
|
const binding = process.electronBinding('dialog')
|
|
|
|
const v8Util = process.electronBinding('v8_util')
|
2016-01-12 02:40:23 +00:00
|
|
|
|
2019-08-13 20:40:07 +00:00
|
|
|
const DialogType = {
|
|
|
|
OPEN: 'OPEN',
|
|
|
|
SAVE: 'SAVE'
|
|
|
|
}
|
|
|
|
|
|
|
|
const saveFileDialogProperties = {
|
|
|
|
createDirectory: 1 << 0,
|
|
|
|
showHiddenFiles: 1 << 1,
|
|
|
|
treatPackageAsDirectory: 1 << 2,
|
|
|
|
showOverwriteConfirmation: 1 << 3,
|
|
|
|
dontAddToRecent: 1 << 4
|
|
|
|
}
|
|
|
|
|
|
|
|
const openFileDialogProperties = {
|
2016-01-12 02:40:23 +00:00
|
|
|
openFile: 1 << 0,
|
|
|
|
openDirectory: 1 << 1,
|
|
|
|
multiSelections: 1 << 2,
|
2019-08-13 15:48:22 +00:00
|
|
|
createDirectory: 1 << 3, // macOS
|
2017-02-02 16:30:02 +00:00
|
|
|
showHiddenFiles: 1 << 4,
|
2019-08-13 15:48:22 +00:00
|
|
|
promptToCreate: 1 << 5, // Windows
|
|
|
|
noResolveAliases: 1 << 6, // macOS
|
|
|
|
treatPackageAsDirectory: 1 << 7, // macOS
|
|
|
|
dontAddToRecent: 1 << 8 // Windows
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|
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-08-13 20:40:07 +00:00
|
|
|
const setupDialogProperties = (type, properties) => {
|
|
|
|
const dialogPropertiesTypes = (type === DialogType.OPEN) ? openFileDialogProperties : saveFileDialogProperties
|
|
|
|
let dialogProperties = 0
|
|
|
|
for (const prop in dialogPropertiesTypes) {
|
|
|
|
if (properties.includes(prop)) {
|
|
|
|
dialogProperties |= dialogPropertiesTypes[prop]
|
|
|
|
}
|
|
|
|
}
|
2019-08-16 15:55:01 +00:00
|
|
|
|
|
|
|
return dialogProperties
|
2019-08-13 20:40:07 +00:00
|
|
|
}
|
|
|
|
|
2019-03-05 21:48:20 +00:00
|
|
|
const saveDialog = (sync, window, options) => {
|
|
|
|
checkAppInitialized()
|
|
|
|
|
2019-05-21 14:08:22 +00:00
|
|
|
if (window && window.constructor !== BrowserWindow) {
|
|
|
|
options = window
|
|
|
|
window = null
|
|
|
|
}
|
|
|
|
|
2019-03-05 21:48:20 +00:00
|
|
|
if (options == null) options = { title: 'Save' }
|
|
|
|
|
|
|
|
const {
|
|
|
|
buttonLabel = '',
|
|
|
|
defaultPath = '',
|
|
|
|
filters = [],
|
2019-08-13 20:40:07 +00:00
|
|
|
properties = [],
|
2019-03-05 21:48:20 +00:00
|
|
|
title = '',
|
|
|
|
message = '',
|
|
|
|
securityScopedBookmarks = false,
|
|
|
|
nameFieldLabel = '',
|
|
|
|
showsTagField = true
|
|
|
|
} = options
|
|
|
|
|
|
|
|
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')
|
|
|
|
if (typeof nameFieldLabel !== 'string') throw new TypeError('Name field label must be a string')
|
|
|
|
|
|
|
|
const settings = { buttonLabel, defaultPath, filters, title, message, securityScopedBookmarks, nameFieldLabel, showsTagField, window }
|
2019-08-13 20:40:07 +00:00
|
|
|
settings.properties = setupDialogProperties(DialogType.SAVE, properties)
|
|
|
|
|
2019-03-05 21:48:20 +00:00
|
|
|
return (sync) ? binding.showSaveDialogSync(settings) : binding.showSaveDialog(settings)
|
|
|
|
}
|
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
const openDialog = (sync, window, options) => {
|
|
|
|
checkAppInitialized()
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-05-21 14:08:22 +00:00
|
|
|
if (window && window.constructor !== BrowserWindow) {
|
|
|
|
options = window
|
|
|
|
window = null
|
|
|
|
}
|
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
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')
|
|
|
|
|
|
|
|
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 }
|
2019-08-13 20:40:07 +00:00
|
|
|
settings.properties = setupDialogProperties(DialogType.OPEN, properties)
|
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-12 18:06:59 +00:00
|
|
|
const messageBox = (sync, window, options) => {
|
|
|
|
checkAppInitialized()
|
|
|
|
|
2019-05-21 14:08:22 +00:00
|
|
|
if (window && window.constructor !== BrowserWindow) {
|
|
|
|
options = window
|
|
|
|
window = null
|
|
|
|
}
|
|
|
|
|
2019-03-12 18:06:59 +00:00
|
|
|
if (options == null) options = { type: 'none' }
|
|
|
|
|
2019-03-18 14:58:03 +00:00
|
|
|
const messageBoxTypes = ['none', 'info', 'warning', 'error', 'question']
|
|
|
|
const messageBoxOptions = { noLink: 1 << 0 }
|
|
|
|
|
2019-03-12 18:06:59 +00:00
|
|
|
let {
|
|
|
|
buttons = [],
|
|
|
|
cancelId,
|
|
|
|
checkboxLabel = '',
|
|
|
|
checkboxChecked,
|
|
|
|
defaultId = -1,
|
|
|
|
detail = '',
|
|
|
|
icon = null,
|
2019-12-05 21:38:57 +00:00
|
|
|
noLink = false,
|
2019-03-12 18:06:59 +00:00
|
|
|
message = '',
|
|
|
|
title = '',
|
|
|
|
type = 'none'
|
|
|
|
} = options
|
|
|
|
|
|
|
|
const messageBoxType = messageBoxTypes.indexOf(type)
|
|
|
|
if (messageBoxType === -1) throw new TypeError('Invalid message box type')
|
|
|
|
if (!Array.isArray(buttons)) throw new TypeError('Buttons must be an array')
|
|
|
|
if (options.normalizeAccessKeys) buttons = buttons.map(normalizeAccessKey)
|
|
|
|
if (typeof title !== 'string') throw new TypeError('Title must be a string')
|
2019-12-05 21:38:57 +00:00
|
|
|
if (typeof noLink !== 'boolean') throw new TypeError('noLink must be a boolean')
|
2019-03-12 18:06:59 +00:00
|
|
|
if (typeof message !== 'string') throw new TypeError('Message must be a string')
|
|
|
|
if (typeof detail !== 'string') throw new TypeError('Detail must be a string')
|
|
|
|
if (typeof checkboxLabel !== 'string') throw new TypeError('checkboxLabel must be a string')
|
|
|
|
|
|
|
|
checkboxChecked = !!checkboxChecked
|
2019-11-27 07:39:07 +00:00
|
|
|
if (checkboxChecked && !checkboxLabel) {
|
|
|
|
throw new Error('checkboxChecked requires that checkboxLabel also be passed')
|
|
|
|
}
|
2019-03-12 18:06:59 +00:00
|
|
|
|
|
|
|
// Choose a default button to get selected when dialog is cancelled.
|
|
|
|
if (cancelId == null) {
|
|
|
|
// If the defaultId is set to 0, ensure the cancel button is a different index (1)
|
|
|
|
cancelId = (defaultId === 0 && buttons.length > 1) ? 1 : 0
|
|
|
|
for (let i = 0; i < buttons.length; i++) {
|
|
|
|
const text = buttons[i].toLowerCase()
|
|
|
|
if (text === 'cancel' || text === 'no') {
|
|
|
|
cancelId = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-14 15:26:25 +00:00
|
|
|
const settings = {
|
|
|
|
window,
|
|
|
|
messageBoxType,
|
|
|
|
buttons,
|
|
|
|
defaultId,
|
|
|
|
cancelId,
|
2019-12-05 21:38:57 +00:00
|
|
|
noLink,
|
2019-06-14 15:26:25 +00:00
|
|
|
title,
|
|
|
|
message,
|
|
|
|
detail,
|
|
|
|
checkboxLabel,
|
|
|
|
checkboxChecked,
|
|
|
|
icon
|
|
|
|
}
|
|
|
|
|
2019-03-12 18:06:59 +00:00
|
|
|
if (sync) {
|
2019-06-14 15:26:25 +00:00
|
|
|
return binding.showMessageBoxSync(settings)
|
2019-03-12 18:06:59 +00:00
|
|
|
} else {
|
2019-06-14 15:26:25 +00:00
|
|
|
return binding.showMessageBox(settings)
|
2019-03-12 18:06:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
module.exports = {
|
|
|
|
showOpenDialog: function (window, options) {
|
|
|
|
return openDialog(false, window, options)
|
|
|
|
},
|
2019-03-05 21:48:20 +00:00
|
|
|
|
2019-03-05 13:54:48 +00:00
|
|
|
showOpenDialogSync: function (window, options) {
|
|
|
|
return openDialog(true, window, options)
|
2016-01-12 02:40:23 +00:00
|
|
|
},
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-05 21:48:20 +00:00
|
|
|
showSaveDialog: function (window, options) {
|
|
|
|
return saveDialog(false, window, options)
|
|
|
|
},
|
2017-02-01 15:34:21 +00:00
|
|
|
|
2019-03-05 21:48:20 +00:00
|
|
|
showSaveDialogSync: function (window, options) {
|
|
|
|
return saveDialog(true, window, options)
|
2016-01-12 02:40:23 +00:00
|
|
|
},
|
2016-03-18 18:51:02 +00:00
|
|
|
|
2019-03-12 18:06:59 +00:00
|
|
|
showMessageBox: function (window, options) {
|
|
|
|
return messageBox(false, window, options)
|
|
|
|
},
|
2016-12-01 23:00:12 +00:00
|
|
|
|
2019-03-12 18:06:59 +00:00
|
|
|
showMessageBoxSync: function (window, options) {
|
|
|
|
return messageBox(true, window, options)
|
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
|
|
|
},
|
|
|
|
|
2019-03-15 00:02:50 +00:00
|
|
|
showCertificateTrustDialog: function (window, options) {
|
|
|
|
if (window && window.constructor !== BrowserWindow) options = window
|
2017-04-04 01:33:21 +00:00
|
|
|
if (options == null || typeof options !== 'object') {
|
|
|
|
throw new TypeError('options must be an object')
|
|
|
|
}
|
|
|
|
|
2019-03-15 00:02:50 +00:00
|
|
|
const { 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')
|
|
|
|
}
|
|
|
|
|
2019-03-15 00:02:50 +00:00
|
|
|
if (typeof message !== 'string') throw new TypeError('message must be a string')
|
2017-04-04 01:33:21 +00:00
|
|
|
|
2019-03-15 00:02:50 +00:00
|
|
|
return binding.showCertificateTrustDialog(window, certificate, message)
|
2016-01-12 02:40:23 +00:00
|
|
|
}
|
2016-03-24 20:15:04 +00:00
|
|
|
}
|