nodejs不用electron實現(xiàn)打開文件資源管理器并選擇文件
前言
最近在開發(fā)一些小腳本,用 nodejs 實現(xiàn)。其中很多功能需要選擇一個/多個文件,或者是選擇一個文件夾。
最初的實現(xiàn)是手動輸入一個目錄(這個只是一個普通的終端文本輸入,所以按下 tab 沒有路徑提示),非常的麻煩,而且很容易輸錯。
這種情況下網(wǎng)上給出的解決方案都是 electron。但是我一個小腳本用 electron 屬實有點夸張了,后來轉(zhuǎn)念一想可以通過 powershell 來實現(xiàn)類似的功能。
通過命令喚醒文件選擇器
通過 cmd / prowershell 喚醒文件選擇器
對 powershell 不熟悉的我唰的一聲打開了 gpt,gpt 不負眾望 很快給出了答案
注意這里有區(qū)別:cmd 終端中需要調(diào)用
powershell.exe如果當前已經(jīng)是在 powershell 終端的話,直接運行對應的指令即可
- 在 cmd 中運行:
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $FileDialog = New-Object System.Windows.Forms.OpenFileDialog; $result = $FileDialog.ShowDialog(); if ($result -eq 'OK') { Write-Output $FileDialog.FileName }}"
- 在 powershell 中運行:
& {Add-Type -AssemblyName System.Windows.Forms; $FileDialog = New-Object System.Windows.Forms.OpenFileDialog; $result = $FileDialog.ShowDialog(); if ($result -eq 'OK') { Write-Output $FileDialog.FileName }}
運行效果:選擇文件后終端會輸出你選擇的文件的全路徑

在 nodejs 調(diào)用 cmd 命令
const { exec, execSync } = require('child_process')
const command = `powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $FileDialog = New-Object System.Windows.Forms.OpenFileDialog; $result = $FileDialog.ShowDialog(); if ($result -eq 'OK') { Write-Output $FileDialog.FileName }}"`
// 異步執(zhí)行
exec(command, (error, file) => {
console.log(error, file)
})
// 同步執(zhí)行
const filePath = execSync(command)
console.log('選擇的文件', filePath)
到這結(jié)束了嗎?并沒有,我選擇的是一個包含中文名稱的路徑,輸入結(jié)果如下:

幾個小問題:
- execSync 同步執(zhí)行的代碼返回的是
Buffer類型- 可以用
filePath.toString()獲取實際的路徑
- 可以用
- 選擇的文件/文件夾包含中文,返回亂碼的問題
- 這個需要設置終端的編碼類型,也就是在執(zhí)行上面的命令執(zhí)行先執(zhí)行
chcp 650
- 這個需要設置終端的編碼類型,也就是在執(zhí)行上面的命令執(zhí)行先執(zhí)行
- 遇到執(zhí)行警告:
libpng warning: iCCP: cHRM chunk does not match sRGB- 卸載 QQ 拼音(雖然我也不知道具體是哪里的問題,不過確實是 QQ 拼音引起的)
調(diào)整后執(zhí)行效果如下:

如何實現(xiàn)多選文件 / 選擇文件夾?
- 選擇目錄
# 加載 Windows.Forms 程序集
Add-Type -AssemblyName System.Windows.Forms
# 創(chuàng)建 FolderBrowserDialog 對象
$folderDialog = New-Object System.Windows.Forms.FolderBrowserDialog
# 設置對話框的屬性
$folderDialog.Description = "請選擇文件夾"
$folderDialog.RootFolder = [System.Environment+SpecialFolder]::MyComputer
# 顯示文件夾選擇對話框
$result = $folderDialog.ShowDialog()
# 檢查用戶是否點擊了 "確定" 按鈕
if ($result -eq [System.Windows.Forms.DialogResult]::OK) {
# 輸出所選文件夾的路徑
Write-Output $folderDialog.SelectedPath
} else {
# 用戶取消選擇,這里輸出空路徑
Write-Output ""
}
合并成一行代碼則是:
- cmd 執(zhí)行:
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $folderDialog = New-Object System.Windows.Forms.FolderBrowserDialog; $folderDialog.Description = '請選擇文件夾'; $folderDialog.RootFolder = [System.Environment+SpecialFolder]::MyComputer; $result = $folderDialog.ShowDialog(); if ($result -eq [System.Windows.Forms.DialogResult]::OK) { Write-Output $folderDialog.SelectedPath } else { Write-Output '' }}"
多選文件同理:
# 加載 Windows.Forms 程序集
Add-Type -AssemblyName System.Windows.Forms
# 創(chuàng)建 OpenFileDialog 對象
$fileDialog = New-Object System.Windows.Forms.OpenFileDialog
# 設置對話框的屬性
$fileDialog.Multiselect = $true
$fileDialog.Title = "請選擇文件"
$fileDialog.Filter = "All Files (*.*)|*.*"
# 顯示文件選擇對話框
$result = $fileDialog.ShowDialog()
# 檢查用戶是否點擊了 "確定" 按鈕
if ($result -eq [System.Windows.Forms.DialogResult]::OK) {
# 輸出所選文件的路徑(數(shù)組)
Write-Output $fileDialog.FileNames
} else {
# 用戶取消選擇
Write-Output ""
}
合并為一行命令:
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $fileDialog = New-Object System.Windows.Forms.OpenFileDialog; $fileDialog.Multiselect = $true; $fileDialog.Title = '請選擇文件'; $fileDialog.Filter = 'All Files (*.*)|*.*'; $result = $fileDialog.ShowDialog(); if ($result -eq [System.Windows.Forms.DialogResult]::OK) { Write-Output $fileDialog.FileNames } else { Write-Output '' }}"
一些細節(jié)
- 如果是選擇單個文件/選擇文件目錄。輸出的結(jié)果會包含一些前后空格和換行,所以需要通過
filePath.trim()處理一下多余的字符 - 如果是多選的文件,返回的是字符串,每個文件以換行隔開的,也是需要自行處理
- 眼尖的朋友可能發(fā)現(xiàn)了在多選的命令中有一段代碼:
$fileDialog.Filter = "All Files (*.*)|*.*"可以用于設置可選擇的文件類型的。
對應的是這個功能:

