2019-06-29 18:59:33 +08:00
|
|
|
|
#include "qtcommon.h"
|
|
|
|
|
#include "extension.h"
|
2019-08-12 22:44:51 +08:00
|
|
|
|
#include "defs.h"
|
2020-02-25 19:39:27 +08:00
|
|
|
|
#include "blockmarkup.h"
|
2019-06-13 16:01:29 +08:00
|
|
|
|
#include "network.h"
|
2020-02-27 19:42:29 +08:00
|
|
|
|
#include <map>
|
2020-02-25 19:39:27 +08:00
|
|
|
|
#include <fstream>
|
2019-06-13 16:01:29 +08:00
|
|
|
|
#include <QTimer>
|
|
|
|
|
|
2020-02-25 19:39:27 +08:00
|
|
|
|
extern const char* NATIVE_LANGUAGE;
|
2019-06-13 16:01:29 +08:00
|
|
|
|
extern const char* SELECT_LANGUAGE;
|
|
|
|
|
extern const char* SELECT_LANGUAGE_MESSAGE;
|
2020-02-12 15:06:01 +08:00
|
|
|
|
extern const char* LANGUAGE_SAVED;
|
2019-06-13 16:01:29 +08:00
|
|
|
|
extern const wchar_t* TOO_MANY_TRANS_REQUESTS;
|
|
|
|
|
|
|
|
|
|
extern const char* TRANSLATION_PROVIDER;
|
|
|
|
|
extern QStringList languages;
|
|
|
|
|
std::pair<bool, std::wstring> Translate(const std::wstring& text);
|
|
|
|
|
|
2019-08-12 22:44:51 +08:00
|
|
|
|
const char* LANGUAGE = u8"Language";
|
2020-02-25 19:39:27 +08:00
|
|
|
|
const std::string TRANSLATION_CACHE_FILE = FormatString("%sCache.txt", TRANSLATION_PROVIDER);
|
2019-08-12 22:44:51 +08:00
|
|
|
|
|
2019-07-03 14:53:10 +08:00
|
|
|
|
Synchronized<std::wstring> translateTo = L"en";
|
2019-08-12 22:44:51 +08:00
|
|
|
|
QSettings settings(CONFIG_FILE, QSettings::IniFormat);
|
2020-02-25 19:39:27 +08:00
|
|
|
|
|
2020-02-27 19:42:29 +08:00
|
|
|
|
Synchronized<std::map<std::wstring, std::wstring>> translationCache;
|
2019-06-29 18:59:33 +08:00
|
|
|
|
int savedSize;
|
2019-06-17 07:43:59 +08:00
|
|
|
|
|
2019-06-29 18:59:33 +08:00
|
|
|
|
void SaveCache()
|
|
|
|
|
{
|
2020-02-25 19:39:27 +08:00
|
|
|
|
std::wstring allTranslations(L"\xfeff");
|
2020-02-27 19:42:29 +08:00
|
|
|
|
for (const auto& [sentence, translation] : translationCache.Acquire().contents)
|
|
|
|
|
allTranslations.append(L"|SENTENCE|").append(sentence).append(L"|TRANSLATION|").append(translation).append(L"|END|\r\n");
|
2020-02-25 19:39:27 +08:00
|
|
|
|
std::ofstream(TRANSLATION_CACHE_FILE, std::ios::binary | std::ios::trunc).write((const char*)allTranslations.c_str(), allTranslations.size() * sizeof(wchar_t));
|
2019-06-29 18:59:33 +08:00
|
|
|
|
savedSize = translationCache->size();
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-13 16:01:29 +08:00
|
|
|
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
|
|
|
|
{
|
|
|
|
|
switch (ul_reason_for_call)
|
|
|
|
|
{
|
|
|
|
|
case DLL_PROCESS_ATTACH:
|
|
|
|
|
{
|
2019-08-12 22:44:51 +08:00
|
|
|
|
settings.beginGroup(TRANSLATION_PROVIDER);
|
|
|
|
|
if (settings.contains(LANGUAGE)) translateTo->assign(S(settings.value(LANGUAGE).toString()));
|
|
|
|
|
else QTimer::singleShot(0, []
|
2019-06-13 16:01:29 +08:00
|
|
|
|
{
|
2019-06-29 18:59:33 +08:00
|
|
|
|
QString language = QInputDialog::getItem(
|
2019-06-13 16:01:29 +08:00
|
|
|
|
nullptr,
|
|
|
|
|
SELECT_LANGUAGE,
|
|
|
|
|
QString(SELECT_LANGUAGE_MESSAGE).arg(TRANSLATION_PROVIDER),
|
|
|
|
|
languages,
|
2020-02-25 19:39:27 +08:00
|
|
|
|
std::find_if(languages.begin(), languages.end(), [](QString language) { return language.startsWith(NATIVE_LANGUAGE); }) - languages.begin(),
|
2019-06-13 16:01:29 +08:00
|
|
|
|
false,
|
|
|
|
|
nullptr,
|
2019-06-29 18:59:33 +08:00
|
|
|
|
Qt::WindowCloseButtonHint
|
2019-06-13 16:01:29 +08:00
|
|
|
|
);
|
2019-06-29 18:59:33 +08:00
|
|
|
|
translateTo->assign(S(language.split(": ")[1]));
|
2019-08-12 22:44:51 +08:00
|
|
|
|
settings.setValue(LANGUAGE, S(translateTo->c_str()));
|
2020-02-12 15:06:01 +08:00
|
|
|
|
QMessageBox::information(nullptr, SELECT_LANGUAGE, QString(LANGUAGE_SAVED).arg(CONFIG_FILE));
|
2019-06-13 16:01:29 +08:00
|
|
|
|
});
|
2019-06-17 07:43:59 +08:00
|
|
|
|
|
2020-02-25 19:39:27 +08:00
|
|
|
|
std::ifstream stream(TRANSLATION_CACHE_FILE, std::ios::binary);
|
|
|
|
|
BlockMarkupIterator savedTranslations(stream, Array<std::wstring_view>{ L"|SENTENCE|", L"|TRANSLATION|" });
|
|
|
|
|
auto translationCache = ::translationCache.Acquire();
|
|
|
|
|
while (auto read = savedTranslations.Next())
|
|
|
|
|
{
|
2020-02-27 19:42:29 +08:00
|
|
|
|
auto& [sentence, translation] = read.value();
|
|
|
|
|
translationCache->try_emplace(std::move(sentence), std::move(translation));
|
2020-02-25 19:39:27 +08:00
|
|
|
|
}
|
2019-06-29 18:59:33 +08:00
|
|
|
|
savedSize = translationCache->size();
|
2019-06-13 16:01:29 +08:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case DLL_PROCESS_DETACH:
|
|
|
|
|
{
|
2019-06-29 18:59:33 +08:00
|
|
|
|
SaveCache();
|
2019-06-13 16:01:29 +08:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
|
|
|
|
{
|
|
|
|
|
if (sentenceInfo["text number"] == 0) return false;
|
|
|
|
|
|
|
|
|
|
static class
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
bool Request()
|
|
|
|
|
{
|
|
|
|
|
auto tokens = this->tokens.Acquire();
|
|
|
|
|
tokens->push_back(GetTickCount());
|
|
|
|
|
if (tokens->size() > tokenCount * 5) tokens->erase(tokens->begin(), tokens->begin() + tokenCount * 3);
|
|
|
|
|
tokens->erase(std::remove_if(tokens->begin(), tokens->end(), [this](DWORD token) { return GetTickCount() - token > delay; }), tokens->end());
|
|
|
|
|
return tokens->size() < tokenCount;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
const int tokenCount = 30, delay = 60 * 1000;
|
|
|
|
|
Synchronized<std::vector<DWORD>> tokens;
|
|
|
|
|
} rateLimiter;
|
|
|
|
|
|
|
|
|
|
bool cache = false;
|
|
|
|
|
std::wstring translation;
|
2019-06-29 18:59:33 +08:00
|
|
|
|
{
|
2020-02-25 19:39:27 +08:00
|
|
|
|
auto translationCache = ::translationCache.Acquire();
|
2020-02-27 19:42:29 +08:00
|
|
|
|
auto translationLocation = translationCache->find(sentence);
|
|
|
|
|
if (translationLocation != translationCache->end()) translation = translationLocation->second;
|
2020-02-25 19:39:27 +08:00
|
|
|
|
else if (!(rateLimiter.Request() || sentenceInfo["current select"])) translation = TOO_MANY_TRANS_REQUESTS;
|
|
|
|
|
else std::tie(cache, translation) = Translate(sentence);
|
2020-02-27 19:42:29 +08:00
|
|
|
|
if (cache && sentenceInfo["current select"]) translationCache->try_emplace(translationLocation, sentence, translation);
|
2019-06-29 18:59:33 +08:00
|
|
|
|
}
|
2020-02-25 19:39:27 +08:00
|
|
|
|
if (cache && translationCache->size() > savedSize + 50) SaveCache();
|
2019-06-13 16:01:29 +08:00
|
|
|
|
|
2019-06-29 18:59:33 +08:00
|
|
|
|
Unescape(translation);
|
2019-06-13 16:01:29 +08:00
|
|
|
|
sentence += L"\n" + translation;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(
|
2019-07-03 14:53:10 +08:00
|
|
|
|
assert(Translate(L"こんにちは").second.find(L"ello") != std::wstring::npos)
|
2019-06-13 16:01:29 +08:00
|
|
|
|
);
|