Почему системный файловый диалог меняет текущий каталог?
Это перевод Why does the common file dialog change the current directory? Автор: Реймонд Чен.
Когда вы перемещаетесь по папкам в системном диалоге открытия (или сохранения) файлов, то диалог вызывает функцию SetCurrentDirectory
, передавая ей каталог, в который вы перешли (эй, не заставляйте меня возрождать уголок зануды).
Окей, возможно, вашей первой реакцией будет: «Правда? Я вообще не знал, что он так делает!» Что ж, это ещё одна страница в истории проклятия текущего каталога.
Хорошо, но теперь возникает вопрос: «А зачем диалог это делает?»
Вообще-то, вы уже должны знать ответ на этот вопрос. Многие программы рассчитывают, что текущий каталог совпадает с каталогом открываемого документа.
Но, оказывается, существует способ сказать: «Нет, я — это не такая ламерская программа. Я не делаю предположений о текущем каталоге, я могу работать с любым. Так что не меняй текущий каталог». Вы можете сделать это, передав флаг OFN_NOCHANGEDIR
(а если ваша программа использует интерфейс IFileDialog
, то опция NOCHANGEDIR
включена перманентно — ура Прогрессу!).
Прим.пер.: для Delphi флаг OFN_NOCHANGEDIR
соответствует значению ofNoChangeDir
в TCommonDialog.Options
(значение fdoNoChangeDir
в TCustomFileDialog.Options
всегда игнорируется), а интерфейс IFileDialog
используется Delphi в «диалогах Vista» (например, TFileOpenDialog
), а также в стандартных файловых диалогах — автоматически на Vista+ при условии подключения манифеста (только в новых IDE, конечно же). В последнем случае опция ofNoChangeDir
игнорируется.
Но теперь, когда вы знаете про второе проклятие текущего каталога, вы можете использовать его против первого проклятия.
Если вы смогли определить программу, удерживающую открытым каталог, и вы подозреваете, что программа пала жертвой проклятия текущего каталога, то вы можете зайти в эту программу и открыть в ней файловый диалог (например, Файл / Открыть… или Файл / Сохранить как…). В этом диалоге вы меняете папку на, скажем, корень диска или на рабочий стол. А затем закрываете (отменяете) диалог.
Поскольку системный файловый диалог изменяет текущий каталог, то вы, фактически, внедрили вызов SetCurrentDirectory
прямо в целевой процесс, уводя, таким образом, текущий каталог с каталога, который вы хотите удалить. Заметьте, однако, что этот трюк сработает только если это приложение не передаёт флаг OFN_NOCHANGEDIR
в вызов GetSaveFileName
.
Для Проводника вы можете вызвать файловый диалог нажатием Win + R и кнопки «Обзор» — и в версиях Windows вплоть до Windows XP Проводник не передаёт флаг OFN_NOCHANGEDIR
.
Ответить
Хотите присоединиться к обсуждению?Не стесняйтесь вносить свой вклад!