123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284 |
- // Copyright 2019-2021 Tauri Programme within The Commons Conservancy
- // SPDX-License-Identifier: Apache-2.0
- // SPDX-License-Identifier: MIT
- /**
- * Native system dialogs for opening and saving files.
- *
- * This package is also accessible with `window.__TAURI__.dialog` when [`build.withGlobalTauri`](https://tauri.app/v1/api/config/#buildconfig.withglobaltauri) in `tauri.conf.json` is set to `true`.
- *
- * The APIs must be added to [`tauri.allowlist.dialog`](https://tauri.app/v1/api/config/#allowlistconfig.dialog) in `tauri.conf.json`:
- * ```json
- * {
- * "tauri": {
- * "allowlist": {
- * "dialog": {
- * "all": true, // enable all dialog APIs
- * "open": true, // enable file open API
- * "save": true // enable file save API
- * }
- * }
- * }
- * }
- * ```
- * It is recommended to allowlist only the APIs you use for optimal bundle size and security.
- * @module
- */
- import { invokeTauriCommand } from './helpers/tauri'
- /** Extension filters for the file dialog. */
- interface DialogFilter {
- /** Filter name. */
- name: string
- /**
- * Extensions to filter, without a `.` prefix.
- * @example
- * ```typescript
- * extensions: ['svg', 'png']
- * ```
- */
- extensions: string[]
- }
- /** Options for the open dialog. */
- interface OpenDialogOptions {
- /** The title of the dialog window. */
- title?: string
- /** The filters of the dialog. */
- filters?: DialogFilter[]
- /** Initial directory or file path. */
- defaultPath?: string
- /** Whether the dialog allows multiple selection or not. */
- multiple?: boolean
- /** Whether the dialog is a directory selection or not. */
- directory?: boolean
- /**
- * If `directory` is true, indicates that it will be read recursively later.
- * Defines whether subdirectories will be allowed on the scope or not.
- */
- recursive?: boolean
- }
- /** Options for the save dialog. */
- interface SaveDialogOptions {
- /** The title of the dialog window. */
- title?: string
- /** The filters of the dialog. */
- filters?: DialogFilter[]
- /**
- * Initial directory or file path.
- * If it's a directory path, the dialog interface will change to that folder.
- * If it's not an existing directory, the file name will be set to the dialog's file name input and the dialog will be set to the parent folder.
- */
- defaultPath?: string
- }
- interface MessageDialogOptions {
- /** The title of the dialog. Defaults to the app name. */
- title?: string
- /** The type of the dialog. Defaults to `info`. */
- type?: 'info' | 'warning' | 'error'
- }
- /**
- * Open a file/directory selection dialog.
- *
- * The selected paths are added to the filesystem and asset protocol allowlist scopes.
- * When security is more important than the easy of use of this API,
- * prefer writing a dedicated command instead.
- *
- * Note that the allowlist scope change is not persisted, so the values are cleared when the application is restarted.
- * You can save it to the filesystem using [tauri-plugin-persisted-scope](https://github.com/tauri-apps/tauri-plugin-persisted-scope).
- * @example
- * ```typescript
- * import { open } from '@tauri-apps/api/dialog';
- * // Open a selection dialog for image files
- * const selected = await open({
- * multiple: true,
- * filters: [{
- * name: 'Image',
- * extensions: ['png', 'jpeg']
- * }]
- * });
- * if (Array.isArray(selected)) {
- * // user selected multiple files
- * } else if (selected === null) {
- * // user cancelled the selection
- * } else {
- * // user selected a single file
- * }
- * ```
- *
- * @example
- * ```typescript
- * import { open } from '@tauri-apps/api/dialog';
- * import { appDir } from '@tauri-apps/api/path';
- * // Open a selection dialog for directories
- * const selected = await open({
- * directory: true,
- * multiple: true,
- * defaultPath: await appDir(),
- * });
- * if (Array.isArray(selected)) {
- * // user selected multiple directories
- * } else if (selected === null) {
- * // user cancelled the selection
- * } else {
- * // user selected a single directory
- * }
- * ```
- *
- * @returns A promise resolving to the selected path(s)
- */
- async function open(
- options: OpenDialogOptions = {}
- ): Promise<null | string | string[]> {
- if (typeof options === 'object') {
- Object.freeze(options)
- }
- return invokeTauriCommand({
- __tauriModule: 'Dialog',
- message: {
- cmd: 'openDialog',
- options
- }
- })
- }
- /**
- * Open a file/directory save dialog.
- *
- * The selected path is added to the filesystem and asset protocol allowlist scopes.
- * When security is more important than the easy of use of this API,
- * prefer writing a dedicated command instead.
- *
- * Note that the allowlist scope change is not persisted, so the values are cleared when the application is restarted.
- * You can save it to the filesystem using [tauri-plugin-persisted-scope](https://github.com/tauri-apps/tauri-plugin-persisted-scope).
- * @example
- * ```typescript
- * import { save } from '@tauri-apps/api/dialog';
- * const filePath = await save({
- * multiple: true,
- * filters: [{
- * name: 'Image',
- * extensions: ['stronghold']
- * }]
- * });
- * ```
- *
- * @returns A promise resolving to the selected path.
- */
- async function save(options: SaveDialogOptions = {}): Promise<string> {
- if (typeof options === 'object') {
- Object.freeze(options)
- }
- return invokeTauriCommand({
- __tauriModule: 'Dialog',
- message: {
- cmd: 'saveDialog',
- options
- }
- })
- }
- /**
- * Shows a message dialog with an `Ok` button.
- * @example
- * ```typescript
- * import { message } from '@tauri-apps/api/dialog';
- * await message('Tauri is awesome', 'Tauri');
- * await message('File not found', { title: 'Tauri', type: 'error' });
- * ```
- *
- * @param {string} message The message to show.
- * @param {string|MessageDialogOptions|undefined} options The dialog's options. If a string, it represents the dialog title.
- *
- * @return {Promise<void>} A promise indicating the success or failure of the operation.
- */
- async function message(
- message: string,
- options?: string | MessageDialogOptions
- ): Promise<void> {
- const opts = typeof options === 'string' ? { title: options } : options
- return invokeTauriCommand({
- __tauriModule: 'Dialog',
- message: {
- cmd: 'messageDialog',
- message: message.toString(),
- title: opts?.title?.toString(),
- type: opts?.type
- }
- })
- }
- /**
- * Shows a question dialog with `Yes` and `No` buttons.
- * @example
- * ```typescript
- * import { ask } from '@tauri-apps/api/dialog';
- * const yes = await ask('Are you sure?', 'Tauri');
- * const yes2 = await ask('This action cannot be reverted. Are you sure?', { title: 'Tauri', type: 'warning' });
- * ```
- *
- * @param {string} message The message to show.
- * @param {string|MessageDialogOptions|undefined} options The dialog's options. If a string, it represents the dialog title.
- *
- * @return {Promise<void>} A promise resolving to a boolean indicating whether `Yes` was clicked or not.
- */
- async function ask(
- message: string,
- options?: string | MessageDialogOptions
- ): Promise<boolean> {
- const opts = typeof options === 'string' ? { title: options } : options
- return invokeTauriCommand({
- __tauriModule: 'Dialog',
- message: {
- cmd: 'askDialog',
- message: message.toString(),
- title: opts?.title?.toString(),
- type: opts?.type
- }
- })
- }
- /**
- * Shows a question dialog with `Ok` and `Cancel` buttons.
- * @example
- * ```typescript
- * import { confirm } from '@tauri-apps/api/dialog';
- * const confirmed = await confirm('Are you sure?', 'Tauri');
- * const confirmed2 = await confirm('This action cannot be reverted. Are you sure?', { title: 'Tauri', type: 'warning' });
- * ```
- *
- * @param {string} message The message to show.
- * @param {string|MessageDialogOptions|undefined} options The dialog's options. If a string, it represents the dialog title.
- *
- * @return {Promise<void>} A promise resolving to a boolean indicating whether `Ok` was clicked or not.
- */
- async function confirm(
- message: string,
- options?: string | MessageDialogOptions
- ): Promise<boolean> {
- const opts = typeof options === 'string' ? { title: options } : options
- return invokeTauriCommand({
- __tauriModule: 'Dialog',
- message: {
- cmd: 'confirmDialog',
- message: message.toString(),
- title: opts?.title?.toString(),
- type: opts?.type
- }
- })
- }
- export type {
- DialogFilter,
- OpenDialogOptions,
- SaveDialogOptions,
- MessageDialogOptions
- }
- export { open, save, message, ask, confirm }
|