2018-07-27 13:42:21 +08:00
|
|
|
#include "extensions.h"
|
|
|
|
|
2018-10-28 14:27:24 +08:00
|
|
|
static std::optional<Extension> LoadExtension(QString extenName)
|
2018-10-10 19:03:15 +08:00
|
|
|
{
|
2018-10-28 14:27:24 +08:00
|
|
|
// Extension is dll and exports "OnNewSentence"
|
|
|
|
HMODULE module = GetModuleHandleW(extenName.toStdWString().c_str());
|
|
|
|
if (!module) module = LoadLibraryW(extenName.toStdWString().c_str());
|
2018-10-10 19:03:15 +08:00
|
|
|
if (!module) return {};
|
2018-10-10 20:16:14 +08:00
|
|
|
FARPROC callback = GetProcAddress(module, "OnNewSentence");
|
2018-10-10 19:03:15 +08:00
|
|
|
if (!callback) return {};
|
2018-10-28 14:27:24 +08:00
|
|
|
return Extension{ extenName, (wchar_t*(*)(const wchar_t*, const InfoForExtension*))callback };
|
2018-10-10 19:03:15 +08:00
|
|
|
}
|
|
|
|
|
2018-10-28 14:27:24 +08:00
|
|
|
void Extension::Load(QString extenName)
|
|
|
|
{
|
|
|
|
std::unique_lock<std::shared_mutex> extenLock(extenMutex);
|
|
|
|
if (auto extension = LoadExtension(extenName)) extensions.push_back(extension.value());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Extension::SendToBack(QString extenName)
|
|
|
|
{
|
|
|
|
std::unique_lock<std::shared_mutex> extenLock(extenMutex);
|
|
|
|
Extension* extenIter = std::find_if(extensions.begin(), extensions.end(), [&](Extension extension) { return extension.name == extenName; });
|
|
|
|
Extension extension = *extenIter;
|
|
|
|
extensions.erase(extenIter);
|
|
|
|
extensions.push_back(extension);
|
|
|
|
}
|
2018-07-27 13:42:21 +08:00
|
|
|
|
2018-10-28 14:27:24 +08:00
|
|
|
void Extension::Unload(QString extenName)
|
2018-07-27 13:42:21 +08:00
|
|
|
{
|
2018-10-08 11:32:31 +08:00
|
|
|
std::unique_lock<std::shared_mutex> extenLock(extenMutex);
|
2018-10-28 14:27:24 +08:00
|
|
|
extensions.erase(std::find_if(extensions.begin(), extensions.end(), [&](Extension extension) { return extension.name == extenName; }));
|
|
|
|
FreeLibrary(GetModuleHandleW(extenName.toStdWString().c_str()));
|
2018-07-27 13:42:21 +08:00
|
|
|
}
|
|
|
|
|
2018-10-28 14:27:24 +08:00
|
|
|
QVector<QString> Extension::GetNames()
|
|
|
|
{
|
|
|
|
QVector<QString> ret;
|
|
|
|
for (auto extension : extensions) ret.push_back(extension.name);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Extension::DispatchSentence(std::wstring& sentence, std::unordered_map<std::string, int64_t> miscInfo)
|
2018-07-27 13:42:21 +08:00
|
|
|
{
|
2018-10-03 02:10:25 +08:00
|
|
|
bool success = true;
|
2018-10-07 23:02:00 +08:00
|
|
|
wchar_t* sentenceBuffer = (wchar_t*)HeapAlloc(GetProcessHeap(), 0, (sentence.size() + 1) * sizeof(wchar_t));
|
2018-09-23 03:45:54 +08:00
|
|
|
wcscpy_s(sentenceBuffer, sentence.size() + 1, sentence.c_str());
|
2018-10-10 19:03:15 +08:00
|
|
|
|
|
|
|
InfoForExtension miscInfoLinkedList{ "", 0, nullptr };
|
|
|
|
InfoForExtension* miscInfoTraverser = &miscInfoLinkedList;
|
|
|
|
for (auto& i : miscInfo) miscInfoTraverser = miscInfoTraverser->next = new InfoForExtension{ i.first.c_str(), i.second, nullptr };
|
|
|
|
|
2018-10-03 02:10:25 +08:00
|
|
|
std::shared_lock<std::shared_mutex> extenLock(extenMutex);
|
2018-10-10 19:03:15 +08:00
|
|
|
for (auto extension : extensions)
|
2018-07-30 15:47:09 +08:00
|
|
|
{
|
2018-10-10 19:03:15 +08:00
|
|
|
wchar_t* nextBuffer = extension.callback(sentenceBuffer, &miscInfoLinkedList);
|
2018-10-10 05:43:33 +08:00
|
|
|
if (nextBuffer == nullptr) { success = false; break; }
|
|
|
|
if (nextBuffer != sentenceBuffer) HeapFree(GetProcessHeap(), 0, sentenceBuffer);
|
|
|
|
sentenceBuffer = nextBuffer;
|
2018-07-30 15:47:09 +08:00
|
|
|
}
|
2018-09-01 16:23:29 +08:00
|
|
|
sentence = std::wstring(sentenceBuffer);
|
2018-10-10 19:03:15 +08:00
|
|
|
|
2018-10-07 23:02:00 +08:00
|
|
|
HeapFree(GetProcessHeap(), 0, sentenceBuffer);
|
2018-10-03 02:10:25 +08:00
|
|
|
return success;
|
2018-07-27 13:42:21 +08:00
|
|
|
}
|