5#include <QtCore/qglobal.h>
7#include <AppKit/AppKit.h>
13#include <QtCore/qbuffer.h>
14#include <QtCore/qdebug.h>
15#include <QtCore/qstringlist.h>
16#include <QtCore/qvarlengtharray.h>
17#include <QtCore/qabstracteventdispatcher.h>
18#include <QtCore/qdir.h>
19#include <QtCore/qregularexpression.h>
20#include <QtCore/qpointer.h>
21#include <QtCore/private/qcore_mac_p.h>
23#include <QtGui/qguiapplication.h>
24#include <QtGui/private/qguiapplication_p.h>
26#include <qpa/qplatformtheme.h>
27#include <qpa/qplatformnativeinterface.h>
29#include <UniformTypeIdentifiers/UniformTypeIdentifiers.h>
33using namespace Qt::StringLiterals;
38 return QPlatformTheme::removeMnemonics(s).trimmed().toNSString();
44 if ([panel isKindOfClass:NSOpenPanel.
class])
45 return static_cast<NSOpenPanel*>(panel);
52@implementation QNSOpenSavePanelDelegate {
55 NSView *m_accessoryView;
56 NSPopUpButton *m_popupButton;
57 NSTextField *m_textField;
58 QPointer<QCocoaFileDialogHelper> m_helper;
60 SharedPointerFileDialogOptions m_options;
61 QString m_currentSelection;
62 QStringList m_nameFilterDropDownList;
63 QStringList m_selectedNameFilter;
66- (instancetype)initWithAcceptMode:(
const QString &)selectFile
67 options:(SharedPointerFileDialogOptions)options
68 helper:(QCocoaFileDialogHelper *)helper
70 if ((self = [super init])) {
73 if (m_options->acceptMode() == QFileDialogOptions::AcceptOpen)
74 m_panel = [[NSOpenPanel openPanel] retain];
76 m_panel = [[NSSavePanel savePanel] retain];
78 m_panel.canSelectHiddenExtension = YES;
79 m_panel.level = NSModalPanelWindowLevel;
83 m_nameFilterDropDownList = m_options->nameFilters();
84 QString selectedVisualNameFilter = m_options->initiallySelectedNameFilter();
85 m_selectedNameFilter = [self findStrippedFilterWithVisualFilterName:selectedVisualNameFilter];
87 m_panel.extensionHidden = [&]{
88 for (
const auto &nameFilter : m_nameFilterDropDownList) {
89 const auto extensions = QPlatformFileDialogHelper::cleanFilterList(nameFilter);
90 for (
const auto &extension : extensions) {
96 if (extension ==
"*"_L1 || extension ==
"*.*"_L1)
110 if (extension.count(
'.') > 1)
117 const QFileInfo sel(selectFile);
118 if (sel.isDir() && !sel.isBundle()){
119 m_panel.directoryURL = [NSURL fileURLWithPath:sel.absoluteFilePath().toNSString()];
120 m_currentSelection.clear();
122 m_panel.directoryURL = [NSURL fileURLWithPath:sel.absolutePath().toNSString()];
123 m_currentSelection = sel.absoluteFilePath();
126 [self createPopUpButton:selectedVisualNameFilter hideDetails:options->testOption(QFileDialogOptions::HideNameFilterDetails)];
127 [self createTextField];
128 [self createAccessory];
130 m_panel.accessoryView = m_nameFilterDropDownList.size() > 1 ? m_accessoryView : nil;
134 m_panel.delegate = self;
136 if (
auto *openPanel = openpanel_cast(m_panel))
137 openPanel.accessoryViewDisclosed = YES;
139 [self updateProperties];
146 [m_panel orderOut:m_panel];
147 m_panel.accessoryView = nil;
148 [m_popupButton release];
149 [m_textField release];
150 [m_accessoryView release];
151 m_panel.delegate = nil;
156- (
bool)showPanel:(Qt::WindowModality) windowModality withParent:(QWindow *)parent
158 const QFileInfo info(m_currentSelection);
159 NSString *filepath = info.filePath().toNSString();
160 NSURL *url = [NSURL fileURLWithPath:filepath isDirectory:info.isDir()];
161 bool selectable = (m_options->acceptMode() == QFileDialogOptions::AcceptSave)
162 || [self panel:m_panel shouldEnableURL:url];
164 m_panel.nameFieldStringValue = selectable ? info.fileName().toNSString() : @
"";
166 [self updateProperties];
168 auto completionHandler = ^(NSInteger result) {
170 m_helper->panelClosed(result);
173 if (windowModality == Qt::WindowModal && parent) {
174 NSView *view =
reinterpret_cast<NSView*>(parent->winId());
175 [m_panel beginSheetModalForWindow:view.window completionHandler:completionHandler];
176 }
else if (windowModality == Qt::ApplicationModal) {
179 [m_panel beginWithCompletionHandler:completionHandler];
185-(
void)runApplicationModalPanel
194 QMacAutoReleasePool pool;
199 qApp->processEvents(QEventLoop::ExcludeUserInputEvents | QEventLoop::ExcludeSocketNotifiers);
202 QCocoaEventDispatcher::clearCurrentThreadCocoaEventDispatcherInterruptFlag();
204 auto result = [m_panel runModal];
205 m_helper->panelClosed(result);
209 QCoreApplication::eventDispatcher()->wakeUp();
214 m_currentSelection = QString::fromNSString(m_panel.URL.path).normalized(QString::NormalizationForm_C);
217 [NSApp endSheet:m_panel];
218 else if (NSApp.modalWindow == m_panel)
224- (BOOL)panel:(id)sender shouldEnableURL:(NSURL *)url
228 NSString *filename = url.path;
229 if (!filename.length)
232 const QFileInfo fileInfo(QString::fromNSString(filename));
236 if (fileInfo.isDir()) {
242 bool treatBundlesAsFiles = !m_panel.treatsFilePackagesAsDirectories;
243 if (!(treatBundlesAsFiles && [NSWorkspace.sharedWorkspace isFilePackageAtPath:filename]))
247 if (![self fileInfoMatchesCurrentNameFilter:fileInfo])
250 QDir::Filters filter = m_options->filter();
251 if ((!(filter & (QDir::Dirs | QDir::AllDirs)) && fileInfo.isDir())
252 || (!(filter & QDir::Files) && (fileInfo.isFile() && !fileInfo.isSymLink()))
253 || ((filter & QDir::NoSymLinks) && fileInfo.isSymLink()))
256 bool filterPermissions = ((filter & QDir::PermissionMask)
257 && (filter & QDir::PermissionMask) != QDir::PermissionMask);
258 if (filterPermissions) {
259 if ((!(filter & QDir::Readable) && fileInfo.isReadable())
260 || (!(filter & QDir::Writable) && fileInfo.isWritable())
261 || (!(filter & QDir::Executable) && fileInfo.isExecutable()))
275- (BOOL)panel:(id)sender validateURL:(NSURL *)url error:(NSError *
_Nullable *)outError
277 Q_ASSERT(sender == m_panel);
279 if (![m_panel.allowedContentTypes count] && !m_selectedNameFilter.isEmpty()) {
284 QFileInfo fileInfo(QString::fromNSString(url.path).normalized(QString::NormalizationForm_C));
286 if ([self fileInfoMatchesCurrentNameFilter:fileInfo])
289 if (fileInfo.suffix().isEmpty()) {
294 fileInfo = [self applyDefaultSuffixFromCurrentNameFilter:fileInfo];
296 if (!fileInfo.exists() || m_options->testOption(QFileDialogOptions::DontConfirmOverwrite))
299 QMacAutoReleasePool pool;
300 auto *alert = [[NSAlert
new] autorelease];
301 alert.alertStyle = NSAlertStyleCritical;
303 alert.messageText = [NSString stringWithFormat:qt_mac_AppKitString(@
"SavePanel",
304 @
"\\U201c%@\\U201d already exists. Do you want to replace it?"),
305 fileInfo.fileName().toNSString()];
306 alert.informativeText = [NSString stringWithFormat:qt_mac_AppKitString(@
"SavePanel",
307 @
"A file or folder with the same name already exists in the folder %@. "
308 "Replacing it will overwrite its current contents."),
309 fileInfo.absoluteDir().dirName().toNSString()];
311 auto *replaceButton = [alert addButtonWithTitle:qt_mac_AppKitString(@
"SavePanel", @
"Replace")];
312 replaceButton.hasDestructiveAction = YES;
313 replaceButton.tag = 1337;
314 [alert addButtonWithTitle:qt_mac_AppKitString(@
"Common", @
"Cancel")];
316 [alert beginSheetModalForWindow:m_panel
317 completionHandler:^(NSModalResponse returnCode) {
318 [NSApp stopModalWithCode:returnCode];
320 return [NSApp runModalForWindow:alert.window] == replaceButton.tag;
322 QFileInfo firstFilter(m_selectedNameFilter.first());
323 auto *domain = qGuiApp->organizationDomain().toNSString();
324 *outError = [NSError errorWithDomain:domain code:0 userInfo:@{
325 NSLocalizedDescriptionKey:[NSString stringWithFormat:qt_mac_AppKitString(@
"SavePanel",
326 @
"You cannot save this document with extension \\U201c.%1$@\\U201d at the end "
327 "of the name. The required extension is \\U201c.%2$@\\U201d."),
328 fileInfo.completeSuffix().toNSString(), firstFilter.completeSuffix().toNSString()]
337- (
QFileInfo)applyDefaultSuffixFromCurrentNameFilter:(
const QFileInfo &)fileInfo
339 QFileInfo filterInfo(m_selectedNameFilter.first());
340 return QFileInfo(fileInfo.absolutePath(),
341 fileInfo.baseName() +
'.' + filterInfo.completeSuffix());
344- (
bool)fileInfoMatchesCurrentNameFilter:(
const QFileInfo &)fileInfo
347 if (m_selectedNameFilter.isEmpty())
351 for (
const auto &filter : m_selectedNameFilter) {
352 if (QDir::match(filter, fileInfo.fileName()))
359- (
void)setNameFilters:(
const QStringList &)filters hideDetails:(BOOL)hideDetails
361 [m_popupButton removeAllItems];
362 m_nameFilterDropDownList = filters;
363 if (filters.size() > 0){
364 for (
int i = 0; i < filters.size(); ++i) {
365 const QString filter = hideDetails ? [self removeExtensions:filters.at(i)] : filters.at(i);
366 [m_popupButton.menu addItemWithTitle:filter.toNSString() action:nil keyEquivalent:@
""];
368 [m_popupButton selectItemAtIndex:0];
369 m_panel.accessoryView = m_accessoryView;
371 m_panel.accessoryView = nil;
374 [self filterChanged:self];
377- (
void)filterChanged:(id)sender
383 const QString selection = m_nameFilterDropDownList.value([m_popupButton indexOfSelectedItem]);
384 m_selectedNameFilter = [self findStrippedFilterWithVisualFilterName:selection];
385 [m_panel validateVisibleColumns];
386 [self updateProperties];
388 const QStringList filters = m_options->nameFilters();
389 const int menuIndex = m_popupButton.indexOfSelectedItem;
390 emit m_helper->filterSelected(menuIndex >= 0 && menuIndex < filters.size() ? filters.at(menuIndex) : QString());
393- (QList<QUrl>)selectedFiles
395 if (
auto *openPanel = openpanel_cast(m_panel)) {
397 for (NSURL *url in openPanel.URLs) {
398 QString path = QString::fromNSString(url.path).normalized(QString::NormalizationForm_C);
399 result << QUrl::fromLocalFile(path);
403 QString filename = QString::fromNSString(m_panel.URL.path).normalized(QString::NormalizationForm_C);
404 QFileInfo fileInfo(filename);
406 if (fileInfo.suffix().isEmpty() && ![self fileInfoMatchesCurrentNameFilter:fileInfo]) {
410 fileInfo = [self applyDefaultSuffixFromCurrentNameFilter:fileInfo];
415 const QString defaultSuffix = m_options->defaultSuffix();
416 if (fileInfo.suffix().isEmpty() && !defaultSuffix.isEmpty()) {
417 fileInfo.setFile(fileInfo.absolutePath(),
418 fileInfo.baseName() +
'.' + defaultSuffix);
421 return { QUrl::fromLocalFile(fileInfo.filePath()) };
425- (
void)updateProperties
427 const QFileDialogOptions::FileMode fileMode = m_options->fileMode();
428 bool chooseFilesOnly = fileMode == QFileDialogOptions::ExistingFile
429 || fileMode == QFileDialogOptions::ExistingFiles;
430 bool chooseDirsOnly = fileMode == QFileDialogOptions::Directory
431 || fileMode == QFileDialogOptions::DirectoryOnly
432 || m_options->testOption(QFileDialogOptions::ShowDirsOnly);
434 m_panel.title = m_options->windowTitle().toNSString();
435 m_panel.canCreateDirectories = !(m_options->testOption(QFileDialogOptions::ReadOnly));
437 if (m_options->isLabelExplicitlySet(QFileDialogOptions::Accept))
438 m_panel.prompt = strippedText(m_options->labelText(QFileDialogOptions::Accept));
439 if (m_options->isLabelExplicitlySet(QFileDialogOptions::FileName))
440 m_panel.nameFieldLabel = strippedText(m_options->labelText(QFileDialogOptions::FileName));
442 if (
auto *openPanel = openpanel_cast(m_panel)) {
443 openPanel.canChooseFiles = !chooseDirsOnly;
444 openPanel.canChooseDirectories = !chooseFilesOnly;
445 openPanel.allowsMultipleSelection = (fileMode == QFileDialogOptions::ExistingFiles);
446 openPanel.resolvesAliases = !(m_options->testOption(QFileDialogOptions::DontResolveSymlinks));
449 m_popupButton.hidden = chooseDirsOnly;
451 m_panel.allowedContentTypes = [self computeAllowedContentTypes];
461 const bool nameFieldHasExtension = m_panel.nameFieldStringValue.pathExtension.length > 0;
462 if (![m_panel.allowedContentTypes count] && !nameFieldHasExtension && !openpanel_cast(m_panel)) {
463 if (!UTTypeDirectory.preferredFilenameExtension) {
464 m_panel.allowedContentTypes = @[ UTTypeDirectory ];
465 m_panel.allowedContentTypes = @[];
467 qWarning() <<
"UTTypeDirectory unexpectedly reported an extension";
471 m_panel.showsHiddenFiles = m_options->filter().testFlag(QDir::Hidden);
474 [m_panel validateVisibleColumns];
477- (
void)panelSelectionDidChange:(id)sender
489 if (!openpanel_cast(m_panel))
492 if (m_panel.visible) {
493 const QString selection = QString::fromNSString(m_panel.URL.path);
494 if (selection != m_currentSelection) {
495 m_currentSelection = selection;
496 emit m_helper->currentChanged(QUrl::fromLocalFile(selection));
501- (
void)panel:(id)sender directoryDidChange:(NSString *)path
508 m_helper->panelDirectoryDidChange(path);
512
513
514
515
516
517
518
519
520- (NSArray<UTType*>*)computeAllowedContentTypes
522 if (m_options->acceptMode() != QFileDialogOptions::AcceptSave)
525 auto *types = [[NSMutableArray<UTType*>
new] autorelease];
526 for (
const QString &filter : std::as_const(m_selectedNameFilter)) {
527 if (!filter.startsWith(
"*."_L1))
530 if (filter.contains(u'?'))
533 if (filter.count(u'*') != 1)
536 auto extensions = filter.split(
'.', Qt::SkipEmptyParts);
537 if (extensions.count() > 2)
540 auto *utType = [UTType typeWithFilenameExtension:extensions.last().toNSString()];
541 [types addObject:utType];
547- (
QString)removeExtensions:(
const QString &)filter
549 QRegularExpression regExp(QString::fromLatin1(QPlatformFileDialogHelper::filterRegExp));
550 QRegularExpressionMatch match = regExp.match(filter);
551 if (match.hasMatch())
552 return match.captured(1).trimmed();
556- (
void)createTextField
558 NSRect textRect = { { 0.0, 3.0 }, { 100.0, 25.0 } };
559 m_textField = [[NSTextField alloc] initWithFrame:textRect];
560 m_textField.cell.font = [NSFont systemFontOfSize:[NSFont systemFontSizeForControlSize:NSControlSizeRegular]];
561 m_textField.alignment = NSTextAlignmentRight;
562 m_textField.editable =
false;
563 m_textField.selectable =
false;
564 m_textField.bordered =
false;
565 m_textField.drawsBackground =
false;
566 if (m_options->isLabelExplicitlySet(QFileDialogOptions::FileType))
567 m_textField.stringValue = strippedText(m_options->labelText(QFileDialogOptions::FileType));
570- (
void)createPopUpButton:(
const QString &)selectedFilter hideDetails:(BOOL)hideDetails
572 NSRect popUpRect = { { 100.0, 5.0 }, { 250.0, 25.0 } };
573 m_popupButton = [[NSPopUpButton alloc] initWithFrame:popUpRect pullsDown:NO];
574 m_popupButton.target = self;
575 m_popupButton.action = @selector(filterChanged:);
577 if (!m_nameFilterDropDownList.isEmpty()) {
578 int filterToUse = -1;
579 for (
int i = 0; i < m_nameFilterDropDownList.size(); ++i) {
580 const QString currentFilter = m_nameFilterDropDownList.at(i);
581 if (selectedFilter == currentFilter ||
582 (filterToUse == -1 && currentFilter.startsWith(selectedFilter)))
584 QString filter = hideDetails ? [self removeExtensions:currentFilter] : currentFilter;
585 [m_popupButton.menu addItemWithTitle:filter.toNSString() action:nil keyEquivalent:@
""];
587 if (filterToUse != -1)
588 [m_popupButton selectItemAtIndex:filterToUse];
592- (
QStringList) findStrippedFilterWithVisualFilterName:(QString)name
594 for (
const QString ¤tFilter : std::as_const(m_nameFilterDropDownList)) {
595 if (currentFilter.startsWith(name))
596 return QPlatformFileDialogHelper::cleanFilterList(currentFilter);
598 return QStringList();
601- (
void)createAccessory
603 NSRect accessoryRect = { { 0.0, 0.0 }, { 450.0, 33.0 } };
604 m_accessoryView = [[NSView alloc] initWithFrame:accessoryRect];
605 [m_accessoryView addSubview:m_textField];
606 [m_accessoryView addSubview:m_popupButton];
613QCocoaFileDialogHelper::QCocoaFileDialogHelper()
622 QMacAutoReleasePool pool;
623 [m_delegate release];
629 if (result == NSModalResponseOK)
637 m_directory = directory;
640 m_delegate->m_panel.directoryURL = [NSURL fileURLWithPath:directory.toLocalFile().toNSString()];
650 if (!path || [path isEqual:NSNull.null] || !path.length)
653 const auto oldDirectory = m_directory;
654 m_directory = QUrl::fromLocalFile(
655 QString::fromNSString(path).normalized(QString::NormalizationForm_C));
657 if (m_directory != oldDirectory) {
659 emit directoryEntered(m_directory);
665 QString filePath = filename.toLocalFile();
666 if (QDir::isRelativePath(filePath))
667 filePath = QFileInfo(directory().toLocalFile(), filePath).filePath();
671 setDirectory(QFileInfo(filePath).absolutePath());
677 return [m_delegate selectedFiles];
678 return QList<QUrl>();
686 [m_delegate updateProperties];
693 const int index = options()->nameFilters().indexOf(filter);
696 options()->setInitiallySelectedNameFilter(filter);
699 [m_delegate->m_popupButton selectItemAtIndex:index];
700 [m_delegate filterChanged:nil];
707 return options()->initiallySelectedNameFilter();
708 int index = [m_delegate->m_popupButton indexOfSelectedItem];
709 if (index >= options()->nameFilters().count())
711 return index != -1 ? options()->nameFilters().at(index) : QString();
719 [m_delegate closePanel];
727 if (windowFlags & Qt::WindowStaysOnTopHint) {
735 createNSOpenSavePanelDelegate();
737 return [m_delegate showPanel:windowModality withParent:parent];
742 QMacAutoReleasePool pool;
745 const QList<QUrl> selectedFiles = opts->initiallySelectedFiles();
746 const QUrl directory = m_directory.isEmpty() ? opts->initialDirectory() : m_directory;
747 const bool selectDir = selectedFiles.isEmpty();
748 QString selection(selectDir ? directory.toLocalFile() : selectedFiles.front().toLocalFile());
749 QNSOpenSavePanelDelegate *delegate = [[QNSOpenSavePanelDelegate alloc]
755 [
static_cast<QNSOpenSavePanelDelegate *>(m_delegate) release];
756 m_delegate = delegate;
761 Q_ASSERT(m_delegate);
763 if (m_delegate->m_panel.visible) {
765 QEventLoop eventLoop;
766 m_eventLoop = &eventLoop;
767 eventLoop.exec(QEventLoop::DialogExec);
768 m_eventLoop =
nullptr;
771 [m_delegate runApplicationModalPanel];
QList< QUrl > selectedFiles() const override
bool defaultNameFilterDisables() const override
void setFilter() override
QString selectedNameFilter() const override
void panelDirectoryDidChange(NSString *path)
QUrl directory() const override
void setDirectory(const QUrl &directory) override
virtual ~QCocoaFileDialogHelper()
bool show(Qt::WindowFlags windowFlags, Qt::WindowModality windowModality, QWindow *parent) override
void selectNameFilter(const QString &filter) override
void selectFile(const QUrl &filename) override
void panelClosed(NSInteger result)
QSharedPointer< QFileDialogOptions > SharedPointerFileDialogOptions
static NSString * strippedText(QString s)
static NSOpenPanel * openpanel_cast(NSSavePanel *panel)
QList< QString > QStringList
Constructs a string list that contains the given string, str.