more little tab complete details in the custom file picker

This commit is contained in:
Adam D. Ruppe 2025-01-03 09:08:40 -05:00
parent 2fe1b06e4f
commit 953690139b
1 changed files with 79 additions and 16 deletions

View File

@ -16669,7 +16669,12 @@ class FilePicker : Dialog {
} }
// returns common prefix // returns common prefix
string loadFiles(string cwd, FileNameFilter filters, bool comingFromHistory = false) { static struct CommonPrefixInfo {
string commonPrefix;
int fileCount;
string exactMatch;
}
CommonPrefixInfo loadFiles(string cwd, FileNameFilter filters, bool comingFromHistory = false) {
if(!comingFromHistory) { if(!comingFromHistory) {
if(historyStack.length) { if(historyStack.length) {
@ -16687,6 +16692,8 @@ class FilePicker : Dialog {
dirs ~= "$PWD"; dirs ~= "$PWD";
string commonPrefix; string commonPrefix;
int commonPrefixCount;
string exactMatch;
bool matchesFilter(string name) { bool matchesFilter(string name) {
foreach(filter; filters.globPatterns) { foreach(filter; filters.globPatterns) {
@ -16714,30 +16721,49 @@ class FilePicker : Dialog {
if(commonPrefix is null) { if(commonPrefix is null) {
commonPrefix = name; commonPrefix = name;
commonPrefixCount = 1;
exactMatch = commonPrefix;
} else { } else {
foreach(idx, char i; name) { foreach(idx, char i; name) {
if(idx >= commonPrefix.length || i != commonPrefix[idx]) { if(idx >= commonPrefix.length || i != commonPrefix[idx]) {
commonPrefix = commonPrefix[0 .. idx]; commonPrefix = commonPrefix[0 .. idx];
commonPrefixCount ++;
exactMatch = null;
break; break;
} }
} }
} }
} }
bool applyFilterToDirectories = true;
bool showDotFiles = false;
foreach(filter; filters.globPatterns) {
if(filter == ".*")
showDotFiles = true;
else foreach(ch; filter)
if(ch == '.') {
// a filter like *.exe should not apply to the directory
applyFilterToDirectories = false;
break;
}
}
try try
getFiles(cwd, (string name, bool isDirectory) { getFiles(cwd, (string name, bool isDirectory) {
if(name == ".") if(name == ".")
return; // skip this as unnecessary return; // skip this as unnecessary
if(isDirectory) { if(isDirectory) {
if(name != ".." && name.length > 1 && name[0] == '.') if(applyFilterToDirectories) {
foreach(filter; filters.globPatterns) { if(matchesFilter(name)) {
if(filter == ".*") { dirs ~= name;
dirs ~= name; considerCommonPrefix(name, false);
considerCommonPrefix(name, false);
break;
}
} }
else { } else if(name != ".." && name.length > 1 && name[0] == '.') {
if(showDotFiles) {
dirs ~= name;
considerCommonPrefix(name, false);
}
} else {
dirs ~= name; dirs ~= name;
considerCommonPrefix(name, false); considerCommonPrefix(name, false);
} }
@ -16788,7 +16814,7 @@ class FilePicker : Dialog {
foreach(name; files) foreach(name; files)
listWidget.addOption(name); listWidget.addOption(name);
return commonPrefix; return CommonPrefixInfo(commonPrefix, commonPrefixCount, exactMatch);
} }
ListWidget listWidget; ListWidget listWidget;
@ -16798,6 +16824,7 @@ class FilePicker : Dialog {
LineEdit directoryHolder; LineEdit directoryHolder;
string currentDirectory_; string currentDirectory_;
FileNameFilter currentNonTabFilter;
FileNameFilter currentFilter; FileNameFilter currentFilter;
FileNameFilterSet filterOptions; FileNameFilterSet filterOptions;
@ -16883,6 +16910,7 @@ class FilePicker : Dialog {
}; };
filesOfType.setSelection(0); filesOfType.setSelection(0);
currentFilter = filterOptions.filters[0]; currentFilter = filterOptions.filters[0];
currentNonTabFilter = currentFilter;
} }
{ {
@ -16890,6 +16918,9 @@ class FilePicker : Dialog {
dirWidget = new ListWidget(mainGrid); dirWidget = new ListWidget(mainGrid);
listWidget = new ListWidget(mainGrid); listWidget = new ListWidget(mainGrid);
listWidget.tabStop = false;
dirWidget.tabStop = false;
FileDialogDelegate.PreviewWidget previewWidget = fileDialogDelegate.makePreviewWidget(mainGrid); FileDialogDelegate.PreviewWidget previewWidget = fileDialogDelegate.makePreviewWidget(mainGrid);
mainGrid.setChildPosition(dirWidget, 0, 0, 1, 1); mainGrid.setChildPosition(dirWidget, 0, 0, 1, 1);
@ -16903,6 +16934,7 @@ class FilePicker : Dialog {
dirWidget.addEventListener((scope DoubleClickEvent dev) { dirWidget.addEventListener((scope DoubleClickEvent dev) {
auto ce = new ChangeEvent!void(dirWidget, () {}); auto ce = new ChangeEvent!void(dirWidget, () {});
ce.dispatch(); ce.dispatch();
lineEdit.focus();
}); });
dirWidget.addEventListener((scope ChangeEvent!void sce) { dirWidget.addEventListener((scope ChangeEvent!void sce) {
@ -16923,6 +16955,7 @@ class FilePicker : Dialog {
} }
dirWidget.focusOn = -1; dirWidget.focusOn = -1;
lineEdit.focus();
}); });
// double click here, on the other hand, selects the file // double click here, on the other hand, selects the file
@ -16950,7 +16983,9 @@ class FilePicker : Dialog {
filesOfType.addEventListener(delegate (ChangeEvent!string ce) { filesOfType.addEventListener(delegate (ChangeEvent!string ce) {
currentFilter = FileNameFilter.fromString(ce.stringValue); currentFilter = FileNameFilter.fromString(ce.stringValue);
currentNonTabFilter = currentFilter;
loadFiles(currentDirectory, currentFilter); loadFiles(currentDirectory, currentFilter);
lineEdit.focus();
}); });
lineEdit.addEventListener((KeyDownEvent event) { lineEdit.addEventListener((KeyDownEvent event) {
@ -16966,13 +17001,32 @@ class FilePicker : Dialog {
else if(newFilter.length == 0) else if(newFilter.length == 0)
newFilter = "*"; newFilter = "*";
currentFilter = FileNameFilter("Custom filter", [newFilter]); auto newFilterObj = FileNameFilter("Custom filter", [newFilter]);
filesOfType.content = currentFilter.toString();
auto commonPrefix = loadFiles(currentDirectory, currentFilter); CommonPrefixInfo commonPrefix = loadFiles(currentDirectory, newFilterObj);
if(commonPrefix.fileCount == 1) {
if(commonPrefix.length) // exactly one file, let's see what it is
lineEdit.content = commonPrefix; auto specificFile = FilePath(commonPrefix.exactMatch).makeAbsolute(FilePath(currentDirectory));
if(getFileType(specificFile.toString) == FileType.dir) {
// a directory means we should change to it and keep the old filter
currentDirectory = specificFile.toString();
lineEdit.content = specificFile.toString() ~ "/";
loadFiles(currentDirectory, currentFilter);
} else {
// any other file should be selected in the list
currentDirectory = specificFile.directoryName;
current = specificFile.filename;
lineEdit.content = current;
loadFiles(currentDirectory, currentFilter);
}
} else if(commonPrefix.fileCount > 1) {
currentFilter = newFilterObj;
filesOfType.content = currentFilter.toString();
lineEdit.content = commonPrefix.commonPrefix;
} else {
// if there were no files, we don't really want to change the filter..
sdpyPrintDebugString("no files");
}
// FIXME: if that is a directory, add the slash? or even go inside? // FIXME: if that is a directory, add the slash? or even go inside?
@ -17029,14 +17083,23 @@ class FilePicker : Dialog {
if(ft == FileType.error && isOpenDialogInsteadOfSave) { if(ft == FileType.error && isOpenDialogInsteadOfSave) {
// FIXME: tell the user why // FIXME: tell the user why
messageBox("Cannot open file: " ~ accepted.toString ~ "\nTry another or cancel."); messageBox("Cannot open file: " ~ accepted.toString ~ "\nTry another or cancel.");
lineEdit.focus();
return; return;
} }
// FIXME: symlinks to dirs should prolly also get this behavior
if(ft == FileType.dir) { if(ft == FileType.dir) {
currentDirectory = accepted.toString; currentDirectory = accepted.toString;
currentFilter = currentNonTabFilter;
filesOfType.content = currentFilter.toString();
loadFiles(currentDirectory, currentFilter); loadFiles(currentDirectory, currentFilter);
lineEdit.content = ""; lineEdit.content = "";
lineEdit.focus();
return; return;
} }