mirror of https://github.com/adamdruppe/arsd.git
more little tab complete details in the custom file picker
This commit is contained in:
parent
2fe1b06e4f
commit
953690139b
95
minigui.d
95
minigui.d
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue