[РЕШЕНО] Программы в Delphi 7 и решение проблем с кодировкой

Иногда так бывает, что надо скопировать текст/код, содержащий кириллические буквы, из редактора кода или текстовых полей разработанного приложения. Проблема состоит в том, что, когда такой текст вставляется в другие текстовые редакторы, кириллические символы вставляются в неверной кодировке. Например, такой код:

// комментарии
procedure TForm1.FormCreate(Sender: TObject);
begin
 showMessage('Всё хорошо')
end;

end.

Может быть вставлен в таком виде:


Несколько решений:

Первое:
Перед тем, как копировать текст, переключите язык ввода на русский. Всё должно скопироваться и вставиться нормально.

Второе:
Переключение языка ввода перед копированием текста – пожалуй, рабочий вариант во всех случаях. Но он не всегда удобен в том плане, что конечному пользователю надо сообщать дополнительно (либо в справочной документации, либо на форме самого приложения) о необходимости переключения языка ввода. Можно переключать язык ввода автоматически перед работой с буфером обмена (далее - БО), но, как я говорил ранее, проблема возникает и тогда, когда происходит копирование из стороннего приложения в наше. Если способ отследить такое и существует, мне он неизвестен. Потом, переключение языка ввода без ведома пользователя - в некотором роде моветон, с моей точки зрения, поэтому такой способ здесь не рассматривается.

Сейчас я расскажу о способе решения проблемы, который был найден мной в ходе разработки своей небольшой программы.

Во-первых, вам нужно подключить модуль RusClipboard.pas (согласно информации в Интернете, автор - Игорь Цысь (Igoreha), igoreha@i.com.ua)
Код модуля:

Код модуля RusClipboard.pas

unit RusClipboard;

interface

uses Clipbrd;

type
  TRusClipboard = class(TClipboard)
private
  procedure SetCodePage(const CodePage: longint);
public
  procedure Open; override;
  procedure Close; override;
end;

implementation

uses Windows;

{ TRusClipboard }

procedure TRusClipboard.Close;
begin
  SetCodePage($0419);
  inherited;
end;

procedure TRusClipboard.Open;
begin
  inherited;
  SetCodePage($0419);
end;

procedure TRusClipboard.SetCodePage(const CodePage: longint);
var
  Data: THandle;
  DataPtr: Pointer;
begin
  // Назначить кодовую страницу для буфера обмена
  Data:= GlobalAlloc(GMEM_MOVEABLE + GMEM_DDESHARE, 4);
  try
    DataPtr := GlobalLock(Data);
    try
      Move(CodePage, DataPtr^, 4);
      SetClipboardData(CF_LOCALE, Data);
    finally
      GlobalUnlock(Data);
    end;
  except
    GlobalFree(Data);
  end;
end;

var
 NewClipboard: TClipboard;
 OldClipboard: TClipboard;

initialization
 NewClipboard := TRusClipboard.Create;
 OldClipboard := SetClipboard(NewClipboard);
 OldClipboard.Free;
end. 

Внимание: далее в тексте добавленные стоки отмечены конструкцией <----

Говорят, его надо просто подключить к проекту. В секции uses укажите имя модуля – rusClipboard. Но у меня так не работало. Здесь может быть неясно, как его использовать. Хорошо, что я сразу догадался использовать класс, написанный в этом модуле – TRusClipboard. Значит, вторым шагом после подключения модуля будет создание переменной типа TRusClipboard в секции private класса вашей формы:

type
  TForm1 = class(TForm)
    ...
  private
    { Private declarations }
    clipboard: TRusClipboard; // <----
  public
    { Public declarations }
  end;

В обработчике создания формы надо выделить для него память

procedure TForm1.FormCreate(Sender: TObject);
begin
  clipboard:=TRusClipboard.Create; // <---
end;

А в обработчике закрытия формы память освободить:

procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
  clipboard.Free; // <---
end;

Решая проблему кодировки при копировании символов, я заметил, что всё копируется нормально, если при помещении информации в БО просто обратиться к свойству asText переменной БО, то есть, clipboard.asText. Из этого следует, что нам нужен обработчик помещения (текстовой) информации в БО. В отличие от двух предыдущих, средства Delphi 7 не позволяют автоматически подготовить для него код, поэтому его надо написать самому. Для этого в секции private класса нашей формы надо записать заголовок обработчика:

