2024-05-31 10:04:22 +00:00
|
|
|
import { app, BaseWindow } from 'electron/main';
|
2020-07-13 16:58:49 +00:00
|
|
|
import type { OpenDialogOptions, OpenDialogReturnValue, MessageBoxOptions, SaveDialogOptions, SaveDialogReturnValue, MessageBoxReturnValue, CertificateTrustDialogOptions } from 'electron/main';
|
2024-10-03 02:10:44 +00:00
|
|
|
|
2020-06-30 22:51:44 +00:00
|
|
|
const dialogBinding = process._linkedBinding('electron_browser_dialog');
|
|
|
|
|
|
|
|
enum SaveFileDialogProperties {
|
|
|
|
createDirectory = 1 << 0,
|
|
|
|
showHiddenFiles = 1 << 1,
|
|
|
|
treatPackageAsDirectory = 1 << 2,
|
|
|
|
showOverwriteConfirmation = 1 << 3,
|
|
|
|
dontAddToRecent = 1 << 4
|
|
|
|
}
|
|
|
|
|
|
|
|
enum OpenFileDialogProperties {
|
|
|
|
openFile = 1 << 0,
|
|
|
|
openDirectory = 1 << 1,
|
|
|
|
multiSelections = 1 << 2,
|
|
|
|
createDirectory = 1 << 3, // macOS
|
|
|
|
showHiddenFiles = 1 << 4,
|
|
|
|
promptToCreate = 1 << 5, // Windows
|
|
|
|
noResolveAliases = 1 << 6, // macOS
|
|
|
|
treatPackageAsDirectory = 1 << 7, // macOS
|
|
|
|
dontAddToRecent = 1 << 8 // Windows
|
|
|
|
}
|
|
|
|
|
2021-07-14 22:59:27 +00:00
|
|
|
let nextId = 0;
|
|
|
|
const getNextId = function () {
|
|
|
|
return ++nextId;
|
|
|
|
};
|
|
|
|
|
2020-06-30 22:51:44 +00:00
|
|
|
const normalizeAccessKey = (text: string) => {
|
|
|
|
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') {
|
2023-09-07 06:50:14 +00:00
|
|
|
return text.replaceAll(/&(&?)/g, '$1');
|
2020-06-30 22:51:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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') {
|
2023-09-07 06:50:14 +00:00
|
|
|
return text.replaceAll('_', '__').replaceAll(/&(.?)/g, (match, after) => {
|
2020-06-30 22:51:44 +00:00
|
|
|
if (after === '&') return after;
|
|
|
|
return `_${after}`;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return text;
|
|
|
|
};
|
|
|
|
|
|
|
|
const checkAppInitialized = function () {
|
|
|
|
if (!app.isReady()) {
|
|
|
|
throw new Error('dialog module can only be used after app is ready');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const setupOpenDialogProperties = (properties: (keyof typeof OpenFileDialogProperties)[]): number => {
|
|
|
|
let dialogProperties = 0;
|
|
|
|
for (const property of properties) {
|
2023-06-27 20:57:33 +00:00
|
|
|
if (Object.hasOwn(OpenFileDialogProperties, property)) { dialogProperties |= OpenFileDialogProperties[property]; }
|
2020-06-30 22:51:44 +00:00
|
|
|
}
|
|
|
|
return dialogProperties;
|
|
|
|
};
|
|
|
|
|
|
|
|
const setupSaveDialogProperties = (properties: (keyof typeof SaveFileDialogProperties)[]): number => {
|
|
|
|
let dialogProperties = 0;
|
|
|
|
for (const property of properties) {
|
2023-06-27 20:57:33 +00:00
|
|
|
if (Object.hasOwn(SaveFileDialogProperties, property)) { dialogProperties |= SaveFileDialogProperties[property]; }
|
2020-06-30 22:51:44 +00:00
|
|
|
}
|
|
|
|
return dialogProperties;
|
|
|
|
};
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
const saveDialog = (sync: boolean, window: BaseWindow | null, options?: SaveDialogOptions) => {
|
2020-06-30 22:51:44 +00:00
|
|
|
checkAppInitialized();
|
|
|
|
|
|
|
|
if (options == null) options = { title: 'Save' };
|
|
|
|
|
|
|
|
const {
|
|
|
|
buttonLabel = '',
|
|
|
|
defaultPath = '',
|
|
|
|
filters = [],
|
|
|
|
properties = [],
|
|
|
|
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,
|
2021-05-05 07:00:41 +00:00
|
|
|
properties: setupSaveDialogProperties(properties)
|
2020-06-30 22:51:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return sync ? dialogBinding.showSaveDialogSync(settings) : dialogBinding.showSaveDialog(settings);
|
|
|
|
};
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
const openDialog = (sync: boolean, window: BaseWindow | null, options?: OpenDialogOptions) => {
|
2020-06-30 22:51:44 +00:00
|
|
|
checkAppInitialized();
|
|
|
|
|
|
|
|
if (options == null) {
|
|
|
|
options = {
|
|
|
|
title: 'Open',
|
|
|
|
properties: ['openFile']
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
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');
|
|
|
|
|
|
|
|
const settings = {
|
|
|
|
title,
|
|
|
|
buttonLabel,
|
|
|
|
defaultPath,
|
|
|
|
filters,
|
|
|
|
message,
|
|
|
|
securityScopedBookmarks,
|
|
|
|
window,
|
2021-05-05 07:00:41 +00:00
|
|
|
properties: setupOpenDialogProperties(properties)
|
2020-06-30 22:51:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return (sync) ? dialogBinding.showOpenDialogSync(settings) : dialogBinding.showOpenDialog(settings);
|
|
|
|
};
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
const messageBox = (sync: boolean, window: BaseWindow | null, options?: MessageBoxOptions) => {
|
2020-06-30 22:51:44 +00:00
|
|
|
checkAppInitialized();
|
|
|
|
|
|
|
|
if (options == null) options = { type: 'none', message: '' };
|
|
|
|
|
|
|
|
const messageBoxTypes = ['none', 'info', 'warning', 'error', 'question'];
|
|
|
|
|
|
|
|
let {
|
|
|
|
buttons = [],
|
|
|
|
cancelId,
|
2021-07-14 22:59:27 +00:00
|
|
|
signal,
|
2020-06-30 22:51:44 +00:00
|
|
|
checkboxLabel = '',
|
|
|
|
checkboxChecked,
|
|
|
|
defaultId = -1,
|
|
|
|
detail = '',
|
|
|
|
icon = null,
|
2021-09-23 10:56:14 +00:00
|
|
|
textWidth = 0,
|
2020-06-30 22:51:44 +00:00
|
|
|
noLink = false,
|
|
|
|
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');
|
|
|
|
if (typeof noLink !== 'boolean') throw new TypeError('noLink must be a boolean');
|
|
|
|
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;
|
|
|
|
if (checkboxChecked && !checkboxLabel) {
|
|
|
|
throw new Error('checkboxChecked requires that checkboxLabel also be passed');
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
2023-08-07 09:30:15 +00:00
|
|
|
for (const [i, button] of buttons.entries()) {
|
|
|
|
const text = button.toLowerCase();
|
2020-06-30 22:51:44 +00:00
|
|
|
if (text === 'cancel' || text === 'no') {
|
|
|
|
cancelId = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-14 22:59:27 +00:00
|
|
|
// AbortSignal processing.
|
|
|
|
let id: number | undefined;
|
|
|
|
if (signal) {
|
|
|
|
// Generate an ID used for closing the message box.
|
|
|
|
id = getNextId();
|
|
|
|
// Close the message box when signal is aborted.
|
|
|
|
if (signal.aborted) { return Promise.resolve({ cancelId, checkboxChecked }); }
|
|
|
|
signal.addEventListener('abort', () => dialogBinding._closeMessageBox(id));
|
|
|
|
}
|
|
|
|
|
2020-06-30 22:51:44 +00:00
|
|
|
const settings = {
|
|
|
|
window,
|
|
|
|
messageBoxType,
|
|
|
|
buttons,
|
2021-07-14 22:59:27 +00:00
|
|
|
id,
|
2020-06-30 22:51:44 +00:00
|
|
|
defaultId,
|
|
|
|
cancelId,
|
|
|
|
noLink,
|
|
|
|
title,
|
|
|
|
message,
|
|
|
|
detail,
|
|
|
|
checkboxLabel,
|
|
|
|
checkboxChecked,
|
2021-09-23 10:56:14 +00:00
|
|
|
icon,
|
|
|
|
textWidth
|
2020-06-30 22:51:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (sync) {
|
|
|
|
return dialogBinding.showMessageBoxSync(settings);
|
|
|
|
} else {
|
|
|
|
return dialogBinding.showMessageBox(settings);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showOpenDialog(window: BaseWindow, options: OpenDialogOptions): OpenDialogReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showOpenDialog(options: OpenDialogOptions): OpenDialogReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showOpenDialog (windowOrOptions: BaseWindow | OpenDialogOptions, maybeOptions?: OpenDialogOptions): OpenDialogReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return openDialog(false, window, options);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showOpenDialogSync(window: BaseWindow, options: OpenDialogOptions): OpenDialogReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showOpenDialogSync(options: OpenDialogOptions): OpenDialogReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showOpenDialogSync (windowOrOptions: BaseWindow | OpenDialogOptions, maybeOptions?: OpenDialogOptions): OpenDialogReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return openDialog(true, window, options);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showSaveDialog(window: BaseWindow, options: SaveDialogOptions): SaveDialogReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showSaveDialog(options: SaveDialogOptions): SaveDialogReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showSaveDialog (windowOrOptions: BaseWindow | SaveDialogOptions, maybeOptions?: SaveDialogOptions): SaveDialogReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return saveDialog(false, window, options);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showSaveDialogSync(window: BaseWindow, options: SaveDialogOptions): SaveDialogReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showSaveDialogSync(options: SaveDialogOptions): SaveDialogReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showSaveDialogSync (windowOrOptions: BaseWindow | SaveDialogOptions, maybeOptions?: SaveDialogOptions): SaveDialogReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return saveDialog(true, window, options);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showMessageBox(window: BaseWindow, options: MessageBoxOptions): MessageBoxReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showMessageBox(options: MessageBoxOptions): MessageBoxReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showMessageBox (windowOrOptions: BaseWindow | MessageBoxOptions, maybeOptions?: MessageBoxOptions): MessageBoxReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return messageBox(false, window, options);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showMessageBoxSync(window: BaseWindow, options: MessageBoxOptions): MessageBoxReturnValue;
|
2020-06-30 22:51:44 +00:00
|
|
|
export function showMessageBoxSync(options: MessageBoxOptions): MessageBoxReturnValue;
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showMessageBoxSync (windowOrOptions: BaseWindow | MessageBoxOptions, maybeOptions?: MessageBoxOptions): MessageBoxReturnValue {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
return messageBox(true, window, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function showErrorBox (...args: any[]) {
|
|
|
|
return dialogBinding.showErrorBox(...args);
|
|
|
|
}
|
|
|
|
|
2024-05-31 10:04:22 +00:00
|
|
|
export function showCertificateTrustDialog (windowOrOptions: BaseWindow | CertificateTrustDialogOptions, maybeOptions?: CertificateTrustDialogOptions) {
|
|
|
|
const window = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? null : windowOrOptions);
|
|
|
|
const options = (windowOrOptions && !(windowOrOptions instanceof BaseWindow) ? windowOrOptions : maybeOptions);
|
2020-06-30 22:51:44 +00:00
|
|
|
|
|
|
|
if (options == null || typeof options !== 'object') {
|
|
|
|
throw new TypeError('options must be an object');
|
|
|
|
}
|
|
|
|
|
|
|
|
const { certificate, message = '' } = options;
|
|
|
|
if (certificate == null || typeof certificate !== 'object') {
|
|
|
|
throw new TypeError('certificate must be an object');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof message !== 'string') throw new TypeError('message must be a string');
|
|
|
|
|
|
|
|
return dialogBinding.showCertificateTrustDialog(window, certificate, message);
|
|
|
|
}
|