Ассоциированный файл запускается с помощью механизма командной строки. В реестр записывается командная строка (вместе с именем приложения), в которой имя открываемого файла заменяется на %1
. Когда пользователь запускает ассоциированный файл (или он запускается приложением через ShellExecute
), система извлекает из реестра соответствующую командную строку, вместо %1
подставляет реальное имя файла и пытается выполнить получившуюся команду. Отметим, что если имя файла содержит пробелы, в кавычки оно автоматически не заключается, поэтому о кавычках приходится заботиться самостоятельно, заключая в них %1
. Таким образом, в реестр в качестве командной строки должно записываться следующее
<Имя программы> "%1"
Если существуют разные варианты запуска одного файла (т. е. как в нашем случае — open и view), они различаться дополнительным параметрами. В частности, в нашем примере для открытия для редактирования не будут требоваться дополнительные параметры, для открытия для просмотра в качестве второго параметра должен передаваться кляч v, т. е. в реестр для этой команды будет записана такая строка:
<Имя программы> "%1" v
Программа должна анализировать переданные ей параметры и открывать соответствующий файл в требуемом режиме. В нашем случае этот код выглядит очень просто (листинг 1.47).
procedure TDKSViewMainForm.FormShow(Sender: TObject);
var
OpenForView: Bооlean;
begin
// Проверяем наличие ключа "/v" в качестве второго параметра
OpenForView:= (ParamCount > 1) and (CompareText(ParamStr(2), '/v') = 0);
if ParamCount > 0 then OpenFile(ParamStr(1), OpenForView);
…
end;
B более сложных случаях (например, при большем числе команд для ассоциированного файла) анализ командной строки будет сложнее, но его принципы останутся теми же.
1.3.2.3. Поиск уже запущенной копии приложения
Во многих случаях желательно не давать пользователю возможности запустить второй экземпляр вашего приложения. В 16-разрядных версиях Windows все приложения выполнялись в одной виртуальной машине, и каждому из них через переменную HPrevInstance
передавался дескриптор предыдущей копии. По значению HPrevInstance
программа легко могла найти свой предыдущий экземпляр или определить, что других экземпляров нет, если HPrevInstance
равна нулю. В 32-разрядных версиях эта переменная для совместимости оставлена, но всегда равна нулю, т. к. предыдущая копия работает в своей виртуальной машине, и ее дескриптор не имеет смысла. Альтернативного механизма обнаружения уже запущенной копии система не предоставляет, приходится выкручиваться своими силами.
Для обнаружения уже запущенного приложения многие авторы предлагают использовать именованные системные объекты (мьютексы, семафоры, атомы и т. п.). При запуске программа пытается создать такой объект с определенным именем. Если оказывается, что такой объект уже создан, программа "понимает", что она — вторая копия, и завершается. Недостаток такого подхода — с его помощью можно установить только сам факт наличия предыдущей копии, но не более того. В нашем случае задача шире: при запуске второго экземпляра приложения должен активизироваться первый, а если второму экземпляру была передана непустая командная строка, первый должен получить эту строку и выполнить соответствующее действие, поэтому описанный способ нам не подходит.
Для решения задачи нам подойдут