type
  TForm1 = class(TForm)
    ...
  private
    { Private declarations }
    clipboard: TRusClipboard;
    procedure drawClipboard(var AMsg: TWMDrawClipboard); message WM_DRAWCLIPBOARD; // <---
  public
    { Public declarations }
  end;

Справа от заголовка процедуры располагается ключевое слово message, за которым следует имя обрабатываемого сообщения. Это означает, что эта процедура будет являться обработчиком сообщения о помещении в буфер обмена информации. Далее, в разделе implementation необходимо написать реализацию этого обработчика. Здесь я рекомендую поставить курсор на строку с заголовком этого обработчика и воспользоваться комбинацией клавиш CTRL+SHIFT+C. В этом случае IDE создаст для вас заготовку для обработчика автоматически. Как я говорил ранее, при помещении текста в БО, надо обратиться к свойству asText переменной clipboard. Таким образом, реализация обработчика будет иметь следующий код:

procedure TForm1.drawClipboard(var AMsg: TWMDrawClipboard);
begin
  clipboard.AsText; // <---
end;

Я не разбирался, как работает класс БО в Delphi, и частности его дочерний TRusClipboard, поэтому для меня это – танцы с бубном, особенно, если учесть, что, asText – это не подпрограмма.

Чтобы это работало, нам надо разместить окно нашего приложения в цепочке наблюдателей БО. Для этого сначала мы создадим приватное поле с типом HWND:

type
  TForm1 = class(TForm)
    ...
  private
    { Private declarations }
    clipboard: TRusClipboard;
    nextWindowHandle: HWND; // <---
    procedure drawClipboard(var AMsg: TWMDrawClipboard); message WM_DRAWCLIPBOARD;
  public
    { Public declarations }
  end;

а в обработчике создания формы вызовем функцию setClipboardViewer, которой передадим дескриптор главного окна программы. Эта функция размещает окно, дескриптор которого мы передаём, в цепочке наблюдателей БО, и возвращает дескриптор следующего окна в цепочке. Имеем:

procedure TForm1.FormCreate(Sender: TObject);
begin
  clipboard:=TRusClipboard.Create;
  nextWindowHandle:=SetClipboardViewer(handle); // <---
end;

Обратите внимание на порядок строк в этом обработчике. Размещать окно в цепочке необходимо ПОСЛЕ выделения памяти для БО.

В обработчике закрытия формы нам надо изъять помещённое ранее окно из цепочки наблюдателей. Для этого вызывается функция changeClipboardChain. Она принимает два аргумента, первый из которых – дескриптор извлекаемого окна, а второй дескриптор замещающего окна. Мы его получали ранее, это окно, следующее за нашем в цепочке. Имеем:

procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
  changeClipboardChain(handle, nextWindowHandle); // <---
  clipboard.Free;
end;

Я считаю, что возвращаемый результат этой функции нам не нужен в данном случае, поэтому я его ничему не присваиваю.

Все эти операции достаточно выполнить в модуле основной формы. Во всех остальных формах вашего проекта работа с БО также будет происходить корректно.

Важное замечание
Как я заметил, неприятность выше описанного метода заключается в том, что иногда при помещении текста в буфер обмена, появляется ошибка “Не могу открыть буфер обмена” (“can’t open clipboard” или как-то так), при этом текст копируется без проблем.

Чтобы избежать появления этой ошибки, я советую тело обработчика помещения текста в буфер обмена обернуть в try/except.
В секции except писать ничего не надо - текст копируется нормально, но задача состоит в том, чтобы избежать появления сообщения об ошибке.
Возможно, это быдлокод, но лучшего / более красивого решения я пока не могу предложить.
Таким образом, этот код будет выглядеть так:


procedure TForm1.drawClipboard(var AMsg: TWMDrawClipboard);
begin
  try
    clipboard.AsText;
  except
  end;
end;

Демонстрацию работы данного способа я размещаю как в приложении, так и во внешней ссылке.

Ссылка на демо-проект на яндексДиске: https://yadi.sk/d/uLvsgE_AARpLrA


Ключевые слова для поиска темы
Текст код из Делфи Delphi вставляется криво крокозябрами кракозябрами в испорченной неверной кодировке

Решение проблемы находится в первом сообщении этой темы