- 就不再細說了~ 自行摸索
MacOS 如何實現(xiàn)用命令打開選擇器
以下的命令完全來自 GPT,并沒有經(jīng)過測試。自行判斷代碼是否正常運行 (原諒我并沒有 mac)
- 選擇一個文件
osascript -e 'POSIX path of (choose file with prompt "請選擇一個文件")'
- 選擇一個目錄
osascript -e 'POSIX path of (choose folder with prompt "請選擇一個目錄")'
- 選擇多個文件
(略)gpt 給出的答案非常的長,而且我沒電腦試驗,所以就不放代碼了,有試驗過的可以告訴我補充一下~
最后
至此,我的小腳本使用體驗已經(jīng)拉滿,再也不用一個個輸入文件路徑了。
總結(jié)下所有用到的命令:
Windows
- 選擇單個文件
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $FileDialog = New-Object System.Windows.Forms.OpenFileDialog; $result = $FileDialog.ShowDialog(); if ($result -eq 'OK') { Write-Output $FileDialog.FileName }}"
- 選擇文件目錄
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $folderDialog = New-Object System.Windows.Forms.FolderBrowserDialog; $folderDialog.Description = '請選擇文件夾'; $folderDialog.RootFolder = [System.Environment+SpecialFolder]::MyComputer; $result = $folderDialog.ShowDialog(); if ($result -eq [System.Windows.Forms.DialogResult]::OK) { Write-Output $folderDialog.SelectedPath } else { Write-Output '' }}"
- 選擇多個文件
powershell.exe -Command "& {Add-Type -AssemblyName System.Windows.Forms; $fileDialog = New-Object System.Windows.Forms.OpenFileDialog; $fileDialog.Multiselect = $true; $fileDialog.Title = '請選擇文件'; $fileDialog.Filter = 'All Files (*.*)|*.*'; $result = $fileDialog.ShowDialog(); if ($result -eq [System.Windows.Forms.DialogResult]::OK) { Write-Output $fileDialog.FileNames } else { Write-Output '' }}"
MacOS
- 選擇一個文件
osascript -e 'POSIX path of (choose file with prompt "請選擇一個文件")'
- 選擇一個目錄
osascript -e 'POSIX path of (choose folder with prompt "請選擇一個目錄")'
- 選擇多個文件
以上就是nodejs不用electron實現(xiàn)打開文件資源管理器并選擇文件的詳細內(nèi)容,更多關于nodejs實現(xiàn)打開文件資源管理器的資料請關注腳本之家其它相關文章!
相關文章
node.js中的path.isAbsolute方法使用說明
這篇文章主要介紹了node.js中的path.isAbsolute方法使用說明,本文介紹了path.isAbsolute的方法說明、語法、使用實例和實現(xiàn)源碼,需要的朋友可以參考下2014-12-12
node.js使用Moment.js js 時間計算方法示例小結(jié)
這篇文章主要介紹了node.js使用Moment.js js 時間計算方法,結(jié)合實例形式分析了Moment.js js模塊時間計算的常用操作技巧與相關注意事項,需要的朋友可以參考下2023-05-05
Node.js創(chuàng)建一個Express服務的方法詳解
這篇文章主要介紹了Node.js創(chuàng)建一個Express服務的方法,結(jié)合實例形式分析了node.js創(chuàng)建Express服務的具體步驟、實現(xiàn)方法及相關操作技巧,需要的朋友可以參考下2020-01-01

