Всем привет.
Не так давно выложил статью про бота.
И вот тут меня зацепил) один комментарий.
Уже в момент создания бота в голове крутилась реализация десктопного GUI-заменителя).
Сложности я там не увидел. Но шо-то народ говорит сложно. Не попробуешь - не узнаешь), а тут удалось выкроить немного свободного времени. Погнали)
Качаю Delphi 11CE (бесплатная, письма с ссылками приходили на яндекс-почту), ставлю Python4Delphi - увязывает между собой ЯП и Delphi4PythonExporter - позволяет экспортировать GUI на Delphi в Python (не пригодился).
Мгновенно набрасываю минимальный интерфейс:
Из кода бота на Python вырезаю все лишнее и получаю вроде как шаблон, который и будет грузиться в компонент memo формы:
# pip install yadisk
# pip install delphivcl
import yadisk
# токен яндекс диска
y = yadisk.YaDisk(token='абра-кадабра от яндекса')
##search_dir = 'GOST'
##
##find_str = '1811'
if y.check_token():
# ищем в папке поиска документ содержащий запрос
if not y.is_dir('/'+search_dir):
print('Папка ' + search_dir +' не обнаружена. Шо-то поломалось. Извините.')
else:
Spis = []
for item in y.listdir(search_dir):
if find_str in item['name']:
if len(Spis) < 7:
print('Обнаружен документ: ')
print(item['name'].encode().decode("ansi"))
Spis.append(item['name'])
if len(Spis) == 0:
print('Извините, пока такого документа в папке '+ search_dir +' не нашлось.')
if len(Spis) == 1:
# ------------даем ссылку на файл-------------------
y.publish('/'+search_dir+'/'+Spis[0]) # делаем публичный файл
# шлем ссылку
print(y.get_meta('/'+search_dir+'/'+Spis[0]).public_url)
if len(Spis) > 1:
print('Найдено документов: '+str(len(Spis)) + '. Уточните запрос:')
else:
print('Извините по каким-то причинам диск не доступен. Попробуйте в другой раз')
Попутно убираю файл с настройками, токен всовываю в скрипт.
В скрипт питона думал сначала передавать папку поиска и запрос через текстовые файлы. Хотя и есть компоненты. Но потом подумал - таки это ж я могу и после загрузки скрипта в компонент memo вставить нужные строки:
Memo2.Lines.Insert(0, 'search_dir = "'+ComboBox1.Text+'"');
Memo2.Lines.Insert(0, 'find_str = "'+Edit1.Text+'"');
Питон гибкий). Таким образом избегаю сложностей по обмену данными между 2-мя системами и использую только 2 компонента пакета P4D - PythonEngine1: TPythonEngine и PythonGUIInputOutput1: TPythonGUIInputOutput.
Получаю структуру для минимально рабочего приложения:
Вся работа с приложением сводится к выбору папки поиска, ввода запроса и нажатия кнопки. После в невидимый memo2 грузится скрипт, добавляются нужные строки, запускается выполнение, все сообщения от скрипта через print возвращаются в memo1.
unit Unit2;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.ExtCtrls, Vcl.StdCtrls,
PythonEngine, Vcl.PythonGUIInputOutput, SynEdit;
type
TForm2 = class(TForm)
Memo1: TMemo;
Edit1: TEdit;
Splitter1: TSplitter;
Panel1: TPanel;
Label1: TLabel;
ComboBox1: TComboBox;
Button1: TButton;
PythonEngine1: TPythonEngine;
PythonGUIInputOutput1: TPythonGUIInputOutput;
Memo2: TMemo;
SynEdit1: TSynEdit;
procedure FormCreate(Sender: TObject);
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form2: TForm2;
implementation
{$R *.dfm}
procedure TForm2.Button1Click(Sender: TObject);
begin
Memo1.Clear;
SynEdit1.Clear;
SynEdit1.Lines.Add('Запускаю поиск по запросу: ');
SynEdit1.Lines.Add(Edit1.Text);
SynEdit1.Lines.Add('В папке: '+ComboBox1.Text);
Application.ProcessMessages;
Memo2.Clear;
Memo2.Lines.LoadFromFile('bot_no_tg.py');
// добавляем в скрипт папку поиска и запрос
Memo2.Lines.Insert(0, 'search_dir = "'+ComboBox1.Text+'"');
Memo2.Lines.Insert(0, 'find_str = "'+Edit1.Text+'"');
PythonEngine1.ExecStrings( Memo2.Lines );
Memo1.Visible:=false;
Application.ProcessMessages;
Memo1.Text:=Utf8ToAnsi(memo1.Text);
Memo1.Visible:=true;
Application.ProcessMessages;
end;
procedure TForm2.FormCreate(Sender: TObject);
begin
Memo1.Clear;
Memo2.Clear;
SynEdit1.Clear;
end;
end.
Практически повторен функционал бота. Рюшечек конечно можно добавить, но пока не интересно).
Из подводный камней:
разрядность систем надо соблюдать: если приложение для х64, то и Python х64,
пришлось повозится с кодировками и добавить 1)Delphi: Memo1.Text:=Utf8ToAnsi(memo1.Text); 2)Python: print(item['name'].encode().decode("ansi")),
debug и release имеют очень разные объемы конечного exe файла, а еще и рабочие папки разные. И папки для версий тоже разные.
изначально Python у меня был установлен, потом поставилась Delphi, потом в нее нужные компоненты. В момент установки они подхватили зарегистрированную в системе версию Python. При переносе на другую систему эти связи теряются и прога не работает. В инете скудновато, но есть инфа как программно сделать соответствующие привязки. Вроде даже пишут, шо можно и портабле версию использовать. Вот работу на другом компе я проверил в виртуал-боксе (см ниже).
На самом деле очень много времени отняла эта борьба с кодировками - кряко-зяблы лезли. Пришлось много полопатить разной инфы и много чего пробовать. Думал уже не победю, но как-то само заработало). Компонент memo туповат для этого - не может строки нормально отображать в разных кодировках. Полез даже SynEdit ставить, но и он не помог. Так что можно было и без него обойтись. Да и компонент memo на 1шт можно сократить.
Ура, заработало:
Кнопку "очистить" - убрал впоследствии, и так все многострочные компоненты чистятся).
Для проверки в виртуал-боксе качаю tiny10 23h1 x64, ставлю, через ISO закидываю экзешник со скриптом. Пробую. Без питоновской dll не запустилось. Да еще и версию другую затребовало (на моем компе 3.11х64 и работает), наверное по умолчанию что-то.