#include "extenwindow.h" #include "ui_extenwindow.h" #include "defs.h" #include "text.h" #include "types.h" #include "misc.h" #include #include #include #include #include namespace { struct InfoForExtension { const char* name; int64_t value; InfoForExtension* next; ~InfoForExtension() { if (next) delete next; }; }; QHash extensions; QStringList extenNames; std::shared_mutex extenMutex; void Load(QString extenName) { if (extenName == ITH_DLL) return; // Extension is dll and exports "OnNewSentence" if (FARPROC callback = GetProcAddress(LoadLibraryOnce(S(extenName)), "OnNewSentence")) { std::scoped_lock writeLock(extenMutex); extensions[extenName] = (wchar_t*(*)(wchar_t*, const InfoForExtension*))callback; extenNames.push_back(extenName); } } void Unload(QString extenName) { std::scoped_lock writeLock(extenMutex); extenNames.erase(std::remove(extenNames.begin(), extenNames.end(), extenName), extenNames.end()); FreeLibrary(GetModuleHandleW(S(extenName).c_str())); } void Reorder(QStringList extenNames) { std::scoped_lock writeLock(extenMutex); ::extenNames = extenNames; } } bool DispatchSentenceToExtensions(std::wstring& sentence, std::unordered_map miscInfo) { wchar_t* sentenceBuffer = (wchar_t*)HeapAlloc(GetProcessHeap(), 0, (sentence.size() + 1) * sizeof(wchar_t)); wcscpy_s(sentenceBuffer, sentence.size() + 1, sentence.c_str()); InfoForExtension miscInfoLinkedList{ "", 0, nullptr }; InfoForExtension* miscInfoTraverser = &miscInfoLinkedList; for (auto[name, value] : miscInfo) miscInfoTraverser = miscInfoTraverser->next = new InfoForExtension{ name, value, nullptr }; std::shared_lock readLock(extenMutex); for (auto extenName : extenNames) { wchar_t* nextBuffer = extensions[extenName](sentenceBuffer, &miscInfoLinkedList); if (nextBuffer != sentenceBuffer) HeapFree(GetProcessHeap(), 0, sentenceBuffer); if (nextBuffer == nullptr) return false; sentenceBuffer = nextBuffer; } sentence = sentenceBuffer; HeapFree(GetProcessHeap(), 0, sentenceBuffer); return true; } ExtenWindow::ExtenWindow(QWidget* parent) : QMainWindow(parent, Qt::WindowCloseButtonHint), ui(new Ui::ExtenWindow) { ui->setupUi(this); ui->vboxLayout->addWidget(new QLabel(EXTEN_WINDOW_INSTRUCTIONS, this)); setWindowTitle(EXTENSIONS); ui->extenList->installEventFilter(this); if (!QFile::exists(EXTEN_SAVE_FILE)) QTextFile(EXTEN_SAVE_FILE, QIODevice::WriteOnly).write(DEFAULT_EXTENSIONS); for (auto extenName : QString(QTextFile(EXTEN_SAVE_FILE, QIODevice::ReadOnly).readAll()).split(">")) Load(extenName); Sync(); } ExtenWindow::~ExtenWindow() { delete ui; } void ExtenWindow::Sync() { ui->extenList->clear(); QTextFile extenSaveFile(EXTEN_SAVE_FILE, QIODevice::WriteOnly | QIODevice::Truncate); std::shared_lock readLock(extenMutex); for (auto extenName : extenNames) { ui->extenList->addItem(extenName); extenSaveFile.write((extenName + ">").toUtf8()); } } void ExtenWindow::Add(QFileInfo extenFile) { if (extenFile.suffix() != "dll") return; QFile::copy(extenFile.fileName(), extenFile.absoluteFilePath()); Load(extenFile.completeBaseName()); Sync(); } bool ExtenWindow::eventFilter(QObject* target, QEvent* event) { // See https://stackoverflow.com/questions/1224432/how-do-i-respond-to-an-internal-drag-and-drop-operation-using-a-qlistwidget/1528215 if (event->type() == QEvent::ChildRemoved) { QStringList extenNames; for (int i = 0; i < ui->extenList->count(); ++i) extenNames.push_back(ui->extenList->item(i)->text()); Reorder(extenNames); Sync(); } return false; } void ExtenWindow::keyPressEvent(QKeyEvent* event) { if (event->key() == Qt::Key_Delete) if (auto extenName = ui->extenList->currentItem()) { Unload(extenName->text()); Sync(); } } void ExtenWindow::dragEnterEvent(QDragEnterEvent* event) { event->acceptProposedAction(); } void ExtenWindow::dropEvent(QDropEvent* event) { for (auto file : event->mimeData()->urls()) Add(file.toLocalFile()); }