183 lines
4.8 KiB
C++
183 lines
4.8 KiB
C++
#include "extension.h"
|
|
#include "util.h"
|
|
#include "network.h"
|
|
#include <ctime>
|
|
#include <QInputDialog>
|
|
#include <QTimer>
|
|
|
|
extern const char* SELECT_LANGUAGE;
|
|
extern const wchar_t* TOO_MANY_TRANS_REQUESTS;
|
|
extern const wchar_t* TRANSLATION_ERROR;
|
|
extern const char* GOOGLE_PROMPT;
|
|
|
|
QStringList languages
|
|
{
|
|
"English: en",
|
|
"Afrikaans: af",
|
|
"Arabic: ar",
|
|
"Albanian: sq",
|
|
"Belarusian: be",
|
|
"Bengali: bn",
|
|
"Bosnian: bs",
|
|
"Bulgarian: bg",
|
|
"Catalan: ca",
|
|
"Chinese(Simplified): zh-CH",
|
|
"Chinese(Traditional): zh-TW",
|
|
"Croatian: hr",
|
|
"Czech: cs",
|
|
"Danish: da",
|
|
"Dutch: nl",
|
|
"Esperanto: eo",
|
|
"Estonian: et",
|
|
"Filipino: tl",
|
|
"Finnish: fi",
|
|
"French: fr",
|
|
"Galician: gl",
|
|
"German: de",
|
|
"Greek: el",
|
|
"Hebrew: iw",
|
|
"Hindi: hi",
|
|
"Hungarian: hu",
|
|
"Icelandic: is",
|
|
"Indonesian: id",
|
|
"Irish: ga",
|
|
"Italian: it",
|
|
"Japanese: ja",
|
|
"Klingon: tlh",
|
|
"Korean: ko",
|
|
"Latin: la",
|
|
"Latvian: lv",
|
|
"Lithuanian: lt",
|
|
"Macedonian: mk",
|
|
"Malay: ms",
|
|
"Maltese: mt",
|
|
"Norwegian: no",
|
|
"Persian: fa",
|
|
"Polish: pl",
|
|
"Portuguese: pt",
|
|
"Romanian: ro",
|
|
"Russian: ru",
|
|
"Serbian: sr",
|
|
"Slovak: sk",
|
|
"Slovenian: sl",
|
|
"Somali: so",
|
|
"Spanish: es",
|
|
"Swahili: sw",
|
|
"Swedish: sv",
|
|
"Thai: th",
|
|
"Turkish: tr",
|
|
"Ukranian: uk",
|
|
"Urdu: ur",
|
|
"Vietnamese: vi",
|
|
"Welsh: cy",
|
|
"Yiddish: yi",
|
|
"Zulu: zu"
|
|
};
|
|
|
|
std::wstring translateTo = L"en";
|
|
|
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
|
{
|
|
switch (ul_reason_for_call)
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
{
|
|
QTimer::singleShot(0, []
|
|
{
|
|
translateTo = QInputDialog::getItem(
|
|
nullptr,
|
|
SELECT_LANGUAGE,
|
|
GOOGLE_PROMPT,
|
|
languages,
|
|
0, false, nullptr,
|
|
Qt::WindowCloseButtonHint
|
|
).split(" ")[1].toStdWString();
|
|
});
|
|
}
|
|
break;
|
|
case DLL_PROCESS_DETACH:
|
|
{
|
|
}
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
std::wstring GetTranslationUri(const std::wstring& text, unsigned TKK)
|
|
{
|
|
// If no TKK available, use this uri. Can't use too much or google will detect unauthorized access
|
|
if (!TKK) return L"/translate_a/single?client=gtx&dt=ld&dt=rm&dt=t&tl=" + translateTo + L"&q=" + text;
|
|
|
|
// Artikash 8/19/2018: reverse engineered from translate.google.com
|
|
std::wstring escapedText;
|
|
unsigned a = _time64(NULL) / 3600, b = a; // <- the first part of TKK
|
|
for (unsigned char ch : WideStringToString(text))
|
|
{
|
|
wchar_t escapedChar[4] = {};
|
|
swprintf_s<4>(escapedChar, L"%%%02X", (int)ch);
|
|
escapedText += escapedChar;
|
|
a += ch;
|
|
a += a << 10;
|
|
a ^= a >> 6;
|
|
}
|
|
a += a << 3;
|
|
a ^= a >> 11;
|
|
a += a << 15;
|
|
a ^= TKK;
|
|
a %= 1000000;
|
|
b ^= a;
|
|
|
|
return L"/translate_a/single?client=t&dt=ld&dt=rm&dt=t&tl=" + translateTo + L"&tk=" + std::to_wstring(a) + L"." + std::to_wstring(b) + L"&q=" + escapedText;
|
|
}
|
|
|
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
|
{
|
|
if (sentenceInfo["text number"] == 0) return false;
|
|
|
|
static std::atomic<HINTERNET> internet = NULL;
|
|
if (!internet) internet = WinHttpOpen(L"Mozilla/5.0 Textractor", WINHTTP_ACCESS_TYPE_DEFAULT_PROXY, NULL, NULL, 0);
|
|
static std::atomic<unsigned> TKK = 0;
|
|
static RateLimiter rateLimiter(30, 60 * 1000);
|
|
|
|
std::wstring translation;
|
|
if (!(rateLimiter.Request() || sentenceInfo["current select"])) translation = TOO_MANY_TRANS_REQUESTS;
|
|
else if (internet)
|
|
{
|
|
if (!TKK)
|
|
if (InternetHandle connection = WinHttpConnect(internet, L"translate.google.com", INTERNET_DEFAULT_HTTPS_PORT, 0))
|
|
if (InternetHandle request = WinHttpOpenRequest(connection, L"GET", L"/", NULL, NULL, NULL, WINHTTP_FLAG_SECURE))
|
|
if (WinHttpSendRequest(request, NULL, 0, NULL, 0, 0, NULL))
|
|
if (auto response = ReceiveHttpRequest(request))
|
|
if (std::wsmatch results; std::regex_search(response.value(), results, std::wregex(L"(\\d{7,})'"))) TKK = stoll(results[1]);
|
|
|
|
if (InternetHandle connection = WinHttpConnect(internet, L"translate.google.com", INTERNET_DEFAULT_HTTPS_PORT, 0))
|
|
if (InternetHandle request = WinHttpOpenRequest(connection, L"GET", GetTranslationUri(sentence, TKK).c_str(), NULL, NULL, NULL, WINHTTP_FLAG_ESCAPE_DISABLE | WINHTTP_FLAG_SECURE))
|
|
if (WinHttpSendRequest(request, NULL, 0, NULL, 0, 0, NULL))
|
|
if (auto response = ReceiveHttpRequest(request))
|
|
// Response formatted as JSON: starts with [[["
|
|
if (response.value()[0] == L'[')
|
|
{
|
|
for (std::wsmatch results; std::regex_search(response.value(), results, std::wregex(L"\\[\"(.*?)\",[n\"]")); response = results.suffix())
|
|
translation += std::wstring(results[1]) + L" ";
|
|
Escape(translation);
|
|
}
|
|
else
|
|
{
|
|
translation = TRANSLATION_ERROR + (L" (TKK=" + std::to_wstring(TKK) + L")");
|
|
TKK = 0;
|
|
}
|
|
}
|
|
|
|
if (translation.empty()) translation = TRANSLATION_ERROR;
|
|
sentence += L"\n" + translation;
|
|
return true;
|
|
}
|
|
|
|
TEST(
|
|
{
|
|
std::wstring test = L"こんにちは";
|
|
ProcessSentence(test, { SentenceInfo::DUMMY });
|
|
assert(test.find(L"Hello") != std::wstring::npos);
|
|
}
|
|
);
|