forked from Public-Mirror/Textractor
Compare commits
136 Commits
regex_repl
...
x221Mod
Author | SHA1 | Date | |
---|---|---|---|
4335a4560a | |||
22946df452 | |||
3b22857d12 | |||
|
df5b830d17 | ||
|
f3fbe04409 | ||
|
5adeb1fab7 | ||
|
5c7b41dd84 | ||
|
92e2e85191 | ||
|
0bc23ba2c7 | ||
|
fd8a090e28 | ||
|
aa00d9e789 | ||
|
cfa11c5646 | ||
|
446f6d44e6 | ||
|
ff1a7a358c | ||
|
60a5d74eda | ||
|
4cebe9462f | ||
|
d97ebd4fbb | ||
|
233e75225c | ||
|
72ad51f039 | ||
|
39fcfaf644 | ||
|
5eb0440f01 | ||
|
c4ca84d39b | ||
|
c7b818aa75 | ||
|
2cb45ab9e3 | ||
|
72132239a5 | ||
|
5eaafbe81c | ||
|
97f2bc6304 | ||
|
f732f488e6 | ||
|
3e283975c8 | ||
|
5b35f121d2 | ||
|
b28aa3d189 | ||
|
faeda02f2b | ||
|
ef55ef0c61 | ||
|
c9240d9e68 | ||
|
6ba4f9c909 | ||
|
7c9f861926 | ||
|
e158219266 | ||
|
3a9fdfb9ae | ||
|
dad036083a | ||
|
b09783d111 | ||
|
306b9f0525 | ||
|
823987e79f | ||
|
fdc8a13b36 | ||
|
90b58da867 | ||
|
2316f59415 | ||
|
06b93e3def | ||
|
fde588d922 | ||
|
8e0e17c59b | ||
|
e6ee5ecf3d | ||
|
49133974b3 | ||
|
4b67b8a5cf | ||
|
663bb97fff | ||
|
953a537cb8 | ||
|
8ac4a072a9 | ||
|
45279d26dc | ||
|
5df90e2c01 | ||
|
0346f95707 | ||
|
1f19a9e74d | ||
|
b236a50a8f | ||
|
2f08ab1e9b | ||
|
ad3f09d8ef | ||
|
44c8e4f94c | ||
|
a17d8d5993 | ||
|
bf49f0bca6 | ||
|
15db478e62 | ||
|
46bc6ec84b | ||
|
b3b665fb65 | ||
|
17b5884149 | ||
|
742b7cacf3 | ||
|
4ff150b674 | ||
|
e83579ed7c | ||
|
febff243d3 | ||
|
1a29f5670a | ||
|
6ed3d9c951 | ||
|
492c843e5c | ||
|
574eeec79e | ||
|
6c74df864a | ||
|
ca93120442 | ||
|
eddf1bf0ee | ||
|
c9ad8880fb | ||
|
fb9fb5d54a | ||
|
085bec6d5b | ||
|
695672ce9c | ||
|
bda08d500e | ||
|
5862bf50ac | ||
|
493e80e568 | ||
|
db31e19997 | ||
|
d64a2c05b4 | ||
|
f8874bf8a0 | ||
|
b28b68d218 | ||
|
50f5b183f1 | ||
|
a566818ad4 | ||
|
ac95d873f1 | ||
|
444974ee8a | ||
|
3b948c3359 | ||
|
1eaa054b33 | ||
|
647058e9aa | ||
|
918c877e04 | ||
|
2cf146652d | ||
|
89198fd4fe | ||
|
9e0c95be98 | ||
|
7425ae770f | ||
|
e0981f89e8 | ||
|
eb1421c143 | ||
|
a1d3abb080 | ||
|
dd4b8cfbb5 | ||
|
615d372eeb | ||
|
e4cea83ae4 | ||
|
9e26c462e7 | ||
|
0df2526560 | ||
|
de3cad37f8 | ||
|
8c9faf9947 | ||
|
76fea31fb3 | ||
|
795ecce45e | ||
|
675695cde1 | ||
|
9f8e523ce3 | ||
|
d74dcdc286 | ||
|
b4aa113fac | ||
|
1e03e69d00 | ||
|
76d716956d | ||
|
1782292662 | ||
|
3c33d11d80 | ||
|
a8ae2156d9 | ||
|
f26f30dce7 | ||
|
1fcacc7bc9 | ||
|
71fe1410c2 | ||
|
aa0c0e0047 | ||
|
74121d7484 | ||
|
8543d49192 | ||
|
5537679442 | ||
|
20f35a02ed | ||
|
1bab6956a8 | ||
|
acc85f3a86 | ||
|
54a285b53b | ||
|
bf97055155 | ||
|
ad83eb290a |
@ -5,26 +5,25 @@ skip_branch_with_pr: false
|
|||||||
|
|
||||||
environment:
|
environment:
|
||||||
matrix:
|
matrix:
|
||||||
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2019
|
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2017
|
||||||
arch: x64
|
arch: x64
|
||||||
msvc_name: Visual Studio 16 2019
|
|
||||||
platform: x64
|
platform: x64
|
||||||
qtbin: msvc2017_64
|
qtbin: msvc2017_64
|
||||||
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2019
|
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2017
|
||||||
arch: x86
|
arch: x86
|
||||||
msvc_name: Visual Studio 16 2019
|
|
||||||
platform: Win32
|
platform: Win32
|
||||||
qtbin: msvc2017
|
qtbin: msvc2017
|
||||||
|
|
||||||
before_build:
|
before_build:
|
||||||
|
- git submodule update --init
|
||||||
- cd C:\
|
- cd C:\
|
||||||
- mkdir %arch%
|
- mkdir %arch%
|
||||||
- cd %arch%
|
- cd %arch%
|
||||||
- cmake -G "%msvc_name%" -A "%platform%" -DQt5_DIR="C:\Qt\5.13\%qtbin%\lib\cmake\Qt5" -DCMAKE_BUILD_TYPE="RelWithDebInfo" ../Textractor
|
- cmake -G "Visual Studio 15 2017" -A %platform% -DQt5_DIR="C:\Qt\5.13.2\%qtbin%\lib\cmake\Qt5" -DCMAKE_BUILD_TYPE="RelWithDebInfo" -DVERSION="" ../Textractor
|
||||||
|
|
||||||
build:
|
build:
|
||||||
project: C:\%arch%\Textractor.sln
|
project: C:\%arch%\Textractor.sln
|
||||||
parallel: true
|
parallel: false
|
||||||
verbosity: normal
|
verbosity: normal
|
||||||
|
|
||||||
after_build:
|
after_build:
|
||||||
|
3
.gitmodules
vendored
Normal file
3
.gitmodules
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[submodule "texthook/minhook"]
|
||||||
|
path = texthook/minhook
|
||||||
|
url = https://github.com/TsudaKageyu/minhook.git
|
@ -1,6 +1,11 @@
|
|||||||
cmake_minimum_required(VERSION 3.16)
|
cmake_minimum_required(VERSION 3.16)
|
||||||
|
|
||||||
project(Textractor)
|
project(Textractor)
|
||||||
|
|
||||||
|
if (NOT MSVC)
|
||||||
|
message(FATAL_ERROR "Textractor can only be built with Visual Studio")
|
||||||
|
endif()
|
||||||
|
|
||||||
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${CMAKE_CURRENT_SOURCE_DIR}/cmake)
|
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${CMAKE_CURRENT_SOURCE_DIR}/cmake)
|
||||||
|
|
||||||
add_compile_options(
|
add_compile_options(
|
||||||
@ -30,15 +35,16 @@ target_precompile_headers(pch PUBLIC include/common.h)
|
|||||||
file(GLOB ASSETS assets/*)
|
file(GLOB ASSETS assets/*)
|
||||||
file(COPY ${ASSETS} DESTINATION ${CMAKE_FINAL_OUTPUT_DIRECTORY})
|
file(COPY ${ASSETS} DESTINATION ${CMAKE_FINAL_OUTPUT_DIRECTORY})
|
||||||
|
|
||||||
|
include(QtUtils)
|
||||||
|
msvc_registry_search()
|
||||||
|
find_qt5(Core Widgets WinExtras WebSockets)
|
||||||
|
|
||||||
add_library(text text.cpp)
|
add_library(text text.cpp)
|
||||||
target_compile_definitions(text PRIVATE ${TEXT_LANGUAGE})
|
target_compile_definitions(text PRIVATE ${TEXT_LANGUAGE})
|
||||||
link_libraries(text)
|
link_libraries(text)
|
||||||
|
|
||||||
add_subdirectory(GUI)
|
add_subdirectory(host)
|
||||||
add_subdirectory(texthook)
|
add_subdirectory(texthook)
|
||||||
|
add_subdirectory(GUI)
|
||||||
add_subdirectory(extensions)
|
add_subdirectory(extensions)
|
||||||
add_subdirectory(test)
|
add_subdirectory(test)
|
||||||
if (DEFINED VERSION)
|
|
||||||
add_subdirectory(GUI/host)
|
|
||||||
endif()
|
|
||||||
#add_subdirectory(GUI/host)
|
|
||||||
|
@ -1,40 +1,31 @@
|
|||||||
{
|
{
|
||||||
// See https://go.microsoft.com//fwlink//?linkid=834763 for more information about this file.
|
|
||||||
"configurations": [
|
"configurations": [
|
||||||
{
|
{
|
||||||
"name": "x86-Debug",
|
"name": "x86-Debug",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "Debug",
|
"configurationType": "Debug",
|
||||||
|
"cmakeCommandArgs": "-DQT_ROOT=/Qt",
|
||||||
"inheritEnvironments": [ "msvc_x86" ]
|
"inheritEnvironments": [ "msvc_x86" ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "x86-RelWithDebInfo",
|
"name": "x86-RelWithDebInfo",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "RelWithDebInfo",
|
"configurationType": "RelWithDebInfo",
|
||||||
"inheritEnvironments": [ "msvc_x86" ]
|
"cmakeCommandArgs": "-DQT_ROOT=/Qt",
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "x86-Release",
|
|
||||||
"generator": "Ninja",
|
|
||||||
"configurationType": "Release",
|
|
||||||
"inheritEnvironments": [ "msvc_x86" ]
|
"inheritEnvironments": [ "msvc_x86" ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "x64-Debug",
|
"name": "x64-Debug",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "Debug",
|
"configurationType": "Debug",
|
||||||
|
"cmakeCommandArgs": "-DQT_ROOT=/Qt",
|
||||||
"inheritEnvironments": [ "msvc_x64" ]
|
"inheritEnvironments": [ "msvc_x64" ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "x64-RelWithDebInfo",
|
"name": "x64-RelWithDebInfo",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "RelWithDebInfo",
|
"configurationType": "RelWithDebInfo",
|
||||||
"inheritEnvironments": [ "msvc_x64" ]
|
"cmakeCommandArgs": "-DQT_ROOT=/Qt",
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "x64-Release",
|
|
||||||
"generator": "Ninja",
|
|
||||||
"configurationType": "Release",
|
|
||||||
"inheritEnvironments": [ "msvc_x64" ]
|
"inheritEnvironments": [ "msvc_x64" ]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
28
CREDITS.md
28
CREDITS.md
@ -1,28 +0,0 @@
|
|||||||
# Developers
|
|
||||||
|
|
||||||
If you're on this list and want your link changed let me know.
|
|
||||||
- Textractor mainly made by [Me](https://github.com/Artikash) with the help of
|
|
||||||
- [DoumanAsh](https://github.com/DoumanAsh)
|
|
||||||
- [Niakr1s](https://github.com/Niakr1s)
|
|
||||||
- [tinyAdapter](https://github.com/tinyAdapter)
|
|
||||||
- [lgztx96](https://github.com/lgztx96)
|
|
||||||
- [Jazzinghen](https://github.com/Jazzinghen)
|
|
||||||
- [luojunyuan](https://github.com/luojunyuan)
|
|
||||||
- French translation by Racky and [Gratusfr](https://github.com/Gratusfr)
|
|
||||||
- Spanish translation by [scese250](https://github.com/scese250)
|
|
||||||
- Turkish translation by [niisokusu](https://reddit.com/u/niisokusu)
|
|
||||||
- Simplified Chinese translation by [tinyAdapter](https://github.com/tinyAdapter) and [lgztx96](https://github.com/lgztx96)
|
|
||||||
- Russian translation by [TokcDK](https://github.com/TokcDK)
|
|
||||||
- Indonesian translation by [Hawxone](https://github.com/Hawxone)
|
|
||||||
- Portuguese translation by [TsumiHokiro](https://github.com/TsumiHokiro)
|
|
||||||
- Thai translation by [azmadoppler](https://github.com/azmadoppler)
|
|
||||||
- Korean translation by O SK
|
|
||||||
- Italian translation by [StarFang208](https://github.com/StarFang208)
|
|
||||||
- ITHVNR updated by [mireado](https://github.com/mireado), [Eguni](https://github.com/Eguni), and [IJEMIN](https://github.com/IJEMIN)
|
|
||||||
- ITHVNR originally made by [Stomp](http://www.hongfire.com/forum/member/325894-stomp)
|
|
||||||
- VNR engine made by [jichi](https://github.com/jichifly)
|
|
||||||
- ITH updated by [Andys](https://github.com/AndyScull)
|
|
||||||
- ITH originally made by [kaosu](https://code.google.com/archive/p/interactive-text-hooker)
|
|
||||||
- Locale Emulator library made by [xupefei](https://github.com/xupefei)
|
|
||||||
- MinHook library made by [TsudaKageyu](https://github.com/TsudaKageyu)
|
|
||||||
- Last but not least, the many people that have reported issues. To everybody mentioned here: Thank You!
|
|
@ -1,20 +1,14 @@
|
|||||||
include(QtUtils)
|
|
||||||
msvc_registry_search()
|
|
||||||
find_qt5(Core Widgets)
|
|
||||||
|
|
||||||
add_executable(Textractor WIN32
|
add_executable(Textractor WIN32
|
||||||
main.cpp
|
main.cpp
|
||||||
|
exception.cpp
|
||||||
mainwindow.cpp
|
mainwindow.cpp
|
||||||
extenwindow.cpp
|
extenwindow.cpp
|
||||||
host/exception.cpp
|
attachprocessdialog.cpp
|
||||||
host/host.cpp
|
|
||||||
host/textthread.cpp
|
|
||||||
host/hookcode.cpp
|
|
||||||
Textractor.rc
|
Textractor.rc
|
||||||
Textractor.ico
|
Textractor.ico
|
||||||
)
|
)
|
||||||
target_precompile_headers(Textractor REUSE_FROM pch)
|
target_precompile_headers(Textractor REUSE_FROM pch)
|
||||||
target_link_libraries(Textractor Qt5::Widgets shell32 winhttp)
|
target_link_libraries(Textractor host Qt5::Widgets Qt5::WinExtras shell32 winhttp)
|
||||||
|
|
||||||
if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Core.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Cored.dll)
|
if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Core.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Cored.dll)
|
||||||
add_custom_command(TARGET Textractor
|
add_custom_command(TARGET Textractor
|
||||||
|
39
GUI/attachprocessdialog.cpp
Normal file
39
GUI/attachprocessdialog.cpp
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
#include "attachprocessdialog.h"
|
||||||
|
#include <QtWinExtras/QtWin>
|
||||||
|
|
||||||
|
extern const char* SELECT_PROCESS;
|
||||||
|
extern const char* ATTACH_INFO;
|
||||||
|
|
||||||
|
AttachProcessDialog::AttachProcessDialog(QWidget* parent, std::vector<std::pair<QString, HICON>> processIcons) :
|
||||||
|
QDialog(parent, Qt::WindowCloseButtonHint),
|
||||||
|
model(this)
|
||||||
|
{
|
||||||
|
ui.setupUi(this);
|
||||||
|
setWindowTitle(SELECT_PROCESS);
|
||||||
|
ui.label->setText(ATTACH_INFO);
|
||||||
|
ui.processList->setModel(&model);
|
||||||
|
|
||||||
|
QPixmap transparent(100, 100);
|
||||||
|
transparent.fill(QColor::fromRgba(0));
|
||||||
|
for (const auto& [process, icon] : processIcons)
|
||||||
|
{
|
||||||
|
auto item = new QStandardItem(icon ? QIcon(QtWin::fromHICON(icon)) : transparent, process);
|
||||||
|
item->setEditable(false);
|
||||||
|
model.appendRow(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
connect(ui.buttonBox, &QDialogButtonBox::accepted, this, &QDialog::accept);
|
||||||
|
connect(ui.buttonBox, &QDialogButtonBox::rejected, this, &QDialog::reject);
|
||||||
|
connect(ui.processList, &QListView::clicked, [this](QModelIndex index) { ui.processEdit->setText(model.item(index.row())->text()); });
|
||||||
|
connect(ui.processList, &QListView::doubleClicked, this, &QDialog::accept);
|
||||||
|
connect(ui.processEdit, &QLineEdit::textEdited, [this](QString process)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < model.rowCount(); ++i) ui.processList->setRowHidden(i, !model.item(i)->text().contains(process, Qt::CaseInsensitive));
|
||||||
|
});
|
||||||
|
connect(ui.processEdit, &QLineEdit::returnPressed, this, &QDialog::accept);
|
||||||
|
}
|
||||||
|
|
||||||
|
QString AttachProcessDialog::SelectedProcess()
|
||||||
|
{
|
||||||
|
return ui.processEdit->text();
|
||||||
|
}
|
17
GUI/attachprocessdialog.h
Normal file
17
GUI/attachprocessdialog.h
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "qtcommon.h"
|
||||||
|
#include "ui_attachprocessdialog.h"
|
||||||
|
#include <QStandardItemModel>
|
||||||
|
|
||||||
|
class AttachProcessDialog : public QDialog
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit AttachProcessDialog(QWidget* parent, std::vector<std::pair<QString, HICON>> processIcons);
|
||||||
|
QString SelectedProcess();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Ui::AttachProcessDialog ui;
|
||||||
|
QStandardItemModel model;
|
||||||
|
QString selectedProcess;
|
||||||
|
};
|
37
GUI/attachprocessdialog.ui
Normal file
37
GUI/attachprocessdialog.ui
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>AttachProcessDialog</class>
|
||||||
|
<widget class="QWidget" name="AttachProcessDialog">
|
||||||
|
<property name="windowModality">
|
||||||
|
<enum>Qt::WindowModal</enum>
|
||||||
|
</property>
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>800</width>
|
||||||
|
<height>400</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<layout class="QVBoxLayout">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="label"/>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QLineEdit" name="processEdit"/>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QListView" name="processList"/>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QDialogButtonBox" name="buttonBox">
|
||||||
|
<property name="standardButtons">
|
||||||
|
<set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
<resources/>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
@ -5,7 +5,7 @@ namespace
|
|||||||
{
|
{
|
||||||
char* GetCppExceptionInfo(EXCEPTION_POINTERS* exception)
|
char* GetCppExceptionInfo(EXCEPTION_POINTERS* exception)
|
||||||
{
|
{
|
||||||
// See https://blogs.msdn.microsoft.com/oldnewthing/20100730-00/?p=13273
|
// https://blogs.msdn.microsoft.com/oldnewthing/20100730-00/?p=13273
|
||||||
// Not very reliable so use __try
|
// Not very reliable so use __try
|
||||||
__try { return ((char****)exception->ExceptionRecord->ExceptionInformation[2])[3][1][1] + 8; }
|
__try { return ((char****)exception->ExceptionRecord->ExceptionInformation[2])[3][1][1] + 8; }
|
||||||
__except (EXCEPTION_EXECUTE_HANDLER) { return "Could not find"; }
|
__except (EXCEPTION_EXECUTE_HANDLER) { return "Could not find"; }
|
||||||
@ -21,12 +21,11 @@ namespace
|
|||||||
|
|
||||||
__declspec(noreturn) void Terminate()
|
__declspec(noreturn) void Terminate()
|
||||||
{
|
{
|
||||||
CreateThread(nullptr, 0, [](void* lastError) -> DWORD
|
WaitForSingleObject(CreateThread(nullptr, 0, [](void* lastError) -> DWORD
|
||||||
{
|
{
|
||||||
MessageBoxW(NULL, (wchar_t*)lastError, L"Textractor ERROR", MB_ICONERROR); // might fail to display if called in main thread and exception was in main event loop
|
MessageBoxW(NULL, (wchar_t*)lastError, L"Textractor ERROR", MB_ICONERROR); // might fail to display if called in main thread and exception was in main event loop
|
||||||
abort();
|
abort();
|
||||||
}, lastError.data(), 0, nullptr);
|
}, lastError.data(), 0, nullptr), INFINITE);
|
||||||
Sleep(MAXDWORD);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
LONG WINAPI ExceptionLogger(EXCEPTION_POINTERS* exception)
|
LONG WINAPI ExceptionLogger(EXCEPTION_POINTERS* exception)
|
@ -1,6 +1,5 @@
|
|||||||
#include "extenwindow.h"
|
#include "extenwindow.h"
|
||||||
#include "ui_extenwindow.h"
|
#include "ui_extenwindow.h"
|
||||||
#include <concrt.h>
|
|
||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
#include <QDragEnterEvent>
|
#include <QDragEnterEvent>
|
||||||
@ -10,6 +9,7 @@
|
|||||||
|
|
||||||
extern const char* EXTENSIONS;
|
extern const char* EXTENSIONS;
|
||||||
extern const char* ADD_EXTENSION;
|
extern const char* ADD_EXTENSION;
|
||||||
|
extern const char* REMOVE_EXTENSION;
|
||||||
extern const char* INVALID_EXTENSION;
|
extern const char* INVALID_EXTENSION;
|
||||||
extern const char* CONFIRM_EXTENSION_OVERWRITE;
|
extern const char* CONFIRM_EXTENSION_OVERWRITE;
|
||||||
extern const char* EXTENSION_WRITE_ERROR;
|
extern const char* EXTENSION_WRITE_ERROR;
|
||||||
@ -18,7 +18,7 @@ extern const char* EXTEN_WINDOW_INSTRUCTIONS;
|
|||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
constexpr auto EXTEN_SAVE_FILE = u8"SavedExtensions.txt";
|
constexpr auto EXTEN_SAVE_FILE = u8"SavedExtensions.txt";
|
||||||
constexpr auto DEFAULT_EXTENSIONS = u8"Remove Repeated Characters>Remove Repeated Phrases>Regex Filter>Copy to Clipboard>Bing Translate>Extra Window>Extra Newlines>Styler";
|
constexpr auto DEFAULT_EXTENSIONS = u8"Remove Repeated Characters>Regex Filter>Copy to Clipboard>Google Translate>Extra Window>Extra Newlines";
|
||||||
|
|
||||||
struct Extension
|
struct Extension
|
||||||
{
|
{
|
||||||
@ -43,7 +43,7 @@ namespace
|
|||||||
{
|
{
|
||||||
if (auto callback = (decltype(Extension::callback))GetProcAddress(module, "OnNewSentence"))
|
if (auto callback = (decltype(Extension::callback))GetProcAddress(module, "OnNewSentence"))
|
||||||
{
|
{
|
||||||
std::scoped_lock writeLock(extenMutex);
|
std::scoped_lock lock(extenMutex);
|
||||||
extensions.push_back({ S(extenName), callback });
|
extensions.push_back({ S(extenName), callback });
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -55,14 +55,14 @@ namespace
|
|||||||
|
|
||||||
void Unload(int index)
|
void Unload(int index)
|
||||||
{
|
{
|
||||||
std::scoped_lock writeLock(extenMutex);
|
std::scoped_lock lock(extenMutex);
|
||||||
FreeLibrary(GetModuleHandleW((extensions.at(index).name + L".xdll").c_str()));
|
FreeLibrary(GetModuleHandleW((extensions.at(index).name + L".xdll").c_str()));
|
||||||
extensions.erase(extensions.begin() + index);
|
extensions.erase(extensions.begin() + index);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Reorder(QStringList extenNames)
|
void Reorder(QStringList extenNames)
|
||||||
{
|
{
|
||||||
std::scoped_lock writeLock(extenMutex);
|
std::scoped_lock lock(extenMutex);
|
||||||
std::vector<Extension> extensions;
|
std::vector<Extension> extensions;
|
||||||
for (auto extenName : extenNames)
|
for (auto extenName : extenNames)
|
||||||
extensions.push_back(*std::find_if(::extensions.begin(), ::extensions.end(), [&](Extension extension) { return extension.name == S(extenName); }));
|
extensions.push_back(*std::find_if(::extensions.begin(), ::extensions.end(), [&](Extension extension) { return extension.name == S(extenName); }));
|
||||||
@ -95,11 +95,21 @@ namespace
|
|||||||
QMessageBox::information(This, EXTENSIONS, QString(INVALID_EXTENSION).arg(extenFile.fileName()));
|
QMessageBox::information(This, EXTENSIONS, QString(INVALID_EXTENSION).arg(extenFile.fileName()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenMenu(QPoint point)
|
void Delete()
|
||||||
{
|
{
|
||||||
QAction addExtension(ADD_EXTENSION);
|
if (ui.extenList->currentItem())
|
||||||
if (QMenu::exec({ &addExtension }, ui.extenList->mapToGlobal(point), nullptr, This))
|
{
|
||||||
if (QString extenFile = QFileDialog::getOpenFileName(This, ADD_EXTENSION, ".", EXTENSIONS + QString(" (*.xdll)\nLibraries (*.dll)")); !extenFile.isEmpty()) Add(extenFile);
|
Unload(ui.extenList->currentIndex().row());
|
||||||
|
Sync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ContextMenu(QPoint point)
|
||||||
|
{
|
||||||
|
QAction addExtension(ADD_EXTENSION), removeExtension(REMOVE_EXTENSION);
|
||||||
|
if (auto action = QMenu::exec({ &addExtension, &removeExtension }, ui.extenList->mapToGlobal(point), nullptr, This))
|
||||||
|
if (action == &removeExtension) Delete();
|
||||||
|
else if (QString extenFile = QFileDialog::getOpenFileName(This, ADD_EXTENSION, ".", EXTENSIONS + QString(" (*.xdll);;Libraries (*.dll)")); !extenFile.isEmpty()) Add(extenFile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -109,7 +119,7 @@ bool DispatchSentenceToExtensions(std::wstring& sentence, const InfoForExtension
|
|||||||
wcscpy_s(sentenceBuffer, sentence.size() + 1, sentence.c_str());
|
wcscpy_s(sentenceBuffer, sentence.size() + 1, sentence.c_str());
|
||||||
concurrency::reader_writer_lock::scoped_lock_read readLock(extenMutex);
|
concurrency::reader_writer_lock::scoped_lock_read readLock(extenMutex);
|
||||||
for (const auto& extension : extensions)
|
for (const auto& extension : extensions)
|
||||||
if (*(sentenceBuffer = extension.callback(sentenceBuffer, sentenceInfo)) == L'\0') break;
|
if (!*(sentenceBuffer = extension.callback(sentenceBuffer, sentenceInfo))) break;
|
||||||
sentence = sentenceBuffer;
|
sentence = sentenceBuffer;
|
||||||
HeapFree(GetProcessHeap(), 0, sentenceBuffer);
|
HeapFree(GetProcessHeap(), 0, sentenceBuffer);
|
||||||
return !sentence.empty();
|
return !sentence.empty();
|
||||||
@ -117,7 +127,7 @@ bool DispatchSentenceToExtensions(std::wstring& sentence, const InfoForExtension
|
|||||||
|
|
||||||
void CleanupExtensions()
|
void CleanupExtensions()
|
||||||
{
|
{
|
||||||
std::scoped_lock writeLock(extenMutex);
|
std::scoped_lock lock(extenMutex);
|
||||||
for (auto extension : extensions) FreeLibrary(GetModuleHandleW((extension.name + L".xdll").c_str()));
|
for (auto extension : extensions) FreeLibrary(GetModuleHandleW((extension.name + L".xdll").c_str()));
|
||||||
extensions.clear();
|
extensions.clear();
|
||||||
}
|
}
|
||||||
@ -129,7 +139,7 @@ ExtenWindow::ExtenWindow(QWidget* parent) : QMainWindow(parent, Qt::WindowCloseB
|
|||||||
ui.vboxLayout->addWidget(new QLabel(EXTEN_WINDOW_INSTRUCTIONS, this));
|
ui.vboxLayout->addWidget(new QLabel(EXTEN_WINDOW_INSTRUCTIONS, this));
|
||||||
setWindowTitle(EXTENSIONS);
|
setWindowTitle(EXTENSIONS);
|
||||||
|
|
||||||
connect(ui.extenList, &QListWidget::customContextMenuRequested, OpenMenu);
|
connect(ui.extenList, &QListWidget::customContextMenuRequested, ContextMenu);
|
||||||
ui.extenList->installEventFilter(this);
|
ui.extenList->installEventFilter(this);
|
||||||
|
|
||||||
if (!QFile::exists(EXTEN_SAVE_FILE)) QTextFile(EXTEN_SAVE_FILE, QIODevice::WriteOnly).write(DEFAULT_EXTENSIONS);
|
if (!QFile::exists(EXTEN_SAVE_FILE)) QTextFile(EXTEN_SAVE_FILE, QIODevice::WriteOnly).write(DEFAULT_EXTENSIONS);
|
||||||
@ -139,7 +149,7 @@ ExtenWindow::ExtenWindow(QWidget* parent) : QMainWindow(parent, Qt::WindowCloseB
|
|||||||
|
|
||||||
bool ExtenWindow::eventFilter(QObject* target, QEvent* event)
|
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
|
// 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)
|
if (event->type() == QEvent::ChildRemoved)
|
||||||
{
|
{
|
||||||
QStringList extenNames;
|
QStringList extenNames;
|
||||||
@ -152,11 +162,7 @@ bool ExtenWindow::eventFilter(QObject* target, QEvent* event)
|
|||||||
|
|
||||||
void ExtenWindow::keyPressEvent(QKeyEvent* event)
|
void ExtenWindow::keyPressEvent(QKeyEvent* event)
|
||||||
{
|
{
|
||||||
if (event->key() == Qt::Key_Delete && ui.extenList->currentItem())
|
if (event->key() == Qt::Key_Delete) Delete();
|
||||||
{
|
|
||||||
Unload(ui.extenList->currentIndex().row());
|
|
||||||
Sync();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExtenWindow::dragEnterEvent(QDragEnterEvent* event)
|
void ExtenWindow::dragEnterEvent(QDragEnterEvent* event)
|
||||||
|
@ -2,11 +2,6 @@
|
|||||||
|
|
||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
|
|
||||||
namespace Ui
|
|
||||||
{
|
|
||||||
class ExtenWindow;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct InfoForExtension
|
struct InfoForExtension
|
||||||
{
|
{
|
||||||
const char* name;
|
const char* name;
|
||||||
|
@ -3,8 +3,9 @@
|
|||||||
#include "defs.h"
|
#include "defs.h"
|
||||||
#include "module.h"
|
#include "module.h"
|
||||||
#include "extenwindow.h"
|
#include "extenwindow.h"
|
||||||
#include "host/host.h"
|
#include "../host/host.h"
|
||||||
#include "host/hookcode.h"
|
#include "../host/hookcode.h"
|
||||||
|
#include "attachprocessdialog.h"
|
||||||
#include <shellapi.h>
|
#include <shellapi.h>
|
||||||
#include <process.h>
|
#include <process.h>
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
@ -14,6 +15,7 @@
|
|||||||
#include <QDialogButtonBox>
|
#include <QDialogButtonBox>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
#include <QFontDialog>
|
#include <QFontDialog>
|
||||||
|
#include <QHash>
|
||||||
|
|
||||||
extern const char* ATTACH;
|
extern const char* ATTACH;
|
||||||
extern const char* LAUNCH;
|
extern const char* LAUNCH;
|
||||||
@ -28,13 +30,13 @@ extern const char* SETTINGS;
|
|||||||
extern const char* EXTENSIONS;
|
extern const char* EXTENSIONS;
|
||||||
extern const char* FONT;
|
extern const char* FONT;
|
||||||
extern const char* SELECT_PROCESS;
|
extern const char* SELECT_PROCESS;
|
||||||
extern const char* ATTACH_INFO;
|
|
||||||
extern const char* SELECT_PROCESS_INFO;
|
extern const char* SELECT_PROCESS_INFO;
|
||||||
extern const char* FROM_COMPUTER;
|
extern const char* FROM_COMPUTER;
|
||||||
extern const char* PROCESSES;
|
extern const char* PROCESSES;
|
||||||
extern const char* CODE_INFODUMP;
|
extern const char* CODE_INFODUMP;
|
||||||
extern const char* FAILED_TO_CREATE_CONFIG_FILE;
|
extern const char* FAILED_TO_CREATE_CONFIG_FILE;
|
||||||
extern const char* HOOK_SEARCH_UNSTABLE_WARNING;
|
extern const char* HOOK_SEARCH_UNSTABLE_WARNING;
|
||||||
|
extern const char* HOOK_SEARCH_STARTING_VIEW_CONSOLE;
|
||||||
extern const char* SEARCH_CJK;
|
extern const char* SEARCH_CJK;
|
||||||
extern const char* SEARCH_PATTERN;
|
extern const char* SEARCH_PATTERN;
|
||||||
extern const char* SEARCH_DURATION;
|
extern const char* SEARCH_DURATION;
|
||||||
@ -86,6 +88,8 @@ namespace
|
|||||||
TextThread* current = nullptr;
|
TextThread* current = nullptr;
|
||||||
MainWindow* This = nullptr;
|
MainWindow* This = nullptr;
|
||||||
|
|
||||||
|
void FindHooks();
|
||||||
|
|
||||||
QString TextThreadString(TextThread& thread)
|
QString TextThreadString(TextThread& thread)
|
||||||
{
|
{
|
||||||
return QString("%1:%2:%3:%4:%5: %6").arg(
|
return QString("%1:%2:%3:%4:%5: %6").arg(
|
||||||
@ -99,11 +103,11 @@ namespace
|
|||||||
|
|
||||||
ThreadParam ParseTextThreadString(QString ttString)
|
ThreadParam ParseTextThreadString(QString ttString)
|
||||||
{
|
{
|
||||||
QStringList threadParam = ttString.split(":");
|
auto threadParam = ttString.splitRef(":");
|
||||||
return { threadParam[1].toUInt(nullptr, 16), threadParam[2].toULongLong(nullptr, 16), threadParam[3].toULongLong(nullptr, 16), threadParam[4].toULongLong(nullptr, 16) };
|
return { threadParam[1].toUInt(nullptr, 16), threadParam[2].toULongLong(nullptr, 16), threadParam[3].toULongLong(nullptr, 16), threadParam[4].toULongLong(nullptr, 16) };
|
||||||
}
|
}
|
||||||
|
|
||||||
std::array<InfoForExtension, 10> GetSentenceInfo(TextThread& thread)
|
std::array<InfoForExtension, 20> GetSentenceInfo(TextThread& thread)
|
||||||
{
|
{
|
||||||
void (*AddText)(int64_t, const wchar_t*) = [](int64_t number, const wchar_t* text)
|
void (*AddText)(int64_t, const wchar_t*) = [](int64_t number, const wchar_t* text)
|
||||||
{
|
{
|
||||||
@ -124,6 +128,9 @@ namespace
|
|||||||
{ "hook address", (int64_t)thread.tp.addr },
|
{ "hook address", (int64_t)thread.tp.addr },
|
||||||
{ "text handle", thread.handle },
|
{ "text handle", thread.handle },
|
||||||
{ "text name", (int64_t)thread.name.c_str() },
|
{ "text name", (int64_t)thread.name.c_str() },
|
||||||
|
{ "add sentence", (int64_t)AddSentence },
|
||||||
|
{ "add text", (int64_t)AddText },
|
||||||
|
{ "get selected process id", (int64_t)GetSelectedProcessId },
|
||||||
{ "void (*AddSentence)(int64_t number, const wchar_t* sentence)", (int64_t)AddSentence },
|
{ "void (*AddSentence)(int64_t number, const wchar_t* sentence)", (int64_t)AddSentence },
|
||||||
{ "void (*AddText)(int64_t number, const wchar_t* text)", (int64_t)AddText },
|
{ "void (*AddText)(int64_t number, const wchar_t* text)", (int64_t)AddText },
|
||||||
{ "DWORD (*GetSelectedProcessId)()", (int64_t)GetSelectedProcessId },
|
{ "DWORD (*GetSelectedProcessId)()", (int64_t)GetSelectedProcessId },
|
||||||
@ -131,6 +138,21 @@ namespace
|
|||||||
} };
|
} };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void AttachSavedProcesses()
|
||||||
|
{
|
||||||
|
std::unordered_set<std::wstring> attachTargets;
|
||||||
|
if (autoAttach)
|
||||||
|
for (auto process : QString(QTextFile(GAME_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts))
|
||||||
|
attachTargets.insert(S(process));
|
||||||
|
if (autoAttachSavedOnly)
|
||||||
|
for (auto process : QString(QTextFile(HOOK_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts))
|
||||||
|
attachTargets.insert(S(process.split(" , ")[0]));
|
||||||
|
|
||||||
|
if (!attachTargets.empty())
|
||||||
|
for (auto [processId, processName] : GetAllProcesses())
|
||||||
|
if (processName && attachTargets.count(processName.value()) > 0 && alreadyAttached.count(processId) == 0) Host::InjectProcess(processId);
|
||||||
|
}
|
||||||
|
|
||||||
std::optional<std::wstring> UserSelectedProcess()
|
std::optional<std::wstring> UserSelectedProcess()
|
||||||
{
|
{
|
||||||
QStringList savedProcesses = QString::fromUtf8(QTextFile(GAME_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts);
|
QStringList savedProcesses = QString::fromUtf8(QTextFile(GAME_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts);
|
||||||
@ -138,23 +160,45 @@ namespace
|
|||||||
savedProcesses.removeDuplicates();
|
savedProcesses.removeDuplicates();
|
||||||
savedProcesses.insert(1, FROM_COMPUTER);
|
savedProcesses.insert(1, FROM_COMPUTER);
|
||||||
QString process = QInputDialog::getItem(This, SELECT_PROCESS, SELECT_PROCESS_INFO, savedProcesses, 0, true, &ok, Qt::WindowCloseButtonHint);
|
QString process = QInputDialog::getItem(This, SELECT_PROCESS, SELECT_PROCESS_INFO, savedProcesses, 0, true, &ok, Qt::WindowCloseButtonHint);
|
||||||
if (process == FROM_COMPUTER) process = QDir::toNativeSeparators(QFileDialog::getOpenFileName(This, SELECT_PROCESS, "C:\\", PROCESSES));
|
if (process == FROM_COMPUTER) process = QDir::toNativeSeparators(QFileDialog::getOpenFileName(This, SELECT_PROCESS, "/", PROCESSES));
|
||||||
if (ok && process.contains('\\')) return S(process);
|
if (ok && process.contains('\\')) return S(process);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ViewThread(int index)
|
||||||
|
{
|
||||||
|
ui.ttCombo->setCurrentIndex(index);
|
||||||
|
ui.textOutput->setPlainText(sanitize(S((current = &Host::GetThread(ParseTextThreadString(ui.ttCombo->itemText(index))))->storage->c_str())));
|
||||||
|
ui.textOutput->moveCursor(QTextCursor::End);
|
||||||
|
}
|
||||||
|
|
||||||
void AttachProcess()
|
void AttachProcess()
|
||||||
{
|
{
|
||||||
QMultiHash<QString, DWORD> allProcesses;
|
QMultiHash<QString, DWORD> processesMap;
|
||||||
|
std::vector<std::pair<QString, HICON>> processIcons;
|
||||||
for (auto [processId, processName] : GetAllProcesses())
|
for (auto [processId, processName] : GetAllProcesses())
|
||||||
|
{
|
||||||
if (processName && (showSystemProcesses || processName->find(L":\\Windows\\") == std::string::npos))
|
if (processName && (showSystemProcesses || processName->find(L":\\Windows\\") == std::string::npos))
|
||||||
allProcesses.insert(QFileInfo(S(processName.value())).fileName(), processId);
|
{
|
||||||
|
QString fileName = QFileInfo(S(processName.value())).fileName();
|
||||||
|
if (!processesMap.contains(fileName))
|
||||||
|
{
|
||||||
|
HICON bigIcon, smallIcon;
|
||||||
|
ExtractIconExW(processName->c_str(), 0, &bigIcon, &smallIcon, 1);
|
||||||
|
processIcons.push_back({ fileName, bigIcon ? bigIcon : smallIcon });
|
||||||
|
}
|
||||||
|
processesMap.insert(fileName, processId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
std::sort(processIcons.begin(), processIcons.end(), [](auto one, auto two) { return QString::compare(one.first, two.first, Qt::CaseInsensitive) < 0; });
|
||||||
|
|
||||||
QStringList processList(allProcesses.uniqueKeys());
|
AttachProcessDialog attachProcessDialog(This, processIcons);
|
||||||
processList.sort(Qt::CaseInsensitive);
|
if (attachProcessDialog.exec())
|
||||||
if (QString process = QInputDialog::getItem(This, SELECT_PROCESS, ATTACH_INFO, processList, 0, true, &ok, Qt::WindowCloseButtonHint); ok)
|
{
|
||||||
if (process.toInt(nullptr, 0)) Host::InjectProcess(process.toInt(nullptr, 0));
|
QString process = attachProcessDialog.SelectedProcess();
|
||||||
else for (auto processId : allProcesses.values(process)) Host::InjectProcess(processId);
|
if (int processId = process.toInt(nullptr, 0)) Host::InjectProcess(processId);
|
||||||
|
else for (int processId : processesMap.values(process)) Host::InjectProcess(processId);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void LaunchProcess()
|
void LaunchProcess()
|
||||||
@ -170,7 +214,7 @@ namespace
|
|||||||
{
|
{
|
||||||
if (HMODULE localeEmulator = LoadLibraryW(L"LoaderDll"))
|
if (HMODULE localeEmulator = LoadLibraryW(L"LoaderDll"))
|
||||||
{
|
{
|
||||||
// see https://github.com/xupefei/Locale-Emulator/blob/aa99dec3b25708e676c90acf5fed9beaac319160/LEProc/LoaderWrapper.cs#L252
|
// https://github.com/xupefei/Locale-Emulator/blob/aa99dec3b25708e676c90acf5fed9beaac319160/LEProc/LoaderWrapper.cs#L252
|
||||||
struct
|
struct
|
||||||
{
|
{
|
||||||
ULONG AnsiCodePage = SHIFT_JIS;
|
ULONG AnsiCodePage = SHIFT_JIS;
|
||||||
@ -229,12 +273,10 @@ namespace
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FindHooks();
|
|
||||||
|
|
||||||
void AddHook(QString hook)
|
void AddHook(QString hook)
|
||||||
{
|
{
|
||||||
if (QString hookCode = QInputDialog::getText(This, ADD_HOOK, CODE_INFODUMP, QLineEdit::Normal, hook, &ok, Qt::WindowCloseButtonHint); ok)
|
if (QString hookCode = QInputDialog::getText(This, ADD_HOOK, CODE_INFODUMP, QLineEdit::Normal, hook, &ok, Qt::WindowCloseButtonHint); ok)
|
||||||
if (hookCode.startsWith("S") || hookCode.startsWith("/S")) FindHooks();
|
if (hookCode.startsWith("S") || hookCode.startsWith("/S")) FindHooks(); // backwards compatibility for old hook search UX
|
||||||
else if (auto hp = HookCode::Parse(S(hookCode))) try { Host::InsertHook(selectedProcessId, hp.value()); } catch (std::out_of_range) {}
|
else if (auto hp = HookCode::Parse(S(hookCode))) try { Host::InsertHook(selectedProcessId, hp.value()); } catch (std::out_of_range) {}
|
||||||
else Host::AddConsoleOutput(INVALID_CODE);
|
else Host::AddConsoleOutput(INVALID_CODE);
|
||||||
}
|
}
|
||||||
@ -334,7 +376,11 @@ namespace
|
|||||||
layout.addRow(&confirm);
|
layout.addRow(&confirm);
|
||||||
if (!dialog.exec()) return;
|
if (!dialog.exec()) return;
|
||||||
wcsncpy_s(sp.text, S(textEdit.text()).c_str(), PATTERN_SIZE - 1);
|
wcsncpy_s(sp.text, S(textEdit.text()).c_str(), PATTERN_SIZE - 1);
|
||||||
try { Host::FindHooks(selectedProcessId, sp); } catch (std::out_of_range) {}
|
try
|
||||||
|
{
|
||||||
|
Host::FindHooks(selectedProcessId, sp);
|
||||||
|
ViewThread(0);
|
||||||
|
} catch (std::out_of_range) {}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -404,6 +450,7 @@ namespace
|
|||||||
[hooks, filter](HookParam hp, std::wstring text) { if (filter.match(S(text)).hasMatch()) *hooks << sanitize(S(HookCode::Generate(hp) + L" => " + text)); });
|
[hooks, filter](HookParam hp, std::wstring text) { if (filter.match(S(text)).hasMatch()) *hooks << sanitize(S(HookCode::Generate(hp) + L" => " + text)); });
|
||||||
}
|
}
|
||||||
catch (std::out_of_range) { return; }
|
catch (std::out_of_range) { return; }
|
||||||
|
ViewThread(0);
|
||||||
std::thread([hooks]
|
std::thread([hooks]
|
||||||
{
|
{
|
||||||
for (int lastSize = 0; hooks->size() == 0 || hooks->size() != lastSize; Sleep(2000)) lastSize = hooks->size();
|
for (int lastSize = 0; hooks->size() == 0 || hooks->size() != lastSize; Sleep(2000)) lastSize = hooks->size();
|
||||||
@ -434,6 +481,7 @@ namespace
|
|||||||
}
|
}
|
||||||
hooks->clear();
|
hooks->clear();
|
||||||
}).detach();
|
}).detach();
|
||||||
|
QMessageBox::information(This, SEARCH_FOR_HOOKS, HOOK_SEARCH_STARTING_VIEW_CONSOLE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenSettings()
|
void OpenSettings()
|
||||||
@ -485,13 +533,6 @@ namespace
|
|||||||
extenWindow->showNormal();
|
extenWindow->showNormal();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ViewThread(int index)
|
|
||||||
{
|
|
||||||
ui.ttCombo->setCurrentIndex(index);
|
|
||||||
ui.textOutput->setPlainText(sanitize(S((current = &Host::GetThread(ParseTextThreadString(ui.ttCombo->itemText(index))))->storage->c_str())));
|
|
||||||
ui.textOutput->moveCursor(QTextCursor::End);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetOutputFont(QString fontString)
|
void SetOutputFont(QString fontString)
|
||||||
{
|
{
|
||||||
QFont font = ui.textOutput->font();
|
QFont font = ui.textOutput->font();
|
||||||
@ -613,10 +654,7 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
|
|||||||
}
|
}
|
||||||
ui.processLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Minimum, QSizePolicy::Expanding));
|
ui.processLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Minimum, QSizePolicy::Expanding));
|
||||||
|
|
||||||
connect(ui.processCombo, qOverload<const QString&>(&QComboBox::currentIndexChanged), [](QString process)
|
connect(ui.processCombo, qOverload<int>(&QComboBox::currentIndexChanged), [] { selectedProcessId = ui.processCombo->currentText().split(":")[0].toULong(nullptr, 16); });
|
||||||
{
|
|
||||||
selectedProcessId = ui.processCombo->currentText().split(":")[0].toULong(nullptr, 16);
|
|
||||||
});
|
|
||||||
connect(ui.ttCombo, qOverload<int>(&QComboBox::activated), this, ViewThread);
|
connect(ui.ttCombo, qOverload<int>(&QComboBox::activated), this, ViewThread);
|
||||||
connect(ui.textOutput, &QPlainTextEdit::selectionChanged, this, CopyUnlessMouseDown);
|
connect(ui.textOutput, &QPlainTextEdit::selectionChanged, this, CopyUnlessMouseDown);
|
||||||
connect(ui.textOutput, &QPlainTextEdit::customContextMenuRequested, this, OutputContextMenu);
|
connect(ui.textOutput, &QPlainTextEdit::customContextMenuRequested, this, OutputContextMenu);
|
||||||
@ -649,23 +687,7 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
|
|||||||
else for (auto [processId, processName] : processes)
|
else for (auto [processId, processName] : processes)
|
||||||
if (processName.value_or(L"").find(L"\\" + arg.substr(2)) != std::string::npos) Host::InjectProcess(processId);
|
if (processName.value_or(L"").find(L"\\" + arg.substr(2)) != std::string::npos) Host::InjectProcess(processId);
|
||||||
|
|
||||||
std::thread([]
|
std::thread([] { for (; ; Sleep(10000)) AttachSavedProcesses(); }).detach();
|
||||||
{
|
|
||||||
for (; ; Sleep(10000))
|
|
||||||
{
|
|
||||||
std::unordered_set<std::wstring> attachTargets;
|
|
||||||
if (autoAttach)
|
|
||||||
for (auto process : QString(QTextFile(GAME_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts))
|
|
||||||
attachTargets.insert(S(process));
|
|
||||||
if (autoAttachSavedOnly)
|
|
||||||
for (auto process : QString(QTextFile(HOOK_SAVE_FILE, QIODevice::ReadOnly).readAll()).split("\n", QString::SkipEmptyParts))
|
|
||||||
attachTargets.insert(S(process.split(" , ")[0]));
|
|
||||||
|
|
||||||
if (!attachTargets.empty())
|
|
||||||
for (auto [processId, processName] : GetAllProcesses())
|
|
||||||
if (processName && attachTargets.count(processName.value()) > 0 && alreadyAttached.count(processId) == 0) Host::InjectProcess(processId);
|
|
||||||
}
|
|
||||||
}).detach();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MainWindow::~MainWindow()
|
MainWindow::~MainWindow()
|
||||||
|
@ -77,7 +77,7 @@
|
|||||||
<widget class="QPlainTextEdit" name="textOutput">
|
<widget class="QPlainTextEdit" name="textOutput">
|
||||||
<property name="font">
|
<property name="font">
|
||||||
<font>
|
<font>
|
||||||
<family>Meiryo UI</family>
|
<family>Arial Unicode MS</family>
|
||||||
<pointsize>13</pointsize>
|
<pointsize>13</pointsize>
|
||||||
</font>
|
</font>
|
||||||
</property>
|
</property>
|
||||||
|
BIN
INSTALL_THIS_UNICODE_FONT.ttf
Normal file
BIN
INSTALL_THIS_UNICODE_FONT.ttf
Normal file
Binary file not shown.
31
README.md
31
README.md
@ -2,28 +2,28 @@
|
|||||||
|
|
||||||
![How it looks](screenshot.png)
|
![How it looks](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a. NextHooker) is an open-source x86/x64 video game text hooker for Windows/Wine based off of [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (a.k.a. NextHooker) is an open-source x86/x64 video game text hooker for Windows 7+ (and Wine) based off of [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Watch the [tutorial video](https://tinyurl.com/textractor-tutorial) for a quick rundown on using it.
|
Watch the [tutorial video](docs/TUTORIAL.md) for a quick rundown on using it.
|
||||||
|
|
||||||
## Download
|
## Download
|
||||||
|
|
||||||
Releases of Textractor can be found [here](https://github.com/Artikash/Textractor/releases).<br>
|
Official stable releases of Textractor can be found [here](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
The last release of ITHVNR can be found [here](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
The last release of ITHVNR can be found [here](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
||||||
Try running vcredist if you get an error when starting Textractor.
|
Experimental builds of Textractor (with debug info) from the latest source can be found [here](https://ci.appveyor.com/project/Artikash/textractor/history) in the 'Artifacts' section of each job.
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Highly extensible and customizable
|
- Highly extensible and customizable
|
||||||
- Auto hook many game engines (including some not supported by VNR!)
|
- Auto hook many game engines (including some not supported by VNR!)
|
||||||
- Hook text using /H "hook" codes (most AGTH codes supported)
|
- Hook text using /H "hook" codes (most AGTH codes supported)
|
||||||
- Directly extract text using /R "read" codes
|
- Automatically search for possible hook codes
|
||||||
|
|
||||||
## Support
|
## Support
|
||||||
|
|
||||||
Please let me know of any bugs, games that Textractor has trouble hooking, feature requests, or other suggestions.<br>
|
Let me know of any bugs, games that Textractor has trouble hooking, feature requests, or other suggestions by posting an issue.<br>
|
||||||
If you have trouble hooking a game please email me a place where I can freely download it, or gift it to me on [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
If you have trouble hooking a game, please show me a way to freely download it or gift it to me on [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
||||||
|
|
||||||
## Extensions
|
## Extensions
|
||||||
|
|
||||||
@ -32,22 +32,21 @@ See the extensions folder for examples of what extensions can do.
|
|||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
All contributions are appreciated! Please email (no, I'm not busy!) me at akashmozumdar@gmail.com if you have any questions about the codebase.<br>
|
All contributions are appreciated! Please email me at akashmozumdar@gmail.com if you have any questions about the codebase.<br>
|
||||||
You should use the standard process of making a pull request (fork, branch, commit changes, make PR from your branch to my master).<br>
|
You should use the standard process of making a pull request (fork, branch, commit changes, make PR from your branch to my master).<br>
|
||||||
Contributing a translation is easy: just translate the strings in text.cpp as well as this README.
|
Contributing a translation is easy: [text.cpp](text.cpp) contains all of the text strings that you need to translate. Translations of this README or the tutorial video transcript are also welcome.
|
||||||
|
|
||||||
## Compiling
|
## Compiling
|
||||||
|
Before compiling Textractor, you need Qt version 5.13 and Visual Studio with CMake support.
|
||||||
Before compiling *Textractor*, you should get Visual Studio with CMake support, as well as Qt version 5.13<br>
|
Clone Textractor's source and initialize submodules with `git clone https://github.com/Artikash/Textractor.git` and `git submodule update --init`.
|
||||||
You should then be able to simply open the folder in Visual Studio, and build. Run Textractor.exe.
|
You should then be able to just open the source folder in Visual Studio and build.
|
||||||
|
|
||||||
## Project Architecture
|
## Project Architecture
|
||||||
|
|
||||||
The host (see GUI/host folder) injects texthook.dll (created from the texthook folder) into the target process and connects to it via 2 pipe files.<br>
|
The host injects texthook into the target process and connects to it via 2 pipe files.
|
||||||
Host writes to hostPipe, texthook writes to hookPipe.<br>
|
|
||||||
texthook waits for the pipe to be connected, then injects a few instructions into any text outputting functions (e.g. TextOut, GetGlyphOutline) that cause their input to be sent through the pipe.<br>
|
texthook waits for the pipe to be connected, then injects a few instructions into any text outputting functions (e.g. TextOut, GetGlyphOutline) that cause their input to be sent through the pipe.<br>
|
||||||
Additional information about hooks is exchanged via shared memory.<br>
|
Additional information about hooks is exchanged via shared memory.<br>
|
||||||
The text that the host receives through the pipe is then processed a little before being dispatched back to the GUI.<br>
|
The text that the host receives through the pipe is then processed a little before being dispatched back to the GUI.<br>
|
||||||
Finally, the GUI dispatches the text to extensions before displaying it.
|
Finally, the GUI dispatches the text to extensions before displaying it.
|
||||||
|
|
||||||
## [Developers](CREDITS.md)
|
## [Developers](docs/CREDITS.md)
|
||||||
|
52
README_DE.md
Normal file
52
README_DE.md
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
# Textractor
|
||||||
|
|
||||||
|
![Wie es aussieht](screenshot.png)
|
||||||
|
|
||||||
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
|
**Textractor** (a.b.a. NextHooker) ist ein open-source x86/x64 Video spiel Text hooker für Windows 7+ (und Wine) basierend auf [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
|
Schau das [Tutorial Video](docs/TUTORIAL.md) (auf Englisch) an für einen schnellen Überblick wie du Textractor verwendest.
|
||||||
|
## Download
|
||||||
|
|
||||||
|
Der offizielle Release ist [hier](https://github.com/Artikash/Textractor/releases) zu finden.<br>
|
||||||
|
Der letzte Release von ITHVNR ist [hier](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
||||||
|
Der experimentelle Release von Textractor (mit debug Informationen) ist [hier](https://ci.appveyor.com/project/Artikash/textractor/history) in der 'Artifacts'
|
||||||
|
Kategorie des jeweiligen Jobs.
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
- Modular und Anpassbar
|
||||||
|
- Automatischen 'hooken' von mehreren Engines (einige davon welche keine VNR Support haben!)
|
||||||
|
- Text 'hooken' mithilfe von /H "hook" Codes (die meisten AGTH codes funktionieren)
|
||||||
|
- Automatische suche nach funktionierenden Hook's
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
Wenn ihr irgendwelche Fehler, Spiele bei denen Textractor nicht funktioniert, oder Fragen/Anmerkungen habt lasst es mich bitte wissen.<br>
|
||||||
|
Falls ihr Probleme mit einem Spiel habt, schickt mir einen kostenlosen download Link von dem Spiel oder schenkt es mir auf [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
||||||
|
|
||||||
|
## Erweiterungen
|
||||||
|
|
||||||
|
Siehe [Example Extension project](https://github.com/Artikash/ExampleExtension) für Anleitungen, wie man eine Erweiterung erstellt.<br>
|
||||||
|
Im 'Extensions' Ordner sind Beispiele für Erweiterungen.
|
||||||
|
|
||||||
|
## Unterstützen
|
||||||
|
|
||||||
|
Ich bin dankbar für alle Unterstützungen! Schickt mir eine E-Mail an akashmozumdar@gmail.com falls ihr Fragen zur Codebasis habt.<br>
|
||||||
|
Verwendet bitte als Standard eine pull Request (fork, branch, commit) zum Master Release.<br>
|
||||||
|
Zu Übersetzungen beizutragen ist einfach: [text.cpp](text.cpp) enthält alle text Strings welche übersetzt werden sollen. Übersetzungen der README oder des Tutorial Video Transkripts sind ebenfalls willkommen.
|
||||||
|
|
||||||
|
## Compiling
|
||||||
|
Zum Compilen braucht ihr Qt Version 5.13 und Visual Studio mit CMake Unterstützung.
|
||||||
|
Erstellt einen Clone vom Quellcode und initialisiert die submodule mit 'git clone https://github.com/Artikash/Textractor.git' und 'git submodule update --init'.
|
||||||
|
Ihr solltet danach in der Lage sein, den Quellordner in Visual Studio zu öffnen und anzufangen.
|
||||||
|
|
||||||
|
## Projekt Architektur
|
||||||
|
|
||||||
|
Der Host injiziert texthooks in den ziel Prozess und verbindet ihn mit 2 Pipe Dateien.
|
||||||
|
Texthook wartet auf die Pipe und injiziert Instruktionen für den Text Output (z.b. TextOut, GetGlyphOutline) welche durch die Pipe gesendet werden.<br>
|
||||||
|
Weitere Informationen werden durch geteilten Speicher ausgetauscht.<br>
|
||||||
|
Der Text, welchen der Host durch die Pipe erhält, wird dann verarbeitet, bevor er wieder an die GUI gesendet wird.<br>
|
||||||
|
Zu guter Letzt, sendet die GUI den Text an die Erweiterung, welche einen lesbaren Output anzeigt.
|
||||||
|
|
||||||
|
## [Entwickler](docs/CREDITS.md)
|
@ -1,12 +1,12 @@
|
|||||||
# Textractor
|
# Textractor
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
## [Video tutorial](https://tinyurl.com/textractor-tutorial)
|
## [Video tutorial](docs/TUTORIAL.md)
|
||||||
|
|
||||||
## Descripción general
|
## Descripción general
|
||||||
|
|
||||||
**Textractor** (también conocido como NextHooker) es un extractor y traductor de texto de código abierto x86/x64 para Windows/Wine bassado en [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (también conocido como NextHooker) es un extractor y traductor de texto de código abierto x86/x64 para Windows/Wine bassado en [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
|
|
||||||
![Cómo se ve](screenshot.png)
|
![Cómo se ve](screenshot.png)
|
||||||
|
|
||||||
|
15
README_FR.md
15
README_FR.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![À quoi ça ressemble](screenshot.png)
|
![À quoi ça ressemble](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a. NextHooker) est un traducteur de jeux-videos basé surtout sur du texte en open source x86/x64 pour Windows/Wine [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (a.k.a. NextHooker) est un traducteur de jeux-videos basé surtout sur du texte en open source x86/x64 pour Windows/Wine [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Regarde le [tutorial video](https://tinyurl.com/textractor-tutorial) pour un aperçu rapide de son utilisation.
|
Regarde le [tutorial video](docs/TUTORIAL.md) pour un aperçu rapide de son utilisation.
|
||||||
|
|
||||||
## Téléchargement
|
## Téléchargement
|
||||||
|
|
||||||
Les versions de Textractor peuvent être trouvées[here](https://github.com/Artikash/Textractor/releases).<br>
|
Les versions de Textractor peuvent être trouvées[here](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
Le denière version de THVNR peut etre trouvé [here](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
Le denière version de THVNR peut etre trouvé [here](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
Essayez de démarrer Vcredist si vous avez une erreur en lançant Textractor.
|
|
||||||
|
|
||||||
## Nouveautés
|
## Nouveautés
|
||||||
- Hautement extensible et personnalisable
|
- Hautement extensible et personnalisable
|
||||||
@ -31,9 +30,9 @@ Voir le dossier des extensions pour des exemples de ce que les extensions peuven
|
|||||||
|
|
||||||
## Contribution
|
## Contribution
|
||||||
|
|
||||||
Toutes les contributions sont appréciées! Veuillez m'envoyer un e-mail (non, je ne suis pas occupé!) À akashmozumdar@gmail.com si vous avez des questions sur la base de code. <br>
|
Toutes les contributions sont appréciées! Veuillez m'envoyer un e-mail À akashmozumdar@gmail.com si vous avez des questions sur la base de code. <br>
|
||||||
Vous devez utiliser le processus standard de création d'une demande d'extraction (fork, branch, commit changes, make PR from your branch to my master). <br>
|
Vous devez utiliser le processus standard de création d'une demande d'extraction (fork, branch, commit changes, make PR from your branch to my master). <br>
|
||||||
Contribuer à une traduction est simple: il suffit de traduire les chaînes dans text.cpp ainsi que ce fichier README.
|
Contribuer à une traduction est simple: il suffit de traduire les chaînes dans [text.cpp](text.cpp) ainsi que ce fichier README.
|
||||||
|
|
||||||
## Compilation
|
## Compilation
|
||||||
|
|
||||||
@ -49,4 +48,4 @@ Des informations supplémentaires sur les hooks sont échangées via la mémoire
|
|||||||
Le texte que l'hôte reçoit via le canal est ensuite traité un peu avant d'être renvoyé à l'interface graphique. <br>
|
Le texte que l'hôte reçoit via le canal est ensuite traité un peu avant d'être renvoyé à l'interface graphique. <br>
|
||||||
Enfin, l'interface graphique envoie le texte aux extensions avant de l'afficher.
|
Enfin, l'interface graphique envoie le texte aux extensions avant de l'afficher.
|
||||||
|
|
||||||
## [Développeurs](CREDITS.md)
|
## [Développeurs](docs/CREDITS.md)
|
38
README_ID.md
38
README_ID.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![How it looks](screenshot.png)
|
![How it looks](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a NextHooker) adalah text hooker video game untuk Windows/Wine x86/x64 berbasis open-source yang didasari oleh [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (a.k.a NextHooker) adalah teks hooker video game untuk Windows/Wine x86/x64 berbasis open-source yang didasari oleh [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Lihat [video tutorial](https://tinyurl.com/textractor-tutorial) untuk mengetahui bagaimana cara menggunakannya.
|
Lihat [video tutorial](docs/TUTORIAL.md) untuk mengetahui bagaimana cara menggunakannya.
|
||||||
|
|
||||||
## Pengunduhan
|
## Pengunduhan
|
||||||
|
|
||||||
Rilisan Textractor dapat diunduh [disini](https://github.com/Artikash/Textractor/releases).<br>
|
Rilisan Textractor dapat diunduh [disini](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
Rilisan Terakhir ITHVNR dapat diunduh [disini](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
Rilisan Terakhir ITHVNR dapat diunduh [disini](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
Coba jalankan vcredist.x86.exe jika kamu memiliki masalah ketika menjalankan Textractor.
|
|
||||||
|
|
||||||
## Fitur
|
## Fitur
|
||||||
|
|
||||||
@ -22,36 +21,33 @@ Coba jalankan vcredist.x86.exe jika kamu memiliki masalah ketika menjalankan Tex
|
|||||||
|
|
||||||
## Dukungan
|
## Dukungan
|
||||||
|
|
||||||
Please let me know of any bugs, games that Textractor has trouble hooking, feature requests, or other suggestions.<br>
|
Tolong beritahu saya jika kamu menemukan kutu, game yang tidak dapat di tempel oleh Textractor, permintaan fitur, atau usulan lain.<br>
|
||||||
If you have trouble hooking a game please email me a place where I can freely download it, or gift it to me on [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
Jika kamu memiliki masalah dalam menempelkan kedalam game tolong email saya link agar saya dapat mengunduh game tersebut, atau hadiahkan game tersebut di [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
||||||
|
|
||||||
Tolong beritahu saya jika kamu menemukan bug, game yang tidak dapat di tempel oleh Textractor, permintaan fitur, atau usulan lain.<br>
|
|
||||||
Jika kamu memiliki masalah dalam menempelkan kedalam game tolong email saya link agar saya dapat mendownload game tersebut, atau gift game tersebut di [Steam](https://steamcommunity.com/profiles/76561198097566313/).
|
|
||||||
|
|
||||||
## Ekstensi
|
## Ekstensi
|
||||||
|
|
||||||
Lihat [project sampel ekstensi saya](https://github.com/Artikash/ExampleExtension) untuk melihat bagaimana cara membuat ekstensi.<br>
|
Lihat [project sampel ekstensi saya](https://github.com/Artikash/ExampleExtension) untuk melihat bagaimana cara membuat ekstensi.<br>
|
||||||
Lihat folder extensions untuk melihat sampel ekstensi.
|
Lihat ekstensi folder untuk melihat sampel ekstensi.
|
||||||
|
|
||||||
## Kontribusi
|
## Kontribusi
|
||||||
|
|
||||||
Seluruh kontribusi diapresiasi! Tolong email (tidak, saya tidak sibuk!) saya di akashmozumdar@gmail.com jika kamu memiliki pertanyaan mengenai codebase nya.<br>
|
Seluruh kontribusi diapresiasi! Tolong email saya di akashmozumdar@gmail.com jika kamu memiliki pertanyaan mengenai kode dasar nya.<br>
|
||||||
Kamu harus menggunakan proses standar dalam membuat permintaan pull(fork, branch, commit changes, membuat PR dari branch kamu ke master saya).<br>
|
Kamu harus menggunakan proses standar dalam membuat permintaan pull(fork, cabang, perubahan commit, membuat PR dari cabang kamu ke master saya).<br>
|
||||||
Berkontribusi dalam penerjemahan dapat dilakukan dengan mudah : cukup terjemahkan string dari text.cpp lalu terjemahkan README ini.
|
Berkontribusi dalam penerjemahan dapat dilakukan dengan mudah : cukup terjemahkan string dari [text.cpp](text.cpp) lalu terjemahkan README ini.
|
||||||
|
|
||||||
## Compiling
|
## Mengcompile
|
||||||
|
|
||||||
Sebelum melakukan proses compile *Textractor*, kamu harus memiliki Visual Studio dengan dukungan Cmake, juga dengan Qt version 5.13<br>
|
Sebelum melakukan proses compile *Textractor*, kamu harus memiliki Visual Studio dengan dukungan Cmake, juga dengan Qt version 5.13<br>
|
||||||
Lalu kamu dapat membuka folder di Visual Studio, dan build. Run Textractor.exe.
|
Lalu kamu dapat membuka folder di Visual Studio, dan build. Jalankan Textractor.exe.
|
||||||
|
|
||||||
|
|
||||||
## Arsitektur Project
|
## Arsitektur Project
|
||||||
|
|
||||||
Host (lihat folder GUI/host) menginject texthook.dll (dibuat dari folder texthook) kedalam target process dan disambungkan lewat 2 file pipe.<br>
|
Host (lihat folder host) menyuntikan texthook.dll (dibuat dari folder texthook) kedalam target proses dan disambungkan lewat 2 file pipe.<br>
|
||||||
Host menulis ke hostPipe, texthook menulis ke hookPipe.<br>
|
Host menulis ke hostPipe, texthook menulis ke hookPipe.<br>
|
||||||
texthook menunggu pipe tersambung, lalu menginject beberapa instruksi ke teks yang menghasilkan fungsi (contoh: TextOut, GetGlyphOutline) yang membuat input dikirim melewati pipa.<br>
|
texthook menunggu pipe tersambung, lalu menyuntikan beberapa instruksi ke teks yang menghasilkan fungsi (contoh: TextOut, GetGlyphOutline) yang membuat input dikirim melewati pipa.<br>
|
||||||
Informasi tambahan tentang hook dipindahkan melewati shared memory.<br>
|
Informasi tambahan tentang hook dipindahkan melewati shared memory.<br>
|
||||||
Text yang diterima host melewati pipe lalu diproses lagi sebelum dikembalikan ke GUI.<br>
|
Teks yang diterima host melewati pipe lalu diproses lagi sebelum dikembalikan ke GUI.<br>
|
||||||
Dan pada akhirnya, GUI melepas text ke ekstensi sebelum menampilkan teks.
|
Dan pada akhirnya, GUI melepas teks ke ekstensi sebelum menampilkan teks.
|
||||||
|
|
||||||
## [Pengembang](CREDITS.md)
|
## [Pengembang](docs/CREDITS.md)
|
||||||
|
15
README_IT.md
15
README_IT.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![Come si vede](screenshot.png)
|
![Come si vede](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a. NextHooker) è un agganciatore di testi di videogiochi open-source per Windows/Wine basato su[ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (a.k.a. NextHooker) è un agganciatore di testi di videogiochi open-source per Windows/Wine basato su[ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Guarda il [video tutorial](https://tinyurl.com/textractor-tutorial) per una sintesi veloce sul suo utilizzo.
|
Guarda il [video tutorial](docs/TUTORIAL.md) per una sintesi veloce sul suo utilizzo.
|
||||||
|
|
||||||
## Scarica
|
## Scarica
|
||||||
|
|
||||||
Le uscite di Textractor possono essere trovate [qui](https://github.com/Artikash/Textractor/releases).<br>
|
Le uscite di Textractor possono essere trovate [qui](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
L'ultima uscita di ITHVNR può essere trovata [qui](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
L'ultima uscita di ITHVNR può essere trovata [qui](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
Prova ad avviare vcredist se ti esce un errore quando avvi Textractor.
|
|
||||||
|
|
||||||
## Caratteristiche
|
## Caratteristiche
|
||||||
|
|
||||||
@ -34,7 +33,7 @@ Guardate la cartella delle estenzioni per esempi di cosa possono fare le estenzi
|
|||||||
|
|
||||||
Tutti i contributi sono apprezzati! Inviatemi un email a akashmozumdar@gmail.com se avete delle domande sul codebase.<br>
|
Tutti i contributi sono apprezzati! Inviatemi un email a akashmozumdar@gmail.com se avete delle domande sul codebase.<br>
|
||||||
Dovreste usare il processo standard di creare una pull request (fork, branch, commit changes, crea PR dal vostro ramo al mio master).<br>
|
Dovreste usare il processo standard di creare una pull request (fork, branch, commit changes, crea PR dal vostro ramo al mio master).<br>
|
||||||
Contribuire alla traduzione è semplice: traduci le stringhe in text.cpp cosi come questo README.
|
Contribuire alla traduzione è semplice: traduci le stringhe in [text.cpp](text.cpp) cosi come questo README.
|
||||||
|
|
||||||
## Compiling
|
## Compiling
|
||||||
|
|
||||||
@ -43,11 +42,11 @@ Dovresti essere in grado di aprire la cartella in Visual Studio, e costruire. Av
|
|||||||
|
|
||||||
## Architettura del progetto
|
## Architettura del progetto
|
||||||
|
|
||||||
L'host (guarda la cartella GUI/host) innietta texthook.dll (creato dalla cartella texthook) nel processo e lo connette attraverso due file pipe.<br>
|
L'host (guarda la cartella host) innietta texthook.dll (creato dalla cartella texthook) nel processo e lo connette attraverso due file pipe.<br>
|
||||||
L'host scrive a hostPipe, texthook scrive a hookPipe.<br>
|
L'host scrive a hostPipe, texthook scrive a hookPipe.<br>
|
||||||
Texthook aspetta per il pipe di essere connesso, poi innietta alcune istruzione in qualunque funzione di immissione del testo (es. TextOut, GetGlyphOutline) che causa il loro input di essere inviato attraverso il pipe.<br>
|
Texthook aspetta per il pipe di essere connesso, poi innietta alcune istruzione in qualunque funzione di immissione del testo (es. TextOut, GetGlyphOutline) che causa il loro input di essere inviato attraverso il pipe.<br>
|
||||||
Informazioni aggiuntive sui ganci soo scambiati attraverso la memorio condivisa.<br>
|
Informazioni aggiuntive sui ganci soo scambiati attraverso la memorio condivisa.<br>
|
||||||
Il testo che l'host riceve attraverso il pipe è poi processato un poco prima di essere rinviato alla GUI.<br>
|
Il testo che l'host riceve attraverso il pipe è poi processato un poco prima di essere rinviato alla GUI.<br>
|
||||||
Infine, la GUI dispone il testo alle estenzioni prima di mostrarle.
|
Infine, la GUI dispone il testo alle estenzioni prima di mostrarle.
|
||||||
|
|
||||||
## [Sviluppatori](CREDITS.md)
|
## [Sviluppatori](docs/CREDITS.md)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Textractor
|
# Textractor
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** はビジュアルノベル文字抽出プログラム。
|
**Textractor** はビジュアルノベル文字抽出プログラム。
|
||||||
|
|
||||||
|
13
README_KR.md
13
README_KR.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![How it looks](screenshot.png)
|
![How it looks](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a. NextHooker)는 Windows/Wine에서 작동하는 [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine)을 기반으로 한 오픈소스 x86/x64 비디오게임 텍스트 후커 입니다.<br>
|
**Textractor** (a.k.a. NextHooker)는 Windows/Wine에서 작동하는 [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine)을 기반으로 한 오픈소스 x86/x64 비디오게임 텍스트 후커 입니다.<br>
|
||||||
빠른 사용법의 이해를 위해 [tutorial video](https://tinyurl.com/textractor-tutorial) 를 참고하세요.
|
빠른 사용법의 이해를 위해 [tutorial video](docs/TUTORIAL.md) 를 참고하세요.
|
||||||
|
|
||||||
## 다운로드
|
## 다운로드
|
||||||
|
|
||||||
[여기](https://github.com/Artikash/Textractor/releases)에서 Textractor 최신버전을 받으실 수 있습니다.<br>
|
[여기](https://github.com/Artikash/Textractor/releases)에서 Textractor 최신버전을 받으실 수 있습니다.<br>
|
||||||
최신버전의 ITHVNR은 [여기](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO)서 받을 수 있습니다.<br>
|
최신버전의 ITHVNR은 [여기](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO)서 받을 수 있습니다.
|
||||||
Textractor 실행오류를 겪는다면 vcredist를 실행해 보시기 바랍니다.
|
|
||||||
|
|
||||||
## 특징
|
## 특징
|
||||||
|
|
||||||
@ -41,11 +40,11 @@ Textractor 실행오류를 겪는다면 vcredist를 실행해 보시기 바랍
|
|||||||
|
|
||||||
## 프로젝트 아키텍쳐
|
## 프로젝트 아키텍쳐
|
||||||
|
|
||||||
The host (see GUI/host folder) injects texthook.dll (created from the texthook folder) into the target process and connects to it via 2 pipe files.<br>
|
The host (see host folder) injects texthook.dll (created from the texthook folder) into the target process and connects to it via 2 pipe files.<br>
|
||||||
Host writes to hostPipe, texthook writes to hookPipe.<br>
|
Host writes to hostPipe, texthook writes to hookPipe.<br>
|
||||||
texthook waits for the pipe to be connected, then injects a few instructions into any text outputting functions (e.g. TextOut, GetGlyphOutline) that cause their input to be sent through the pipe.<br>
|
texthook waits for the pipe to be connected, then injects a few instructions into any text outputting functions (e.g. TextOut, GetGlyphOutline) that cause their input to be sent through the pipe.<br>
|
||||||
Additional information about hooks is exchanged via shared memory.<br>
|
Additional information about hooks is exchanged via shared memory.<br>
|
||||||
The text that the host receives through the pipe is then processed a little before being dispatched back to the GUI.<br>
|
The text that the host receives through the pipe is then processed a little before being dispatched back to the GUI.<br>
|
||||||
Finally, the GUI dispatches the text to extensions before displaying it.
|
Finally, the GUI dispatches the text to extensions before displaying it.
|
||||||
|
|
||||||
## [개발자들](CREDITS.md)
|
## [개발자들](docs/CREDITS.md)
|
||||||
|
17
README_PT.md
17
README_PT.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![Como se Parece](screenshot.png)
|
![Como se Parece](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (também conhecido como NextHooker) é um extrator de textos de video-games x86/x64 para Windows/Wine baseado no [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (também conhecido como NextHooker) é um extrator de textos de video-games x86/x64 para Windows/Wine baseado no [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Assista ao [vídeo tutorial](https://tinyurl.com/textractor-tutorial) para uma rápida apresentação de como utilizá-lo.
|
Assista ao [vídeo tutorial](docs/TUTORIAL.md) para uma rápida apresentação de como utilizá-lo.
|
||||||
|
|
||||||
## Download
|
## Download
|
||||||
|
|
||||||
As versões lançadas podem ser encontradas [aqui](https://github.com/Artikash/Textractor/releases).<br>
|
As versões lançadas podem ser encontradas [aqui](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
A última versão lançada do ITHVNR pode ser encontrada [aqui](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
A última versão lançada do ITHVNR pode ser encontrada [aqui](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
Tente rodar o vcredist se você encontrar algum erro ao iniciar o Textractor.
|
|
||||||
|
|
||||||
## Recursos e Funções
|
## Recursos e Funções
|
||||||
|
|
||||||
@ -32,9 +31,9 @@ Veja a pasta de extensões para mais exemplos do que as extensões são capazes
|
|||||||
|
|
||||||
## Contribuindo
|
## Contribuindo
|
||||||
|
|
||||||
Todas contribuições são bem-vindas! Por favor, me mande um e-mail (não, não sou ocupado!) no endereço akashmozumdar@gmail.com caso tenha alguma dúvida quanto ao codebase.<br>
|
Todas contribuições são bem-vindas! Por favor, me mande um e-mail no endereço akashmozumdar@gmail.com caso tenha alguma dúvida quanto ao codebase.<br>
|
||||||
Você deve seguir o processo padrão de fazer um pull request (fork, branch, realizar mudanças, realizar o PR do seu branch para o meu master).<br>
|
Você deve seguir o processo padrão de fazer um pull request (fork, branch, realizar mudanças, realizar o PR do seu branch para o meu master).<br>
|
||||||
Contribuir com uma tradução é fácil: basta traduzir as linhas do text.cpp assim como esse README.
|
Contribuir com uma tradução é fácil: basta traduzir as linhas do [text.cpp](text.cpp) assim como esse README.
|
||||||
|
|
||||||
## Compilando
|
## Compilando
|
||||||
|
|
||||||
@ -43,11 +42,11 @@ Você deverá então ser capaz de simplesmente abrir uma pasta no Visual Studio
|
|||||||
|
|
||||||
## Arquitetura do Projeto
|
## Arquitetura do Projeto
|
||||||
|
|
||||||
O host (veja a pasta GUI/host) injeta o texthook.dll (criado a partir da pasta texthook) dentro do processo-alvo e se conecta a ele por meio de 2 arquivos pipe.<br>
|
O host (veja a pasta host) injeta o texthook.dll (criado a partir da pasta texthook) dentro do processo-alvo e se conecta a ele por meio de 2 arquivos pipe.<br>
|
||||||
O Host escreve para hostPipe, o texthook escreve para hookPipe.<br>
|
O Host escreve para hostPipe, o texthook escreve para hookPipe.<br>
|
||||||
O texthook espera pelo pipe estar conectado e então injeta algumas intruções dentro de quaisquer funções que produzam texto (por exemplo: TextOut, GetGlyphOutline) o que faz com que seu produto seja mandado por meio do pipe.<br>
|
O texthook espera pelo pipe estar conectado e então injeta algumas intruções dentro de quaisquer funções que produzam texto (por exemplo: TextOut, GetGlyphOutline) o que faz com que seu produto seja mandado por meio do pipe.<br>
|
||||||
Informação adicional sobre os hooks é trocada por meio da memória compartilhada.<br>
|
Informação adicional sobre os hooks é trocada por meio da memória compartilhada.<br>
|
||||||
O texto que o host recebe por meio do pipe é então processado um pouco antes de ser despachado devolta para a IGU/GUI.<br>
|
O texto que o host recebe por meio do pipe é então processado um pouco antes de ser despachado devolta para a IGU/GUI.<br>
|
||||||
Finalmente, a IGU/GUI despacha o texto para as extensões antes de mostrá-lo.
|
Finalmente, a IGU/GUI despacha o texto para as extensões antes de mostrá-lo.
|
||||||
|
|
||||||
## [Desenvolvedores](CREDITS.md)
|
## [Desenvolvedores](docs/CREDITS.md)
|
||||||
|
17
README_RU.md
17
README_RU.md
@ -2,16 +2,15 @@
|
|||||||
|
|
||||||
![Как это выглядит](screenshot.png)
|
![Как это выглядит](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor** (a.k.a. NextHooker) это проект x86/x64 Windows/Wine программы для захвата текста из видеоигр, основанный на [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
**Textractor** (a.k.a. NextHooker) это проект x86/x64 Windows/Wine программы для захвата текста из видеоигр, основанный на [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
Смотреть [обучающее видео](https://tinyurl.com/textractor-tutorial) для быстрого ознакомления.
|
Смотреть [обучающее видео](docs/TUTORIAL.md) для быстрого ознакомления.
|
||||||
|
|
||||||
## Загрузка
|
## Загрузка
|
||||||
|
|
||||||
Выпуски Textractor могут быть найдены [здесь](https://github.com/Artikash/Textractor/releases).<br>
|
Выпуски Textractor могут быть найдены [здесь](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
Последний выпуск ITHVNR может быть найден [здесь](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
Последний выпуск ITHVNR может быть найден [здесь](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
Установите библиотеки Visual C redist(vcredist.x86.exe), если получаете ошибку при запуске Textractor.
|
|
||||||
|
|
||||||
## Возможности
|
## Возможности
|
||||||
|
|
||||||
@ -32,9 +31,9 @@
|
|||||||
|
|
||||||
## Вклад
|
## Вклад
|
||||||
|
|
||||||
Любой вклад приветствуется! Пишите мне(автору)(нет, я не занят!) на akashmozumdar@gmail.com, если у вас есть любые вопросы о кодовой базе.<br>
|
Любой вклад приветствуется! Пишите мне(автору) на akashmozumdar@gmail.com, если у вас есть любые вопросы о кодовой базе.<br>
|
||||||
Используйте стандартные действия для создания pull request (fork, branch, commit changes, создайте PR из своей ветки branch на мой master).<br>
|
Используйте стандартные действия для создания pull request (fork, branch, commit changes, создайте PR из своей ветки branch на мой master).<br>
|
||||||
Вклад в перевод совсем не сложен: просто переведите строки в text.cpp, также, как и этот README.
|
Вклад в перевод совсем не сложен: просто переведите строки в [text.cpp](text.cpp), также, как и этот README.
|
||||||
|
|
||||||
## Компиляция
|
## Компиляция
|
||||||
|
|
||||||
@ -43,11 +42,11 @@
|
|||||||
|
|
||||||
## Архитектура проекта
|
## Архитектура проекта
|
||||||
|
|
||||||
Хост (смотрите папку GUI/host) внедряет texthook.dll (созданной из папки texthook) в целевой процесс и подключается к нему через два файла-канала (pipe).<br>
|
Хост (смотрите папку host) внедряет texthook.dll (созданной из папки texthook) в целевой процесс и подключается к нему через два файла-канала (pipe).<br>
|
||||||
Хост пишет в hostPipe, texthook пишет в hookPipe.<br>
|
Хост пишет в hostPipe, texthook пишет в hookPipe.<br>
|
||||||
texthook ждет присоединения канала, тогда внедряет некоторые инструкции в любые выводящие текст функции (такие как TextOut, GetGlyphOutline), что вызывает пересылку поступающего в них текста через канал.<br>
|
texthook ждет присоединения канала, тогда внедряет некоторые инструкции в любые выводящие текст функции (такие как TextOut, GetGlyphOutline), что вызывает пересылку поступающего в них текста через канал.<br>
|
||||||
Дополнительная информация о хуках размещена через файл просмотра (a.k.a. section object), который сопоставлен с ссылкой на класс TextHook.<br>
|
Дополнительная информация о хуках размещена через файл просмотра (a.k.a. section object), который сопоставлен с ссылкой на класс TextHook.<br>
|
||||||
Текст, который хост получает через канал, затем немного обрабатывается перед отправкой обратно в графический интерфейс (GUI).<br>
|
Текст, который хост получает через канал, затем немного обрабатывается перед отправкой обратно в графический интерфейс (GUI).<br>
|
||||||
Наконец, GUI отправляет текст расширениям, перед его отображением.
|
Наконец, GUI отправляет текст расширениям, перед его отображением.
|
||||||
|
|
||||||
## [Разработчики](CREDITS.md)
|
## [Разработчики](docs/CREDITS.md)
|
||||||
|
12
README_SC.md
12
README_SC.md
@ -1,10 +1,10 @@
|
|||||||
# Textractor
|
# Textractor
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
## 概述
|
## 概述
|
||||||
|
|
||||||
**Textractor** (曾用名: NextHooker) 是一个基于 [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine), 为 Windows/Wine 开发的开源 x86/x64 文本提取器.<br>
|
**Textractor** (曾用名: NextHooker) 是一个基于 [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine), 为 Windows/Wine 开发的开源 x86/x64 文本提取器.<br>
|
||||||
|
|
||||||
![它工作起来的样子](screenshot.png)
|
![它工作起来的样子](screenshot.png)
|
||||||
|
|
||||||
@ -28,9 +28,9 @@ Textractor 的发行版可以在[这里](https://github.com/Artikash/Textractor/
|
|||||||
|
|
||||||
## 贡献
|
## 贡献
|
||||||
|
|
||||||
欢迎一切贡献!如有任何关于代码的疑问,请向 akashmozumdar@gmail.com 发邮件 (不,我并不忙!).<br>
|
欢迎一切贡献!如有任何关于代码的疑问,请向 akashmozumdar@gmail.com 发邮件.<br>
|
||||||
你应当使用创建 PR 的标准过程 (分岔 (fork), 分支 (branch), 提交变化, 创建从你的分支到我的 master 分支的 PR).<br>
|
你应当使用创建 PR 的标准过程 (分岔 (fork), 分支 (branch), 提交变化, 创建从你的分支到我的 master 分支的 PR).<br>
|
||||||
提供翻译贡献很简单: 只需翻译 text.cpp 中的字符串和这份 README 即可.
|
提供翻译贡献很简单: 只需翻译 [text.cpp](text.cpp) 中的字符串和这份 README 即可.
|
||||||
|
|
||||||
## 编译
|
## 编译
|
||||||
|
|
||||||
@ -39,11 +39,11 @@ Textractor 的发行版可以在[这里](https://github.com/Artikash/Textractor/
|
|||||||
|
|
||||||
## 项目架构
|
## 项目架构
|
||||||
|
|
||||||
宿主 (位于 GUI/host 文件夹) 向目标进程注入 texthook.dll (由 texthook 文件夹创建) 并通过两个管道文件互联.<br>
|
宿主 (位于 host 文件夹) 向目标进程注入 texthook.dll (由 texthook 文件夹创建) 并通过两个管道文件互联.<br>
|
||||||
宿主向 hostPipe 写入, texthook 向 hookPipe 写入.<br>
|
宿主向 hostPipe 写入, texthook 向 hookPipe 写入.<br>
|
||||||
texthook 等待管道连接, 之后向一些文本输出函数 (如 TextOut, GetGlyphOutline) 注入一系列指令, 使得它们的输入被沿着管道发送.<br>
|
texthook 等待管道连接, 之后向一些文本输出函数 (如 TextOut, GetGlyphOutline) 注入一系列指令, 使得它们的输入被沿着管道发送.<br>
|
||||||
其它关于钩子的信息通过一个被 TextHook 类保有引用的文件视图 (曾用名: 段对象) 共享.<br>
|
其它关于钩子的信息通过一个被 TextHook 类保有引用的文件视图 (曾用名: 段对象) 共享.<br>
|
||||||
之后, 宿主通过管道接收到的文本在传回 GUI 前被简单处理.<br>
|
之后, 宿主通过管道接收到的文本在传回 GUI 前被简单处理.<br>
|
||||||
最后, GUI 在显示文本前将其分发给扩展.
|
最后, GUI 在显示文本前将其分发给扩展.
|
||||||
|
|
||||||
## [开发者](CREDITS.md)
|
## [开发者](docs/CREDITS.md)
|
||||||
|
13
README_TH.md
13
README_TH.md
@ -2,17 +2,16 @@
|
|||||||
|
|
||||||
![How it looks](screenshot.png)
|
![How it looks](screenshot.png)
|
||||||
|
|
||||||
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa](README_ID.md) ● [Português](README_PT.md)
|
[English](README.md) ● [Español](README_ES.md) ● [简体中文](README_SC.md) ● [Русский](README_RU.md) ● [한국어](README_KR.md) ● [ภาษาไทย](README_TH.md) ● [Français](README_FR.md) ● [Italiano](README_IT.md) ● [日本語](README_JP.md) ● [Bahasa Indonesia](README_ID.md) ● [Português](README_PT.md) ● [Deutsch](README_DE.md)
|
||||||
|
|
||||||
**Textractor**
|
**Textractor**
|
||||||
(หรือ NextHooker) คือโปรแกรมโอเพนซอร์ซสำหรับปฏิบัติการที่มีหน้าที่เพื่อเชื่อมกับตัวอักษรกับเกมจากที่มาจากระบบปฏิบัติการ Window/Wine โดยมีแบบดังเดิมมาจาก [ITHVNR](http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
(หรือ NextHooker) คือโปรแกรมโอเพนซอร์ซสำหรับปฏิบัติการที่มีหน้าที่เพื่อเชื่อมกับตัวอักษรกับเกมจากที่มาจากระบบปฏิบัติการ Window/Wine โดยมีแบบดังเดิมมาจาก [ITHVNR](https://web.archive.org/web/20160202084144/http://www.hongfire.com/forum/showthread.php/438331-ITHVNR-ITH-with-the-VNR-engine).<br>
|
||||||
สามารถดูตัวอย่างวิธีการใช้งาน [วีดีโอตัวอย่างการใช้งาน](https://tinyurl.com/textractor-tutorial) เพื่อที่จะแสดงความเข้าใจคร่าวๆเกี่ยวกับโปรแกรม.
|
สามารถดูตัวอย่างวิธีการใช้งาน [วีดีโอตัวอย่างการใช้งาน](docs/TUTORIAL.md) เพื่อที่จะแสดงความเข้าใจคร่าวๆเกี่ยวกับโปรแกรม.
|
||||||
|
|
||||||
## ดาวน์โหลด
|
## ดาวน์โหลด
|
||||||
|
|
||||||
Textractor รุ่นล่าสุดสามารถดาวน์โหลดจาก [ที่นี้](https://github.com/Artikash/Textractor/releases).<br>
|
Textractor รุ่นล่าสุดสามารถดาวน์โหลดจาก [ที่นี้](https://github.com/Artikash/Textractor/releases).<br>
|
||||||
ITHVNR รุ่นสุดท้ายสามารถดาวน์โหลดได้ [ที่นี้](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).<br>
|
ITHVNR รุ่นสุดท้ายสามารถดาวน์โหลดได้ [ที่นี้](https://drive.google.com/open?id=13aHF4uIXWn-3YML_k2YCDWhtGgn5-tnO).
|
||||||
ถ้าหากมีปัญหาขณะที่เปิด Textractor ลองเปิด vcredist
|
|
||||||
|
|
||||||
## คุณสมบัติ
|
## คุณสมบัติ
|
||||||
|
|
||||||
@ -38,7 +37,7 @@ ITHVNR รุ่นสุดท้ายสามารถดาวน์โห
|
|||||||
|
|
||||||
## โครงสร้างโปรแกรม
|
## โครงสร้างโปรแกรม
|
||||||
|
|
||||||
ฐานของโปรแกรม (โฟลเดอร์ GUI/host) ส่งข้อมูลจาก texthook.dll (ที่ถูกสร้างจาก texthook โฟลเดอร์) ไปยังเกมเป้าหมาย และ เชื่อมทั่งสองอย่างเข้าด้วยกัน<br>
|
ฐานของโปรแกรม (โฟลเดอร์ host) ส่งข้อมูลจาก texthook.dll (ที่ถูกสร้างจาก texthook โฟลเดอร์) ไปยังเกมเป้าหมาย และ เชื่อมทั่งสองอย่างเข้าด้วยกัน<br>
|
||||||
ฐานโปรแกรมเขียนผ่านฝั่ง hostPipe(ท่อเชื่อมฝั่งฐานข้อมูล) ในขณะที่ตัวดึงตัวอักษรที่ทางฝั่ง hookPipe(ท่อเชื่อมฝั่งดึงข้อมูล).<br>
|
ฐานโปรแกรมเขียนผ่านฝั่ง hostPipe(ท่อเชื่อมฝั่งฐานข้อมูล) ในขณะที่ตัวดึงตัวอักษรที่ทางฝั่ง hookPipe(ท่อเชื่อมฝั่งดึงข้อมูล).<br>
|
||||||
ตัวดึงตัวอักษรรอการเชื่อมเข้ากับของทั่งสองท่อ หลังจากนั่นส่งคำสั่งไปยังข้อมูลนั่น (เช่น แสดงผลข้อมูล เป็นต้น) และทำให้ข้อมูลส่งผ่านต่อมาออกมาได้ถูกต้อง<br>
|
ตัวดึงตัวอักษรรอการเชื่อมเข้ากับของทั่งสองท่อ หลังจากนั่นส่งคำสั่งไปยังข้อมูลนั่น (เช่น แสดงผลข้อมูล เป็นต้น) และทำให้ข้อมูลส่งผ่านต่อมาออกมาได้ถูกต้อง<br>
|
||||||
ข้อมูลบางอย่างเกี่ยวกับการเชื่อมจะถูกแลกเปลี่ยนผ่านความทรงจำของระบบ (shared memory)
|
ข้อมูลบางอย่างเกี่ยวกับการเชื่อมจะถูกแลกเปลี่ยนผ่านความทรงจำของระบบ (shared memory)
|
||||||
@ -46,4 +45,4 @@ ITHVNR รุ่นสุดท้ายสามารถดาวน์โห
|
|||||||
ตัวอักษรที่ฐานโปรแกรมรับผ่านท่อจะถูกแปลงเล็กน้อยก่อนที่จะแสดงผ่าน GUI <br>
|
ตัวอักษรที่ฐานโปรแกรมรับผ่านท่อจะถูกแปลงเล็กน้อยก่อนที่จะแสดงผ่าน GUI <br>
|
||||||
สุดท้ายแล้ว GUI จะส่งข้อมูลตัวอักษรไปยังส่วนขยายต่างๆก่อนที่จะแสดงให้เห็นในหน้าจอ
|
สุดท้ายแล้ว GUI จะส่งข้อมูลตัวอักษรไปยังส่วนขยายต่างๆก่อนที่จะแสดงให้เห็นในหน้าจอ
|
||||||
|
|
||||||
## [นักพัฒนา](CREDITS.md)
|
## [นักพัฒนา](docs/CREDITS.md)
|
||||||
|
@ -1,49 +1,24 @@
|
|||||||
macro(msvc_registry_search)
|
macro(msvc_registry_search)
|
||||||
if(NOT DEFINED Qt5_DIR AND MSVC)
|
if(NOT DEFINED Qt5_DIR)
|
||||||
|
if (NOT EXISTS ${QT_ROOT})
|
||||||
# look for user-registry pointing to qtcreator
|
# look for user-registry pointing to qtcreator
|
||||||
get_filename_component(QT_BIN [HKEY_CURRENT_USER\\Software\\Classes\\Applications\\QtProject.QtCreator.pro\\shell\\Open\\Command] PATH)
|
get_filename_component(QT_ROOT [HKEY_CURRENT_USER\\Software\\Classes\\Applications\\QtProject.QtCreator.pro\\shell\\Open\\Command] PATH)
|
||||||
|
|
||||||
# get root path so we can search for 5.3, 5.4, 5.5, etc
|
# get root path
|
||||||
string(REPLACE "/Tools" ";" QT_BIN "${QT_BIN}")
|
string(REPLACE "/Tools" ";" QT_ROOT "${QT_ROOT}")
|
||||||
list(GET QT_BIN 0 QT_BIN)
|
list(GET QT_ROOT 0 QT_ROOT)
|
||||||
file(GLOB QT_VERSIONS "${QT_BIN}/5.1*")
|
|
||||||
list(SORT QT_VERSIONS)
|
|
||||||
|
|
||||||
# assume the latest version will be last alphabetically
|
|
||||||
list(REVERSE QT_VERSIONS)
|
|
||||||
|
|
||||||
list(GET QT_VERSIONS 0 QT_VERSION)
|
|
||||||
|
|
||||||
# fix any double slashes which seem to be common
|
|
||||||
string(REPLACE "//" "/" QT_VERSION "${QT_VERSION}")
|
|
||||||
|
|
||||||
if(MSVC_VERSION GREATER_EQUAL 1920)
|
|
||||||
set(QT_MSVC 2019)
|
|
||||||
elseif(MSVC_VERSION GREATER_EQUAL 1910)
|
|
||||||
set(QT_MSVC 2017)
|
|
||||||
elseif(MSVC_VERSION GREATER_EQUAL 1900)
|
|
||||||
set(QT_MSVC 2015)
|
|
||||||
else()
|
|
||||||
message(WARNING "Unsupported MSVC toolchain version")
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
set(QT_VERSION 5.13.2)
|
||||||
|
set(QT_MSVC 2019)
|
||||||
|
|
||||||
if(QT_MSVC)
|
if(QT_MSVC)
|
||||||
if(CMAKE_CL_64)
|
if(CMAKE_CL_64)
|
||||||
SET(QT_SUFFIX "_64")
|
SET(QT_SUFFIX "_64")
|
||||||
else()
|
else()
|
||||||
set(QT_SUFFIX "")
|
set(QT_SUFFIX "")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# MSVC 2015+ is only backwards compatible
|
|
||||||
if(EXISTS "${QT_VERSION}/msvc${QT_MSVC}${QT_SUFFIX}")
|
|
||||||
set(Qt5_DIR "${QT_VERSION}/msvc${QT_MSVC}${QT_SUFFIX}/lib/cmake/Qt5")
|
|
||||||
elseif(QT_MSVC GREATER_EQUAL 2019 AND EXISTS "${QT_VERSION}/msvc2017${QT_SUFFIX}")
|
|
||||||
set(Qt5_DIR "${QT_VERSION}/msvc2017${QT_SUFFIX}/lib/cmake/Qt5")
|
set(Qt5_DIR "${QT_VERSION}/msvc2017${QT_SUFFIX}/lib/cmake/Qt5")
|
||||||
elseif(QT_MSVC GREATER_EQUAL 2017 AND EXISTS "${QT_VERSION}/msvc2015${QT_SUFFIX}")
|
|
||||||
set(Qt5_DIR "${QT_VERSION}/msvc2015${QT_SUFFIX}/lib/cmake/Qt5")
|
|
||||||
else()
|
|
||||||
message(WARNING "Required QT5 toolchain is not installed")
|
|
||||||
endif()
|
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
endmacro()
|
endmacro()
|
||||||
@ -52,7 +27,12 @@ macro(find_qt5)
|
|||||||
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
||||||
#set(CMAKE_AUTOMOC ON)
|
#set(CMAKE_AUTOMOC ON)
|
||||||
set(CMAKE_AUTOUIC ON)
|
set(CMAKE_AUTOUIC ON)
|
||||||
add_definitions(-DQT_DEPRECATED_WARNINGS -DQT_DISABLE_DEPRECATED_BEFORE=0x060000)
|
#add_definitions(-DQT_DEPRECATED_WARNINGS -DQT_DISABLE_DEPRECATED_BEFORE=0x060000)
|
||||||
|
if(CMAKE_SIZEOF_VOID_P EQUAL 8)
|
||||||
|
set(Qt5_DIR "C:/Qt/Qt5.13.2/5.13.2/msvc2017_64/lib/cmake/Qt5")
|
||||||
|
else()
|
||||||
|
set(Qt5_DIR "C:/Qt/Qt5.13.2/5.13.2/msvc2017/lib/cmake/Qt5")
|
||||||
|
endif()
|
||||||
find_package(Qt5 COMPONENTS ${ARGN})
|
find_package(Qt5 COMPONENTS ${ARGN})
|
||||||
|
|
||||||
if(Qt5_FOUND)
|
if(Qt5_FOUND)
|
||||||
|
14
deploy.ps1
14
deploy.ps1
@ -1,8 +1,8 @@
|
|||||||
param([string]$version)
|
param([string]$version)
|
||||||
|
|
||||||
cd $PSScriptRoot;
|
cd $PSScriptRoot;
|
||||||
mkdir -Force -Verbose Builds;
|
mkdir -Force -Verbose builds;
|
||||||
cd Builds;
|
cd builds;
|
||||||
mkdir -Force -Verbose x86;
|
mkdir -Force -Verbose x86;
|
||||||
mkdir -Force -Verbose x64;
|
mkdir -Force -Verbose x64;
|
||||||
|
|
||||||
@ -45,11 +45,14 @@ foreach ($language in @{
|
|||||||
"Copy to Clipboard",
|
"Copy to Clipboard",
|
||||||
"DeepL Translate",
|
"DeepL Translate",
|
||||||
"DevTools DeepL Translate",
|
"DevTools DeepL Translate",
|
||||||
|
"DevTools Papago Translate",
|
||||||
|
"DevTools Systran Translate",
|
||||||
"Extra Newlines",
|
"Extra Newlines",
|
||||||
"Extra Window",
|
"Extra Window",
|
||||||
"Google Translate",
|
"Google Translate",
|
||||||
"Lua",
|
"Lua",
|
||||||
"Regex Filter",
|
"Regex Filter",
|
||||||
|
"Regex Replacer",
|
||||||
"Remove Repeated Characters",
|
"Remove Repeated Characters",
|
||||||
"Remove Repeated Phrases",
|
"Remove Repeated Phrases",
|
||||||
"Remove Repeated Phrases 2",
|
"Remove Repeated Phrases 2",
|
||||||
@ -62,10 +65,9 @@ foreach ($language in @{
|
|||||||
copy -Force -Recurse -Verbose -Destination "$folder/$arch/$extension.xdll" -Path "Release_$arch/$extension.dll";
|
copy -Force -Recurse -Verbose -Destination "$folder/$arch/$extension.xdll" -Path "Release_$arch/$extension.dll";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
&"C:\Program Files (x86)\Windows Kits\10\App Certification Kit\signtool.exe" sign /a /v /t "http://timestamp.digicert.com" /fd SHA256 @(dir "$folder\**\*");
|
||||||
}
|
}
|
||||||
|
|
||||||
&"C:\Program Files (x86)\Windows Kits\10\App Certification Kit\signtool.exe" sign /a /v /t "http://timestamp.digicert.com" /fd SHA256 @(dir "Textractor-*-$version\**\*");
|
|
||||||
|
|
||||||
rm -Force -Recurse -Verbose "Runtime";
|
rm -Force -Recurse -Verbose "Runtime";
|
||||||
mkdir -Force -Verbose "Runtime";
|
mkdir -Force -Verbose "Runtime";
|
||||||
foreach ($arch in @("x86", "x64"))
|
foreach ($arch in @("x86", "x64"))
|
||||||
@ -78,6 +80,7 @@ foreach ($arch in @("x86", "x64"))
|
|||||||
"Qt5Gui.dll",
|
"Qt5Gui.dll",
|
||||||
"Qt5Network.dll",
|
"Qt5Network.dll",
|
||||||
"Qt5WebSockets.dll",
|
"Qt5WebSockets.dll",
|
||||||
|
"Qt5WinExtras.dll"
|
||||||
"Qt5Widgets.dll",
|
"Qt5Widgets.dll",
|
||||||
"platforms",
|
"platforms",
|
||||||
"styles"
|
"styles"
|
||||||
@ -85,12 +88,13 @@ foreach ($arch in @("x86", "x64"))
|
|||||||
{
|
{
|
||||||
copy -Force -Recurse -Verbose -Destination "Runtime/$arch/$file" -Path "Release_$arch/$file";
|
copy -Force -Recurse -Verbose -Destination "Runtime/$arch/$file" -Path "Release_$arch/$file";
|
||||||
}
|
}
|
||||||
|
copy -Force -Recurse -Verbose -Destination "Runtime/$arch" -Path "C:/Program Files (x86)/Microsoft Visual Studio/2019/Community/VC/Redist/MSVC/**/$arch/Microsoft.VC142.CRT/*"
|
||||||
}
|
}
|
||||||
|
|
||||||
rm -Force -Recurse -Verbose "Textractor";
|
rm -Force -Recurse -Verbose "Textractor";
|
||||||
mkdir -Force -Verbose "Textractor";
|
mkdir -Force -Verbose "Textractor";
|
||||||
copy -Force -Recurse -Verbose -Destination "Textractor" -Path @("Runtime/*", "Textractor--$version/*");
|
copy -Force -Recurse -Verbose -Destination "Textractor" -Path @("Runtime/*", "Textractor--$version/*");
|
||||||
&"C:\Program Files\7-Zip\7z" a "Textractor-$version-Zip-Version-English-Only.zip" Textractor/
|
&"C:\Program Files\7-Zip\7z" a "Textractor-$version-Zip-Version-English-Only.zip" Textractor/ ../INSTALL_THIS_UNICODE_FONT.ttf
|
||||||
|
|
||||||
cd ..;
|
cd ..;
|
||||||
&"C:\Program Files (x86)\Inno Setup 6\iscc" -DVERSION="$version" installer.iss;
|
&"C:\Program Files (x86)\Inno Setup 6\iscc" -DVERSION="$version" installer.iss;
|
||||||
|
32
docs/CREDITS.md
Normal file
32
docs/CREDITS.md
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# Developers
|
||||||
|
|
||||||
|
- Textractor mainly made by [Artikash](https://github.com/Artikash) with the contributions of
|
||||||
|
- Build system improvements by [DoumanAsh](https://github.com/DoumanAsh)
|
||||||
|
- Original text removal in Extra Window by [Niakr1s](https://github.com/Niakr1s)
|
||||||
|
- CLI bugfixes by [tinyAdapter](https://github.com/tinyAdapter)
|
||||||
|
- Hook codes for WAFFLE and WillPlus by [lgztx96](https://github.com/lgztx96)
|
||||||
|
- Hook codes for TokyoNecro by [Jazzinghen](https://github.com/Jazzinghen)
|
||||||
|
- Hook codes for ShinyDays and Artemis by [luojunyuan](https://github.com/luojunyuan)
|
||||||
|
- Google Chrome DevTools integration by [zeheyler](https://github.com/zeheyler)
|
||||||
|
- AppVeyor setup by [silas1037](https://github.com/silas1037)
|
||||||
|
- Improved attach dialog by [tera8m4](https://github.com/tera8m4)
|
||||||
|
- Regex Replacer, Papago, Systran, improvements to Extra Window, and bugfixes by [Blu3train](https://github.com/Blu3train)
|
||||||
|
- French translation by [Racky](mailto:maitrenoah@gmail.com) and [Gratusfr](https://github.com/Gratusfr)
|
||||||
|
- Spanish translation by [scese250](https://github.com/scese250)
|
||||||
|
- Turkish translation by [niisokusu](https://reddit.com/u/niisokusu)
|
||||||
|
- Simplified Chinese translation by [tinyAdapter](https://github.com/tinyAdapter), [lgztx96](https://github.com/lgztx96) and [chinanoahli](https://github.com/chinanoahli)
|
||||||
|
- Russian translation by [TokcDK](https://github.com/TokcDK)
|
||||||
|
- Indonesian translation by [Hawxone](https://github.com/Hawxone)
|
||||||
|
- Portuguese translation by [TsumiHokiro](https://github.com/TsumiHokiro)
|
||||||
|
- Thai translation by [azmadoppler](https://github.com/azmadoppler)
|
||||||
|
- Korean translation by [O SK](mailto:afkl11@outlook.kr)
|
||||||
|
- Italian translation by [StarFang208](https://github.com/StarFang208)
|
||||||
|
- ITHVNR updated by [mireado](https://github.com/mireado), [Eguni](https://github.com/Eguni), and [IJEMIN](https://github.com/IJEMIN)
|
||||||
|
- ITHVNR originally made by [Stomp](mailto:zorkzero@hotmail.com)
|
||||||
|
- VNR engine made by [jichi](https://github.com/jichifly)
|
||||||
|
- ITH updated by [Andys](https://github.com/AndyScull)
|
||||||
|
- ITH originally made by [kaosu](https://code.google.com/archive/p/interactive-text-hooker)
|
||||||
|
|
||||||
|
If you're on this list and want your link changed let Artikash know.
|
||||||
|
|
||||||
|
# THANK YOU!!
|
7
docs/TUTORIAL.md
Normal file
7
docs/TUTORIAL.md
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
# Tutorial Video
|
||||||
|
|
||||||
|
https://www.youtube.com/watch?v=eecEOacF6mw
|
||||||
|
|
||||||
|
## Updates/Corrections
|
||||||
|
|
||||||
|
Automatically finding hooks is now done via the `Search for hooks` button, the method shown in the video is found in the `Search for specific text` option.
|
@ -1,17 +1,17 @@
|
|||||||
include(QtUtils)
|
|
||||||
msvc_registry_search()
|
|
||||||
find_qt5(Core Widgets WebSockets)
|
|
||||||
cmake_policy(SET CMP0037 OLD)
|
cmake_policy(SET CMP0037 OLD)
|
||||||
|
|
||||||
add_library(Bing\ Translate MODULE bingtranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
add_library(Bing\ Translate MODULE bingtranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
add_library(Copy\ to\ Clipboard MODULE copyclipboard.cpp extensionimpl.cpp)
|
add_library(Copy\ to\ Clipboard MODULE copyclipboard.cpp extensionimpl.cpp)
|
||||||
add_library(DeepL\ Translate MODULE deepltranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
add_library(DeepL\ Translate MODULE deepltranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
add_library(DevTools\ DeepL\ Translate MODULE devtoolsdeepltranslate.cpp devtools.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
add_library(DevTools\ DeepL\ Translate MODULE devtoolsdeepltranslate.cpp devtools.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
|
add_library(DevTools\ Papago\ Translate MODULE devtoolspapagotranslate.cpp devtools.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
|
add_library(DevTools\ Systran\ Translate MODULE devtoolssystrantranslate.cpp devtools.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
add_library(Extra\ Newlines MODULE extranewlines.cpp extensionimpl.cpp)
|
add_library(Extra\ Newlines MODULE extranewlines.cpp extensionimpl.cpp)
|
||||||
add_library(Extra\ Window MODULE extrawindow.cpp extensionimpl.cpp)
|
add_library(Extra\ Window MODULE extrawindow.cpp extensionimpl.cpp)
|
||||||
add_library(Google\ Translate MODULE googletranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
add_library(Google\ Translate MODULE googletranslate.cpp translatewrapper.cpp network.cpp extensionimpl.cpp)
|
||||||
add_library(Lua MODULE lua.cpp extensionimpl.cpp)
|
add_library(Lua MODULE lua.cpp extensionimpl.cpp)
|
||||||
add_library(Regex\ Filter MODULE regexfilter.cpp extensionimpl.cpp)
|
add_library(Regex\ Filter MODULE regexfilter.cpp extensionimpl.cpp)
|
||||||
|
add_library(Regex\ Replacer MODULE regexreplacer.cpp extensionimpl.cpp)
|
||||||
add_library(Remove\ Repeated\ Characters MODULE removerepeatchar.cpp extensionimpl.cpp)
|
add_library(Remove\ Repeated\ Characters MODULE removerepeatchar.cpp extensionimpl.cpp)
|
||||||
add_library(Remove\ Repeated\ Phrases MODULE removerepeatphrase.cpp extensionimpl.cpp)
|
add_library(Remove\ Repeated\ Phrases MODULE removerepeatphrase.cpp extensionimpl.cpp)
|
||||||
add_library(Remove\ Repeated\ Phrases\ 2 MODULE removerepeatphrase2.cpp extensionimpl.cpp)
|
add_library(Remove\ Repeated\ Phrases\ 2 MODULE removerepeatphrase2.cpp extensionimpl.cpp)
|
||||||
@ -24,11 +24,14 @@ target_precompile_headers(Bing\ Translate REUSE_FROM pch)
|
|||||||
target_precompile_headers(Copy\ to\ Clipboard REUSE_FROM pch)
|
target_precompile_headers(Copy\ to\ Clipboard REUSE_FROM pch)
|
||||||
target_precompile_headers(DeepL\ Translate REUSE_FROM pch)
|
target_precompile_headers(DeepL\ Translate REUSE_FROM pch)
|
||||||
target_precompile_headers(DevTools\ DeepL\ Translate REUSE_FROM pch)
|
target_precompile_headers(DevTools\ DeepL\ Translate REUSE_FROM pch)
|
||||||
|
target_precompile_headers(DevTools\ Papago\ Translate REUSE_FROM pch)
|
||||||
|
target_precompile_headers(DevTools\ Systran\ Translate REUSE_FROM pch)
|
||||||
target_precompile_headers(Extra\ Newlines REUSE_FROM pch)
|
target_precompile_headers(Extra\ Newlines REUSE_FROM pch)
|
||||||
target_precompile_headers(Extra\ Window REUSE_FROM pch)
|
target_precompile_headers(Extra\ Window REUSE_FROM pch)
|
||||||
target_precompile_headers(Google\ Translate REUSE_FROM pch)
|
target_precompile_headers(Google\ Translate REUSE_FROM pch)
|
||||||
target_precompile_headers(Lua REUSE_FROM pch)
|
target_precompile_headers(Lua REUSE_FROM pch)
|
||||||
target_precompile_headers(Regex\ Filter REUSE_FROM pch)
|
target_precompile_headers(Regex\ Filter REUSE_FROM pch)
|
||||||
|
target_precompile_headers(Regex\ Replacer REUSE_FROM pch)
|
||||||
target_precompile_headers(Remove\ Repeated\ Characters REUSE_FROM pch)
|
target_precompile_headers(Remove\ Repeated\ Characters REUSE_FROM pch)
|
||||||
target_precompile_headers(Remove\ Repeated\ Phrases REUSE_FROM pch)
|
target_precompile_headers(Remove\ Repeated\ Phrases REUSE_FROM pch)
|
||||||
target_precompile_headers(Remove\ Repeated\ Phrases\ 2 REUSE_FROM pch)
|
target_precompile_headers(Remove\ Repeated\ Phrases\ 2 REUSE_FROM pch)
|
||||||
@ -40,6 +43,8 @@ target_precompile_headers(Thread\ Linker REUSE_FROM pch)
|
|||||||
target_link_libraries(Bing\ Translate winhttp Qt5::Widgets)
|
target_link_libraries(Bing\ Translate winhttp Qt5::Widgets)
|
||||||
target_link_libraries(DeepL\ Translate winhttp Qt5::Widgets)
|
target_link_libraries(DeepL\ Translate winhttp Qt5::Widgets)
|
||||||
target_link_libraries(DevTools\ DeepL\ Translate shell32 winhttp Qt5::Widgets Qt5::WebSockets)
|
target_link_libraries(DevTools\ DeepL\ Translate shell32 winhttp Qt5::Widgets Qt5::WebSockets)
|
||||||
|
target_link_libraries(DevTools\ Papago\ Translate shell32 winhttp Qt5::Widgets Qt5::WebSockets)
|
||||||
|
target_link_libraries(DevTools\ Systran\ Translate shell32 winhttp Qt5::Widgets Qt5::WebSockets)
|
||||||
target_link_libraries(Extra\ Window Qt5::Widgets)
|
target_link_libraries(Extra\ Window Qt5::Widgets)
|
||||||
target_link_libraries(Google\ Translate winhttp Qt5::Widgets)
|
target_link_libraries(Google\ Translate winhttp Qt5::Widgets)
|
||||||
target_link_libraries(Lua lua53 Qt5::Widgets)
|
target_link_libraries(Lua lua53 Qt5::Widgets)
|
||||||
@ -47,6 +52,8 @@ target_link_libraries(Regex\ Filter Qt5::Widgets)
|
|||||||
target_link_libraries(Styler Qt5::Widgets)
|
target_link_libraries(Styler Qt5::Widgets)
|
||||||
target_link_libraries(Thread\ Linker Qt5::Widgets)
|
target_link_libraries(Thread\ Linker Qt5::Widgets)
|
||||||
|
|
||||||
|
add_custom_target(Cleaner ALL COMMAND del *.xdll WORKING_DIRECTORY ${CMAKE_FINAL_OUTPUT_DIRECTORY})
|
||||||
|
|
||||||
if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSockets.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSocketsd.dll)
|
if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSockets.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSocketsd.dll)
|
||||||
add_custom_command(TARGET DevTools\ DeepL\ Translate
|
add_custom_command(TARGET DevTools\ DeepL\ Translate
|
||||||
POST_BUILD
|
POST_BUILD
|
||||||
|
@ -1,113 +1,241 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
#include "network.h"
|
#include "network.h"
|
||||||
|
|
||||||
extern const wchar_t* TRANSLATION_ERROR;
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
|
|
||||||
extern Synchronized<std::wstring> translateTo, translateFrom, authKey;
|
|
||||||
|
|
||||||
const char* TRANSLATION_PROVIDER = "Bing Translate";
|
const char* TRANSLATION_PROVIDER = "Bing Translate";
|
||||||
const char* GET_API_KEY_FROM = "https://www.microsoft.com/en-us/translator/business/trial/#get-started";
|
const char* GET_API_KEY_FROM = "https://www.microsoft.com/en-us/translator/business/trial/#get-started";
|
||||||
QStringList languages
|
extern const QStringList languagesTo
|
||||||
{
|
{
|
||||||
"Afrikaans: af",
|
"Afrikaans",
|
||||||
"Arabic: ar",
|
"Albanian",
|
||||||
"Bangla: bn",
|
"Amharic",
|
||||||
"Bosnian: bs",
|
"Arabic",
|
||||||
"Bulgarian: bg",
|
"Armenian",
|
||||||
"Cantonese (traditional): yue",
|
"Assamese",
|
||||||
"Catalan: ca",
|
"Azerbaijani",
|
||||||
"Chinese (simplified): zh-Hans",
|
"Bangla",
|
||||||
"Chinese (traditional): zh-Hant",
|
"Bosnian (Latin)",
|
||||||
"Croatian: hr",
|
"Bulgarian",
|
||||||
"Czech: cs",
|
"Cantonese (Traditional)",
|
||||||
"Danish: da",
|
"Catalan",
|
||||||
"Dutch: nl",
|
"Chinese (Simplified)",
|
||||||
"English: en",
|
"Chinese (Traditional)",
|
||||||
"Estonian: et",
|
"Croatian",
|
||||||
"Fijian: fj",
|
"Czech",
|
||||||
"Filipino: fil",
|
"Danish",
|
||||||
"Finnish: fi",
|
"Dari",
|
||||||
"French: fr",
|
"Dutch",
|
||||||
"German: de",
|
"English",
|
||||||
"Greek: el",
|
"Estonian",
|
||||||
"Haitian Creole: ht",
|
"Fijian",
|
||||||
"Hebrew: he",
|
"Filipino",
|
||||||
"Hindi: hi",
|
"Finnish",
|
||||||
"Hmong Daw: mww",
|
"French",
|
||||||
"Hungarian: hu",
|
"French (Canada)",
|
||||||
"Icelandic: is",
|
"German",
|
||||||
"Indonesian: id",
|
"Greek",
|
||||||
"Irish: ga",
|
"Gujarati",
|
||||||
"Italian: it",
|
"Haitian Creole",
|
||||||
"Japanese: ja",
|
"Hebrew",
|
||||||
"Kannada: kn",
|
"Hindi",
|
||||||
"Klingon: tlh",
|
"Hmong Daw",
|
||||||
"Korean: ko",
|
"Hungarian",
|
||||||
"Latvian: lv",
|
"Icelandic",
|
||||||
"Lithuanian: lt",
|
"Indonesian",
|
||||||
"Malagasy: mg",
|
"Inuktitut",
|
||||||
"Malay: ms",
|
"Irish",
|
||||||
"Malayalam: ml",
|
"Italian",
|
||||||
"Maltese: mt",
|
"Japanese",
|
||||||
"Maori: mi",
|
"Kannada",
|
||||||
"Norwegian: nb",
|
"Kazakh",
|
||||||
"Persian: fa",
|
"Khmer",
|
||||||
"Polish: pl",
|
"Klingon",
|
||||||
"Portuguese (Brazil): pt",
|
"Korean",
|
||||||
"Portuguese (Portugal): pt-pt",
|
"Kurdish (Central)",
|
||||||
"Punjabi: pa",
|
"Kurdish (Northern)",
|
||||||
"Romanian: ro",
|
"Lao",
|
||||||
"Russian: ru",
|
"Latvian",
|
||||||
"Samoan: sm",
|
"Lithuanian",
|
||||||
"Serbian (Cyrillic): sr-Cyrl",
|
"Malagasy",
|
||||||
"Serbian (Latin): sr-Latn",
|
"Malay",
|
||||||
"Slovak: sk",
|
"Malayalam",
|
||||||
"Slovenian: sl",
|
"Maltese",
|
||||||
"Spanish: es",
|
"Maori",
|
||||||
"Swahili: sw",
|
"Marathi",
|
||||||
"Swedish: sv",
|
"Myanmar",
|
||||||
"Tahitian: ty",
|
"Nepali",
|
||||||
"Tamil: ta",
|
"Norwegian",
|
||||||
"Telugu: te",
|
"Odia",
|
||||||
"Thai: th",
|
"Pashto",
|
||||||
"Tongan: to",
|
"Persian",
|
||||||
"Turkish: tr",
|
"Polish",
|
||||||
"Ukrainian: uk",
|
"Portuguese (Brazil)",
|
||||||
"Urdu: ur",
|
"Portuguese (Portugal)",
|
||||||
"Vietnamese: vi",
|
"Punjabi",
|
||||||
"Welsh: cy",
|
"Queretaro Otomi",
|
||||||
"Yucatec Maya: yua"
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Samoan",
|
||||||
|
"Serbian (Cyrillic)",
|
||||||
|
"Serbian (Latin)",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Spanish",
|
||||||
|
"Swahili",
|
||||||
|
"Swedish",
|
||||||
|
"Tahitian",
|
||||||
|
"Tamil",
|
||||||
|
"Telugu",
|
||||||
|
"Thai",
|
||||||
|
"Tigrinya",
|
||||||
|
"Tongan",
|
||||||
|
"Turkish",
|
||||||
|
"Ukrainian",
|
||||||
|
"Urdu",
|
||||||
|
"Vietnamese",
|
||||||
|
"Welsh",
|
||||||
|
"Yucatec Maya"
|
||||||
|
}, languagesFrom = languagesTo;
|
||||||
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Afrikaans" }, { L"af" } },
|
||||||
|
{ { L"Albanian" }, { L"sq" } },
|
||||||
|
{ { L"Amharic" }, { L"am" } },
|
||||||
|
{ { L"Arabic" }, { L"ar" } },
|
||||||
|
{ { L"Armenian" }, { L"hy" } },
|
||||||
|
{ { L"Assamese" }, { L"as" } },
|
||||||
|
{ { L"Azerbaijani" }, { L"az" } },
|
||||||
|
{ { L"Bangla" }, { L"bn" } },
|
||||||
|
{ { L"Bosnian (Latin)" }, { L"bs" } },
|
||||||
|
{ { L"Bulgarian" }, { L"bg" } },
|
||||||
|
{ { L"Cantonese (Traditional)" }, { L"yue" } },
|
||||||
|
{ { L"Catalan" }, { L"ca" } },
|
||||||
|
{ { L"Chinese (Simplified)" }, { L"zh-Hans" } },
|
||||||
|
{ { L"Chinese (Traditional)" }, { L"zh-Hant" } },
|
||||||
|
{ { L"Croatian" }, { L"hr" } },
|
||||||
|
{ { L"Czech" }, { L"cs" } },
|
||||||
|
{ { L"Danish" }, { L"da" } },
|
||||||
|
{ { L"Dari" }, { L"prs" } },
|
||||||
|
{ { L"Dutch" }, { L"nl" } },
|
||||||
|
{ { L"English" }, { L"en" } },
|
||||||
|
{ { L"Estonian" }, { L"et" } },
|
||||||
|
{ { L"Fijian" }, { L"fj" } },
|
||||||
|
{ { L"Filipino" }, { L"fil" } },
|
||||||
|
{ { L"Finnish" }, { L"fi" } },
|
||||||
|
{ { L"French" }, { L"fr" } },
|
||||||
|
{ { L"French (Canada)" }, { L"fr-ca" } },
|
||||||
|
{ { L"German" }, { L"de" } },
|
||||||
|
{ { L"Greek" }, { L"el" } },
|
||||||
|
{ { L"Gujarati" }, { L"gu" } },
|
||||||
|
{ { L"Haitian Creole" }, { L"ht" } },
|
||||||
|
{ { L"Hebrew" }, { L"he" } },
|
||||||
|
{ { L"Hindi" }, { L"hi" } },
|
||||||
|
{ { L"Hmong Daw" }, { L"mww" } },
|
||||||
|
{ { L"Hungarian" }, { L"hu" } },
|
||||||
|
{ { L"Icelandic" }, { L"is" } },
|
||||||
|
{ { L"Indonesian" }, { L"id" } },
|
||||||
|
{ { L"Inuktitut" }, { L"iu" } },
|
||||||
|
{ { L"Irish" }, { L"ga" } },
|
||||||
|
{ { L"Italian" }, { L"it" } },
|
||||||
|
{ { L"Japanese" }, { L"ja" } },
|
||||||
|
{ { L"Kannada" }, { L"kn" } },
|
||||||
|
{ { L"Kazakh" }, { L"kk" } },
|
||||||
|
{ { L"Khmer" }, { L"km" } },
|
||||||
|
{ { L"Klingon" }, { L"tlh-Latn" } },
|
||||||
|
{ { L"Korean" }, { L"ko" } },
|
||||||
|
{ { L"Kurdish (Central)" }, { L"ku" } },
|
||||||
|
{ { L"Kurdish (Northern)" }, { L"kmr" } },
|
||||||
|
{ { L"Lao" }, { L"lo" } },
|
||||||
|
{ { L"Latvian" }, { L"lv" } },
|
||||||
|
{ { L"Lithuanian" }, { L"lt" } },
|
||||||
|
{ { L"Malagasy" }, { L"mg" } },
|
||||||
|
{ { L"Malay" }, { L"ms" } },
|
||||||
|
{ { L"Malayalam" }, { L"ml" } },
|
||||||
|
{ { L"Maltese" }, { L"mt" } },
|
||||||
|
{ { L"Maori" }, { L"mi" } },
|
||||||
|
{ { L"Marathi" }, { L"mr" } },
|
||||||
|
{ { L"Myanmar" }, { L"my" } },
|
||||||
|
{ { L"Nepali" }, { L"ne" } },
|
||||||
|
{ { L"Norwegian" }, { L"nb" } },
|
||||||
|
{ { L"Odia" }, { L"or" } },
|
||||||
|
{ { L"Pashto" }, { L"ps" } },
|
||||||
|
{ { L"Persian" }, { L"fa" } },
|
||||||
|
{ { L"Polish" }, { L"pl" } },
|
||||||
|
{ { L"Portuguese (Brazil)" }, { L"pt" } },
|
||||||
|
{ { L"Portuguese (Portugal)" }, { L"pt-pt" } },
|
||||||
|
{ { L"Punjabi" }, { L"pa" } },
|
||||||
|
{ { L"Queretaro Otomi" }, { L"otq" } },
|
||||||
|
{ { L"Romanian" }, { L"ro" } },
|
||||||
|
{ { L"Russian" }, { L"ru" } },
|
||||||
|
{ { L"Samoan" }, { L"sm" } },
|
||||||
|
{ { L"Serbian (Cyrillic)" }, { L"sr-Cyrl" } },
|
||||||
|
{ { L"Serbian (Latin)" }, { L"sr-Latn" } },
|
||||||
|
{ { L"Slovak" }, { L"sk" } },
|
||||||
|
{ { L"Slovenian" }, { L"sl" } },
|
||||||
|
{ { L"Spanish" }, { L"es" } },
|
||||||
|
{ { L"Swahili" }, { L"sw" } },
|
||||||
|
{ { L"Swedish" }, { L"sv" } },
|
||||||
|
{ { L"Tahitian" }, { L"ty" } },
|
||||||
|
{ { L"Tamil" }, { L"ta" } },
|
||||||
|
{ { L"Telugu" }, { L"te" } },
|
||||||
|
{ { L"Thai" }, { L"th" } },
|
||||||
|
{ { L"Tigrinya" }, { L"ti" } },
|
||||||
|
{ { L"Tongan" }, { L"to" } },
|
||||||
|
{ { L"Turkish" }, { L"tr" } },
|
||||||
|
{ { L"Ukrainian" }, { L"uk" } },
|
||||||
|
{ { L"Urdu" }, { L"ur" } },
|
||||||
|
{ { L"Vietnamese" }, { L"vi" } },
|
||||||
|
{ { L"Welsh" }, { L"cy" } },
|
||||||
|
{ { L"Yucatec Maya" }, { L"yua" } },
|
||||||
|
{ { L"?" }, { L"auto-detect" } }
|
||||||
};
|
};
|
||||||
std::wstring autoDetectLanguage = L"auto-detect";
|
|
||||||
|
|
||||||
bool translateSelectedOnly = false, rateLimitAll = true, rateLimitSelected = false, useCache = true;
|
bool translateSelectedOnly = false, useRateLimiter = true, rateLimitSelected = false, useCache = true, useFilter = true;
|
||||||
int tokenCount = 30, tokenRestoreDelay = 60000, maxSentenceSize = 1000;
|
int tokenCount = 30, rateLimitTimespan = 60000, maxSentenceSize = 1000;
|
||||||
|
|
||||||
std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
{
|
{
|
||||||
if (!authKey->empty())
|
if (!tlp.authKey.empty())
|
||||||
{
|
{
|
||||||
std::wstring translateFromComponent = translateFrom.Copy() == autoDetectLanguage ? L"" : L"&from=" + translateFrom.Copy();
|
std::wstring translateFromComponent = tlp.translateFrom == L"?" ? L"" : L"&from=" + codes.at(tlp.translateFrom);
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"api.cognitive.microsofttranslator.com",
|
L"api.cognitive.microsofttranslator.com",
|
||||||
L"POST",
|
L"POST",
|
||||||
FormatString(L"/translate?api-version=3.0&to=%s%s", translateTo.Copy(), translateFromComponent).c_str(),
|
FormatString(L"/translate?api-version=3.0&to=%s%s", codes.at(tlp.translateTo), translateFromComponent).c_str(),
|
||||||
FormatString(R"([{"text":"%s"}])", JSON::Escape(WideStringToString(text))),
|
FormatString(R"([{"text":"%s"}])", JSON::Escape(WideStringToString(text))),
|
||||||
FormatString(L"Content-Type: application/json; charset=UTF-8\r\nOcp-Apim-Subscription-Key:%s", authKey.Copy()).c_str()
|
FormatString(L"Content-Type: application/json; charset=UTF-8\r\nOcp-Apim-Subscription-Key:%s", tlp.authKey).c_str()
|
||||||
})
|
})
|
||||||
if (auto translation = Copy(JSON::Parse(httpRequest.response)[0][L"translations"][0][L"text"].String())) return { true, translation.value() };
|
if (auto translation = Copy(JSON::Parse(httpRequest.response)[0][L"translations"][0][L"text"].String())) return { true, translation.value() };
|
||||||
else return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, httpRequest.response) };
|
else return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, httpRequest.response) };
|
||||||
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static std::atomic<int> i = 0;
|
||||||
|
static Synchronized<std::wstring> token;
|
||||||
|
if (token->empty()) if (HttpRequest httpRequest{ L"Mozilla/5.0 Textractor", L"www.bing.com", L"GET", L"translator" })
|
||||||
|
{
|
||||||
|
std::wstring tokenBuilder;
|
||||||
|
if (auto tokenPos = httpRequest.response.find(L"[" + std::to_wstring(time(nullptr) / 100)); tokenPos != std::string::npos)
|
||||||
|
tokenBuilder = FormatString(L"&key=%s&token=%s", httpRequest.response.substr(tokenPos + 1, 13), httpRequest.response.substr(tokenPos + 16, 32));
|
||||||
|
if (auto tokenPos = httpRequest.response.find(L"IG:\""); tokenPos != std::string::npos)
|
||||||
|
tokenBuilder += L"&IG=" + httpRequest.response.substr(tokenPos + 4, 32);
|
||||||
|
if (auto tokenPos = httpRequest.response.find(L"data-iid=\""); tokenPos != std::string::npos)
|
||||||
|
tokenBuilder += L"&IID=" + httpRequest.response.substr(tokenPos + 10, 15);
|
||||||
|
if (!tokenBuilder.empty()) token->assign(tokenBuilder);
|
||||||
|
else return { false, FormatString(L"%s: %s\ntoken not found", TRANSLATION_ERROR, httpRequest.response) };
|
||||||
|
}
|
||||||
|
else return { false, FormatString(L"%s: could not acquire token", TRANSLATION_ERROR) };
|
||||||
|
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"www.bing.com",
|
L"www.bing.com",
|
||||||
L"POST",
|
L"POST",
|
||||||
FormatString(L"/ttranslatev3?fromLang=%s&to=%s&text=%s", translateFrom.Copy(), translateTo.Copy(), Escape(text)).c_str()
|
FormatString(L"/ttranslatev3?fromLang=%s&to=%s&text=%s%s.%d", codes.at(tlp.translateFrom), codes.at(tlp.translateTo), Escape(text), token.Copy(), i++).c_str()
|
||||||
})
|
})
|
||||||
if (auto translation = Copy(JSON::Parse(httpRequest.response)[0][L"translations"][0][L"text"].String())) return { true, translation.value() };
|
if (auto translation = Copy(JSON::Parse(httpRequest.response)[0][L"translations"][0][L"text"].String())) return { true, translation.value() };
|
||||||
else return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, httpRequest.response) };
|
else return { false, FormatString(L"%s (token=%s): %s", TRANSLATION_ERROR, std::exchange(token.Acquire().contents, L""), httpRequest.response) };
|
||||||
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,7 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
static constexpr C endImpl[5] = { '|', 'E', 'N', 'D', '|' };
|
static constexpr C endImpl[5] = { '|', 'E', 'N', 'D', '|' };
|
||||||
static constexpr std::basic_string_view end{ endImpl, 5 };
|
static constexpr std::basic_string_view<C> end{ endImpl, 5 };
|
||||||
|
|
||||||
std::basic_streambuf<char>& streambuf;
|
std::basic_streambuf<char>& streambuf;
|
||||||
std::basic_string<C> buffer;
|
std::basic_string<C> buffer;
|
||||||
|
@ -1,54 +1,153 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
#include "network.h"
|
#include "network.h"
|
||||||
#include <random>
|
#include <random>
|
||||||
|
|
||||||
extern const wchar_t* TRANSLATION_ERROR;
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
extern const char* USE_PREV_SENTENCE_CONTEXT;
|
|
||||||
|
|
||||||
extern Synchronized<std::wstring> translateTo, translateFrom, authKey;
|
|
||||||
|
|
||||||
const char* TRANSLATION_PROVIDER = "DeepL Translate";
|
const char* TRANSLATION_PROVIDER = "DeepL Translate";
|
||||||
const char* GET_API_KEY_FROM = "https://www.deepl.com/pro.html";
|
const char* GET_API_KEY_FROM = "https://www.deepl.com/pro.html#developer";
|
||||||
QStringList languages
|
extern const QStringList languagesTo
|
||||||
{
|
{
|
||||||
"Chinese (simplified): ZH",
|
"Arabic",
|
||||||
"Dutch: NL",
|
"Bulgarian",
|
||||||
"English: EN",
|
"Czech",
|
||||||
"French: FR",
|
"Danish",
|
||||||
"German: DE",
|
"German",
|
||||||
"Italian: IT",
|
"Greek",
|
||||||
"Japanese: JA",
|
"English (backward compatibility)",
|
||||||
"Polish: PL",
|
"English (British)",
|
||||||
"Portuguese: PT",
|
"English (American)",
|
||||||
"Russian: RU",
|
"Spanish",
|
||||||
"Spanish: ES",
|
"Estonian",
|
||||||
|
"Finnish",
|
||||||
|
"French",
|
||||||
|
"Hungarian",
|
||||||
|
"Indonesian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Korean",
|
||||||
|
"Lithuanian",
|
||||||
|
"Latvian",
|
||||||
|
"Norwegian Bokmål",
|
||||||
|
"Dutch",
|
||||||
|
"Polish",
|
||||||
|
"Portuguese (backward compatibility)",
|
||||||
|
"Portuguese (Brazilian)",
|
||||||
|
"Portuguese (all Portuguese variants excluding Brazilian Portuguese)",
|
||||||
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Swedish",
|
||||||
|
"Turkish",
|
||||||
|
"Ukrainian",
|
||||||
|
"Chinese (backward compatibility)",
|
||||||
|
"Chinese (simplified)",
|
||||||
|
"Chinese (traditional)"
|
||||||
|
},
|
||||||
|
languagesFrom
|
||||||
|
{
|
||||||
|
"Arabic",
|
||||||
|
"Bulgarian",
|
||||||
|
"Chinese (all Chinese variants)",
|
||||||
|
"Czech",
|
||||||
|
"Danish",
|
||||||
|
"Dutch",
|
||||||
|
"English (all English variants)",
|
||||||
|
"Estonian",
|
||||||
|
"Finnish",
|
||||||
|
"French",
|
||||||
|
"German",
|
||||||
|
"Greek",
|
||||||
|
"Hungarian",
|
||||||
|
"Indonesian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Korean",
|
||||||
|
"Latvian",
|
||||||
|
"Lithuanian",
|
||||||
|
"Norwegian Bokmål",
|
||||||
|
"Polish",
|
||||||
|
"Portuguese (all Portuguese variants)",
|
||||||
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Spanish",
|
||||||
|
"Swedish",
|
||||||
|
"Turkish",
|
||||||
|
"Ukrainian"
|
||||||
};
|
};
|
||||||
std::wstring autoDetectLanguage = L"auto";
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Arabic" }, { L"AR" } },
|
||||||
|
{ { L"Bulgarian" }, { L"BG" } },
|
||||||
|
{ { L"Czech" }, { L"CS" } },
|
||||||
|
{ { L"Danish" }, { L"DA" } },
|
||||||
|
{ { L"German" }, { L"DE" } },
|
||||||
|
{ { L"Greek" }, { L"EL" } },
|
||||||
|
{ { L"English (all English variants)" }, { L"EN" } },
|
||||||
|
{ { L"English (backward compatibility)" }, { L"EN" } },
|
||||||
|
{ { L"English (British)" }, { L"EN-GB" } },
|
||||||
|
{ { L"English (American)" }, { L"EN-US" } },
|
||||||
|
{ { L"Spanish" }, { L"ES" } },
|
||||||
|
{ { L"Estonian" }, { L"ET" } },
|
||||||
|
{ { L"Finnish" }, { L"FI" } },
|
||||||
|
{ { L"French" }, { L"FR" } },
|
||||||
|
{ { L"Hungarian" }, { L"HU" } },
|
||||||
|
{ { L"Indonesian" }, { L"ID" } },
|
||||||
|
{ { L"Italian" }, { L"IT" } },
|
||||||
|
{ { L"Japanese" }, { L"JA" } },
|
||||||
|
{ { L"Korean" }, { L"KO" } },
|
||||||
|
{ { L"Lithuanian" }, { L"LT" } },
|
||||||
|
{ { L"Latvian" }, { L"LV" } },
|
||||||
|
{ { L"Norwegian Bokmål" }, { L"NB" } },
|
||||||
|
{ { L"Dutch" }, { L"NL" } },
|
||||||
|
{ { L"Polish" }, { L"PL" } },
|
||||||
|
{ { L"Portuguese (all Portuguese variants)" }, { L"PT" } },
|
||||||
|
{ { L"Portuguese (backward compatibility)" }, { L"PT" } },
|
||||||
|
{ { L"Portuguese (Brazilian)" }, { L"PT-BR" } },
|
||||||
|
{ { L"Portuguese (all Portuguese variants excluding Brazilian Portuguese)" }, { L"PT-PT" } },
|
||||||
|
{ { L"Romanian" }, { L"RO" } },
|
||||||
|
{ { L"Russian" }, { L"RU" } },
|
||||||
|
{ { L"Slovak" }, { L"SK" } },
|
||||||
|
{ { L"Slovenian" }, { L"SL" } },
|
||||||
|
{ { L"Swedish" }, { L"SV" } },
|
||||||
|
{ { L"Turkish" }, { L"TR" } },
|
||||||
|
{ { L"Ukrainian" }, { L"UK" } },
|
||||||
|
{ { L"Chinese (all Chinese variants)" }, { L"ZH" } },
|
||||||
|
{ { L"Chinese (backward compatibility)" }, { L"ZH" } },
|
||||||
|
{ { L"Chinese (simplified)" }, { L"ZH-HANS" } },
|
||||||
|
{ { L"Chinese (traditional)" }, { L"ZH-HANT" } },
|
||||||
|
{ { L"?" }, { L"auto" } }
|
||||||
|
|
||||||
bool translateSelectedOnly = true, rateLimitAll = true, rateLimitSelected = true, useCache = true;
|
};
|
||||||
int tokenCount = 10, tokenRestoreDelay = 60000, maxSentenceSize = 1000;
|
|
||||||
|
bool translateSelectedOnly = true, useRateLimiter = true, rateLimitSelected = true, useCache = true, useFilter = true;
|
||||||
|
int tokenCount = 10, rateLimitTimespan = 60000, maxSentenceSize = 1000;
|
||||||
|
|
||||||
enum KeyType { CAT, REST };
|
enum KeyType { CAT, REST };
|
||||||
int keyType = CAT;
|
int keyType = REST;
|
||||||
|
|
||||||
std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
{
|
{
|
||||||
if (!authKey->empty())
|
if (!tlp.authKey.empty())
|
||||||
{
|
{
|
||||||
std::string translateFromComponent = translateFrom.Copy() == autoDetectLanguage ? "" : "&source_lang=" + WideStringToString(translateFrom.Copy());
|
std::string translateFromComponent = tlp.translateFrom == L"?" ? "" : "&source_lang=" + WideStringToString(codes.at(tlp.translateFrom));
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"api.deepl.com",
|
tlp.authKey.find(L":fx") == std::string::npos ? L"api.deepl.com" : L"api-free.deepl.com",
|
||||||
L"POST",
|
L"POST",
|
||||||
keyType == CAT ? L"/v1/translate" : L"/v2/translate",
|
keyType == CAT ? L"/v1/translate" : L"/v2/translate",
|
||||||
FormatString("text=%S&auth_key=%S&target_lang=%S", Escape(text), authKey.Copy(), translateTo.Copy()) + translateFromComponent,
|
FormatString("text=%S&auth_key=%S&target_lang=%S", Escape(text), tlp.authKey, codes.at(tlp.translateTo)) + translateFromComponent,
|
||||||
L"Content-Type: application/x-www-form-urlencoded"
|
L"Content-Type: application/x-www-form-urlencoded"
|
||||||
}; httpRequest && (!httpRequest.response.empty() || (httpRequest = HttpRequest{
|
}; httpRequest && (httpRequest.response.find(L"translations") != std::string::npos || (httpRequest = HttpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"api.deepl.com",
|
tlp.authKey.find(L":fx") == std::string::npos ? L"api.deepl.com" : L"api-free.deepl.com",
|
||||||
L"POST",
|
L"POST",
|
||||||
(keyType = !keyType) == CAT ? L"/v1/translate" : L"/v2/translate",
|
(keyType = !keyType) == CAT ? L"/v1/translate" : L"/v2/translate",
|
||||||
FormatString("text=%S&auth_key=%S&target_lang=%S", Escape(text), authKey.Copy(), translateTo.Copy()) + translateFromComponent,
|
FormatString("text=%S&auth_key=%S&target_lang=%S", Escape(text), tlp.authKey, codes.at(tlp.translateTo)) + translateFromComponent,
|
||||||
L"Content-Type: application/x-www-form-urlencoded"
|
L"Content-Type: application/x-www-form-urlencoded"
|
||||||
})))
|
})))
|
||||||
// Response formatted as JSON: translation starts with text":" and ends with "}]
|
// Response formatted as JSON: translation starts with text":" and ends with "}]
|
||||||
@ -58,9 +157,8 @@ std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// the following code was reverse engineered from the DeepL website; it's as close as I could make it but I'm not sure what parts of this could be removed and still have it work
|
// the following code was reverse engineered from the DeepL website; it's as close as I could make it but I'm not sure what parts of this could be removed and still have it work
|
||||||
|
int id = 10000 * std::uniform_int_distribution(0, 9999)(std::random_device()) + 1;
|
||||||
int64_t r = _time64(nullptr), n = std::count(text.begin(), text.end(), L'i') + 1;
|
int64_t r = _time64(nullptr), n = std::count(text.begin(), text.end(), L'i') + 1;
|
||||||
thread_local auto generator = std::mt19937(std::random_device()());
|
|
||||||
int id = 10000 * std::uniform_int_distribution(0, 9999)(generator) + 1;
|
|
||||||
// user_preferred_langs? what should priority be? does timestamp do anything? other translation quality options?
|
// user_preferred_langs? what should priority be? does timestamp do anything? other translation quality options?
|
||||||
auto body = FormatString(R"(
|
auto body = FormatString(R"(
|
||||||
{
|
{
|
||||||
@ -71,7 +169,7 @@ std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
|||||||
"priority": -1,
|
"priority": -1,
|
||||||
"timestamp": %lld,
|
"timestamp": %lld,
|
||||||
"lang": {
|
"lang": {
|
||||||
"target_lang": "%S",
|
"target_lang": "%.2S",
|
||||||
"source_lang_user_selected": "%S"
|
"source_lang_user_selected": "%S"
|
||||||
},
|
},
|
||||||
"jobs": [{
|
"jobs": [{
|
||||||
@ -84,7 +182,7 @@ std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
|||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)", id, r + (n - r % n), translateTo.Copy(), translateFrom.Copy(), JSON::Escape(WideStringToString(text)));
|
)", id, r + (n - r % n), codes.at(tlp.translateTo), codes.at(tlp.translateFrom), JSON::Escape(WideStringToString(text)));
|
||||||
// missing accept-encoding header since it fucks up HttpRequest
|
// missing accept-encoding header since it fucks up HttpRequest
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
|
@ -1,51 +1,53 @@
|
|||||||
#include "devtools.h"
|
#include "devtools.h"
|
||||||
|
#include "module.h"
|
||||||
|
#include <ppltasks.h>
|
||||||
|
#include <ShlObj.h>
|
||||||
#include <QWebSocket>
|
#include <QWebSocket>
|
||||||
#include <QMetaEnum>
|
#include <QMetaEnum>
|
||||||
#include <ppltasks.h>
|
#include <QFileDialog>
|
||||||
|
#include <QMouseEvent>
|
||||||
|
|
||||||
|
extern const char* CHROME_LOCATION;
|
||||||
|
extern const char* START_DEVTOOLS;
|
||||||
|
extern const char* STOP_DEVTOOLS;
|
||||||
|
extern const char* HIDE_CHROME;
|
||||||
|
extern const char* DEVTOOLS_STATUS;
|
||||||
|
extern const char* AUTO_START;
|
||||||
|
|
||||||
|
extern const char* TRANSLATION_PROVIDER;
|
||||||
|
|
||||||
|
extern QFormLayout* display;
|
||||||
|
extern Settings settings;
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
std::function<void(QString)> OnStatusChanged = Swallow;
|
QLabel* statusLabel;
|
||||||
PROCESS_INFORMATION processInfo = {};
|
AutoHandle<> process = NULL;
|
||||||
std::atomic<int> idCounter = 0;
|
|
||||||
std::mutex devToolsMutex;
|
|
||||||
QWebSocket webSocket;
|
QWebSocket webSocket;
|
||||||
std::unordered_map<int, concurrency::task_completion_event<JSON::Value<wchar_t>>> mapQueue;
|
std::atomic<int> idCounter = 0;
|
||||||
auto _ = ([]
|
Synchronized<std::unordered_map<int, concurrency::task_completion_event<JSON::Value<wchar_t>>>> mapQueue;
|
||||||
{
|
|
||||||
QObject::connect(&webSocket, &QWebSocket::stateChanged,
|
|
||||||
[](QAbstractSocket::SocketState state) { OnStatusChanged(QMetaEnum::fromType<QAbstractSocket::SocketState>().valueToKey(state)); }
|
|
||||||
);
|
|
||||||
QObject::connect(&webSocket, &QWebSocket::textMessageReceived, [](QString message)
|
|
||||||
{
|
|
||||||
auto result = JSON::Parse(S(message));
|
|
||||||
std::scoped_lock lock(devToolsMutex);
|
|
||||||
if (auto id = result[L"id"].Number()) if (auto request = mapQueue.find((int)*id); request != mapQueue.end())
|
|
||||||
{
|
|
||||||
request->second.set(result);
|
|
||||||
mapQueue.erase(request);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}(), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace DevTools
|
void StatusChanged(QString status)
|
||||||
{
|
{
|
||||||
void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless)
|
QMetaObject::invokeMethod(statusLabel, std::bind(&QLabel::setText, statusLabel, status));
|
||||||
|
}
|
||||||
|
void Start(std::wstring chromePath, bool headless)
|
||||||
{
|
{
|
||||||
OnStatusChanged = statusChanged;
|
if (process) DevTools::Close();
|
||||||
DWORD exitCode = 0;
|
|
||||||
auto args = FormatString(
|
auto args = FormatString(
|
||||||
L"%s --proxy-server=direct:// --disable-extensions --disable-gpu --user-data-dir=%s\\devtoolscache --remote-debugging-port=9222",
|
L"%s --proxy-server=direct:// --disable-extensions --disable-gpu --no-first-run --user-data-dir=\"%s\\devtoolscache\" --remote-debugging-port=9222",
|
||||||
path,
|
chromePath,
|
||||||
std::filesystem::current_path().wstring()
|
std::filesystem::current_path().wstring()
|
||||||
);
|
);
|
||||||
if (headless) args += L" --headless";
|
args += headless ? L" --window-size=1920,1080 --headless" : L" --window-size=850,900";
|
||||||
|
DWORD exitCode = 0;
|
||||||
STARTUPINFOW DUMMY = { sizeof(DUMMY) };
|
STARTUPINFOW DUMMY = { sizeof(DUMMY) };
|
||||||
if ((GetExitCodeProcess(processInfo.hProcess, &exitCode) && exitCode == STILL_ACTIVE) ||
|
PROCESS_INFORMATION processInfo = {};
|
||||||
CreateProcessW(NULL, args.data(), nullptr, nullptr, FALSE, 0, nullptr, nullptr, &DUMMY, &processInfo)
|
if (!CreateProcessW(NULL, args.data(), nullptr, nullptr, FALSE, 0, nullptr, nullptr, &DUMMY, &processInfo)) return StatusChanged("StartupFailed");
|
||||||
)
|
CloseHandle(processInfo.hThread);
|
||||||
{
|
process = processInfo.hProcess;
|
||||||
|
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"127.0.0.1",
|
L"127.0.0.1",
|
||||||
@ -57,59 +59,115 @@ namespace DevTools
|
|||||||
NULL,
|
NULL,
|
||||||
WINHTTP_FLAG_ESCAPE_DISABLE
|
WINHTTP_FLAG_ESCAPE_DISABLE
|
||||||
})
|
})
|
||||||
{
|
|
||||||
if (auto list = Copy(JSON::Parse(httpRequest.response).Array())) if (auto it = std::find_if(
|
if (auto list = Copy(JSON::Parse(httpRequest.response).Array())) if (auto it = std::find_if(
|
||||||
list->begin(),
|
list->begin(),
|
||||||
list->end(),
|
list->end(),
|
||||||
[](const JSON::Value<wchar_t>& object) { return object[L"type"].String() && *object[L"type"].String() == L"page" && object[L"webSocketDebuggerUrl"].String(); }
|
[](const JSON::Value<wchar_t>& object) { return object[L"type"].String() && *object[L"type"].String() == L"page" && object[L"webSocketDebuggerUrl"].String(); }
|
||||||
); it != list->end())
|
); it != list->end()) return webSocket.open(S(*(*it)[L"webSocketDebuggerUrl"].String()));
|
||||||
|
|
||||||
|
StatusChanged("ConnectingFailed");
|
||||||
|
}
|
||||||
|
|
||||||
|
auto _ = ([]
|
||||||
{
|
{
|
||||||
std::scoped_lock lock(devToolsMutex);
|
QObject::connect(&webSocket, &QWebSocket::stateChanged,
|
||||||
webSocket.open(S(*(*it)[L"webSocketDebuggerUrl"].String()));
|
[](QAbstractSocket::SocketState state) { StatusChanged(QMetaEnum::fromType<QAbstractSocket::SocketState>().valueToKey(state)); });
|
||||||
return;
|
QObject::connect(&webSocket, &QWebSocket::textMessageReceived, [](QString message)
|
||||||
|
{
|
||||||
|
auto result = JSON::Parse(S(message));
|
||||||
|
auto mapQueue = ::mapQueue.Acquire();
|
||||||
|
if (auto id = result[L"id"].Number()) if (auto request = mapQueue->find((int)*id); request != mapQueue->end())
|
||||||
|
{
|
||||||
|
request->second.set(result);
|
||||||
|
mapQueue->erase(request);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace DevTools
|
||||||
|
{
|
||||||
|
void Initialize()
|
||||||
|
{
|
||||||
|
QString chromePath = settings.value(CHROME_LOCATION).toString();
|
||||||
|
if (chromePath.isEmpty())
|
||||||
|
{
|
||||||
|
for (auto [_, process] : GetAllProcesses())
|
||||||
|
if (process && (process->find(L"\\chrome.exe") != std::string::npos || process->find(L"\\msedge.exe") != std::string::npos)) chromePath = S(process.value());
|
||||||
|
wchar_t programFiles[MAX_PATH + 100] = {};
|
||||||
|
for (auto folder : { CSIDL_PROGRAM_FILESX86, CSIDL_PROGRAM_FILES, CSIDL_LOCAL_APPDATA })
|
||||||
|
{
|
||||||
|
SHGetFolderPathW(NULL, folder, NULL, SHGFP_TYPE_CURRENT, programFiles);
|
||||||
|
wcscat_s(programFiles, L"/Google/Chrome/Application/chrome.exe");
|
||||||
|
if (std::filesystem::exists(programFiles)) chromePath = S(programFiles);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
OnStatusChanged("Failed Connection");
|
auto chromePathEdit = new QLineEdit(chromePath);
|
||||||
|
static struct : QObject
|
||||||
|
{
|
||||||
|
bool eventFilter(QObject* object, QEvent* event)
|
||||||
|
{
|
||||||
|
if (auto mouseEvent = dynamic_cast<QMouseEvent*>(event))
|
||||||
|
if (mouseEvent->button() == Qt::LeftButton)
|
||||||
|
if (QString chromePath = QFileDialog::getOpenFileName(nullptr, TRANSLATION_PROVIDER, "/", "Google Chrome (*.exe)"); !chromePath.isEmpty())
|
||||||
|
((QLineEdit*)object)->setText(chromePath);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
else OnStatusChanged("Failed Startup");
|
} chromeSelector;
|
||||||
|
chromePathEdit->installEventFilter(&chromeSelector);
|
||||||
|
QObject::connect(chromePathEdit, &QLineEdit::textChanged, [chromePathEdit](QString path) { settings.setValue(CHROME_LOCATION, path); });
|
||||||
|
display->addRow(CHROME_LOCATION, chromePathEdit);
|
||||||
|
auto headlessCheck = new QCheckBox();
|
||||||
|
auto startButton = new QPushButton(START_DEVTOOLS), stopButton = new QPushButton(STOP_DEVTOOLS);
|
||||||
|
headlessCheck->setChecked(settings.value(HIDE_CHROME, true).toBool());
|
||||||
|
QObject::connect(headlessCheck, &QCheckBox::clicked, [](bool headless) { settings.setValue(HIDE_CHROME, headless); });
|
||||||
|
QObject::connect(startButton, &QPushButton::clicked, [chromePathEdit, headlessCheck] { Start(S(chromePathEdit->text()), headlessCheck->isChecked()); });
|
||||||
|
QObject::connect(stopButton, &QPushButton::clicked, &Close);
|
||||||
|
auto buttons = new QHBoxLayout();
|
||||||
|
buttons->addWidget(startButton);
|
||||||
|
buttons->addWidget(stopButton);
|
||||||
|
display->addRow(HIDE_CHROME, headlessCheck);
|
||||||
|
auto autoStartCheck = new QCheckBox();
|
||||||
|
autoStartCheck->setChecked(settings.value(AUTO_START, false).toBool());
|
||||||
|
QObject::connect(autoStartCheck, &QCheckBox::clicked, [](bool autoStart) { settings.setValue(AUTO_START, autoStart); });
|
||||||
|
display->addRow(AUTO_START, autoStartCheck);
|
||||||
|
display->addRow(buttons);
|
||||||
|
statusLabel = new QLabel("Stopped");
|
||||||
|
statusLabel->setFrameStyle(QFrame::Panel | QFrame::Sunken);
|
||||||
|
display->addRow(DEVTOOLS_STATUS, statusLabel);
|
||||||
|
if (autoStartCheck->isChecked()) QMetaObject::invokeMethod(startButton, &QPushButton::click, Qt::QueuedConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Close()
|
void Close()
|
||||||
{
|
{
|
||||||
std::scoped_lock lock(devToolsMutex);
|
|
||||||
for (const auto& [_, task] : mapQueue) task.set_exception(std::runtime_error("closed"));
|
|
||||||
webSocket.close();
|
webSocket.close();
|
||||||
mapQueue.clear();
|
for (const auto& [_, task] : mapQueue.Acquire().contents) task.set_exception(std::runtime_error("closed"));
|
||||||
DWORD exitCode = 0;
|
mapQueue->clear();
|
||||||
if (GetExitCodeProcess(processInfo.hProcess, &exitCode) && exitCode == STILL_ACTIVE)
|
|
||||||
|
if (process)
|
||||||
{
|
{
|
||||||
TerminateProcess(processInfo.hProcess, 0);
|
TerminateProcess(process, 0);
|
||||||
WaitForSingleObject(processInfo.hProcess, 100);
|
WaitForSingleObject(process, 1000);
|
||||||
CloseHandle(processInfo.hProcess);
|
for (int retry = 0; ++retry < 20; Sleep(100))
|
||||||
CloseHandle(processInfo.hThread);
|
try { std::filesystem::remove_all(L"devtoolscache"); break; }
|
||||||
|
catch (std::filesystem::filesystem_error) { continue; }
|
||||||
}
|
}
|
||||||
try { std::filesystem::remove_all(L"devtoolscache"); } catch (std::filesystem::filesystem_error) {}
|
process = NULL;
|
||||||
OnStatusChanged("Stopped");
|
StatusChanged("Stopped");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Connected()
|
bool Connected()
|
||||||
{
|
{
|
||||||
std::scoped_lock lock(devToolsMutex);
|
|
||||||
return webSocket.state() == QAbstractSocket::ConnectedState;
|
return webSocket.state() == QAbstractSocket::ConnectedState;
|
||||||
}
|
}
|
||||||
|
|
||||||
JSON::Value<wchar_t> SendRequest(const std::wstring& method, const std::wstring& params)
|
JSON::Value<wchar_t> SendRequest(const char* method, const std::wstring& params)
|
||||||
{
|
{
|
||||||
concurrency::task_completion_event<JSON::Value<wchar_t>> response;
|
concurrency::task_completion_event<JSON::Value<wchar_t>> response;
|
||||||
int id = idCounter += 1;
|
int id = idCounter += 1;
|
||||||
auto message = FormatString(LR"({"id":%d,"method":"%s","params":%s})", id, method, params);
|
if (!Connected()) return {};
|
||||||
{
|
mapQueue->try_emplace(id, response);
|
||||||
std::scoped_lock lock(devToolsMutex);
|
QMetaObject::invokeMethod(&webSocket, std::bind(&QWebSocket::sendTextMessage, &webSocket, S(FormatString(LR"({"id":%d,"method":"%S","params":%s})", id, method, params))));
|
||||||
if (webSocket.state() != QAbstractSocket::ConnectedState) return {};
|
|
||||||
mapQueue.try_emplace(id, response);
|
|
||||||
webSocket.sendTextMessage(S(message));
|
|
||||||
webSocket.flush();
|
|
||||||
}
|
|
||||||
try { if (auto result = create_task(response).get()[L"result"]) return result; } catch (...) {}
|
try { if (auto result = create_task(response).get()[L"result"]) return result; } catch (...) {}
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
@ -3,8 +3,8 @@
|
|||||||
|
|
||||||
namespace DevTools
|
namespace DevTools
|
||||||
{
|
{
|
||||||
void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless);
|
void Initialize();
|
||||||
void Close();
|
void Close();
|
||||||
bool Connected();
|
bool Connected();
|
||||||
JSON::Value<wchar_t> SendRequest(const std::wstring& method, const std::wstring& params = L"{}");
|
JSON::Value<wchar_t> SendRequest(const char* method, const std::wstring& params = L"{}");
|
||||||
}
|
}
|
||||||
|
@ -1,40 +1,104 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
#include "devtools.h"
|
#include "devtools.h"
|
||||||
#include <ShlObj.h>
|
|
||||||
|
|
||||||
|
extern const wchar_t* ERROR_START_CHROME;
|
||||||
extern const wchar_t* TRANSLATION_ERROR;
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
extern Synchronized<std::wstring> translateTo, translateFrom;
|
|
||||||
extern QFormLayout* display;
|
|
||||||
extern Settings settings;
|
|
||||||
|
|
||||||
const char* TRANSLATION_PROVIDER = "DevTools DeepL Translate";
|
const char* TRANSLATION_PROVIDER = "DevTools DeepL Translate";
|
||||||
const char* GET_API_KEY_FROM = nullptr;
|
const char* GET_API_KEY_FROM = nullptr;
|
||||||
bool translateSelectedOnly = true, rateLimitAll = false, rateLimitSelected = false, useCache = true;
|
|
||||||
int tokenCount = 30, tokenRestoreDelay = 60000, maxSentenceSize = 2500;
|
|
||||||
|
|
||||||
extern const char* CHROME_LOCATION;
|
extern const QStringList languagesTo
|
||||||
extern const char* START_DEVTOOLS;
|
|
||||||
extern const char* STOP_DEVTOOLS;
|
|
||||||
extern const char* HEADLESS_MODE;
|
|
||||||
extern const char* DEVTOOLS_STATUS;
|
|
||||||
extern const char* AUTO_START;
|
|
||||||
extern const wchar_t* ERROR_START_CHROME;
|
|
||||||
|
|
||||||
QStringList languages
|
|
||||||
{
|
{
|
||||||
"Chinese (simplified): zh",
|
"Bulgarian",
|
||||||
"Dutch: nl",
|
"Chinese (Simplified)",
|
||||||
"English: en",
|
"Czech",
|
||||||
"French: fr",
|
"Danish",
|
||||||
"German: de",
|
"Dutch",
|
||||||
"Italian: it",
|
"English (American)",
|
||||||
"Japanese: ja",
|
"English (British)",
|
||||||
"Polish: pl",
|
"Estonian",
|
||||||
"Portuguese: pt",
|
"Finnish",
|
||||||
"Russian: ru",
|
"French",
|
||||||
"Spanish: es",
|
"German",
|
||||||
|
"Greek",
|
||||||
|
"Hungarian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Latvian",
|
||||||
|
"Lithuanian",
|
||||||
|
"Polish",
|
||||||
|
"Portuguese",
|
||||||
|
"Portuguese (Brazilian)",
|
||||||
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Spanish",
|
||||||
|
"Swedish"
|
||||||
|
},
|
||||||
|
languagesFrom =
|
||||||
|
{
|
||||||
|
"Bulgarian",
|
||||||
|
"Chinese",
|
||||||
|
"Czech",
|
||||||
|
"Danish",
|
||||||
|
"Dutch",
|
||||||
|
"English",
|
||||||
|
"Estonian",
|
||||||
|
"Finnish",
|
||||||
|
"French",
|
||||||
|
"German",
|
||||||
|
"Greek",
|
||||||
|
"Hungarian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Latvian",
|
||||||
|
"Lithuanian",
|
||||||
|
"Polish",
|
||||||
|
"Portuguese",
|
||||||
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Spanish",
|
||||||
|
"Swedish"
|
||||||
};
|
};
|
||||||
std::wstring autoDetectLanguage = L"auto";
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Bulgarian" }, { L"Bulgarian" } },
|
||||||
|
{ { L"Chinese" }, { L"Chinese" } },
|
||||||
|
{ { L"Chinese (Simplified)" }, { L"Chinese (simplified)" } },
|
||||||
|
{ { L"Czech" }, { L"Czech" } },
|
||||||
|
{ { L"Danish" }, { L"Danish" } },
|
||||||
|
{ { L"Dutch" }, { L"Dutch" } },
|
||||||
|
{ { L"English" }, { L"English" } },
|
||||||
|
{ { L"English (American)" }, { L"English (American)" } },
|
||||||
|
{ { L"English (British)" }, { L"English (British)" } },
|
||||||
|
{ { L"Estonian" }, { L"Estonian" } },
|
||||||
|
{ { L"Finnish" }, { L"Finnish" } },
|
||||||
|
{ { L"French" }, { L"French" } },
|
||||||
|
{ { L"German" }, { L"German" } },
|
||||||
|
{ { L"Greek" }, { L"Greek" } },
|
||||||
|
{ { L"Hungarian" }, { L"Hungarian" } },
|
||||||
|
{ { L"Italian" }, { L"Italian" } },
|
||||||
|
{ { L"Japanese" }, { L"Japanese" } },
|
||||||
|
{ { L"Latvian" }, { L"Latvian" } },
|
||||||
|
{ { L"Lithuanian" }, { L"Lithuanian" } },
|
||||||
|
{ { L"Polish" }, { L"Polish" } },
|
||||||
|
{ { L"Portuguese" }, { L"Portuguese" } },
|
||||||
|
{ { L"Portuguese (Brazilian)" }, { L"Portuguese (Brazilian)" } },
|
||||||
|
{ { L"Romanian" }, { L"Romanian" } },
|
||||||
|
{ { L"Russian" }, { L"Russian" } },
|
||||||
|
{ { L"Slovak" }, { L"Slovak" } },
|
||||||
|
{ { L"Slovenian" }, { L"Slovenian" } },
|
||||||
|
{ { L"Spanish" }, { L"Spanish" } },
|
||||||
|
{ { L"Swedish" }, { L"Swedish" } },
|
||||||
|
{ { L"?" }, { L"Detect language" } }
|
||||||
|
};
|
||||||
|
|
||||||
|
bool translateSelectedOnly = true, useRateLimiter = true, rateLimitSelected = false, useCache = true, useFilter = true;
|
||||||
|
int tokenCount = 30, rateLimitTimespan = 60000, maxSentenceSize = 2500;
|
||||||
|
|
||||||
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
||||||
{
|
{
|
||||||
@ -42,63 +106,7 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved
|
|||||||
{
|
{
|
||||||
case DLL_PROCESS_ATTACH:
|
case DLL_PROCESS_ATTACH:
|
||||||
{
|
{
|
||||||
QString chromePath = settings.value(CHROME_LOCATION).toString();
|
DevTools::Initialize();
|
||||||
wchar_t programFiles[MAX_PATH + 100] = {};
|
|
||||||
if (chromePath.isEmpty()) for (auto folder : { CSIDL_PROGRAM_FILESX86, CSIDL_PROGRAM_FILES, CSIDL_LOCAL_APPDATA })
|
|
||||||
{
|
|
||||||
SHGetFolderPathW(NULL, folder, NULL, SHGFP_TYPE_CURRENT, programFiles);
|
|
||||||
wcscat_s(programFiles, L"/Google/Chrome/Application/chrome.exe");
|
|
||||||
if (std::filesystem::exists(programFiles)) chromePath = S(programFiles);
|
|
||||||
}
|
|
||||||
auto chromePathEdit = new QLineEdit(chromePath);
|
|
||||||
QObject::connect(chromePathEdit, &QLineEdit::textChanged, [chromePathEdit](QString path) { settings.setValue(CHROME_LOCATION, path); });
|
|
||||||
display->addRow(CHROME_LOCATION, chromePathEdit);
|
|
||||||
auto statusLabel = new QLabel("Stopped");
|
|
||||||
auto startButton = new QPushButton(START_DEVTOOLS), stopButton = new QPushButton(STOP_DEVTOOLS);
|
|
||||||
auto headlessCheck = new QCheckBox();
|
|
||||||
headlessCheck->setChecked(settings.value(HEADLESS_MODE, true).toBool());
|
|
||||||
QObject::connect(headlessCheck, &QCheckBox::clicked, [](bool headless) { settings.setValue(HEADLESS_MODE, headless); });
|
|
||||||
QObject::connect(startButton, &QPushButton::clicked, [statusLabel, chromePathEdit, headlessCheck] {
|
|
||||||
DevTools::Start(
|
|
||||||
S(chromePathEdit->text()),
|
|
||||||
[statusLabel](QString status)
|
|
||||||
{
|
|
||||||
QMetaObject::invokeMethod(statusLabel, std::bind(&QLabel::setText, statusLabel, status));
|
|
||||||
if (status == "ConnectedState") std::thread([]
|
|
||||||
{
|
|
||||||
if (HttpRequest httpRequest{
|
|
||||||
L"Mozilla/5.0 Textractor",
|
|
||||||
L"127.0.0.1",
|
|
||||||
L"POST",
|
|
||||||
L"/json/version",
|
|
||||||
"",
|
|
||||||
NULL,
|
|
||||||
9222,
|
|
||||||
NULL,
|
|
||||||
WINHTTP_FLAG_ESCAPE_DISABLE
|
|
||||||
})
|
|
||||||
if (auto userAgent = Copy(JSON::Parse(httpRequest.response)[L"User-Agent"].String()))
|
|
||||||
if (userAgent->find(L"Headless") != std::string::npos)
|
|
||||||
DevTools::SendRequest(L"Network.setUserAgentOverride",
|
|
||||||
FormatString(LR"({"userAgent":"%s"})", userAgent->replace(userAgent->find(L"Headless"), 8, L"")));
|
|
||||||
}).detach();
|
|
||||||
},
|
|
||||||
headlessCheck->isChecked()
|
|
||||||
);
|
|
||||||
});
|
|
||||||
QObject::connect(stopButton, &QPushButton::clicked, &DevTools::Close);
|
|
||||||
auto buttons = new QHBoxLayout();
|
|
||||||
buttons->addWidget(startButton);
|
|
||||||
buttons->addWidget(stopButton);
|
|
||||||
display->addRow(HEADLESS_MODE, headlessCheck);
|
|
||||||
auto autoStartButton = new QCheckBox();
|
|
||||||
autoStartButton->setChecked(settings.value(AUTO_START, false).toBool());
|
|
||||||
QObject::connect(autoStartButton, &QCheckBox::clicked, [](bool autoStart) { settings.setValue(AUTO_START, autoStart); });
|
|
||||||
display->addRow(AUTO_START, autoStartButton);
|
|
||||||
display->addRow(buttons);
|
|
||||||
statusLabel->setFrameStyle(QFrame::Panel | QFrame::Sunken);
|
|
||||||
display->addRow(DEVTOOLS_STATUS, statusLabel);
|
|
||||||
if (autoStartButton->isChecked()) QMetaObject::invokeMethod(startButton, &QPushButton::click, Qt::QueuedConnection);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case DLL_PROCESS_DETACH:
|
case DLL_PROCESS_DETACH:
|
||||||
@ -110,22 +118,31 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved
|
|||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
{
|
{
|
||||||
if (!DevTools::Connected()) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, ERROR_START_CHROME) };
|
if (!DevTools::Connected()) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, ERROR_START_CHROME) };
|
||||||
// DevTools can't handle concurrent translations yet
|
// DevTools can't handle concurrent translations yet
|
||||||
static std::mutex translationMutex;
|
static std::mutex translationMutex;
|
||||||
std::scoped_lock lock(translationMutex);
|
std::scoped_lock lock(translationMutex);
|
||||||
DevTools::SendRequest(L"Page.navigate", FormatString(LR"({"url":"https://www.deepl.com/translator#%s/%s/%s"})", translateFrom.Copy(), translateTo.Copy(), Escape(text)));
|
std::wstring escaped; // DeepL breaks with slash in input
|
||||||
|
for (auto ch : text) ch == '/' ? escaped += L"\\/" : escaped += ch;
|
||||||
|
DevTools::SendRequest("Page.navigate", FormatString(LR"({"url":"https://www.deepl.com/en/translator#en/en/%s"})", Escape(escaped)));
|
||||||
|
for (int retry = 0; ++retry < 20; Sleep(100))
|
||||||
|
if (Copy(DevTools::SendRequest("Runtime.evaluate", LR"({"expression":"document.readyState"})")[L"result"][L"value"].String()) == L"complete") break;
|
||||||
|
|
||||||
|
DevTools::SendRequest("Runtime.evaluate", FormatString(LR"({"expression":"
|
||||||
|
document.querySelector('.lmt__language_select--source').querySelector('button').click();
|
||||||
|
document.evaluate(`//*[text()='%s']`,document.querySelector('.lmt__language_select__menu'),null,XPathResult.FIRST_ORDERED_NODE_TYPE,null).singleNodeValue.click();
|
||||||
|
document.querySelector('.lmt__language_select--target').querySelector('button').click();
|
||||||
|
document.evaluate(`//*[text()='%s']`,document.querySelector('.lmt__language_select__menu'),null,XPathResult.FIRST_ORDERED_NODE_TYPE,null).singleNodeValue.click();
|
||||||
|
"})", codes.at(tlp.translateFrom), codes.at(tlp.translateTo)));
|
||||||
|
|
||||||
for (int retry = 0; ++retry < 100; Sleep(100))
|
for (int retry = 0; ++retry < 100; Sleep(100))
|
||||||
if (auto translation = Copy(
|
if (auto translation = Copy(DevTools::SendRequest("Runtime.evaluate",
|
||||||
DevTools::SendRequest(L"Runtime.evaluate", LR"({"expression":"document.querySelector('#target-dummydiv').innerHTML","returnByValue":true})")[L"result"][L"value"].String()
|
LR"({"expression":"document.querySelector('#target-dummydiv').innerHTML.trim() ","returnByValue":true})"
|
||||||
)) if (!translation->empty()) return { true, translation.value() };
|
)[L"result"][L"value"].String())) if (!translation->empty()) return { true, translation.value() };
|
||||||
if (auto errorMessage = Copy(
|
if (auto errorMessage = Copy(DevTools::SendRequest("Runtime.evaluate",
|
||||||
DevTools::SendRequest(
|
|
||||||
L"Runtime.evaluate",
|
|
||||||
LR"({"expression":"document.querySelector('div.lmt__system_notification').innerHTML","returnByValue":true})"
|
LR"({"expression":"document.querySelector('div.lmt__system_notification').innerHTML","returnByValue":true})"
|
||||||
)[L"result"][L"value"].String()
|
)[L"result"][L"value"].String())) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, errorMessage.value()) };
|
||||||
)) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, errorMessage.value()) };
|
|
||||||
return { false, TRANSLATION_ERROR };
|
return { false, TRANSLATION_ERROR };
|
||||||
}
|
}
|
||||||
|
82
extensions/devtoolspapagotranslate.cpp
Normal file
82
extensions/devtoolspapagotranslate.cpp
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
|
#include "devtools.h"
|
||||||
|
|
||||||
|
extern const wchar_t* ERROR_START_CHROME;
|
||||||
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
|
|
||||||
|
const char* TRANSLATION_PROVIDER = "DevTools Papago Translate";
|
||||||
|
const char* GET_API_KEY_FROM = nullptr;
|
||||||
|
|
||||||
|
extern const QStringList languagesTo
|
||||||
|
{
|
||||||
|
"Chinese (Simplified)",
|
||||||
|
"Chinese (Traditional)",
|
||||||
|
"English",
|
||||||
|
"French",
|
||||||
|
"German",
|
||||||
|
"Hindi",
|
||||||
|
"Indonesian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Korean",
|
||||||
|
"Portuguese",
|
||||||
|
"Russian",
|
||||||
|
"Spanish",
|
||||||
|
"Thai",
|
||||||
|
"Vietnamese",
|
||||||
|
}, languagesFrom = languagesTo;
|
||||||
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Chinese (Simplified)" }, { L"zh-CN" } },
|
||||||
|
{ { L"Chinese (Traditional)" }, { L"zt-TW" } },
|
||||||
|
{ { L"English" }, { L"en" } },
|
||||||
|
{ { L"French" }, { L"fr" } },
|
||||||
|
{ { L"German" }, { L"de" } },
|
||||||
|
{ { L"Hindi" }, { L"hi" } },
|
||||||
|
{ { L"Indonesian" }, { L"id" } },
|
||||||
|
{ { L"Italian" }, { L"it" } },
|
||||||
|
{ { L"Japanese" }, { L"ja" } },
|
||||||
|
{ { L"Korean" }, { L"ko" } },
|
||||||
|
{ { L"Portuguese" }, { L"pt" } },
|
||||||
|
{ { L"Russian" }, { L"ru" } },
|
||||||
|
{ { L"Spanish" }, { L"es" } },
|
||||||
|
{ { L"Thai" }, { L"th" } },
|
||||||
|
{ { L"Vietnamese" }, { L"vi" } },
|
||||||
|
{ { L"?" }, { L"auto" } }
|
||||||
|
};
|
||||||
|
|
||||||
|
bool translateSelectedOnly = true, useRateLimiter = true, rateLimitSelected = false, useCache = true, useFilter = true;
|
||||||
|
int tokenCount = 30, rateLimitTimespan = 60000, maxSentenceSize = 2500;
|
||||||
|
|
||||||
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
||||||
|
{
|
||||||
|
switch (ul_reason_for_call)
|
||||||
|
{
|
||||||
|
case DLL_PROCESS_ATTACH:
|
||||||
|
{
|
||||||
|
DevTools::Initialize();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case DLL_PROCESS_DETACH:
|
||||||
|
{
|
||||||
|
DevTools::Close();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
|
{
|
||||||
|
if (!DevTools::Connected()) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, ERROR_START_CHROME) };
|
||||||
|
// DevTools can't handle concurrent translations yet
|
||||||
|
static std::mutex translationMutex;
|
||||||
|
std::scoped_lock lock(translationMutex);
|
||||||
|
DevTools::SendRequest("Page.navigate", FormatString(LR"({"url":"https://papago.naver.com/?sk=%s&tk=%s&st=%s"})", codes.at(tlp.translateFrom), codes.at(tlp.translateTo), Escape(text)));
|
||||||
|
for (int retry = 0; ++retry < 100; Sleep(100))
|
||||||
|
if (auto translation = Copy(DevTools::SendRequest("Runtime.evaluate",
|
||||||
|
LR"({"expression":"document.querySelector('#txtTarget').textContent.trim() ","returnByValue":true})"
|
||||||
|
)[L"result"][L"value"].String())) if (!translation->empty()) return { true, translation.value() };
|
||||||
|
return { false, TRANSLATION_ERROR };
|
||||||
|
}
|
152
extensions/devtoolssystrantranslate.cpp
Normal file
152
extensions/devtoolssystrantranslate.cpp
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
|
#include "devtools.h"
|
||||||
|
|
||||||
|
extern const wchar_t* ERROR_START_CHROME;
|
||||||
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
|
|
||||||
|
const char* TRANSLATION_PROVIDER = "DevTools Systran Translate";
|
||||||
|
const char* GET_API_KEY_FROM = nullptr;
|
||||||
|
|
||||||
|
extern const QStringList languagesTo
|
||||||
|
{
|
||||||
|
"Albanian",
|
||||||
|
"Arabic",
|
||||||
|
"Bengali",
|
||||||
|
"Bulgarian",
|
||||||
|
"Burmese",
|
||||||
|
"Catalan",
|
||||||
|
"Chinese (Simplified)",
|
||||||
|
"Chinese (Traditional)",
|
||||||
|
"Croatian",
|
||||||
|
"Czech",
|
||||||
|
"Danish",
|
||||||
|
"Dutch",
|
||||||
|
"English",
|
||||||
|
"Estonian",
|
||||||
|
"Finnish",
|
||||||
|
"French",
|
||||||
|
"German",
|
||||||
|
"Greek",
|
||||||
|
"Hebrew",
|
||||||
|
"Hindi",
|
||||||
|
"Hungarian",
|
||||||
|
"Indonesian",
|
||||||
|
"Italian",
|
||||||
|
"Japanese",
|
||||||
|
"Korean",
|
||||||
|
"Latvian",
|
||||||
|
"Lithuanian",
|
||||||
|
"Malay",
|
||||||
|
"Norwegian",
|
||||||
|
"Pashto",
|
||||||
|
"Persian",
|
||||||
|
"Polish",
|
||||||
|
"Portuguese",
|
||||||
|
"Romanian",
|
||||||
|
"Russian",
|
||||||
|
"Serbian",
|
||||||
|
"Slovak",
|
||||||
|
"Slovenian",
|
||||||
|
"Somali",
|
||||||
|
"Spanish",
|
||||||
|
"Swedish",
|
||||||
|
"Tagalog",
|
||||||
|
"Tamil",
|
||||||
|
"Thai",
|
||||||
|
"Turkish",
|
||||||
|
"Ukrainian",
|
||||||
|
"Urdu",
|
||||||
|
"Vietnamese"
|
||||||
|
}, languagesFrom = languagesTo;
|
||||||
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Albanian" }, { L"sq" } },
|
||||||
|
{ { L"Arabic" }, { L"ar" } },
|
||||||
|
{ { L"Bengali" }, { L"bn" } },
|
||||||
|
{ { L"Bulgarian" }, { L"bg" } },
|
||||||
|
{ { L"Burmese" }, { L"my" } },
|
||||||
|
{ { L"Catalan" }, { L"ca" } },
|
||||||
|
{ { L"Chinese (Simplified)" }, { L"zh" } },
|
||||||
|
{ { L"Chinese (Traditional)" }, { L"zt" } },
|
||||||
|
{ { L"Croatian" }, { L"hr" } },
|
||||||
|
{ { L"Czech" }, { L"cs" } },
|
||||||
|
{ { L"Danish" }, { L"da" } },
|
||||||
|
{ { L"Dutch" }, { L"nl" } },
|
||||||
|
{ { L"English" }, { L"en" } },
|
||||||
|
{ { L"Estonian" }, { L"et" } },
|
||||||
|
{ { L"Finnish" }, { L"fi" } },
|
||||||
|
{ { L"French" }, { L"fr" } },
|
||||||
|
{ { L"German" }, { L"de" } },
|
||||||
|
{ { L"Greek" }, { L"el" } },
|
||||||
|
{ { L"Hebrew" }, { L"he" } },
|
||||||
|
{ { L"Hindi" }, { L"hi" } },
|
||||||
|
{ { L"Hungarian" }, { L"hu" } },
|
||||||
|
{ { L"Indonesian" }, { L"id" } },
|
||||||
|
{ { L"Italian" }, { L"it" } },
|
||||||
|
{ { L"Japanese" }, { L"ja" } },
|
||||||
|
{ { L"Korean" }, { L"ko" } },
|
||||||
|
{ { L"Latvian" }, { L"lv" } },
|
||||||
|
{ { L"Lithuanian" }, { L"lt" } },
|
||||||
|
{ { L"Malay" }, { L"ms" } },
|
||||||
|
{ { L"Norwegian" }, { L"no" } },
|
||||||
|
{ { L"Pashto" }, { L"ps" } },
|
||||||
|
{ { L"Persian" }, { L"fa" } },
|
||||||
|
{ { L"Polish" }, { L"pl" } },
|
||||||
|
{ { L"Portuguese" }, { L"pt" } },
|
||||||
|
{ { L"Romanian" }, { L"ro" } },
|
||||||
|
{ { L"Russian" }, { L"ru" } },
|
||||||
|
{ { L"Serbian" }, { L"sr" } },
|
||||||
|
{ { L"Slovak" }, { L"sk" } },
|
||||||
|
{ { L"Slovenian" }, { L"sl" } },
|
||||||
|
{ { L"Somali" }, { L"so" } },
|
||||||
|
{ { L"Spanish" }, { L"es" } },
|
||||||
|
{ { L"Swedish" }, { L"sv" } },
|
||||||
|
{ { L"Tagalog" }, { L"tl" } },
|
||||||
|
{ { L"Tamil" }, { L"ta" } },
|
||||||
|
{ { L"Thai" }, { L"th" } },
|
||||||
|
{ { L"Turkish" }, { L"tr" } },
|
||||||
|
{ { L"Ukrainian" }, { L"uk" } },
|
||||||
|
{ { L"Urdu" }, { L"ur" } },
|
||||||
|
{ { L"Vietnamese" }, { L"vi" } },
|
||||||
|
{ { L"?" }, { L"autodetect" } }
|
||||||
|
};
|
||||||
|
|
||||||
|
bool translateSelectedOnly = true, useRateLimiter = true, rateLimitSelected = false, useCache = true, useFilter = true;
|
||||||
|
int tokenCount = 30, rateLimitTimespan = 60000, maxSentenceSize = 2500;
|
||||||
|
|
||||||
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
||||||
|
{
|
||||||
|
switch (ul_reason_for_call)
|
||||||
|
{
|
||||||
|
case DLL_PROCESS_ATTACH:
|
||||||
|
{
|
||||||
|
DevTools::Initialize();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case DLL_PROCESS_DETACH:
|
||||||
|
{
|
||||||
|
DevTools::Close();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
|
{
|
||||||
|
if (!DevTools::Connected()) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, ERROR_START_CHROME) };
|
||||||
|
// DevTools can't handle concurrent translations yet
|
||||||
|
static std::mutex translationMutex;
|
||||||
|
std::scoped_lock lock(translationMutex);
|
||||||
|
|
||||||
|
DevTools::SendRequest(
|
||||||
|
"Page.navigate",
|
||||||
|
FormatString(LR"({"url":"https://translate.systran.net/?source=%s&target=%s&input=%s"})", codes.at(tlp.translateFrom), codes.at(tlp.translateTo), Escape(text))
|
||||||
|
);
|
||||||
|
for (int retry = 0; ++retry < 100; Sleep(100))
|
||||||
|
if (auto translation = Copy(DevTools::SendRequest("Runtime.evaluate",
|
||||||
|
LR"({"expression":"document.querySelector('#outputEditor').textContent.trim() ","returnByValue":true})"
|
||||||
|
)[L"result"][L"value"].String())) if (!translation->empty()) return { true, translation.value() };
|
||||||
|
return { false, TRANSLATION_ERROR };
|
||||||
|
}
|
@ -17,12 +17,12 @@ extern "C" __declspec(dllexport) wchar_t* OnNewSentence(wchar_t* sentence, const
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
std::wstring sentenceStr(sentence);
|
std::wstring sentenceCopy(sentence);
|
||||||
int origLength = sentenceStr.size();
|
int oldSize = sentenceCopy.size();
|
||||||
if (ProcessSentence(sentenceStr, SentenceInfo{ sentenceInfo }))
|
if (ProcessSentence(sentenceCopy, SentenceInfo{ sentenceInfo }))
|
||||||
{
|
{
|
||||||
if (sentenceStr.size() > origLength) sentence = (wchar_t*)HeapReAlloc(GetProcessHeap(), HEAP_GENERATE_EXCEPTIONS, sentence, (sentenceStr.size() + 1) * sizeof(wchar_t));
|
if (sentenceCopy.size() > oldSize) sentence = (wchar_t*)HeapReAlloc(GetProcessHeap(), HEAP_GENERATE_EXCEPTIONS, sentence, (sentenceCopy.size() + 1) * sizeof(wchar_t));
|
||||||
wcscpy_s(sentence, sentenceStr.size() + 1, sentenceStr.c_str());
|
wcscpy_s(sentence, sentenceCopy.size() + 1, sentenceCopy.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (SKIP)
|
catch (SKIP)
|
||||||
|
@ -13,15 +13,20 @@
|
|||||||
#include <QFontMetrics>
|
#include <QFontMetrics>
|
||||||
#include <QMouseEvent>
|
#include <QMouseEvent>
|
||||||
#include <QWheelEvent>
|
#include <QWheelEvent>
|
||||||
|
#include <QScrollArea>
|
||||||
|
#include <QAbstractNativeEventFilter>
|
||||||
|
|
||||||
extern const char* EXTRA_WINDOW_INFO;
|
extern const char* EXTRA_WINDOW_INFO;
|
||||||
extern const char* SENTENCE_TOO_BIG;
|
|
||||||
extern const char* MAX_SENTENCE_SIZE;
|
|
||||||
extern const char* TOPMOST;
|
extern const char* TOPMOST;
|
||||||
extern const char* OPACITY;
|
extern const char* OPACITY;
|
||||||
extern const char* SHOW_ORIGINAL;
|
extern const char* SHOW_ORIGINAL;
|
||||||
extern const char* SHOW_ORIGINAL_INFO;
|
extern const char* ORIGINAL_AFTER_TRANSLATION;
|
||||||
extern const char* SIZE_LOCK;
|
extern const char* SIZE_LOCK;
|
||||||
|
extern const char* POSITION_LOCK;
|
||||||
|
extern const char* CENTERED_TEXT;
|
||||||
|
extern const char* AUTO_RESIZE_WINDOW_HEIGHT;
|
||||||
|
extern const char* CLICK_THROUGH;
|
||||||
|
extern const char* HIDE_MOUSEOVER;
|
||||||
extern const char* DICTIONARY;
|
extern const char* DICTIONARY;
|
||||||
extern const char* DICTIONARY_INSTRUCTIONS;
|
extern const char* DICTIONARY_INSTRUCTIONS;
|
||||||
extern const char* BG_COLOR;
|
extern const char* BG_COLOR;
|
||||||
@ -31,9 +36,9 @@ extern const char* OUTLINE_COLOR;
|
|||||||
extern const char* OUTLINE_SIZE;
|
extern const char* OUTLINE_SIZE;
|
||||||
extern const char* OUTLINE_SIZE_INFO;
|
extern const char* OUTLINE_SIZE_INFO;
|
||||||
extern const char* FONT;
|
extern const char* FONT;
|
||||||
extern const char* SAVE_SETTINGS;
|
|
||||||
|
|
||||||
constexpr auto DICTIONARY_SAVE_FILE = u8"SavedDictionary.txt";
|
constexpr auto DICTIONARY_SAVE_FILE = u8"SavedDictionary.txt";
|
||||||
|
constexpr int CLICK_THROUGH_HOTKEY = 0xc0d0;
|
||||||
|
|
||||||
QColor colorPrompt(QWidget* parent, QColor default, const QString& title, bool customOpacity = true)
|
QColor colorPrompt(QWidget* parent, QColor default, const QString& title, bool customOpacity = true)
|
||||||
{
|
{
|
||||||
@ -47,7 +52,7 @@ struct PrettyWindow : QDialog, Localizer
|
|||||||
PrettyWindow(const char* name)
|
PrettyWindow(const char* name)
|
||||||
{
|
{
|
||||||
ui.setupUi(this);
|
ui.setupUi(this);
|
||||||
ui.display->setGraphicsEffect(&outliner);
|
ui.display->setGraphicsEffect(outliner = new Outliner);
|
||||||
setWindowFlags(Qt::FramelessWindowHint);
|
setWindowFlags(Qt::FramelessWindowHint);
|
||||||
setAttribute(Qt::WA_TranslucentBackground);
|
setAttribute(Qt::WA_TranslucentBackground);
|
||||||
|
|
||||||
@ -56,15 +61,21 @@ struct PrettyWindow : QDialog, Localizer
|
|||||||
if (font.fromString(settings.value(FONT, font.toString()).toString())) ui.display->setFont(font);
|
if (font.fromString(settings.value(FONT, font.toString()).toString())) ui.display->setFont(font);
|
||||||
SetBackgroundColor(settings.value(BG_COLOR, backgroundColor).value<QColor>());
|
SetBackgroundColor(settings.value(BG_COLOR, backgroundColor).value<QColor>());
|
||||||
SetTextColor(settings.value(TEXT_COLOR, TextColor()).value<QColor>());
|
SetTextColor(settings.value(TEXT_COLOR, TextColor()).value<QColor>());
|
||||||
outliner.color = settings.value(OUTLINE_COLOR, outliner.color).value<QColor>();
|
outliner->color = settings.value(OUTLINE_COLOR, outliner->color).value<QColor>();
|
||||||
outliner.size = settings.value(OUTLINE_SIZE, outliner.size).toDouble();
|
outliner->size = settings.value(OUTLINE_SIZE, outliner->size).toDouble();
|
||||||
|
autoHide = settings.value(HIDE_MOUSEOVER, autoHide).toBool();
|
||||||
menu.addAction(FONT, this, &PrettyWindow::RequestFont);
|
menu.addAction(FONT, this, &PrettyWindow::RequestFont);
|
||||||
menu.addAction(BG_COLOR, [this] { SetBackgroundColor(colorPrompt(this, backgroundColor, BG_COLOR)); });
|
menu.addAction(BG_COLOR, [this] { SetBackgroundColor(colorPrompt(this, backgroundColor, BG_COLOR)); });
|
||||||
menu.addAction(TEXT_COLOR, [this] { SetTextColor(colorPrompt(this, TextColor(), TEXT_COLOR)); });
|
menu.addAction(TEXT_COLOR, [this] { SetTextColor(colorPrompt(this, TextColor(), TEXT_COLOR)); });
|
||||||
QAction* outlineAction = menu.addAction(TEXT_OUTLINE, this, &PrettyWindow::SetOutline);
|
QAction* outlineAction = menu.addAction(TEXT_OUTLINE, this, &PrettyWindow::SetOutline);
|
||||||
outlineAction->setCheckable(true);
|
outlineAction->setCheckable(true);
|
||||||
outlineAction->setChecked(outliner.size >= 0);
|
outlineAction->setChecked(outliner->size >= 0);
|
||||||
connect(ui.display, &QLabel::customContextMenuRequested, [this](QPoint point) { menu.exec(mapToGlobal(point)); });
|
QAction* autoHideAction = menu.addAction(HIDE_MOUSEOVER, this, [this](bool autoHide) { settings.setValue(HIDE_MOUSEOVER, this->autoHide = autoHide); });
|
||||||
|
autoHideAction->setCheckable(true);
|
||||||
|
autoHideAction->setChecked(autoHide);
|
||||||
|
connect(this, &QDialog::customContextMenuRequested, [this](QPoint point) { menu.exec(mapToGlobal(point)); });
|
||||||
|
connect(ui.display, &QLabel::customContextMenuRequested, [this](QPoint point) { menu.exec(ui.display->mapToGlobal(point)); });
|
||||||
|
startTimer(50);
|
||||||
}
|
}
|
||||||
|
|
||||||
~PrettyWindow()
|
~PrettyWindow()
|
||||||
@ -75,6 +86,31 @@ struct PrettyWindow : QDialog, Localizer
|
|||||||
Ui::ExtraWindow ui;
|
Ui::ExtraWindow ui;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
void timerEvent(QTimerEvent*) override
|
||||||
|
{
|
||||||
|
if (autoHide && geometry().contains(QCursor::pos()))
|
||||||
|
{
|
||||||
|
if (!hidden)
|
||||||
|
{
|
||||||
|
if (backgroundColor.alphaF() > 0.05) backgroundColor.setAlphaF(0.05);
|
||||||
|
if (outliner->color.alphaF() > 0.05) outliner->color.setAlphaF(0.05);
|
||||||
|
QColor hiddenTextColor = TextColor();
|
||||||
|
if (hiddenTextColor.alphaF() > 0.05) hiddenTextColor.setAlphaF(0.05);
|
||||||
|
ui.display->setPalette(QPalette(hiddenTextColor, {}, {}, {}, {}, {}, {}));
|
||||||
|
hidden = true;
|
||||||
|
repaint();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (hidden)
|
||||||
|
{
|
||||||
|
backgroundColor.setAlpha(settings.value(BG_COLOR).value<QColor>().alpha());
|
||||||
|
outliner->color.setAlpha(settings.value(OUTLINE_COLOR).value<QColor>().alpha());
|
||||||
|
ui.display->setPalette(QPalette(settings.value(TEXT_COLOR).value<QColor>(), {}, {}, {}, {}, {}, {}));
|
||||||
|
hidden = false;
|
||||||
|
repaint();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
QMenu menu{ ui.display };
|
QMenu menu{ ui.display };
|
||||||
Settings settings{ this };
|
Settings settings{ this };
|
||||||
|
|
||||||
@ -113,13 +149,13 @@ private:
|
|||||||
{
|
{
|
||||||
if (enable)
|
if (enable)
|
||||||
{
|
{
|
||||||
QColor color = colorPrompt(this, outliner.color, OUTLINE_COLOR);
|
QColor color = colorPrompt(this, outliner->color, OUTLINE_COLOR);
|
||||||
if (color.isValid()) outliner.color = color;
|
if (color.isValid()) outliner->color = color;
|
||||||
outliner.size = QInputDialog::getDouble(this, OUTLINE_SIZE, OUTLINE_SIZE_INFO, 0.5, 0, INT_MAX, 2, nullptr, Qt::WindowCloseButtonHint);
|
outliner->size = QInputDialog::getDouble(this, OUTLINE_SIZE, OUTLINE_SIZE_INFO, -outliner->size, 0, INT_MAX, 2, nullptr, Qt::WindowCloseButtonHint);
|
||||||
}
|
}
|
||||||
else outliner.size = -1;
|
else outliner->size = -outliner->size;
|
||||||
settings.setValue(OUTLINE_COLOR, outliner.color.name(QColor::HexArgb));
|
settings.setValue(OUTLINE_COLOR, outliner->color.name(QColor::HexArgb));
|
||||||
settings.setValue(OUTLINE_SIZE, outliner.size);
|
settings.setValue(OUTLINE_SIZE, outliner->size);
|
||||||
}
|
}
|
||||||
|
|
||||||
void paintEvent(QPaintEvent*) override
|
void paintEvent(QPaintEvent*) override
|
||||||
@ -127,8 +163,9 @@ private:
|
|||||||
QPainter(this).fillRect(rect(), backgroundColor);
|
QPainter(this).fillRect(rect(), backgroundColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool autoHide = false, hidden = false;
|
||||||
QColor backgroundColor{ palette().window().color() };
|
QColor backgroundColor{ palette().window().color() };
|
||||||
struct : QGraphicsEffect
|
struct Outliner : QGraphicsEffect
|
||||||
{
|
{
|
||||||
void draw(QPainter* painter) override
|
void draw(QPainter* painter) override
|
||||||
{
|
{
|
||||||
@ -147,23 +184,26 @@ private:
|
|||||||
painter->drawPixmap(offset, pixmap);
|
painter->drawPixmap(offset, pixmap);
|
||||||
}
|
}
|
||||||
QColor color{ Qt::black };
|
QColor color{ Qt::black };
|
||||||
double size = -1;
|
double size = -0.5;
|
||||||
} outliner;
|
}* outliner;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ExtraWindow : public PrettyWindow
|
class ExtraWindow : public PrettyWindow, QAbstractNativeEventFilter
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
ExtraWindow() : PrettyWindow("Extra Window")
|
ExtraWindow() : PrettyWindow("Extra Window")
|
||||||
{
|
{
|
||||||
ui.display->setTextFormat(Qt::PlainText);
|
ui.display->setTextFormat(Qt::PlainText);
|
||||||
if (settings.contains(WINDOW) && QApplication::screenAt(settings.value(WINDOW).toRect().bottomRight())) setGeometry(settings.value(WINDOW).toRect());
|
if (settings.contains(WINDOW) && QApplication::screenAt(settings.value(WINDOW).toRect().bottomRight())) setGeometry(settings.value(WINDOW).toRect());
|
||||||
maxSentenceSize = settings.value(MAX_SENTENCE_SIZE, maxSentenceSize).toInt();
|
|
||||||
|
|
||||||
for (auto [name, default, slot] : Array<const char*, bool, void(ExtraWindow::*)(bool)>{
|
for (auto [name, default, slot] : Array<const char*, bool, void(ExtraWindow::*)(bool)>{
|
||||||
{ TOPMOST, false, &ExtraWindow::SetTopmost },
|
{ TOPMOST, false, &ExtraWindow::SetTopmost },
|
||||||
{ SIZE_LOCK, false, &ExtraWindow::SetLock },
|
{ SIZE_LOCK, false, &ExtraWindow::SetSizeLock },
|
||||||
|
{ POSITION_LOCK, false, &ExtraWindow::SetPositionLock },
|
||||||
|
{ CENTERED_TEXT, false, &ExtraWindow::SetCenteredText },
|
||||||
|
{ AUTO_RESIZE_WINDOW_HEIGHT, false, &ExtraWindow::SetAutoResize },
|
||||||
{ SHOW_ORIGINAL, true, &ExtraWindow::SetShowOriginal },
|
{ SHOW_ORIGINAL, true, &ExtraWindow::SetShowOriginal },
|
||||||
|
{ ORIGINAL_AFTER_TRANSLATION, true, &ExtraWindow::SetShowOriginalAfterTranslation },
|
||||||
{ DICTIONARY, false, &ExtraWindow::SetUseDictionary },
|
{ DICTIONARY, false, &ExtraWindow::SetUseDictionary },
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
@ -173,15 +213,15 @@ public:
|
|||||||
action->setCheckable(true);
|
action->setCheckable(true);
|
||||||
action->setChecked(default);
|
action->setChecked(default);
|
||||||
}
|
}
|
||||||
menu.addAction(MAX_SENTENCE_SIZE, this, [this]
|
|
||||||
{
|
menu.addAction(CLICK_THROUGH, this, &ExtraWindow::ToggleClickThrough);
|
||||||
settings.setValue(MAX_SENTENCE_SIZE, maxSentenceSize = QInputDialog::getInt(this, MAX_SENTENCE_SIZE, "", maxSentenceSize, 0, INT_MAX, 1, nullptr, Qt::WindowCloseButtonHint));
|
|
||||||
});
|
|
||||||
ui.display->installEventFilter(this);
|
ui.display->installEventFilter(this);
|
||||||
ui.display->setMouseTracking(true);
|
qApp->installNativeEventFilter(this);
|
||||||
|
|
||||||
QMetaObject::invokeMethod(this, [this]
|
QMetaObject::invokeMethod(this, [this]
|
||||||
{
|
{
|
||||||
|
RegisterHotKey((HWND)winId(), CLICK_THROUGH_HOTKEY, MOD_ALT | MOD_NOREPEAT, 0x58);
|
||||||
show();
|
show();
|
||||||
AddSentence(EXTRA_WINDOW_INFO);
|
AddSentence(EXTRA_WINDOW_INFO);
|
||||||
}, Qt::QueuedConnection);
|
}, Qt::QueuedConnection);
|
||||||
@ -194,16 +234,47 @@ public:
|
|||||||
|
|
||||||
void AddSentence(QString sentence)
|
void AddSentence(QString sentence)
|
||||||
{
|
{
|
||||||
if (sentence.size() > maxSentenceSize) sentence = SENTENCE_TOO_BIG;
|
|
||||||
if (!showOriginal && sentence.contains(u8"\x200b \n")) sentence = sentence.split(u8"\x200b \n")[1];
|
|
||||||
sanitize(sentence);
|
sanitize(sentence);
|
||||||
sentence.chop(std::distance(std::remove(sentence.begin(), sentence.end(), QChar::Tabulation), sentence.end()));
|
sentence.chop(std::distance(std::remove(sentence.begin(), sentence.end(), QChar::Tabulation), sentence.end()));
|
||||||
sentenceHistory.push_back(sentence);
|
sentenceHistory.push_back(sentence);
|
||||||
|
if (sentenceHistory.size() > 1000) sentenceHistory.erase(sentenceHistory.begin());
|
||||||
historyIndex = sentenceHistory.size() - 1;
|
historyIndex = sentenceHistory.size() - 1;
|
||||||
ui.display->setText(sentence);
|
DisplaySentence();
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void DisplaySentence()
|
||||||
|
{
|
||||||
|
if (sentenceHistory.empty()) return;
|
||||||
|
QString sentence = sentenceHistory[historyIndex];
|
||||||
|
if (sentence.contains(u8"\x200b \n"))
|
||||||
|
if (!showOriginal) sentence = sentence.split(u8"\x200b \n")[1];
|
||||||
|
else if (showOriginalAfterTranslation) sentence = sentence.split(u8"\x200b \n")[1] + "\n" + sentence.split(u8"\x200b \n")[0];
|
||||||
|
|
||||||
|
if (sizeLock && !autoResize)
|
||||||
|
{
|
||||||
|
QFontMetrics fontMetrics(ui.display->font(), ui.display);
|
||||||
|
int low = 0, high = sentence.size(), last = 0;
|
||||||
|
while (low <= high)
|
||||||
|
{
|
||||||
|
int mid = (low + high) / 2;
|
||||||
|
if (fontMetrics.boundingRect(0, 0, ui.display->width(), INT_MAX, Qt::TextWordWrap, sentence.left(mid)).height() <= ui.display->height())
|
||||||
|
{
|
||||||
|
last = mid;
|
||||||
|
low = mid + 1;
|
||||||
|
}
|
||||||
|
else high = mid - 1;
|
||||||
|
}
|
||||||
|
sentence = sentence.left(last);
|
||||||
|
}
|
||||||
|
|
||||||
|
ui.display->setText(sentence);
|
||||||
|
if (autoResize)
|
||||||
|
resize(width(), height() - ui.display->height() +
|
||||||
|
QFontMetrics(ui.display->font(), ui.display).boundingRect(0, 0, ui.display->width(), INT_MAX, Qt::TextWordWrap, sentence).height()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void SetTopmost(bool topmost)
|
void SetTopmost(bool topmost)
|
||||||
{
|
{
|
||||||
for (auto window : { winId(), dictionaryWindow.winId() })
|
for (auto window : { winId(), dictionaryWindow.winId() })
|
||||||
@ -211,16 +282,39 @@ private:
|
|||||||
settings.setValue(TOPMOST, topmost);
|
settings.setValue(TOPMOST, topmost);
|
||||||
};
|
};
|
||||||
|
|
||||||
void SetLock(bool locked)
|
void SetPositionLock(bool locked)
|
||||||
|
{
|
||||||
|
settings.setValue(POSITION_LOCK, posLock = locked);
|
||||||
|
};
|
||||||
|
|
||||||
|
void SetSizeLock(bool locked)
|
||||||
{
|
{
|
||||||
setSizeGripEnabled(!locked);
|
setSizeGripEnabled(!locked);
|
||||||
settings.setValue(SIZE_LOCK, this->locked = locked);
|
settings.setValue(SIZE_LOCK, sizeLock = locked);
|
||||||
|
};
|
||||||
|
|
||||||
|
void SetCenteredText(bool centeredText)
|
||||||
|
{
|
||||||
|
ui.display->setAlignment(centeredText ? Qt::AlignHCenter : Qt::AlignLeft);
|
||||||
|
settings.setValue(CENTERED_TEXT, this->centeredText = centeredText);
|
||||||
|
};
|
||||||
|
|
||||||
|
void SetAutoResize(bool autoResize)
|
||||||
|
{
|
||||||
|
settings.setValue(AUTO_RESIZE_WINDOW_HEIGHT, this->autoResize = autoResize);
|
||||||
|
DisplaySentence();
|
||||||
};
|
};
|
||||||
|
|
||||||
void SetShowOriginal(bool showOriginal)
|
void SetShowOriginal(bool showOriginal)
|
||||||
{
|
{
|
||||||
if (!showOriginal && settings.value(SHOW_ORIGINAL, false).toBool()) QMessageBox::information(this, SHOW_ORIGINAL, SHOW_ORIGINAL_INFO);
|
|
||||||
settings.setValue(SHOW_ORIGINAL, this->showOriginal = showOriginal);
|
settings.setValue(SHOW_ORIGINAL, this->showOriginal = showOriginal);
|
||||||
|
DisplaySentence();
|
||||||
|
};
|
||||||
|
|
||||||
|
void SetShowOriginalAfterTranslation(bool showOriginalAfterTranslation)
|
||||||
|
{
|
||||||
|
settings.setValue(ORIGINAL_AFTER_TRANSLATION, this->showOriginalAfterTranslation = showOriginalAfterTranslation);
|
||||||
|
DisplaySentence();
|
||||||
};
|
};
|
||||||
|
|
||||||
void SetUseDictionary(bool useDictionary)
|
void SetUseDictionary(bool useDictionary)
|
||||||
@ -237,27 +331,40 @@ private:
|
|||||||
settings.setValue(DICTIONARY, this->useDictionary = useDictionary);
|
settings.setValue(DICTIONARY, this->useDictionary = useDictionary);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ComputeDictionaryPosition(QPoint mouse)
|
void ToggleClickThrough()
|
||||||
|
{
|
||||||
|
clickThrough = !clickThrough;
|
||||||
|
for (auto window : { winId(), dictionaryWindow.winId() })
|
||||||
|
{
|
||||||
|
unsigned exStyle = GetWindowLongPtrW((HWND)window, GWL_EXSTYLE);
|
||||||
|
if (clickThrough) exStyle |= WS_EX_TRANSPARENT;
|
||||||
|
else exStyle &= ~WS_EX_TRANSPARENT;
|
||||||
|
SetWindowLongPtrW((HWND)window, GWL_EXSTYLE, exStyle);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void ShowDictionary(QPoint mouse)
|
||||||
{
|
{
|
||||||
QString sentence = ui.display->text();
|
QString sentence = ui.display->text();
|
||||||
const QFont& font = ui.display->font();
|
const QFont& font = ui.display->font();
|
||||||
if (cachedDisplayInfo.CompareExchange(ui.display))
|
if (cachedDisplayInfo.CompareExchange(ui.display))
|
||||||
{
|
{
|
||||||
QFontMetrics fontMetrics(font, ui.display);
|
QFontMetrics fontMetrics(font, ui.display);
|
||||||
|
int flags = Qt::TextWordWrap | (ui.display->alignment() & (Qt::AlignLeft | Qt::AlignHCenter));
|
||||||
textPositionMap.clear();
|
textPositionMap.clear();
|
||||||
for (int i = 0, height = 0, lineBreak = 0; i < sentence.size(); ++i)
|
for (int i = 0, height = 0, lineBreak = 0; i < sentence.size(); ++i)
|
||||||
{
|
{
|
||||||
int block = 1;
|
int block = 1;
|
||||||
for (int charHeight = fontMetrics.boundingRect(0, 0, 1, INT_MAX, Qt::TextWordWrap, sentence.mid(i, 1)).height();
|
for (int charHeight = fontMetrics.boundingRect(0, 0, 1, INT_MAX, flags, sentence.mid(i, 1)).height();
|
||||||
i + block < sentence.size() && fontMetrics.boundingRect(0, 0, 1, INT_MAX, Qt::TextWordWrap, sentence.mid(i, block + 1)).height() < charHeight * 1.5; ++block);
|
i + block < sentence.size() && fontMetrics.boundingRect(0, 0, 1, INT_MAX, flags, sentence.mid(i, block + 1)).height() < charHeight * 1.5; ++block);
|
||||||
auto boundingRect = fontMetrics.boundingRect(0, 0, ui.display->width(), INT_MAX, Qt::TextWordWrap, sentence.left(i + block));
|
auto boundingRect = fontMetrics.boundingRect(0, 0, ui.display->width(), INT_MAX, flags, sentence.left(i + block));
|
||||||
if (boundingRect.height() > height)
|
if (boundingRect.height() > height)
|
||||||
{
|
{
|
||||||
height = boundingRect.height();
|
height = boundingRect.height();
|
||||||
lineBreak = i;
|
lineBreak = i;
|
||||||
}
|
}
|
||||||
textPositionMap.push_back({
|
textPositionMap.push_back({
|
||||||
fontMetrics.boundingRect(0, 0, ui.display->width(), INT_MAX, Qt::TextWordWrap, sentence.mid(lineBreak, i - lineBreak + 1)).width(),
|
fontMetrics.boundingRect(0, 0, ui.display->width(), INT_MAX, flags, sentence.mid(lineBreak, i - lineBreak + 1)).right() + 1,
|
||||||
height
|
height
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -274,13 +381,27 @@ private:
|
|||||||
dictionaryWindow.move(ui.display->mapToGlobal(QPoint(x, y - dictionaryWindow.height())));
|
dictionaryWindow.move(ui.display->mapToGlobal(QPoint(x, y - dictionaryWindow.height())));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool nativeEventFilter(const QByteArray&, void* message, long* result) override
|
||||||
|
{
|
||||||
|
auto msg = (MSG*)message;
|
||||||
|
if (msg->message == WM_HOTKEY)
|
||||||
|
if (msg->wParam == CLICK_THROUGH_HOTKEY) return ToggleClickThrough(), true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool eventFilter(QObject*, QEvent* event) override
|
bool eventFilter(QObject*, QEvent* event) override
|
||||||
{
|
{
|
||||||
if (useDictionary && event->type() == QEvent::MouseMove) ComputeDictionaryPosition(((QMouseEvent*)event)->localPos().toPoint());
|
if (event->type() == QEvent::MouseButtonPress) mousePressEvent((QMouseEvent*)event);
|
||||||
if (event->type() == QEvent::MouseButtonPress) dictionaryWindow.hide();
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void timerEvent(QTimerEvent* event) override
|
||||||
|
{
|
||||||
|
if (useDictionary && QCursor::pos() != oldPos && (!dictionaryWindow.isVisible() || !dictionaryWindow.geometry().contains(QCursor::pos())))
|
||||||
|
ShowDictionary(ui.display->mapFromGlobal(QCursor::pos()));
|
||||||
|
PrettyWindow::timerEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
void mousePressEvent(QMouseEvent* event) override
|
void mousePressEvent(QMouseEvent* event) override
|
||||||
{
|
{
|
||||||
dictionaryWindow.hide();
|
dictionaryWindow.hide();
|
||||||
@ -289,19 +410,19 @@ private:
|
|||||||
|
|
||||||
void mouseMoveEvent(QMouseEvent* event) override
|
void mouseMoveEvent(QMouseEvent* event) override
|
||||||
{
|
{
|
||||||
if (!locked) move(pos() + event->globalPos() - oldPos);
|
if (!posLock) move(pos() + event->globalPos() - oldPos);
|
||||||
oldPos = event->globalPos();
|
oldPos = event->globalPos();
|
||||||
}
|
}
|
||||||
|
|
||||||
void wheelEvent(QWheelEvent* event) override
|
void wheelEvent(QWheelEvent* event) override
|
||||||
{
|
{
|
||||||
int scroll = event->angleDelta().y();
|
int scroll = event->angleDelta().y();
|
||||||
if (scroll > 0 && historyIndex > 0) ui.display->setText(sentenceHistory[--historyIndex]);
|
if (scroll > 0 && historyIndex > 0) --historyIndex;
|
||||||
if (scroll < 0 && historyIndex + 1 < sentenceHistory.size()) ui.display->setText(sentenceHistory[++historyIndex]);
|
if (scroll < 0 && historyIndex + 1 < sentenceHistory.size()) ++historyIndex;
|
||||||
|
DisplaySentence();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool locked, showOriginal, useDictionary;
|
bool sizeLock, posLock, centeredText, autoResize, showOriginal, showOriginalAfterTranslation, useDictionary, clickThrough;
|
||||||
int maxSentenceSize = 1000;
|
|
||||||
QPoint oldPos;
|
QPoint oldPos;
|
||||||
|
|
||||||
class
|
class
|
||||||
@ -309,10 +430,11 @@ private:
|
|||||||
public:
|
public:
|
||||||
bool CompareExchange(QLabel* display)
|
bool CompareExchange(QLabel* display)
|
||||||
{
|
{
|
||||||
if (display->text() == text && display->font() == font && display->width() == width) return false;
|
if (display->text() == text && display->font() == font && display->width() == width && display->alignment() == alignment) return false;
|
||||||
text = display->text();
|
text = display->text();
|
||||||
font = display->font();
|
font = display->font();
|
||||||
width = display->width();
|
width = display->width();
|
||||||
|
alignment = display->alignment();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -320,6 +442,7 @@ private:
|
|||||||
QString text;
|
QString text;
|
||||||
QFont font;
|
QFont font;
|
||||||
int width;
|
int width;
|
||||||
|
Qt::Alignment alignment;
|
||||||
} cachedDisplayInfo;
|
} cachedDisplayInfo;
|
||||||
std::vector<QPoint> textPositionMap;
|
std::vector<QPoint> textPositionMap;
|
||||||
|
|
||||||
|
@ -10,6 +10,9 @@
|
|||||||
<height>300</height>
|
<height>300</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
|
<property name="contextMenuPolicy">
|
||||||
|
<enum>Qt::CustomContextMenu</enum>
|
||||||
|
</property>
|
||||||
<layout class="QHBoxLayout">
|
<layout class="QHBoxLayout">
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="display">
|
<widget class="QLabel" name="display">
|
||||||
|
@ -1,140 +1,250 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
#include "network.h"
|
#include "network.h"
|
||||||
#include <ctime>
|
|
||||||
|
|
||||||
extern const wchar_t* TRANSLATION_ERROR;
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
|
|
||||||
extern Synchronized<std::wstring> translateTo, translateFrom, authKey;
|
|
||||||
|
|
||||||
const char* TRANSLATION_PROVIDER = "Google Translate";
|
const char* TRANSLATION_PROVIDER = "Google Translate";
|
||||||
const char* GET_API_KEY_FROM = "https://codelabs.developers.google.com/codelabs/cloud-translation-intro";
|
const char* GET_API_KEY_FROM = "https://console.cloud.google.com/marketplace/product/google/translate.googleapis.com";
|
||||||
QStringList languages
|
extern const QStringList languagesTo
|
||||||
{
|
{
|
||||||
"Afrikaans: af",
|
"Afrikaans",
|
||||||
"Albanian: sq",
|
"Albanian",
|
||||||
"Amharic: am",
|
"Amharic",
|
||||||
"Arabic: ar",
|
"Arabic",
|
||||||
"Armenian: hy",
|
"Armenian",
|
||||||
"Azerbaijani: az",
|
"Azerbaijani",
|
||||||
"Basque: eu",
|
"Basque",
|
||||||
"Belarusian: be",
|
"Belarusian",
|
||||||
"Bengali: bn",
|
"Bengali",
|
||||||
"Bosnian: bs",
|
"Bosnian",
|
||||||
"Bulgarian: bg",
|
"Bulgarian",
|
||||||
"Catalan: ca",
|
"Catalan",
|
||||||
"Cebuano: ceb",
|
"Cebuano",
|
||||||
"Chichewa: ny",
|
"Chichewa",
|
||||||
"Chinese (simplified): zh",
|
"Chinese (Simplified)",
|
||||||
"Chinese (traditional): zh-TW",
|
"Chinese (Traditional)",
|
||||||
"Corsican: co",
|
"Corsican",
|
||||||
"Croatian: hr",
|
"Croatian",
|
||||||
"Czech: cs",
|
"Czech",
|
||||||
"Danish: da",
|
"Danish",
|
||||||
"Dutch: nl",
|
"Dutch",
|
||||||
"English: en",
|
"English",
|
||||||
"Esperanto: eo",
|
"Esperanto",
|
||||||
"Estonian: et",
|
"Estonian",
|
||||||
"Filipino: tl",
|
"Filipino",
|
||||||
"Finnish: fi",
|
"Finnish",
|
||||||
"French: fr",
|
"French",
|
||||||
"Frisian: fy",
|
"Frisian",
|
||||||
"Galician: gl",
|
"Galician",
|
||||||
"Georgian: ka",
|
"Georgian",
|
||||||
"German: de",
|
"German",
|
||||||
"Greek: el",
|
"Greek",
|
||||||
"Gujarati: gu",
|
"Gujarati",
|
||||||
"Haitian Creole: ht",
|
"Haitian Creole",
|
||||||
"Hausa: ha",
|
"Hausa",
|
||||||
"Hawaiian: haw",
|
"Hawaiian",
|
||||||
"Hebrew: iw",
|
"Hebrew",
|
||||||
"Hindi: hi",
|
"Hindi",
|
||||||
"Hmong: hmn",
|
"Hmong",
|
||||||
"Hungarian: hu",
|
"Hungarian",
|
||||||
"Icelandic: is",
|
"Icelandic",
|
||||||
"Igbo: ig",
|
"Igbo",
|
||||||
"Indonesian: id",
|
"Indonesian",
|
||||||
"Irish: ga",
|
"Irish",
|
||||||
"Italian: it",
|
"Italian",
|
||||||
"Japanese: ja",
|
"Japanese",
|
||||||
"Javanese: jw",
|
"Javanese",
|
||||||
"Kannada: kn",
|
"Kannada",
|
||||||
"Kazakh: kk",
|
"Kazakh",
|
||||||
"Khmer: km",
|
"Khmer",
|
||||||
"Kinyarwanda: rw",
|
"Kinyarwanda",
|
||||||
"Korean: ko",
|
"Korean",
|
||||||
"Kurdish (Kurmanji): ku",
|
"Kurdish (Kurmanji)",
|
||||||
"Kyrgyz: ky",
|
"Kyrgyz",
|
||||||
"Lao: lo",
|
"Lao",
|
||||||
"Latin: la",
|
"Latin",
|
||||||
"Latvian: lv",
|
"Latvian",
|
||||||
"Lithuanian: lt",
|
"Lithuanian",
|
||||||
"Luxembourgish: lb",
|
"Luxembourgish",
|
||||||
"Macedonian: mk",
|
"Macedonian",
|
||||||
"Malagasy: mg",
|
"Malagasy",
|
||||||
"Malay: ms",
|
"Malay",
|
||||||
"Malayalam: ml",
|
"Malayalam",
|
||||||
"Maltese: mt",
|
"Maltese",
|
||||||
"Maori: mi",
|
"Maori",
|
||||||
"Marathi: mr",
|
"Marathi",
|
||||||
"Mongolian: mn",
|
"Mongolian",
|
||||||
"Myanmar (Burmese): my",
|
"Myanmar (Burmese)",
|
||||||
"Nepali: ne",
|
"Nepali",
|
||||||
"Norwegian: no",
|
"Norwegian",
|
||||||
"Odia (Oriya): or",
|
"Odia (Oriya)",
|
||||||
"Pashto: ps",
|
"Pashto",
|
||||||
"Persian: fa",
|
"Persian",
|
||||||
"Polish: pl",
|
"Polish",
|
||||||
"Portuguese: pt",
|
"Portuguese",
|
||||||
"Punjabi: pa",
|
"Punjabi",
|
||||||
"Romanian: ro",
|
"Romanian",
|
||||||
"Russian: ru",
|
"Russian",
|
||||||
"Samoan: sm",
|
"Samoan",
|
||||||
"Scots Gaelic: gd",
|
"Scots Gaelic",
|
||||||
"Serbian: sr",
|
"Serbian",
|
||||||
"Sesotho: st",
|
"Sesotho",
|
||||||
"Shona: sn",
|
"Shona",
|
||||||
"Sindhi: sd",
|
"Sindhi",
|
||||||
"Sinhala: si",
|
"Sinhala",
|
||||||
"Slovak: sk",
|
"Slovak",
|
||||||
"Slovenian: sl",
|
"Slovenian",
|
||||||
"Somali: so",
|
"Somali",
|
||||||
"Spanish: es",
|
"Spanish",
|
||||||
"Sundanese: su",
|
"Sundanese",
|
||||||
"Swahili: sw",
|
"Swahili",
|
||||||
"Swedish: sv",
|
"Swedish",
|
||||||
"Tajik: tg",
|
"Tajik",
|
||||||
"Tamil: ta",
|
"Tamil",
|
||||||
"Tatar: tt",
|
"Tatar",
|
||||||
"Telugu: te",
|
"Telugu",
|
||||||
"Thai: th",
|
"Thai",
|
||||||
"Turkish: tr",
|
"Turkish",
|
||||||
"Turkmen: tk",
|
"Turkmen",
|
||||||
"Ukrainian: uk",
|
"Ukrainian",
|
||||||
"Urdu: ur",
|
"Urdu",
|
||||||
"Uyghur: ug",
|
"Uyghur",
|
||||||
"Uzbek: uz",
|
"Uzbek",
|
||||||
"Vietnamese: vi",
|
"Vietnamese",
|
||||||
"Welsh: cy",
|
"Welsh",
|
||||||
"Xhosa: xh",
|
"Xhosa",
|
||||||
"Yiddish: yi",
|
"Yiddish",
|
||||||
"Yoruba: yo",
|
"Yoruba",
|
||||||
"Zulu: zu"
|
"Zulu",
|
||||||
|
}, languagesFrom = languagesTo;
|
||||||
|
extern const std::unordered_map<std::wstring, std::wstring> codes
|
||||||
|
{
|
||||||
|
{ { L"Afrikaans" }, { L"af" } },
|
||||||
|
{ { L"Albanian" }, { L"sq" } },
|
||||||
|
{ { L"Amharic" }, { L"am" } },
|
||||||
|
{ { L"Arabic" }, { L"ar" } },
|
||||||
|
{ { L"Armenian" }, { L"hy" } },
|
||||||
|
{ { L"Azerbaijani" }, { L"az" } },
|
||||||
|
{ { L"Basque" }, { L"eu" } },
|
||||||
|
{ { L"Belarusian" }, { L"be" } },
|
||||||
|
{ { L"Bengali" }, { L"bn" } },
|
||||||
|
{ { L"Bosnian" }, { L"bs" } },
|
||||||
|
{ { L"Bulgarian" }, { L"bg" } },
|
||||||
|
{ { L"Catalan" }, { L"ca" } },
|
||||||
|
{ { L"Cebuano" }, { L"ceb" } },
|
||||||
|
{ { L"Chichewa" }, { L"ny" } },
|
||||||
|
{ { L"Chinese (Simplified)" }, { L"zh-CN" } },
|
||||||
|
{ { L"Chinese (Traditional)" }, { L"zh-TW" } },
|
||||||
|
{ { L"Corsican" }, { L"co" } },
|
||||||
|
{ { L"Croatian" }, { L"hr" } },
|
||||||
|
{ { L"Czech" }, { L"cs" } },
|
||||||
|
{ { L"Danish" }, { L"da" } },
|
||||||
|
{ { L"Dutch" }, { L"nl" } },
|
||||||
|
{ { L"English" }, { L"en" } },
|
||||||
|
{ { L"Esperanto" }, { L"eo" } },
|
||||||
|
{ { L"Estonian" }, { L"et" } },
|
||||||
|
{ { L"Filipino" }, { L"tl" } },
|
||||||
|
{ { L"Finnish" }, { L"fi" } },
|
||||||
|
{ { L"French" }, { L"fr" } },
|
||||||
|
{ { L"Frisian" }, { L"fy" } },
|
||||||
|
{ { L"Galician" }, { L"gl" } },
|
||||||
|
{ { L"Georgian" }, { L"ka" } },
|
||||||
|
{ { L"German" }, { L"de" } },
|
||||||
|
{ { L"Greek" }, { L"el" } },
|
||||||
|
{ { L"Gujarati" }, { L"gu" } },
|
||||||
|
{ { L"Haitian Creole" }, { L"ht" } },
|
||||||
|
{ { L"Hausa" }, { L"ha" } },
|
||||||
|
{ { L"Hawaiian" }, { L"haw" } },
|
||||||
|
{ { L"Hebrew" }, { L"iw" } },
|
||||||
|
{ { L"Hindi" }, { L"hi" } },
|
||||||
|
{ { L"Hmong" }, { L"hmn" } },
|
||||||
|
{ { L"Hungarian" }, { L"hu" } },
|
||||||
|
{ { L"Icelandic" }, { L"is" } },
|
||||||
|
{ { L"Igbo" }, { L"ig" } },
|
||||||
|
{ { L"Indonesian" }, { L"id" } },
|
||||||
|
{ { L"Irish" }, { L"ga" } },
|
||||||
|
{ { L"Italian" }, { L"it" } },
|
||||||
|
{ { L"Japanese" }, { L"ja" } },
|
||||||
|
{ { L"Javanese" }, { L"jw" } },
|
||||||
|
{ { L"Kannada" }, { L"kn" } },
|
||||||
|
{ { L"Kazakh" }, { L"kk" } },
|
||||||
|
{ { L"Khmer" }, { L"km" } },
|
||||||
|
{ { L"Kinyarwanda" }, { L"rw" } },
|
||||||
|
{ { L"Korean" }, { L"ko" } },
|
||||||
|
{ { L"Kurdish (Kurmanji)" }, { L"ku" } },
|
||||||
|
{ { L"Kyrgyz" }, { L"ky" } },
|
||||||
|
{ { L"Lao" }, { L"lo" } },
|
||||||
|
{ { L"Latin" }, { L"la" } },
|
||||||
|
{ { L"Latvian" }, { L"lv" } },
|
||||||
|
{ { L"Lithuanian" }, { L"lt" } },
|
||||||
|
{ { L"Luxembourgish" }, { L"lb" } },
|
||||||
|
{ { L"Macedonian" }, { L"mk" } },
|
||||||
|
{ { L"Malagasy" }, { L"mg" } },
|
||||||
|
{ { L"Malay" }, { L"ms" } },
|
||||||
|
{ { L"Malayalam" }, { L"ml" } },
|
||||||
|
{ { L"Maltese" }, { L"mt" } },
|
||||||
|
{ { L"Maori" }, { L"mi" } },
|
||||||
|
{ { L"Marathi" }, { L"mr" } },
|
||||||
|
{ { L"Mongolian" }, { L"mn" } },
|
||||||
|
{ { L"Myanmar (Burmese)" }, { L"my" } },
|
||||||
|
{ { L"Nepali" }, { L"ne" } },
|
||||||
|
{ { L"Norwegian" }, { L"no" } },
|
||||||
|
{ { L"Odia (Oriya)" }, { L"or" } },
|
||||||
|
{ { L"Pashto" }, { L"ps" } },
|
||||||
|
{ { L"Persian" }, { L"fa" } },
|
||||||
|
{ { L"Polish" }, { L"pl" } },
|
||||||
|
{ { L"Portuguese" }, { L"pt" } },
|
||||||
|
{ { L"Punjabi" }, { L"pa" } },
|
||||||
|
{ { L"Romanian" }, { L"ro" } },
|
||||||
|
{ { L"Russian" }, { L"ru" } },
|
||||||
|
{ { L"Samoan" }, { L"sm" } },
|
||||||
|
{ { L"Scots Gaelic" }, { L"gd" } },
|
||||||
|
{ { L"Serbian" }, { L"sr" } },
|
||||||
|
{ { L"Sesotho" }, { L"st" } },
|
||||||
|
{ { L"Shona" }, { L"sn" } },
|
||||||
|
{ { L"Sindhi" }, { L"sd" } },
|
||||||
|
{ { L"Sinhala" }, { L"si" } },
|
||||||
|
{ { L"Slovak" }, { L"sk" } },
|
||||||
|
{ { L"Slovenian" }, { L"sl" } },
|
||||||
|
{ { L"Somali" }, { L"so" } },
|
||||||
|
{ { L"Spanish" }, { L"es" } },
|
||||||
|
{ { L"Sundanese" }, { L"su" } },
|
||||||
|
{ { L"Swahili" }, { L"sw" } },
|
||||||
|
{ { L"Swedish" }, { L"sv" } },
|
||||||
|
{ { L"Tajik" }, { L"tg" } },
|
||||||
|
{ { L"Tamil" }, { L"ta" } },
|
||||||
|
{ { L"Tatar" }, { L"tt" } },
|
||||||
|
{ { L"Telugu" }, { L"te" } },
|
||||||
|
{ { L"Thai" }, { L"th" } },
|
||||||
|
{ { L"Turkish" }, { L"tr" } },
|
||||||
|
{ { L"Turkmen" }, { L"tk" } },
|
||||||
|
{ { L"Ukrainian" }, { L"uk" } },
|
||||||
|
{ { L"Urdu" }, { L"ur" } },
|
||||||
|
{ { L"Uyghur" }, { L"ug" } },
|
||||||
|
{ { L"Uzbek" }, { L"uz" } },
|
||||||
|
{ { L"Vietnamese" }, { L"vi" } },
|
||||||
|
{ { L"Welsh" }, { L"cy" } },
|
||||||
|
{ { L"Xhosa" }, { L"xh" } },
|
||||||
|
{ { L"Yiddish" }, { L"yi" } },
|
||||||
|
{ { L"Yoruba" }, { L"yo" } },
|
||||||
|
{ { L"Zulu" }, { L"zu" } },
|
||||||
|
{ { L"?" }, { L"auto" } }
|
||||||
};
|
};
|
||||||
std::wstring autoDetectLanguage = L"auto";
|
|
||||||
|
|
||||||
bool translateSelectedOnly = false, rateLimitAll = true, rateLimitSelected = false, useCache = true;
|
bool translateSelectedOnly = false, useRateLimiter = true, rateLimitSelected = false, useCache = true, useFilter = true;
|
||||||
int tokenCount = 30, tokenRestoreDelay = 60000, maxSentenceSize = 1000;
|
int tokenCount = 30, rateLimitTimespan = 60000, maxSentenceSize = 1000;
|
||||||
|
|
||||||
std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp)
|
||||||
{
|
{
|
||||||
if (!authKey->empty())
|
if (!tlp.authKey.empty())
|
||||||
{
|
{
|
||||||
std::wstring translateFromComponent = translateFrom.Copy() == autoDetectLanguage ? L"" : L"&source=" + translateFrom.Copy();
|
std::wstring translateFromComponent = tlp.translateFrom == L"?" ? L"" : L"&source=" + codes.at(tlp.translateFrom);
|
||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"translation.googleapis.com",
|
L"translation.googleapis.com",
|
||||||
L"POST",
|
L"POST",
|
||||||
FormatString(L"/language/translate/v2?format=text&target=%s&key=%s%s", translateTo.Copy(), authKey.Copy(), translateFromComponent).c_str(),
|
FormatString(L"/language/translate/v2?format=text&target=%s&key=%s%s", codes.at(tlp.translateTo), tlp.authKey, translateFromComponent).c_str(),
|
||||||
FormatString(R"({"q":["%s"]})", JSON::Escape(WideStringToString(text)))
|
FormatString(R"({"q":["%s"]})", JSON::Escape(WideStringToString(text)))
|
||||||
})
|
})
|
||||||
if (auto translation = Copy(JSON::Parse(httpRequest.response)[L"data"][L"translations"][0][L"translatedText"].String())) return { true, translation.value() };
|
if (auto translation = Copy(JSON::Parse(httpRequest.response)[L"data"][L"translations"][0][L"translatedText"].String())) return { true, translation.value() };
|
||||||
@ -145,34 +255,12 @@ std::pair<bool, std::wstring> Translate(const std::wstring& text)
|
|||||||
if (HttpRequest httpRequest{
|
if (HttpRequest httpRequest{
|
||||||
L"Mozilla/5.0 Textractor",
|
L"Mozilla/5.0 Textractor",
|
||||||
L"translate.google.com",
|
L"translate.google.com",
|
||||||
L"POST",
|
L"GET",
|
||||||
L"/_/TranslateWebserverUi/data/batchexecute?rpcids=MkEWBc",
|
FormatString(L"/m?sl=%s&tl=%s&q=%s", codes.at(tlp.translateFrom), codes.at(tlp.translateTo), Escape(text)).c_str()
|
||||||
"f.req=" + Escape(WideStringToString(
|
|
||||||
FormatString(LR"([[["MkEWBc","[[\"%s\",\"%s\",\"%s\",true],[null]]",null,"generic"]]])", JSON::Escape((JSON::Escape(text))), translateFrom.Copy(), translateTo.Copy())
|
|
||||||
)),
|
|
||||||
L"Content-Type: application/x-www-form-urlencoded"
|
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
if (auto start = httpRequest.response.find(L"[["); start != std::string::npos)
|
auto start = httpRequest.response.find(L"result-container\">"), end = httpRequest.response.find(L'<', start);
|
||||||
{
|
if (end != std::string::npos) return { true, HTML::Unescape(httpRequest.response.substr(start + 18, end - start - 18)) };
|
||||||
if (auto blob = Copy(JSON::Parse(httpRequest.response.substr(start))[0][2].String())) if (auto translations = Copy(JSON::Parse(blob.value())[1][0].Array()))
|
|
||||||
{
|
|
||||||
std::wstring translation;
|
|
||||||
if (translations->size() == 1)
|
|
||||||
{
|
|
||||||
if (translations = Copy(translations.value()[0][5].Array()))
|
|
||||||
for (const auto& sentence : translations.value())
|
|
||||||
if (sentence[0].String()) (translation += *sentence[0].String()) += L" ";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
for (const auto& conjugation : translations.value())
|
|
||||||
if (auto sentence = conjugation[0].String()) if (auto gender = conjugation[2].String()) translation += FormatString(L"%s %s\n", *sentence, *gender);
|
|
||||||
}
|
|
||||||
if (!translation.empty()) return { true, translation };
|
|
||||||
return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, blob.value()) };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, httpRequest.response) };
|
return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, httpRequest.response) };
|
||||||
}
|
}
|
||||||
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
else return { false, FormatString(L"%s (code=%u)", TRANSLATION_ERROR, httpRequest.errorCode) };
|
||||||
|
@ -62,4 +62,4 @@ std::string Escape(const std::string& text)
|
|||||||
return escaped;
|
return escaped;
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(assert(JSON::Parse<wchar_t>(LR"([{"string":"hello world","boolean":false,"number":1.67e+4,"null":null,"array":[]},"hello world"])")))
|
TEST(assert(JSON::Parse<wchar_t>(LR"([{"string":"hello world","boolean":false,"number":1.67e+4,"null":null,"array":[]},"hello world"])")));
|
||||||
|
@ -32,6 +32,36 @@ struct HttpRequest
|
|||||||
std::wstring Escape(const std::wstring& text);
|
std::wstring Escape(const std::wstring& text);
|
||||||
std::string Escape(const std::string& text);
|
std::string Escape(const std::string& text);
|
||||||
|
|
||||||
|
namespace HTML
|
||||||
|
{
|
||||||
|
template <typename C>
|
||||||
|
std::basic_string<C> Unescape(std::basic_string<C> text)
|
||||||
|
{
|
||||||
|
constexpr C
|
||||||
|
lt[] = { '&', 'l', 't', ';' },
|
||||||
|
gt[] = { '&', 'g', 't', ';' },
|
||||||
|
apos1[] = { '&', 'a', 'p', 'o', 's', ';' },
|
||||||
|
apos2[] = { '&', '#', '3', '9', ';' },
|
||||||
|
apos3[] = { '&', '#', 'x', '2', '7', ';' },
|
||||||
|
apos4[] = { '&', '#', 'X', '2', '7', ';' },
|
||||||
|
quot[] = { '&', 'q', 'u', 'o', 't', ';' },
|
||||||
|
amp[] = { '&', 'a', 'm', 'p', ';' };
|
||||||
|
for (int i = 0; i < text.size(); ++i)
|
||||||
|
if (text[i] == '&')
|
||||||
|
for (auto [original, length, replacement] : Array<const C*, size_t, C>{
|
||||||
|
{ lt, std::size(lt), '<' },
|
||||||
|
{ gt, std::size(gt), '>' },
|
||||||
|
{ apos1, std::size(apos1), '\'' },
|
||||||
|
{ apos2, std::size(apos2), '\'' },
|
||||||
|
{ apos3, std::size(apos3), '\'' },
|
||||||
|
{ apos4, std::size(apos4), '\'' },
|
||||||
|
{ quot, std::size(quot), '"' },
|
||||||
|
{ amp, std::size(amp), '&' }
|
||||||
|
}) if (std::char_traits<C>::compare(text.data() + i, original, length) == 0) text.replace(i, length, 1, replacement);
|
||||||
|
return text;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
namespace JSON
|
namespace JSON
|
||||||
{
|
{
|
||||||
template <typename C>
|
template <typename C>
|
||||||
@ -116,7 +146,7 @@ namespace JSON
|
|||||||
ch = text[i + 1];
|
ch = text[i + 1];
|
||||||
if (ch == 'u' && isxdigit(text[i + 2]) && isxdigit(text[i + 3]) && isxdigit(text[i + 4]) && isxdigit(text[i + 5]))
|
if (ch == 'u' && isxdigit(text[i + 2]) && isxdigit(text[i + 3]) && isxdigit(text[i + 4]) && isxdigit(text[i + 5]))
|
||||||
{
|
{
|
||||||
char charCode[] = { text[i + 2], text[i + 3], text[i + 4], text[i + 5], 0 };
|
char charCode[] = { (char)text[i + 2], (char)text[i + 3], (char)text[i + 4], (char)text[i + 5], 0 };
|
||||||
unescaped += UTF<C>::FromCodepoint(strtoul(charCode, nullptr, 16));
|
unescaped += UTF<C>::FromCodepoint(strtoul(charCode, nullptr, 16));
|
||||||
i += 5;
|
i += 5;
|
||||||
continue;
|
continue;
|
||||||
@ -136,7 +166,7 @@ namespace JSON
|
|||||||
|
|
||||||
if (SkipWhitespace()) return {};
|
if (SkipWhitespace()) return {};
|
||||||
|
|
||||||
static C nullStr[] = { 'n', 'u', 'l', 'l' }, trueStr[] = { 't', 'r', 'u', 'e' }, falseStr[] = { 'f', 'a', 'l', 's', 'e' };
|
constexpr C nullStr[] = { 'n', 'u', 'l', 'l' }, trueStr[] = { 't', 'r', 'u', 'e' }, falseStr[] = { 'f', 'a', 'l', 's', 'e' };
|
||||||
if (ch == nullStr[0])
|
if (ch == nullStr[0])
|
||||||
if (std::char_traits<C>::compare(text.data() + i, nullStr, std::size(nullStr)) == 0) return i += std::size(nullStr), nullptr;
|
if (std::char_traits<C>::compare(text.data() + i, nullStr, std::size(nullStr)) == 0) return i += std::size(nullStr), nullptr;
|
||||||
else return {};
|
else return {};
|
||||||
|
@ -12,9 +12,9 @@ extern const char* CURRENT_FILTER;
|
|||||||
const char* REGEX_SAVE_FILE = "SavedRegexFilters.txt";
|
const char* REGEX_SAVE_FILE = "SavedRegexFilters.txt";
|
||||||
|
|
||||||
std::optional<std::wregex> regex;
|
std::optional<std::wregex> regex;
|
||||||
std::wstring replace;
|
std::wstring replace = L"$1";
|
||||||
std::shared_mutex m;
|
concurrency::reader_writer_lock m;
|
||||||
DWORD (*GetSelectedProcessId)() = nullptr;
|
DWORD (*GetSelectedProcessId)() = [] { return 0UL; };
|
||||||
|
|
||||||
class Window : public QDialog, Localizer
|
class Window : public QDialog, Localizer
|
||||||
{
|
{
|
||||||
@ -24,7 +24,6 @@ public:
|
|||||||
ui.setupUi(this);
|
ui.setupUi(this);
|
||||||
|
|
||||||
connect(ui.regexEdit, &QLineEdit::textEdited, this, &Window::SetRegex);
|
connect(ui.regexEdit, &QLineEdit::textEdited, this, &Window::SetRegex);
|
||||||
connect(ui.replaceEdit, &QLineEdit::textEdited, this, &Window::SetReplace);
|
|
||||||
connect(ui.saveButton, &QPushButton::clicked, this, &Window::Save);
|
connect(ui.saveButton, &QPushButton::clicked, this, &Window::Save);
|
||||||
|
|
||||||
setWindowTitle(REGEX_FILTER);
|
setWindowTitle(REGEX_FILTER);
|
||||||
@ -41,21 +40,13 @@ public:
|
|||||||
ui.output->setText(QString(CURRENT_FILTER).arg(regex));
|
ui.output->setText(QString(CURRENT_FILTER).arg(regex));
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetReplace(QString replace)
|
|
||||||
{
|
|
||||||
ui.replaceEdit->setText(replace);
|
|
||||||
std::scoped_lock lock(m);
|
|
||||||
::replace = S(replace);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void Save()
|
void Save()
|
||||||
{
|
{
|
||||||
auto formatted = FormatString(
|
auto formatted = FormatString(
|
||||||
L"\xfeff|PROCESS|%s|FILTER|%s|REPLACE|%s|END|\r\n",
|
L"\xfeff|PROCESS|%s|FILTER|%s|END|\r\n",
|
||||||
GetModuleFilename(GetSelectedProcessId()).value_or(FormatString(L"Error getting name of process 0x%X", GetSelectedProcessId())),
|
GetModuleFilename(GetSelectedProcessId()).value_or(FormatString(L"Error getting name of process 0x%X", GetSelectedProcessId())),
|
||||||
S(ui.regexEdit->text()),
|
S(ui.regexEdit->text())
|
||||||
S(ui.replaceEdit->text())
|
|
||||||
);
|
);
|
||||||
std::ofstream(REGEX_SAVE_FILE, std::ios::binary | std::ios::app).write((const char*)formatted.c_str(), formatted.size() * sizeof(wchar_t));
|
std::ofstream(REGEX_SAVE_FILE, std::ios::binary | std::ios::app).write((const char*)formatted.c_str(), formatted.size() * sizeof(wchar_t));
|
||||||
}
|
}
|
||||||
@ -65,21 +56,17 @@ private:
|
|||||||
|
|
||||||
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
||||||
{
|
{
|
||||||
static auto _ = GetSelectedProcessId = (DWORD(*)())sentenceInfo["DWORD (*GetSelectedProcessId)()"];
|
static auto _ = GetSelectedProcessId = (DWORD(*)())sentenceInfo["get selected process id"];
|
||||||
if (sentenceInfo["text number"] == 0) return false;
|
if (sentenceInfo["text number"] == 0) return false;
|
||||||
if (sentenceInfo["current select"] && !regex) if (auto processName = GetModuleFilename(sentenceInfo["process id"]))
|
if (/*sentenceInfo["current select"] && */!regex) if (auto processName = GetModuleFilename(sentenceInfo["process id"]))
|
||||||
{
|
{
|
||||||
std::ifstream stream(REGEX_SAVE_FILE, std::ios::binary);
|
std::ifstream stream(REGEX_SAVE_FILE, std::ios::binary);
|
||||||
BlockMarkupIterator savedFilters(stream, Array<std::wstring_view>{ L"|PROCESS|", L"|FILTER|", L"|REPLACE|" });
|
BlockMarkupIterator savedFilters(stream, Array<std::wstring_view>{ L"|PROCESS|", L"|FILTER|" });
|
||||||
std::vector<std::array<std::wstring, 3>> regexes;
|
std::vector<std::wstring> regexes;
|
||||||
while (auto read = savedFilters.Next()) if (read->at(0) == processName) regexes.push_back(std::move(read.value()));
|
while (auto read = savedFilters.Next()) if (read->at(0) == processName) regexes.push_back(std::move(read->at(1)));
|
||||||
if (!regexes.empty()) QMetaObject::invokeMethod(&window, [regex = S(regexes.back()[1]), replace = S(regexes.back()[2])]
|
if (!regexes.empty()) QMetaObject::invokeMethod(&window, std::bind(&Window::SetRegex, &window, S(regexes.back())), Qt::BlockingQueuedConnection);
|
||||||
{
|
|
||||||
window.SetRegex(regex);
|
|
||||||
window.SetReplace(replace);
|
|
||||||
}, Qt::BlockingQueuedConnection);
|
|
||||||
}
|
}
|
||||||
std::shared_lock lock(m);
|
concurrency::reader_writer_lock::scoped_lock_read readLock(m);
|
||||||
if (regex) sentence = std::regex_replace(sentence, regex.value(), replace);
|
if (regex) sentence = std::regex_replace(sentence, regex.value(), replace);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>500</width>
|
<width>500</width>
|
||||||
<height>107</height>
|
<height>80</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout">
|
<layout class="QVBoxLayout">
|
||||||
@ -17,15 +17,12 @@
|
|||||||
<widget class="QLineEdit" name="regexEdit"/>
|
<widget class="QLineEdit" name="regexEdit"/>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel">
|
<widget class="QPushButton" name="saveButton">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>=></string>
|
<string>Save</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
|
||||||
<widget class="QLineEdit" name="replaceEdit"/>
|
|
||||||
</item>
|
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
@ -38,13 +35,6 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
|
||||||
<widget class="QPushButton" name="saveButton">
|
|
||||||
<property name="text">
|
|
||||||
<string>Save</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel">
|
<widget class="QLabel">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
|
72
extensions/regexreplacer.cpp
Normal file
72
extensions/regexreplacer.cpp
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
#include "extension.h"
|
||||||
|
#include "module.h"
|
||||||
|
#include "blockmarkup.h"
|
||||||
|
#include <fstream>
|
||||||
|
#include <process.h>
|
||||||
|
|
||||||
|
extern const wchar_t* REGEX_REPLACER_INSTRUCTIONS;
|
||||||
|
|
||||||
|
const char* REPLACE_SAVE_FILE = "SavedRegexReplacements.txt";
|
||||||
|
|
||||||
|
std::atomic<std::filesystem::file_time_type> replaceFileLastWrite = {};
|
||||||
|
concurrency::reader_writer_lock m;
|
||||||
|
std::vector<std::tuple<std::wregex, std::wstring, std::regex_constants::match_flag_type>> replacements;
|
||||||
|
|
||||||
|
void UpdateReplacements()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (replaceFileLastWrite.exchange(std::filesystem::last_write_time(REPLACE_SAVE_FILE)) == std::filesystem::last_write_time(REPLACE_SAVE_FILE)) return;
|
||||||
|
std::scoped_lock lock(m);
|
||||||
|
replacements.clear();
|
||||||
|
std::ifstream stream(REPLACE_SAVE_FILE, std::ios::binary);
|
||||||
|
BlockMarkupIterator savedFilters(stream, Array<std::wstring_view>{ L"|REGEX|", L"|BECOMES|", L"|MODIFIER|" });
|
||||||
|
while (auto read = savedFilters.Next())
|
||||||
|
{
|
||||||
|
const auto& [regex, replacement, modifier] = read.value();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
replacements.emplace_back(
|
||||||
|
std::wregex(regex, modifier.find(L'i') == std::string::npos ? std::regex::ECMAScript : std::regex::icase),
|
||||||
|
replacement,
|
||||||
|
modifier.find(L'g') == std::string::npos ? std::regex_constants::format_first_only : std::regex_constants::format_default
|
||||||
|
);
|
||||||
|
}
|
||||||
|
catch (std::regex_error) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (std::filesystem::filesystem_error) { replaceFileLastWrite.store({}); }
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
||||||
|
{
|
||||||
|
switch (ul_reason_for_call)
|
||||||
|
{
|
||||||
|
case DLL_PROCESS_ATTACH:
|
||||||
|
{
|
||||||
|
UpdateReplacements();
|
||||||
|
if (replacements.empty())
|
||||||
|
{
|
||||||
|
auto file = std::ofstream(REPLACE_SAVE_FILE, std::ios::binary) << "\xff\xfe";
|
||||||
|
for (auto ch : std::wstring_view(REGEX_REPLACER_INSTRUCTIONS))
|
||||||
|
file << (ch == L'\n' ? std::string_view("\r\0\n", 4) : std::string_view((char*)&ch, 2));
|
||||||
|
SpawnThread([] { _spawnlp(_P_DETACH, "notepad", "notepad", REPLACE_SAVE_FILE, NULL); }); // show file to user
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case DLL_PROCESS_DETACH:
|
||||||
|
{
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
||||||
|
{
|
||||||
|
UpdateReplacements();
|
||||||
|
|
||||||
|
concurrency::reader_writer_lock::scoped_lock_read readLock(m);
|
||||||
|
for (const auto& [regex, replacement, flags] : replacements) sentence = std::regex_replace(sentence, regex, replacement, flags);
|
||||||
|
return true;
|
||||||
|
}
|
@ -5,30 +5,25 @@ bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
|||||||
if (sentenceInfo["text number"] == 0) return false;
|
if (sentenceInfo["text number"] == 0) return false;
|
||||||
|
|
||||||
std::vector<int> repeatNumbers(sentence.size() + 1, 0);
|
std::vector<int> repeatNumbers(sentence.size() + 1, 0);
|
||||||
int repeatNumber = 1;
|
for (int i = 0; i < sentence.size(); ++i)
|
||||||
wchar_t prevChar = L'\0';
|
|
||||||
for (auto nextChar : sentence)
|
|
||||||
{
|
{
|
||||||
if (nextChar == prevChar)
|
if (sentence[i] != sentence[i + 1])
|
||||||
{
|
{
|
||||||
repeatNumber += 1;
|
int j = i;
|
||||||
}
|
while (sentence[j] == sentence[i] && --j >= 0);
|
||||||
else
|
repeatNumbers[i - j] += 1;
|
||||||
{
|
|
||||||
prevChar = nextChar;
|
|
||||||
repeatNumbers.at(repeatNumber) += 1;
|
|
||||||
repeatNumber = 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ((repeatNumber = std::distance(repeatNumbers.begin(), std::max_element(repeatNumbers.begin(), repeatNumbers.end()))) == 1) return false;
|
int repeatNumber = std::distance(repeatNumbers.begin(), std::max_element(repeatNumbers.rbegin(), repeatNumbers.rend()).base() - 1);
|
||||||
|
if (repeatNumber < 2) return false;
|
||||||
|
|
||||||
std::wstring newSentence;
|
std::wstring newSentence;
|
||||||
for (int i = 0; i < sentence.size();)
|
for (int i = 0; i < sentence.size();)
|
||||||
{
|
{
|
||||||
newSentence.push_back(sentence.at(i));
|
newSentence.push_back(sentence[i]);
|
||||||
for (int j = i; j <= sentence.size(); ++j)
|
for (int j = i; j <= sentence.size(); ++j)
|
||||||
{
|
{
|
||||||
if (j == sentence.size() || sentence.at(i) != sentence.at(j))
|
if (j == sentence.size() || sentence[i] != sentence[j])
|
||||||
{
|
{
|
||||||
i += (j - i) % repeatNumber == 0 ? repeatNumber : 1;
|
i += (j - i) % repeatNumber == 0 ? repeatNumber : 1;
|
||||||
break;
|
break;
|
||||||
@ -44,8 +39,11 @@ TEST(
|
|||||||
InfoForExtension nonConsole[] = { { "text number", 1 }, {} };
|
InfoForExtension nonConsole[] = { { "text number", 1 }, {} };
|
||||||
|
|
||||||
std::wstring repeatedChars = L"aaaaaaaaaaaabbbbbbcccdddaabbbcccddd";
|
std::wstring repeatedChars = L"aaaaaaaaaaaabbbbbbcccdddaabbbcccddd";
|
||||||
|
std::wstring someRepeatedChars = L"abcdefaabbccddeeff";
|
||||||
ProcessSentence(repeatedChars, { nonConsole });
|
ProcessSentence(repeatedChars, { nonConsole });
|
||||||
|
ProcessSentence(someRepeatedChars, { nonConsole });
|
||||||
assert(repeatedChars.find(L"aaaabbcd") == 0);
|
assert(repeatedChars.find(L"aaaabbcd") == 0);
|
||||||
|
assert(someRepeatedChars == L"abcdefabcdef");
|
||||||
|
|
||||||
std::wstring empty = L"", one = L" ", normal = L"This is a normal sentence. はい";
|
std::wstring empty = L"", one = L" ", normal = L"This is a normal sentence. はい";
|
||||||
ProcessSentence(empty, { nonConsole });
|
ProcessSentence(empty, { nonConsole });
|
||||||
|
@ -16,10 +16,10 @@ std::vector<int> GenerateSuffixArray(const std::wstring& text)
|
|||||||
eqClasses[suffixArray[0]] = 0;
|
eqClasses[suffixArray[0]] = 0;
|
||||||
for (int i = 1; i < text.size(); ++i)
|
for (int i = 1; i < text.size(); ++i)
|
||||||
{
|
{
|
||||||
int currentSuffix = suffixArray[i];
|
int currentSuffix = suffixArray[i], lastSuffix = suffixArray[i - 1];
|
||||||
int lastSuffix = suffixArray[i - 1];
|
|
||||||
if (currentSuffix + length < text.size() && prevEqClasses[currentSuffix] == prevEqClasses[lastSuffix] &&
|
if (currentSuffix + length < text.size() && prevEqClasses[currentSuffix] == prevEqClasses[lastSuffix] &&
|
||||||
prevEqClasses[currentSuffix + length / 2] == prevEqClasses.at(lastSuffix + length / 2)) // not completely certain that this will stay in range
|
prevEqClasses[currentSuffix + length / 2] == prevEqClasses[lastSuffix + length / 2]
|
||||||
|
)
|
||||||
eqClasses[currentSuffix] = eqClasses[lastSuffix];
|
eqClasses[currentSuffix] = eqClasses[lastSuffix];
|
||||||
else eqClasses[currentSuffix] = i;
|
else eqClasses[currentSuffix] = i;
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved
|
|||||||
{
|
{
|
||||||
wchar_t filePath[MAX_PATH];
|
wchar_t filePath[MAX_PATH];
|
||||||
GetModuleFileNameW(hModule, filePath, MAX_PATH);
|
GetModuleFileNameW(hModule, filePath, MAX_PATH);
|
||||||
if (wchar_t* fileName = wcsrchr(filePath, L'\\')) swscanf_s(fileName, L"\\Remove %d Repeated Sentences.dll", &sentenceCacheSize);
|
if (wchar_t* fileName = wcsrchr(filePath, L'\\')) swscanf_s(fileName, L"\\Remove %d Repeated Sentences.xdll", &sentenceCacheSize);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case DLL_PROCESS_DETACH:
|
case DLL_PROCESS_DETACH:
|
||||||
@ -30,7 +30,7 @@ bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
|||||||
static std::mutex m;
|
static std::mutex m;
|
||||||
m.lock();
|
m.lock();
|
||||||
if (textNumber + 1 > cache.size()) cache.resize(textNumber + 1);
|
if (textNumber + 1 > cache.size()) cache.resize(textNumber + 1);
|
||||||
auto prevSentences = cache.at(textNumber).Acquire();
|
auto prevSentences = cache[textNumber].Acquire();
|
||||||
m.unlock();
|
m.unlock();
|
||||||
auto& inserted = prevSentences->emplace_back(sentence);
|
auto& inserted = prevSentences->emplace_back(sentence);
|
||||||
auto firstLocation = std::find(prevSentences->begin(), prevSentences->end(), sentence);
|
auto firstLocation = std::find(prevSentences->begin(), prevSentences->end(), sentence);
|
||||||
|
@ -10,7 +10,7 @@ extern const wchar_t* REPLACER_INSTRUCTIONS;
|
|||||||
constexpr auto REPLACE_SAVE_FILE = u8"SavedReplacements.txt";
|
constexpr auto REPLACE_SAVE_FILE = u8"SavedReplacements.txt";
|
||||||
|
|
||||||
std::atomic<std::filesystem::file_time_type> replaceFileLastWrite = {};
|
std::atomic<std::filesystem::file_time_type> replaceFileLastWrite = {};
|
||||||
std::shared_mutex m;
|
concurrency::reader_writer_lock m;
|
||||||
|
|
||||||
class Trie
|
class Trie
|
||||||
{
|
{
|
||||||
@ -105,7 +105,7 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved
|
|||||||
auto file = std::ofstream(REPLACE_SAVE_FILE, std::ios::binary) << "\xff\xfe";
|
auto file = std::ofstream(REPLACE_SAVE_FILE, std::ios::binary) << "\xff\xfe";
|
||||||
for (auto ch : std::wstring_view(REPLACER_INSTRUCTIONS))
|
for (auto ch : std::wstring_view(REPLACER_INSTRUCTIONS))
|
||||||
file << (ch == L'\n' ? std::string_view("\r\0\n", 4) : std::string_view((char*)&ch, 2));
|
file << (ch == L'\n' ? std::string_view("\r\0\n", 4) : std::string_view((char*)&ch, 2));
|
||||||
_spawnlp(_P_DETACH, "notepad", "notepad", REPLACE_SAVE_FILE, NULL); // show file to user
|
SpawnThread([] { _spawnlp(_P_DETACH, "notepad", "notepad", REPLACE_SAVE_FILE, NULL); }); // show file to user
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -121,7 +121,7 @@ bool ProcessSentence(std::wstring& sentence, SentenceInfo)
|
|||||||
{
|
{
|
||||||
UpdateReplacements();
|
UpdateReplacements();
|
||||||
|
|
||||||
std::shared_lock lock(m);
|
concurrency::reader_writer_lock::scoped_lock_read readLock(m);
|
||||||
sentence = trie.Replace(sentence);
|
sentence = trie.Replace(sentence);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
extern const char* LOAD_SCRIPT;
|
extern const char* LOAD_SCRIPT;
|
||||||
|
|
||||||
constexpr auto STYLE_SAVE_FILE = u8"Textractor.css";
|
constexpr auto STYLE_SAVE_FILE = u8"Textractor.qss";
|
||||||
|
|
||||||
class Window : public QDialog, Localizer
|
class Window : public QDialog, Localizer
|
||||||
{
|
{
|
||||||
@ -13,7 +13,8 @@ public:
|
|||||||
{
|
{
|
||||||
connect(&loadButton, &QPushButton::clicked, this, &Window::LoadScript);
|
connect(&loadButton, &QPushButton::clicked, this, &Window::LoadScript);
|
||||||
|
|
||||||
if (scriptEditor.toPlainText().isEmpty()) scriptEditor.setPlainText("/*https://doc.qt.io/qt-5/stylesheet-syntax.html*/");
|
if (scriptEditor.toPlainText().isEmpty())
|
||||||
|
scriptEditor.setPlainText("/*\nhttps://www.google.com/search?q=Qt+stylesheet+gallery\nhttps://doc.qt.io/qt-5/stylesheet-syntax.html\n*/");
|
||||||
layout.addWidget(&scriptEditor);
|
layout.addWidget(&scriptEditor);
|
||||||
layout.addWidget(&loadButton);
|
layout.addWidget(&loadButton);
|
||||||
|
|
||||||
|
@ -1,25 +1,30 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
#include "extension.h"
|
#include "extension.h"
|
||||||
|
#include "ui_threadlinker.h"
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
|
||||||
extern const char* THREAD_LINKER;
|
extern const char* THREAD_LINKER;
|
||||||
extern const char* LINK;
|
extern const char* LINK;
|
||||||
|
extern const char* UNLINK;
|
||||||
extern const char* THREAD_LINK_FROM;
|
extern const char* THREAD_LINK_FROM;
|
||||||
extern const char* THREAD_LINK_TO;
|
extern const char* THREAD_LINK_TO;
|
||||||
extern const char* HEXADECIMAL;
|
extern const char* HEXADECIMAL;
|
||||||
|
|
||||||
std::unordered_map<int64_t, std::unordered_multiset<int64_t>> linkedTextHandles;
|
std::unordered_map<int64_t, std::unordered_set<int64_t>> links;
|
||||||
std::shared_mutex m;
|
std::unordered_set<int64_t> universalLinks, empty;
|
||||||
|
bool separateSentences = false; // allow user to change?
|
||||||
|
concurrency::reader_writer_lock m;
|
||||||
|
|
||||||
class Window : public QDialog, Localizer
|
class Window : public QDialog, Localizer
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Window() : QDialog(nullptr, Qt::WindowMinMaxButtonsHint)
|
Window() : QDialog(nullptr, Qt::WindowMinMaxButtonsHint)
|
||||||
{
|
{
|
||||||
connect(&linkButton, &QPushButton::clicked, this, &Window::Link);
|
ui.setupUi(this);
|
||||||
|
ui.linkButton->setText(LINK);
|
||||||
layout.addWidget(&linkList);
|
ui.unlinkButton->setText(UNLINK);
|
||||||
layout.addWidget(&linkButton);
|
connect(ui.linkButton, &QPushButton::clicked, this, &Window::Link);
|
||||||
|
connect(ui.unlinkButton, &QPushButton::clicked, this, &Window::Unlink);
|
||||||
|
|
||||||
setWindowTitle(THREAD_LINKER);
|
setWindowTitle(THREAD_LINKER);
|
||||||
QMetaObject::invokeMethod(this, &QWidget::show, Qt::QueuedConnection);
|
QMetaObject::invokeMethod(this, &QWidget::show, Qt::QueuedConnection);
|
||||||
@ -29,39 +34,46 @@ private:
|
|||||||
void Link()
|
void Link()
|
||||||
{
|
{
|
||||||
bool ok1, ok2, ok3, ok4;
|
bool ok1, ok2, ok3, ok4;
|
||||||
int from = QInputDialog::getText(this, THREAD_LINK_FROM, HEXADECIMAL, QLineEdit::Normal, "", &ok1, Qt::WindowCloseButtonHint).toInt(&ok2, 16);
|
QString fromInput = QInputDialog::getText(this, THREAD_LINK_FROM, HEXADECIMAL, QLineEdit::Normal, "All", &ok1, Qt::WindowCloseButtonHint);
|
||||||
|
int from = fromInput.toInt(&ok2, 16);
|
||||||
|
if (ok1 && (fromInput == "All" || ok2))
|
||||||
|
{
|
||||||
int to = QInputDialog::getText(this, THREAD_LINK_TO, HEXADECIMAL, QLineEdit::Normal, "", &ok3, Qt::WindowCloseButtonHint).toInt(&ok4, 16);
|
int to = QInputDialog::getText(this, THREAD_LINK_TO, HEXADECIMAL, QLineEdit::Normal, "", &ok3, Qt::WindowCloseButtonHint).toInt(&ok4, 16);
|
||||||
if (ok1 && ok2 && ok3 && ok4)
|
if (ok3 && ok4)
|
||||||
{
|
{
|
||||||
std::scoped_lock lock(m);
|
std::scoped_lock lock(m);
|
||||||
linkedTextHandles[from].insert(to);
|
if ((ok2 ? links[from] : universalLinks).insert(to).second)
|
||||||
linkList.addItem(QString::number(from, 16) + "->" + QString::number(to, 16));
|
ui.linkList->addItem((ok2 ? QString::number(from, 16) : "All") + "->" + QString::number(to, 16));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Unlink()
|
||||||
|
{
|
||||||
|
if (ui.linkList->currentItem())
|
||||||
|
{
|
||||||
|
QStringList link = ui.linkList->currentItem()->text().split("->");
|
||||||
|
ui.linkList->takeItem(ui.linkList->currentRow());
|
||||||
|
std::scoped_lock lock(m);
|
||||||
|
(link[0] == "All" ? universalLinks : links[link[0].toInt(nullptr, 16)]).erase(link[1].toInt(nullptr, 16));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void keyPressEvent(QKeyEvent* event) override
|
void keyPressEvent(QKeyEvent* event) override
|
||||||
{
|
{
|
||||||
if (event->key() == Qt::Key_Delete && linkList.currentItem())
|
if (event->key() == Qt::Key_Delete) Unlink();
|
||||||
{
|
|
||||||
QStringList link = linkList.currentItem()->text().split("->");
|
|
||||||
linkList.takeItem(linkList.currentRow());
|
|
||||||
std::scoped_lock lock(m);
|
|
||||||
linkedTextHandles[link[0].toInt(nullptr, 16)].erase(link[1].toInt(nullptr, 16));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QHBoxLayout layout{ this };
|
Ui::LinkWindow ui;
|
||||||
QListWidget linkList{ this };
|
|
||||||
QPushButton linkButton{ LINK, this };
|
|
||||||
} window;
|
} window;
|
||||||
|
|
||||||
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
||||||
{
|
{
|
||||||
std::shared_lock lock(m);
|
concurrency::reader_writer_lock::scoped_lock_read readLock(m);
|
||||||
int64_t textHandle = sentenceInfo["text number"];
|
auto action = separateSentences ? sentenceInfo["add sentence"] : sentenceInfo["add text"];
|
||||||
|
auto it = links.find(sentenceInfo["text number"]);
|
||||||
for (auto linkedHandle : linkedTextHandles[textHandle])
|
for (const auto& linkSet : { it != links.end() ? it->second : empty, sentenceInfo["text number"] > 1 ? universalLinks : empty })
|
||||||
((void(*)(int64_t, const wchar_t*))sentenceInfo["void (*AddText)(int64_t number, const wchar_t* text)"])(linkedHandle, sentence.c_str());
|
for (auto link : linkSet)
|
||||||
|
((void(*)(int64_t, const wchar_t*))action)(link, sentence.c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
47
extensions/threadlinker.ui
Normal file
47
extensions/threadlinker.ui
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>LinkWindow</class>
|
||||||
|
<widget class="QDialog">
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>400</width>
|
||||||
|
<height>300</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<layout class="QHBoxLayout">
|
||||||
|
<item>
|
||||||
|
<widget class="QListWidget" name="linkList"/>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<layout class="QVBoxLayout">
|
||||||
|
<item>
|
||||||
|
<spacer>
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="linkButton">
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="unlinkButton">
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer>
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
<resources/>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
@ -1,8 +1,8 @@
|
|||||||
#include "qtcommon.h"
|
#include "qtcommon.h"
|
||||||
#include "extension.h"
|
#include "extension.h"
|
||||||
|
#include "translatewrapper.h"
|
||||||
#include "blockmarkup.h"
|
#include "blockmarkup.h"
|
||||||
#include "network.h"
|
#include <concurrent_priority_queue.h>
|
||||||
#include <map>
|
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
|
|
||||||
@ -13,39 +13,52 @@ extern const char* TRANSLATE_SELECTED_THREAD_ONLY;
|
|||||||
extern const char* RATE_LIMIT_ALL_THREADS;
|
extern const char* RATE_LIMIT_ALL_THREADS;
|
||||||
extern const char* RATE_LIMIT_SELECTED_THREAD;
|
extern const char* RATE_LIMIT_SELECTED_THREAD;
|
||||||
extern const char* USE_TRANS_CACHE;
|
extern const char* USE_TRANS_CACHE;
|
||||||
extern const char* RATE_LIMIT_TOKEN_COUNT;
|
extern const char* FILTER_GARBAGE;
|
||||||
extern const char* RATE_LIMIT_TOKEN_RESTORE_DELAY;
|
extern const char* MAX_TRANSLATIONS_IN_TIMESPAN;
|
||||||
|
extern const char* TIMESPAN;
|
||||||
extern const char* MAX_SENTENCE_SIZE;
|
extern const char* MAX_SENTENCE_SIZE;
|
||||||
extern const char* API_KEY;
|
extern const char* API_KEY;
|
||||||
|
extern const wchar_t* SENTENCE_TOO_LARGE_TO_TRANS;
|
||||||
|
extern const wchar_t* TRANSLATION_ERROR;
|
||||||
extern const wchar_t* TOO_MANY_TRANS_REQUESTS;
|
extern const wchar_t* TOO_MANY_TRANS_REQUESTS;
|
||||||
|
|
||||||
extern const char* TRANSLATION_PROVIDER;
|
extern const char* TRANSLATION_PROVIDER;
|
||||||
extern const char* GET_API_KEY_FROM;
|
extern const char* GET_API_KEY_FROM;
|
||||||
extern QStringList languages;
|
extern const QStringList languagesTo, languagesFrom;
|
||||||
extern std::wstring autoDetectLanguage;
|
extern bool translateSelectedOnly, useRateLimiter, rateLimitSelected, useCache, useFilter;
|
||||||
extern bool translateSelectedOnly, rateLimitAll, rateLimitSelected, useCache;
|
extern int tokenCount, rateLimitTimespan, maxSentenceSize;
|
||||||
extern int tokenCount, tokenRestoreDelay, maxSentenceSize;
|
std::pair<bool, std::wstring> Translate(const std::wstring& text, TranslationParam tlp);
|
||||||
std::pair<bool, std::wstring> Translate(const std::wstring& text);
|
|
||||||
|
|
||||||
// backwards compatibility
|
|
||||||
const char* LANGUAGE = u8"Language";
|
|
||||||
const std::string TRANSLATION_CACHE_FILE = FormatString("%s Translation Cache.txt", TRANSLATION_PROVIDER);
|
|
||||||
|
|
||||||
QFormLayout* display;
|
QFormLayout* display;
|
||||||
Settings settings;
|
Settings settings;
|
||||||
Synchronized<std::wstring> translateTo = L"en", translateFrom = L"auto", authKey;
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
Synchronized<std::map<std::wstring, std::wstring>> translationCache;
|
Synchronized<TranslationParam> tlp;
|
||||||
int savedSize;
|
Synchronized<std::unordered_map<std::wstring, std::wstring>> translationCache;
|
||||||
|
|
||||||
|
std::string CacheFile()
|
||||||
|
{
|
||||||
|
return FormatString("%s Cache (%S).txt", TRANSLATION_PROVIDER, tlp->translateTo);
|
||||||
|
}
|
||||||
void SaveCache()
|
void SaveCache()
|
||||||
{
|
{
|
||||||
std::wstring allTranslations(L"\xfeff");
|
std::wstring allTranslations(L"\xfeff");
|
||||||
for (const auto& [sentence, translation] : translationCache.Acquire().contents)
|
for (const auto& [sentence, translation] : translationCache.Acquire().contents)
|
||||||
allTranslations.append(L"|SENTENCE|").append(sentence).append(L"|TRANSLATION|").append(translation).append(L"|END|\r\n");
|
allTranslations.append(L"|SENTENCE|").append(sentence).append(L"|TRANSLATION|").append(translation).append(L"|END|\r\n");
|
||||||
std::ofstream(TRANSLATION_CACHE_FILE, std::ios::binary | std::ios::trunc).write((const char*)allTranslations.c_str(), allTranslations.size() * sizeof(wchar_t));
|
std::ofstream(CacheFile(), std::ios::binary | std::ios::trunc).write((const char*)allTranslations.c_str(), allTranslations.size() * sizeof(wchar_t));
|
||||||
savedSize = translationCache->size();
|
}
|
||||||
|
void LoadCache()
|
||||||
|
{
|
||||||
|
translationCache->clear();
|
||||||
|
std::ifstream stream(CacheFile(), std::ios::binary);
|
||||||
|
BlockMarkupIterator savedTranslations(stream, Array<std::wstring_view>{ L"|SENTENCE|", L"|TRANSLATION|" });
|
||||||
|
auto translationCache = ::translationCache.Acquire();
|
||||||
|
while (auto read = savedTranslations.Next())
|
||||||
|
{
|
||||||
|
auto& [sentence, translation] = read.value();
|
||||||
|
translationCache->try_emplace(std::move(sentence), std::move(translation));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,31 +72,31 @@ public:
|
|||||||
settings.beginGroup(TRANSLATION_PROVIDER);
|
settings.beginGroup(TRANSLATION_PROVIDER);
|
||||||
|
|
||||||
auto translateToCombo = new QComboBox(this);
|
auto translateToCombo = new QComboBox(this);
|
||||||
translateToCombo->addItems(languages);
|
translateToCombo->addItems(languagesTo);
|
||||||
int language = -1;
|
int i = -1;
|
||||||
if (settings.contains(LANGUAGE)) language = translateToCombo->findText(settings.value(LANGUAGE).toString(), Qt::MatchEndsWith);
|
if (settings.contains(TRANSLATE_TO)) i = translateToCombo->findText(settings.value(TRANSLATE_TO).toString());
|
||||||
if (settings.contains(TRANSLATE_TO)) language = translateToCombo->findText(settings.value(TRANSLATE_TO).toString(), Qt::MatchEndsWith);
|
if (i < 0) i = translateToCombo->findText(NATIVE_LANGUAGE, Qt::MatchStartsWith);
|
||||||
if (language < 0) language = translateToCombo->findText(NATIVE_LANGUAGE, Qt::MatchStartsWith);
|
if (i < 0) i = translateToCombo->findText("English", Qt::MatchStartsWith);
|
||||||
if (language < 0) language = translateToCombo->findText("English", Qt::MatchStartsWith);
|
translateToCombo->setCurrentIndex(i);
|
||||||
translateToCombo->setCurrentIndex(language);
|
|
||||||
SaveTranslateTo(translateToCombo->currentText());
|
SaveTranslateTo(translateToCombo->currentText());
|
||||||
display->addRow(TRANSLATE_TO, translateToCombo);
|
display->addRow(TRANSLATE_TO, translateToCombo);
|
||||||
connect(translateToCombo, &QComboBox::currentTextChanged, this, &Window::SaveTranslateTo);
|
connect(translateToCombo, &QComboBox::currentTextChanged, this, &Window::SaveTranslateTo);
|
||||||
languages.push_front("?: " + S(autoDetectLanguage));
|
|
||||||
auto translateFromCombo = new QComboBox(this);
|
auto translateFromCombo = new QComboBox(this);
|
||||||
translateFromCombo->addItems(languages);
|
translateFromCombo->addItem("?");
|
||||||
language = -1;
|
translateFromCombo->addItems(languagesFrom);
|
||||||
if (settings.contains(TRANSLATE_FROM)) language = translateFromCombo->findText(settings.value(TRANSLATE_FROM).toString(), Qt::MatchEndsWith);
|
i = -1;
|
||||||
if (language < 0) language = translateFromCombo->findText("?", Qt::MatchStartsWith);
|
if (settings.contains(TRANSLATE_FROM)) i = translateFromCombo->findText(settings.value(TRANSLATE_FROM).toString());
|
||||||
translateFromCombo->setCurrentIndex(language);
|
if (i < 0) i = 0;
|
||||||
|
translateFromCombo->setCurrentIndex(i);
|
||||||
SaveTranslateFrom(translateFromCombo->currentText());
|
SaveTranslateFrom(translateFromCombo->currentText());
|
||||||
display->addRow(TRANSLATE_FROM, translateFromCombo);
|
display->addRow(TRANSLATE_FROM, translateFromCombo);
|
||||||
connect(translateFromCombo, &QComboBox::currentTextChanged, this, &Window::SaveTranslateFrom);
|
connect(translateFromCombo, &QComboBox::currentTextChanged, this, &Window::SaveTranslateFrom);
|
||||||
for (auto [value, label] : Array<bool&, const char*>{
|
for (auto [value, label] : Array<bool&, const char*>{
|
||||||
{ translateSelectedOnly, TRANSLATE_SELECTED_THREAD_ONLY },
|
{ translateSelectedOnly, TRANSLATE_SELECTED_THREAD_ONLY },
|
||||||
{ rateLimitAll, RATE_LIMIT_ALL_THREADS },
|
{ useRateLimiter, RATE_LIMIT_ALL_THREADS },
|
||||||
{ rateLimitSelected, RATE_LIMIT_SELECTED_THREAD },
|
{ rateLimitSelected, RATE_LIMIT_SELECTED_THREAD },
|
||||||
{ useCache, USE_TRANS_CACHE },
|
{ useCache, USE_TRANS_CACHE },
|
||||||
|
{ useFilter, FILTER_GARBAGE }
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
value = settings.value(label, value).toBool();
|
value = settings.value(label, value).toBool();
|
||||||
@ -93,8 +106,8 @@ public:
|
|||||||
connect(checkBox, &QCheckBox::clicked, [label, &value](bool checked) { settings.setValue(label, value = checked); });
|
connect(checkBox, &QCheckBox::clicked, [label, &value](bool checked) { settings.setValue(label, value = checked); });
|
||||||
}
|
}
|
||||||
for (auto [value, label] : Array<int&, const char*>{
|
for (auto [value, label] : Array<int&, const char*>{
|
||||||
{ tokenCount, RATE_LIMIT_TOKEN_COUNT },
|
{ tokenCount, MAX_TRANSLATIONS_IN_TIMESPAN },
|
||||||
{ tokenRestoreDelay, RATE_LIMIT_TOKEN_RESTORE_DELAY },
|
{ rateLimitTimespan, TIMESPAN },
|
||||||
{ maxSentenceSize, MAX_SENTENCE_SIZE },
|
{ maxSentenceSize, MAX_SENTENCE_SIZE },
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
@ -108,8 +121,8 @@ public:
|
|||||||
if (GET_API_KEY_FROM)
|
if (GET_API_KEY_FROM)
|
||||||
{
|
{
|
||||||
auto keyEdit = new QLineEdit(settings.value(API_KEY).toString(), this);
|
auto keyEdit = new QLineEdit(settings.value(API_KEY).toString(), this);
|
||||||
authKey->assign(S(keyEdit->text()));
|
tlp->authKey = S(keyEdit->text());
|
||||||
QObject::connect(keyEdit, &QLineEdit::textChanged, [](QString key) { settings.setValue(API_KEY, S(authKey->assign(S(key)))); });
|
QObject::connect(keyEdit, &QLineEdit::textChanged, [](QString key) { settings.setValue(API_KEY, S(tlp->authKey = S(key))); });
|
||||||
auto keyLabel = new QLabel(QString("<a href=\"%1\">%2</a>").arg(GET_API_KEY_FROM, API_KEY), this);
|
auto keyLabel = new QLabel(QString("<a href=\"%1\">%2</a>").arg(GET_API_KEY_FROM, API_KEY), this);
|
||||||
keyLabel->setOpenExternalLinks(true);
|
keyLabel->setOpenExternalLinks(true);
|
||||||
display->addRow(keyLabel, keyEdit);
|
display->addRow(keyLabel, keyEdit);
|
||||||
@ -117,16 +130,6 @@ public:
|
|||||||
|
|
||||||
setWindowTitle(TRANSLATION_PROVIDER);
|
setWindowTitle(TRANSLATION_PROVIDER);
|
||||||
QMetaObject::invokeMethod(this, &QWidget::show, Qt::QueuedConnection);
|
QMetaObject::invokeMethod(this, &QWidget::show, Qt::QueuedConnection);
|
||||||
|
|
||||||
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())
|
|
||||||
{
|
|
||||||
auto& [sentence, translation] = read.value();
|
|
||||||
translationCache->try_emplace(std::move(sentence), std::move(translation));
|
|
||||||
}
|
|
||||||
savedSize = translationCache->size();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
~Window()
|
~Window()
|
||||||
@ -137,58 +140,73 @@ public:
|
|||||||
private:
|
private:
|
||||||
void SaveTranslateTo(QString language)
|
void SaveTranslateTo(QString language)
|
||||||
{
|
{
|
||||||
settings.setValue(TRANSLATE_TO, S(translateTo->assign(S(language.split(": ")[1]))));
|
SaveCache();
|
||||||
|
settings.setValue(TRANSLATE_TO, S(tlp->translateTo = S(language)));
|
||||||
|
LoadCache();
|
||||||
}
|
}
|
||||||
void SaveTranslateFrom(QString language)
|
void SaveTranslateFrom(QString language)
|
||||||
{
|
{
|
||||||
settings.setValue(TRANSLATE_FROM, S(translateFrom->assign(S(language.split(": ")[1]))));
|
settings.setValue(TRANSLATE_FROM, S(tlp->translateFrom = S(language)));
|
||||||
}
|
}
|
||||||
} window;
|
} window;
|
||||||
|
|
||||||
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
bool ProcessSentence(std::wstring& sentence, SentenceInfo sentenceInfo)
|
||||||
{
|
{
|
||||||
if (sentenceInfo["text number"] == 0 || sentence.size() > maxSentenceSize) return false;
|
if (sentenceInfo["text number"] == 0) return false;
|
||||||
|
|
||||||
static class
|
static class
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
bool Request()
|
bool Request()
|
||||||
{
|
{
|
||||||
auto tokens = this->tokens.Acquire();
|
DWORD64 current = GetTickCount64(), token;
|
||||||
tokens->push_back(GetTickCount());
|
while (tokens.try_pop(token)) if (token > current - rateLimitTimespan)
|
||||||
if (tokens->size() > tokenCount * 5) tokens->erase(tokens->begin(), tokens->begin() + tokenCount * 3);
|
{
|
||||||
tokens->erase(std::remove_if(tokens->begin(), tokens->end(), [](DWORD token) { return GetTickCount() - token > tokenRestoreDelay; }), tokens->end());
|
tokens.push(token); // popped one too many
|
||||||
return tokens->size() < tokenCount;
|
break;
|
||||||
|
}
|
||||||
|
bool available = tokens.size() < tokenCount;
|
||||||
|
if (available) tokens.push(current);
|
||||||
|
return available;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Synchronized<std::vector<DWORD>> tokens;
|
concurrency::concurrent_priority_queue<DWORD64, std::greater<DWORD64>> tokens;
|
||||||
} rateLimiter;
|
} rateLimiter;
|
||||||
|
|
||||||
auto StripWhitespace = [](std::wstring& text)
|
|
||||||
{
|
|
||||||
text.erase(text.begin(), std::find_if_not(text.begin(), text.end(), iswspace));
|
|
||||||
text.erase(std::find_if_not(text.rbegin(), text.rend(), iswspace).base(), text.end());
|
|
||||||
};
|
|
||||||
|
|
||||||
bool cache = false;
|
bool cache = false;
|
||||||
std::wstring translation;
|
std::wstring translation;
|
||||||
StripWhitespace(sentence);
|
if (useFilter)
|
||||||
|
{
|
||||||
|
Trim(sentence);
|
||||||
|
sentence.erase(std::remove_if(sentence.begin(), sentence.end(), [](wchar_t ch) { return ch < ' ' && ch != '\n'; }), sentence.end());
|
||||||
|
}
|
||||||
|
if (sentence.empty()) return true;
|
||||||
|
if (sentence.size() > maxSentenceSize) translation = SENTENCE_TOO_LARGE_TO_TRANS;
|
||||||
if (useCache)
|
if (useCache)
|
||||||
{
|
{
|
||||||
auto translationCache = ::translationCache.Acquire();
|
auto translationCache = ::translationCache.Acquire();
|
||||||
if (auto it = translationCache->find(sentence); it != translationCache->end()) translation = it->second + L"\x200b"; // dumb hack to not try to translate if stored empty translation
|
if (auto it = translationCache->find(sentence); it != translationCache->end()) translation = it->second;
|
||||||
}
|
}
|
||||||
if (translation.empty() && (!translateSelectedOnly || sentenceInfo["current select"]))
|
if (translation.empty() && (!translateSelectedOnly || sentenceInfo["current select"]))
|
||||||
if (rateLimiter.Request() || !rateLimitAll || (!rateLimitSelected && sentenceInfo["current select"])) std::tie(cache, translation) = Translate(sentence);
|
if (rateLimiter.Request() || !useRateLimiter || (!rateLimitSelected && sentenceInfo["current select"])) std::tie(cache, translation) = Translate(sentence, tlp.Copy());
|
||||||
else translation = TOO_MANY_TRANS_REQUESTS;
|
else translation = TOO_MANY_TRANS_REQUESTS;
|
||||||
StripWhitespace(translation);
|
if (cache) translationCache->operator[](sentence) = translation;
|
||||||
if (cache) translationCache->try_emplace(sentence, translation);
|
|
||||||
if (cache && translationCache->size() > savedSize + 50) SaveCache();
|
|
||||||
|
|
||||||
|
if (useFilter) Trim(translation);
|
||||||
for (int i = 0; i < translation.size(); ++i) if (translation[i] == '\r' && translation[i + 1] == '\n') translation[i] = 0x200b; // for some reason \r appears as newline - no need to double
|
for (int i = 0; i < translation.size(); ++i) if (translation[i] == '\r' && translation[i + 1] == '\n') translation[i] = 0x200b; // for some reason \r appears as newline - no need to double
|
||||||
if (!translation.empty()) (sentence += L"\x200b \n") += translation;
|
if (translation.empty()) translation = TRANSLATION_ERROR;
|
||||||
|
(sentence += L"\x200b \n") += translation;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(assert(Translate(L"こんにちは").second.find(L"ello") != std::string::npos));
|
extern const std::unordered_map<std::wstring, std::wstring> codes;
|
||||||
|
TEST(
|
||||||
|
{
|
||||||
|
assert(Translate(L"こんにちは", { L"English", L"?", L"" }).second.find(L"ello") == 1 || strstr(TRANSLATION_PROVIDER, "DevTools"));
|
||||||
|
|
||||||
|
for (auto languages : { languagesFrom, languagesTo }) for (auto language : languages)
|
||||||
|
assert(codes.count(S(language)));
|
||||||
|
assert(codes.count(L"?"));
|
||||||
|
}
|
||||||
|
);
|
||||||
|
6
extensions/translatewrapper.h
Normal file
6
extensions/translatewrapper.h
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
struct TranslationParam
|
||||||
|
{
|
||||||
|
std::wstring translateTo, translateFrom, authKey;
|
||||||
|
};
|
@ -1,9 +1,4 @@
|
|||||||
# The CLI isn't used by Textractor itself, but is here for other people that want to build projects on top of Textractor
|
# The CLI isn't used by Textractor itself, but is here for other people that want to build projects on top of Textractor
|
||||||
add_executable(TextractorCLI
|
add_executable(TextractorCLI main.cpp)
|
||||||
cli.cpp
|
|
||||||
exception.cpp
|
|
||||||
host.cpp
|
|
||||||
textthread.cpp
|
|
||||||
hookcode.cpp
|
|
||||||
)
|
|
||||||
target_precompile_headers(TextractorCLI REUSE_FROM pch)
|
target_precompile_headers(TextractorCLI REUSE_FROM pch)
|
||||||
|
target_link_libraries(TextractorCLI host)
|
@ -1,5 +1,5 @@
|
|||||||
#include "host.h"
|
#include "../host.h"
|
||||||
#include "hookcode.h"
|
#include "../hookcode.h"
|
||||||
#include <io.h>
|
#include <io.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <iostream>
|
#include <iostream>
|
8
host/CMakeLists.txt
Normal file
8
host/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
add_library(host
|
||||||
|
host.cpp
|
||||||
|
textthread.cpp
|
||||||
|
hookcode.cpp
|
||||||
|
)
|
||||||
|
target_precompile_headers(host REUSE_FROM pch)
|
||||||
|
|
||||||
|
add_subdirectory(CLI)
|
@ -276,7 +276,9 @@ namespace HookCode
|
|||||||
{
|
{
|
||||||
std::optional<HookParam> Parse(std::wstring code)
|
std::optional<HookParam> Parse(std::wstring code)
|
||||||
{
|
{
|
||||||
if (code[0] == L'/') code.erase(0, 1); // legacy/AGTH compatibility
|
if (code[0] == L'/') code.erase(0, 1);
|
||||||
|
code.erase(std::find(code.begin(), code.end(), L'/'), code.end()); // legacy/AGTH compatibility
|
||||||
|
Trim(code);
|
||||||
if (code[0] == L'R') return ParseRCode(code.erase(0, 1));
|
if (code[0] == L'R') return ParseRCode(code.erase(0, 1));
|
||||||
else if (code[0] == L'H') return ParseHCode(code.erase(0, 1));
|
else if (code[0] == L'H') return ParseHCode(code.erase(0, 1));
|
||||||
return {};
|
return {};
|
||||||
@ -292,6 +294,7 @@ namespace HookCode
|
|||||||
assert(HexString(-12) == L"-C"),
|
assert(HexString(-12) == L"-C"),
|
||||||
assert(HexString(12) == L"C"),
|
assert(HexString(12) == L"C"),
|
||||||
assert(Parse(L"/HQN936#-c*C:C*1C@4AA:gdi.dll:GetTextOutA")),
|
assert(Parse(L"/HQN936#-c*C:C*1C@4AA:gdi.dll:GetTextOutA")),
|
||||||
|
assert(Parse(L"/HQN936#-c*C:C*1C@4AA:gdi.dll:GetTextOutA /KF")),
|
||||||
assert(Parse(L"HB4@0")),
|
assert(Parse(L"HB4@0")),
|
||||||
assert(Parse(L"/RS65001#@44")),
|
assert(Parse(L"/RS65001#@44")),
|
||||||
assert(Parse(L"HQ@4")),
|
assert(Parse(L"HQ@4")),
|
@ -63,7 +63,7 @@ void TextThread::Push(BYTE* data, int length)
|
|||||||
else if (auto converted = StringToWideString(std::string((char*)data, length), hp.codepage ? hp.codepage : Host::defaultCodepage)) buffer.append(converted.value());
|
else if (auto converted = StringToWideString(std::string((char*)data, length), hp.codepage ? hp.codepage : Host::defaultCodepage)) buffer.append(converted.value());
|
||||||
else Host::AddConsoleOutput(INVALID_CODEPAGE);
|
else Host::AddConsoleOutput(INVALID_CODEPAGE);
|
||||||
if (hp.type & FULL_STRING) buffer.push_back(L'\n');
|
if (hp.type & FULL_STRING) buffer.push_back(L'\n');
|
||||||
lastPushTime = GetTickCount();
|
lastPushTime = GetTickCount64();
|
||||||
|
|
||||||
if (filterRepetition)
|
if (filterRepetition)
|
||||||
{
|
{
|
||||||
@ -87,7 +87,7 @@ void TextThread::Push(const wchar_t* data)
|
|||||||
{
|
{
|
||||||
std::scoped_lock lock(bufferMutex);
|
std::scoped_lock lock(bufferMutex);
|
||||||
// not sure if this should filter repetition
|
// not sure if this should filter repetition
|
||||||
lastPushTime = GetTickCount();
|
lastPushTime = GetTickCount64();
|
||||||
buffer += data;
|
buffer += data;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -104,13 +104,13 @@ void TextThread::Flush()
|
|||||||
for (auto& sentence : sentences)
|
for (auto& sentence : sentences)
|
||||||
{
|
{
|
||||||
totalSize += sentence.size();
|
totalSize += sentence.size();
|
||||||
sentence.erase(std::remove(sentence.begin(), sentence.end(), L'\0'), sentence.end());
|
sentence.erase(std::remove(sentence.begin(), sentence.end(), 0), sentence.end());
|
||||||
if (Output(*this, sentence)) storage->append(sentence);
|
if (Output(*this, sentence)) storage->append(sentence);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::scoped_lock lock(bufferMutex);
|
std::scoped_lock lock(bufferMutex);
|
||||||
if (buffer.empty()) return;
|
if (buffer.empty()) return;
|
||||||
if (buffer.size() > maxBufferSize || GetTickCount() - lastPushTime > flushDelay)
|
if (buffer.size() > maxBufferSize || GetTickCount64() - lastPushTime > flushDelay)
|
||||||
{
|
{
|
||||||
AddSentence(std::move(buffer));
|
AddSentence(std::move(buffer));
|
||||||
buffer.clear();
|
buffer.clear();
|
@ -8,9 +8,9 @@ public:
|
|||||||
using OutputCallback = bool(*)(TextThread&, std::wstring&);
|
using OutputCallback = bool(*)(TextThread&, std::wstring&);
|
||||||
inline static OutputCallback Output;
|
inline static OutputCallback Output;
|
||||||
|
|
||||||
inline static bool filterRepetition = true;
|
inline static bool filterRepetition = false;
|
||||||
inline static int flushDelay = 400; // flush every 400ms by default
|
inline static int flushDelay = 500; // flush every 500ms by default
|
||||||
inline static int maxBufferSize = 1000;
|
inline static int maxBufferSize = 3000;
|
||||||
inline static int maxHistorySize = 10'000'000;
|
inline static int maxHistorySize = 10'000'000;
|
||||||
|
|
||||||
TextThread(ThreadParam tp, HookParam hp, std::optional<std::wstring> name = {});
|
TextThread(ThreadParam tp, HookParam hp, std::optional<std::wstring> name = {});
|
||||||
@ -36,7 +36,7 @@ private:
|
|||||||
BYTE leadByte = 0;
|
BYTE leadByte = 0;
|
||||||
std::unordered_set<wchar_t> repeatingChars;
|
std::unordered_set<wchar_t> repeatingChars;
|
||||||
std::mutex bufferMutex;
|
std::mutex bufferMutex;
|
||||||
DWORD lastPushTime = 0;
|
DWORD64 lastPushTime = 0;
|
||||||
Synchronized<std::vector<std::wstring>> queuedSentences;
|
Synchronized<std::vector<std::wstring>> queuedSentences;
|
||||||
struct TimerDeleter { void operator()(HANDLE h) { DeleteTimerQueueTimer(NULL, h, INVALID_HANDLE_VALUE); } };
|
struct TimerDeleter { void operator()(HANDLE h) { DeleteTimerQueueTimer(NULL, h, INVALID_HANDLE_VALUE); } };
|
||||||
AutoHandle<TimerDeleter> timer = NULL;
|
AutoHandle<TimerDeleter> timer = NULL;
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
#define WIN32_LEAN_AND_MEAN
|
#define WIN32_LEAN_AND_MEAN
|
||||||
#include <Windows.h>
|
#include <Windows.h>
|
||||||
|
#include <concrt.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
@ -15,7 +16,6 @@
|
|||||||
#include <optional>
|
#include <optional>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <shared_mutex>
|
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
@ -76,7 +76,19 @@ private:
|
|||||||
M m;
|
M m;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct // should be inline but MSVC (linker) is bugged
|
template <typename F>
|
||||||
|
void SpawnThread(const F& f) // works in DllMain unlike std thread
|
||||||
|
{
|
||||||
|
F* copy = new F(f);
|
||||||
|
CloseHandle(CreateThread(nullptr, 0, [](void* copy)
|
||||||
|
{
|
||||||
|
(*(F*)copy)();
|
||||||
|
delete (F*)copy;
|
||||||
|
return 0UL;
|
||||||
|
}, copy, 0, nullptr));
|
||||||
|
}
|
||||||
|
|
||||||
|
inline struct
|
||||||
{
|
{
|
||||||
inline static BYTE DUMMY[100];
|
inline static BYTE DUMMY[100];
|
||||||
template <typename T> operator T*() { static_assert(sizeof(T) < sizeof(DUMMY)); return (T*)DUMMY; }
|
template <typename T> operator T*() { static_assert(sizeof(T) < sizeof(DUMMY)); return (T*)DUMMY; }
|
||||||
@ -108,6 +120,12 @@ inline std::wstring FormatString(const wchar_t* format, const Args&... args)
|
|||||||
}
|
}
|
||||||
#pragma warning(pop)
|
#pragma warning(pop)
|
||||||
|
|
||||||
|
inline void Trim(std::wstring& text)
|
||||||
|
{
|
||||||
|
text.erase(text.begin(), std::find_if_not(text.begin(), text.end(), iswspace));
|
||||||
|
text.erase(std::find_if_not(text.rbegin(), text.rend(), iswspace).base(), text.end());
|
||||||
|
}
|
||||||
|
|
||||||
inline std::optional<std::wstring> StringToWideString(const std::string& text, UINT encoding)
|
inline std::optional<std::wstring> StringToWideString(const std::string& text, UINT encoding)
|
||||||
{
|
{
|
||||||
std::vector<wchar_t> buffer(text.size() + 1);
|
std::vector<wchar_t> buffer(text.size() + 1);
|
||||||
@ -134,12 +152,12 @@ template <typename... Args>
|
|||||||
inline void TEXTRACTOR_MESSAGE(const wchar_t* format, const Args&... args) { MessageBoxW(NULL, FormatString(format, args...).c_str(), L"Textractor", MB_OK); }
|
inline void TEXTRACTOR_MESSAGE(const wchar_t* format, const Args&... args) { MessageBoxW(NULL, FormatString(format, args...).c_str(), L"Textractor", MB_OK); }
|
||||||
|
|
||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
inline void TEXTRACTOR_DEBUG(const wchar_t* format, const Args&... args) { std::thread([=] { TEXTRACTOR_MESSAGE(format, args...); }).detach(); }
|
inline void TEXTRACTOR_DEBUG(const wchar_t* format, const Args&... args) { SpawnThread([=] { TEXTRACTOR_MESSAGE(format, args...); }); }
|
||||||
|
|
||||||
void Localize();
|
void Localize();
|
||||||
|
|
||||||
#ifdef _DEBUG
|
#ifdef _DEBUG
|
||||||
#define TEST(...) static auto _ = CreateThread(nullptr, 0, [](auto) { __VA_ARGS__; return 0UL; }, NULL, 0, nullptr);
|
#define TEST(...) static auto _ = CreateThread(nullptr, 0, [](auto) { __VA_ARGS__; return 0UL; }, NULL, 0, nullptr)
|
||||||
#else
|
#else
|
||||||
#define TEST(...)
|
#define TEST(...)
|
||||||
#endif
|
#endif
|
||||||
|
@ -62,7 +62,7 @@ struct SearchParam
|
|||||||
BYTE pattern[PATTERN_SIZE] = { x64 ? 0xcc : 0x55, x64 ? 0xcc : 0x8b, x64 ? 0x48 : 0xec, 0x89 }; // pattern in memory to search for
|
BYTE pattern[PATTERN_SIZE] = { x64 ? 0xcc : 0x55, x64 ? 0xcc : 0x8b, x64 ? 0x48 : 0xec, 0x89 }; // pattern in memory to search for
|
||||||
int length = x64 ? 4 : 3, // length of pattern (zero means this SearchParam is invalid and the default should be used)
|
int length = x64 ? 4 : 3, // length of pattern (zero means this SearchParam is invalid and the default should be used)
|
||||||
offset = x64 ? 2 : 0, // offset from start of pattern to add hook
|
offset = x64 ? 2 : 0, // offset from start of pattern to add hook
|
||||||
searchTime = 20000, // ms
|
searchTime = 30000, // ms
|
||||||
maxRecords = 100000,
|
maxRecords = 100000,
|
||||||
codepage = SHIFT_JIS;
|
codepage = SHIFT_JIS;
|
||||||
uintptr_t padding = 0, // same as hook param padding
|
uintptr_t padding = 0, // same as hook param padding
|
||||||
|
@ -10,7 +10,8 @@ DefaultGroupName=Textractor
|
|||||||
MinVersion=6.1
|
MinVersion=6.1
|
||||||
OutputBaseFilename=Textractor-{#VERSION}-Setup
|
OutputBaseFilename=Textractor-{#VERSION}-Setup
|
||||||
OutputDir=Builds
|
OutputDir=Builds
|
||||||
PrivilegesRequired=lowest
|
PrivilegesRequired=admin
|
||||||
|
PrivilegesRequiredOverridesAllowed=dialog
|
||||||
SolidCompression=yes
|
SolidCompression=yes
|
||||||
Uninstallable=no
|
Uninstallable=no
|
||||||
|
|
||||||
@ -41,3 +42,4 @@ Source: "Builds\Textractor-Thai-{#VERSION}\*"; DestDir: "{app}"; Languages: th;
|
|||||||
Source: "Builds\Textractor-Korean-{#VERSION}\*"; DestDir: "{app}"; Languages: ko; Flags: recursesubdirs ignoreversion
|
Source: "Builds\Textractor-Korean-{#VERSION}\*"; DestDir: "{app}"; Languages: ko; Flags: recursesubdirs ignoreversion
|
||||||
Source: "Builds\Textractor-Italian-{#VERSION}\*"; DestDir: "{app}"; Languages: it; Flags: recursesubdirs ignoreversion
|
Source: "Builds\Textractor-Italian-{#VERSION}\*"; DestDir: "{app}"; Languages: it; Flags: recursesubdirs ignoreversion
|
||||||
Source: "Builds\Textractor-French-{#VERSION}\*"; DestDir: "{app}"; Languages: fr; Flags: recursesubdirs ignoreversion
|
Source: "Builds\Textractor-French-{#VERSION}\*"; DestDir: "{app}"; Languages: fr; Flags: recursesubdirs ignoreversion
|
||||||
|
Source: "INSTALL_THIS_UNICODE_FONT.ttf"; DestDir: "{autofonts}"; DestName: "ARIAL_UNICODE_MS.ttf"; FontInstall: "Arial Unicode MS";
|
||||||
|
264
text.cpp
264
text.cpp
@ -33,8 +33,8 @@ const char* SETTINGS = u8"Settings";
|
|||||||
const char* EXTENSIONS = u8"Extensions";
|
const char* EXTENSIONS = u8"Extensions";
|
||||||
const char* SELECT_PROCESS = u8"Select process";
|
const char* SELECT_PROCESS = u8"Select process";
|
||||||
const char* ATTACH_INFO = u8R"(If you don't see the process you want to attach, try running with admin rights
|
const char* ATTACH_INFO = u8R"(If you don't see the process you want to attach, try running with admin rights
|
||||||
You can also type in the process id)";
|
You can also type in the process ID)";
|
||||||
const char* SELECT_PROCESS_INFO = u8"If you manually type in the process file name, please use the absolute path";
|
const char* SELECT_PROCESS_INFO = u8"If you manually type in the process file name, use the absolute path";
|
||||||
const char* FROM_COMPUTER = u8"Select from computer";
|
const char* FROM_COMPUTER = u8"Select from computer";
|
||||||
const char* PROCESSES = u8"Processes (*.exe)";
|
const char* PROCESSES = u8"Processes (*.exe)";
|
||||||
const char* CODE_INFODUMP = u8R"(Enter read code
|
const char* CODE_INFODUMP = u8R"(Enter read code
|
||||||
@ -57,18 +57,18 @@ Negatives for data_offset/split_offset refer to registers
|
|||||||
-C for RAX, -14 for RBX, -1C for RCX, -24 for RDX, and so on for RSP, RBP, RSI, RDI, R8-R15
|
-C for RAX, -14 for RBX, -1C for RCX, -24 for RDX, and so on for RSP, RBP, RSI, RDI, R8-R15
|
||||||
* means dereference pointer+deref_offset)";
|
* means dereference pointer+deref_offset)";
|
||||||
const char* SAVE_SETTINGS = u8"Save settings";
|
const char* SAVE_SETTINGS = u8"Save settings";
|
||||||
const char* EXTEN_WINDOW_INSTRUCTIONS = u8R"(To add an extension, right click the extension list
|
const char* EXTEN_WINDOW_INSTRUCTIONS = u8R"(Right click the list to add or remove extensions
|
||||||
Alternatively, drag and drop the extension file from your computer
|
Drag and drop extensions within the list to reorder them
|
||||||
To reorder extensions, drag and drop them within the list
|
(Extensions are used from top to bottom: order DOES matter))";
|
||||||
(Extensions are used from top to bottom: order DOES matter)
|
|
||||||
To remove an extension, select it and press delete)";
|
|
||||||
const char* ADD_EXTENSION = u8"Add extension";
|
const char* ADD_EXTENSION = u8"Add extension";
|
||||||
|
const char* REMOVE_EXTENSION = u8"Remove extension";
|
||||||
const char* INVALID_EXTENSION = u8"%1 is an invalid extension";
|
const char* INVALID_EXTENSION = u8"%1 is an invalid extension";
|
||||||
const char* CONFIRM_EXTENSION_OVERWRITE = u8"Another version of this extension already exists, do you want to delete and overwrite it?";
|
const char* CONFIRM_EXTENSION_OVERWRITE = u8"Another version of this extension already exists, do you want to delete and overwrite it?";
|
||||||
const char* EXTENSION_WRITE_ERROR = u8"Failed to save extension";
|
const char* EXTENSION_WRITE_ERROR = u8"Failed to save extension";
|
||||||
const char* USE_JP_LOCALE = u8"Emulate japanese locale?";
|
const char* USE_JP_LOCALE = u8"Emulate japanese locale?";
|
||||||
const char* FAILED_TO_CREATE_CONFIG_FILE = u8"Failed to create config file \"%1\"";
|
const char* FAILED_TO_CREATE_CONFIG_FILE = u8"Failed to create config file \"%1\"";
|
||||||
const char* HOOK_SEARCH_UNSTABLE_WARNING = u8"Searching for hooks is unstable! Be prepared for your game to crash!";
|
const char* HOOK_SEARCH_UNSTABLE_WARNING = u8"Searching for hooks is unstable! Be prepared for your game to crash!";
|
||||||
|
const char* HOOK_SEARCH_STARTING_VIEW_CONSOLE = u8"Initializing hook search - please check console for further instructions";
|
||||||
const char* SEARCH_CJK = u8"Search for Chinese/Japanese/Korean";
|
const char* SEARCH_CJK = u8"Search for Chinese/Japanese/Korean";
|
||||||
const char* SEARCH_PATTERN = u8"Search pattern (hex byte array)";
|
const char* SEARCH_PATTERN = u8"Search pattern (hex byte array)";
|
||||||
const char* SEARCH_DURATION = u8"Search duration (ms)";
|
const char* SEARCH_DURATION = u8"Search duration (ms)";
|
||||||
@ -97,16 +97,16 @@ const char* MAX_HISTORY_SIZE = u8"Max history size";
|
|||||||
const char* CONFIG_JP_LOCALE = u8"Launch with JP locale";
|
const char* CONFIG_JP_LOCALE = u8"Launch with JP locale";
|
||||||
const wchar_t* CONSOLE = L"Console";
|
const wchar_t* CONSOLE = L"Console";
|
||||||
const wchar_t* CLIPBOARD = L"Clipboard";
|
const wchar_t* CLIPBOARD = L"Clipboard";
|
||||||
const wchar_t* ABOUT = L"Textractor " ARCH L" v" VERSION LR"( made by me: Artikash (email: akashmozumdar@gmail.com)
|
const wchar_t* ABOUT = L"Textractor " ARCH L" v" VERSION LR"( made by Artikash (email: akashmozumdar@gmail.com)
|
||||||
Project homepage: https://github.com/Artikash/Textractor
|
Project homepage: https://github.com/Artikash/Textractor
|
||||||
Tutorial video: https://tinyurl.com/textractor-tutorial
|
Tutorial video: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
FAQ: https://github.com/Artikash/Textractor/wiki/FAQ
|
FAQ: https://github.com/Artikash/Textractor/wiki/FAQ
|
||||||
Please contact me with any problems, feature requests, or questions relating to Textractor
|
Please contact Artikash with any problems, feature requests, or questions relating to Textractor
|
||||||
You can do so via the project homepage (issues section) or via email
|
You can do so via the project homepage (issues section) or via email
|
||||||
Source code available under GPLv3 at project homepage
|
Source code available under GPLv3 at project homepage
|
||||||
If you like this project, please tell everyone about it :))";
|
If you like this project, please tell everyone about it! It's time to put AGTH down :))";
|
||||||
const wchar_t* CL_OPTIONS = LR"(usage: Textractor [-p{process id|"process name"}]...
|
const wchar_t* CL_OPTIONS = LR"(usage: Textractor [-p{process ID|"process name"}]...
|
||||||
example: Textractor -p4466 -p"My Game.exe" tries to inject processes with id 4466 or with name My Game.exe)";
|
example: Textractor -p4466 -p"My Game.exe" tries to inject processes with ID 4466 or with name My Game.exe)";
|
||||||
const wchar_t* UPDATE_AVAILABLE = L"Update available: download it from https://github.com/Artikash/Textractor/releases";
|
const wchar_t* UPDATE_AVAILABLE = L"Update available: download it from https://github.com/Artikash/Textractor/releases";
|
||||||
const wchar_t* ALREADY_INJECTED = L"Textractor: already injected";
|
const wchar_t* ALREADY_INJECTED = L"Textractor: already injected";
|
||||||
const wchar_t* NEED_32_BIT = L"Textractor: architecture mismatch: only Textractor x86 can inject this process";
|
const wchar_t* NEED_32_BIT = L"Textractor: architecture mismatch: only Textractor x86 can inject this process";
|
||||||
@ -120,9 +120,10 @@ const char* INSERTING_HOOK = u8"Textractor: inserting hook: %s";
|
|||||||
const char* REMOVING_HOOK = u8"Textractor: removing hook: %s";
|
const char* REMOVING_HOOK = u8"Textractor: removing hook: %s";
|
||||||
const char* HOOK_FAILED = u8"Textractor: failed to insert hook";
|
const char* HOOK_FAILED = u8"Textractor: failed to insert hook";
|
||||||
const char* TOO_MANY_HOOKS = u8"Textractor: too many hooks: can't insert";
|
const char* TOO_MANY_HOOKS = u8"Textractor: too many hooks: can't insert";
|
||||||
const char* STARTING_SEARCH = u8"Textractor: starting search";
|
const char* HOOK_SEARCH_STARTING = u8"Textractor: starting hook search";
|
||||||
|
const char* HOOK_SEARCH_INITIALIZING = u8"Textractor: initializing hook search (%f%%)";
|
||||||
const char* NOT_ENOUGH_TEXT = u8"Textractor: not enough text to search accurately";
|
const char* NOT_ENOUGH_TEXT = u8"Textractor: not enough text to search accurately";
|
||||||
const char* HOOK_SEARCH_INITIALIZED = u8"Textractor: search initialized with %zd hooks";
|
const char* HOOK_SEARCH_INITIALIZED = u8"Textractor: initialized hook search with %zd hooks";
|
||||||
const char* MAKE_GAME_PROCESS_TEXT = u8"Textractor: please click around in the game to force it to process text during the next %d seconds";
|
const char* MAKE_GAME_PROCESS_TEXT = u8"Textractor: please click around in the game to force it to process text during the next %d seconds";
|
||||||
const char* HOOK_SEARCH_FINISHED = u8"Textractor: hook search finished, %d results found";
|
const char* HOOK_SEARCH_FINISHED = u8"Textractor: hook search finished, %d results found";
|
||||||
const char* OUT_OF_RECORDS_RETRY = u8"Textractor: out of search records, please retry if results are poor (default record count increased)";
|
const char* OUT_OF_RECORDS_RETRY = u8"Textractor: out of search records, please retry if results are poor (default record count increased)";
|
||||||
@ -135,26 +136,27 @@ const char* HIJACK_ERROR = u8"Textractor: Hijack ERROR";
|
|||||||
const char* COULD_NOT_FIND = u8"Textractor: could not find text";
|
const char* COULD_NOT_FIND = u8"Textractor: could not find text";
|
||||||
const char* TRANSLATE_TO = u8"Translate to";
|
const char* TRANSLATE_TO = u8"Translate to";
|
||||||
const char* TRANSLATE_FROM = u8"Translate from";
|
const char* TRANSLATE_FROM = u8"Translate from";
|
||||||
|
const char* FILTER_GARBAGE = u8"Filter garbage characters";
|
||||||
const char* TRANSLATE_SELECTED_THREAD_ONLY = u8"Translate selected text thread only";
|
const char* TRANSLATE_SELECTED_THREAD_ONLY = u8"Translate selected text thread only";
|
||||||
const char* RATE_LIMIT_ALL_THREADS = u8"Rate limit all text threads";
|
const char* RATE_LIMIT_ALL_THREADS = u8"Use rate limiter";
|
||||||
const char* RATE_LIMIT_SELECTED_THREAD = u8"Rate limit selected text thread";
|
const char* RATE_LIMIT_SELECTED_THREAD = u8"Rate limit selected text thread";
|
||||||
const char* USE_TRANS_CACHE = u8"Use translation cache";
|
const char* USE_TRANS_CACHE = u8"Use translation cache";
|
||||||
const char* RATE_LIMIT_TOKEN_COUNT = u8"Rate limiter token count";
|
const char* MAX_TRANSLATIONS_IN_TIMESPAN = u8"Max translation requests in timespan";
|
||||||
const char* RATE_LIMIT_TOKEN_RESTORE_DELAY = u8"Rate limiter token restore delay (ms)";
|
const char* TIMESPAN = u8"Timespan (ms)";
|
||||||
|
const wchar_t* SENTENCE_TOO_LARGE_TO_TRANS = L"Sentence too large to translate";
|
||||||
const wchar_t* TOO_MANY_TRANS_REQUESTS = L"Rate limit exceeded: refuse to make more translation requests";
|
const wchar_t* TOO_MANY_TRANS_REQUESTS = L"Rate limit exceeded: refuse to make more translation requests";
|
||||||
const wchar_t* TRANSLATION_ERROR = L"Error while translating";
|
const wchar_t* TRANSLATION_ERROR = L"Error while translating";
|
||||||
const char* USE_PREV_SENTENCE_CONTEXT = u8"Use previous sentence as context";
|
const char* USE_PREV_SENTENCE_CONTEXT = u8"Use previous sentence as context";
|
||||||
const char* API_KEY = u8"API key";
|
const char* API_KEY = u8"API key";
|
||||||
const char* CHROME_LOCATION = "Google Chrome location";
|
const char* CHROME_LOCATION = u8"Google Chrome file location";
|
||||||
const char* START_DEVTOOLS = u8"Start DevTools";
|
const char* START_DEVTOOLS = u8"Start DevTools";
|
||||||
const char* STOP_DEVTOOLS = u8"Stop DevTools";
|
const char* STOP_DEVTOOLS = u8"Stop DevTools";
|
||||||
const char* HEADLESS_MODE = u8"Headless mode";
|
const char* HIDE_CHROME = u8"Hide Chrome window";
|
||||||
const char* DEVTOOLS_STATUS = u8"DevTools status";
|
const char* DEVTOOLS_STATUS = u8"DevTools status";
|
||||||
const char* AUTO_START = u8"Start automatically";
|
const char* AUTO_START = u8"Start automatically";
|
||||||
const wchar_t* ERROR_START_CHROME = L"failed to start Chrome or to connect to it";
|
const wchar_t* ERROR_START_CHROME = L"failed to start Chrome or to connect to it";
|
||||||
const char* EXTRA_WINDOW_INFO = u8R"(Right click to change settings
|
const char* EXTRA_WINDOW_INFO = u8R"(Right click to change settings
|
||||||
Click and drag on window edges to move, or the bottom right corner to resize)";
|
Click and drag on window edges to move, or the bottom right corner to resize)";
|
||||||
const char* SENTENCE_TOO_BIG = u8"Sentence too large to display";
|
|
||||||
const char* MAX_SENTENCE_SIZE = u8"Max sentence size";
|
const char* MAX_SENTENCE_SIZE = u8"Max sentence size";
|
||||||
const char* TOPMOST = u8"Always on top";
|
const char* TOPMOST = u8"Always on top";
|
||||||
const char* DICTIONARY = u8"Dictionary";
|
const char* DICTIONARY = u8"Dictionary";
|
||||||
@ -176,7 +178,13 @@ This file must be encoded in UTF-8.)";
|
|||||||
const char* SHOW_ORIGINAL = u8"Original text";
|
const char* SHOW_ORIGINAL = u8"Original text";
|
||||||
const char* SHOW_ORIGINAL_INFO = u8R"(Original text will not be shown
|
const char* SHOW_ORIGINAL_INFO = u8R"(Original text will not be shown
|
||||||
Only works if this extension is used directly after a translation extension)";
|
Only works if this extension is used directly after a translation extension)";
|
||||||
|
const char* ORIGINAL_AFTER_TRANSLATION = u8"Original text after translation";
|
||||||
const char* SIZE_LOCK = u8"Size lock";
|
const char* SIZE_LOCK = u8"Size lock";
|
||||||
|
const char* POSITION_LOCK = u8"Position lock";
|
||||||
|
const char* CENTERED_TEXT = u8"Centered text";
|
||||||
|
const char* AUTO_RESIZE_WINDOW_HEIGHT = u8"Auto resize window height";
|
||||||
|
const char* CLICK_THROUGH = u8"Click through\tAlt+X";
|
||||||
|
const char* HIDE_MOUSEOVER = u8"Hide while mouse on top";
|
||||||
const char* OPACITY = u8"Opacity";
|
const char* OPACITY = u8"Opacity";
|
||||||
const char* BG_COLOR = u8"Background color";
|
const char* BG_COLOR = u8"Background color";
|
||||||
const char* TEXT_COLOR = u8"Text color";
|
const char* TEXT_COLOR = u8"Text color";
|
||||||
@ -199,7 +207,7 @@ Modifications to global variables from ProcessSentence are not guaranteed to per
|
|||||||
|
|
||||||
Properties in sentenceInfo:
|
Properties in sentenceInfo:
|
||||||
"current select": 0 unless sentence is in the text thread currently selected by the user.
|
"current select": 0 unless sentence is in the text thread currently selected by the user.
|
||||||
"process id": process id that the sentence is coming from. 0 for console and clipboard.
|
"process id": process ID that the sentence is coming from. 0 for console and clipboard.
|
||||||
"text number": number of the current text thread. Counts up one by one as text threads are created. 0 for console, 1 for clipboard.
|
"text number": number of the current text thread. Counts up one by one as text threads are created. 0 for console, 1 for clipboard.
|
||||||
--]]
|
--]]
|
||||||
function ProcessSentence(sentence, sentenceInfo)
|
function ProcessSentence(sentence, sentenceInfo)
|
||||||
@ -217,8 +225,20 @@ All text in this file outside of a replacement command is ignored.
|
|||||||
A caret (^) acts as a wildcard that matches any other single character.
|
A caret (^) acts as a wildcard that matches any other single character.
|
||||||
Whitespace in original_text is ignored, but replacement_text can contain spaces, newlines, etc.
|
Whitespace in original_text is ignored, but replacement_text can contain spaces, newlines, etc.
|
||||||
This file must be encoded in Unicode (UTF-16 Little Endian).)";
|
This file must be encoded in Unicode (UTF-16 Little Endian).)";
|
||||||
|
const wchar_t* REGEX_REPLACER_INSTRUCTIONS = LR"(This file only does anything when the "Regex Replacer" extension is used.
|
||||||
|
Replacement commands must be formatted like this:
|
||||||
|
|REGEX|regular_expression|BECOMES|replacement_text|MODIFIER|modifiers|END|
|
||||||
|
replacement_text can reference capture groups with a $ followed by their number (e.g. $1 references first capture group).
|
||||||
|
modifiers can contain the following:
|
||||||
|
"g" the replacement is global.
|
||||||
|
"i" the replacement ignores the case.
|
||||||
|
If empty the replacement is only for the first match and case sensitive.
|
||||||
|
All text in this file outside of a replacement command is ignored.
|
||||||
|
This file must be encoded in Unicode (UTF-16 Little Endian).
|
||||||
|
Learn, build, & test Regular Expressions: https://regexr.com/)";
|
||||||
const char* THREAD_LINKER = u8"Thread Linker";
|
const char* THREAD_LINKER = u8"Thread Linker";
|
||||||
const char* LINK = u8"Link";
|
const char* LINK = u8"Link";
|
||||||
|
const char* UNLINK = u8"Unlink";
|
||||||
const char* THREAD_LINK_FROM = u8"Thread number to link from";
|
const char* THREAD_LINK_FROM = u8"Thread number to link from";
|
||||||
const char* THREAD_LINK_TO = u8"Thread number to link to";
|
const char* THREAD_LINK_TO = u8"Thread number to link to";
|
||||||
const char* HEXADECIMAL = u8"Hexadecimal";
|
const char* HEXADECIMAL = u8"Hexadecimal";
|
||||||
@ -285,7 +305,7 @@ Presiona supr en una extension seleccionada para removerla)";
|
|||||||
CLIPBOARD = L"Portapapeles";
|
CLIPBOARD = L"Portapapeles";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( hecho por mí: Artikash (correo: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( hecho por mí: Artikash (correo: akashmozumdar@gmail.com)
|
||||||
Página del proyecto: https://github.com/Artikash/Textractor
|
Página del proyecto: https://github.com/Artikash/Textractor
|
||||||
Video tutorial: https://tinyurl.com/textractor-tutorial
|
Video tutorial: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
No dudes en conectarme si tienes algún problema, petición de característica o preguntas relacionadas con Textractor
|
No dudes en conectarme si tienes algún problema, petición de característica o preguntas relacionadas con Textractor
|
||||||
Puedes hacerlo en la página del proyecto (en el apartado de "Issues") o por correo. Usa el inglés para comunicarte.
|
Puedes hacerlo en la página del proyecto (en el apartado de "Issues") o por correo. Usa el inglés para comunicarte.
|
||||||
Código fuente disponible bajo GPLv3 en la página del proyecto)";
|
Código fuente disponible bajo GPLv3 en la página del proyecto)";
|
||||||
@ -321,7 +341,7 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
#endif // SPANISH
|
#endif // SPANISH
|
||||||
|
|
||||||
#ifdef SIMPLIFIED_CHINESE
|
#ifdef SIMPLIFIED_CHINESE
|
||||||
NATIVE_LANGUAGE = "Chinese (simplified)";
|
NATIVE_LANGUAGE = "Chinese (Simplified)";
|
||||||
ATTACH = u8"附加到游戏";
|
ATTACH = u8"附加到游戏";
|
||||||
LAUNCH = u8"启动游戏";
|
LAUNCH = u8"启动游戏";
|
||||||
CONFIG = u8"配置游戏";
|
CONFIG = u8"配置游戏";
|
||||||
@ -334,28 +354,30 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
SETTINGS = u8"设置";
|
SETTINGS = u8"设置";
|
||||||
EXTENSIONS = u8"扩展";
|
EXTENSIONS = u8"扩展";
|
||||||
SELECT_PROCESS = u8"选择进程";
|
SELECT_PROCESS = u8"选择进程";
|
||||||
ATTACH_INFO = u8R"(如果没看见想要附加的进程,尝试使用管理员权限运行
|
ATTACH_INFO = u8R"(如果没看见想要附加的进程,请尝试使用管理员权限运行Textractor
|
||||||
也可以手动输入进程ID)";
|
也可以手动输入进程ID)";
|
||||||
SELECT_PROCESS_INFO = u8"如果手动输入游戏名,请使用绝对路径";
|
SELECT_PROCESS_INFO = u8"如果手动输入游戏名,请使用绝对路径";
|
||||||
FROM_COMPUTER = u8"从计算机中选择";
|
FROM_COMPUTER = u8"从资源管理器中选择";
|
||||||
PROCESSES = u8"进程 (*.exe)";
|
PROCESSES = u8"可执行文件 (*.exe)";
|
||||||
SAVE_SETTINGS = u8"保存设置";
|
SAVE_SETTINGS = u8"保存设置";
|
||||||
EXTEN_WINDOW_INSTRUCTIONS = u8R"(从计算机拖拽扩展 (.dll) 文件到这里来添加
|
EXTEN_WINDOW_INSTRUCTIONS = u8R"(在列表中单击右键以添加或移除扩展
|
||||||
(如果使用超级管理员运行,则无法工作)
|
在列表中拖拽扩展可以调整插件的执行顺序, 扩展会从上往下依次执行
|
||||||
在列表中拖拽来重新排序
|
排在下方的扩展只能接收到由排在上方的扩展处理过后的文本)";
|
||||||
使用 delete 键移除选中的扩展)";
|
|
||||||
ADD_EXTENSION = u8"添加扩展";
|
ADD_EXTENSION = u8"添加扩展";
|
||||||
|
REMOVE_EXTENSION = u8"移除扩展";
|
||||||
INVALID_EXTENSION = u8"%1 是一个无效扩展";
|
INVALID_EXTENSION = u8"%1 是一个无效扩展";
|
||||||
CONFIRM_EXTENSION_OVERWRITE = u8"此扩展的另一个版本已启用,是否删除并覆盖它?";
|
CONFIRM_EXTENSION_OVERWRITE = u8"此扩展的另一个版本已启用,是否删除并覆盖它?";
|
||||||
EXTENSION_WRITE_ERROR = u8"保存扩展失败";
|
EXTENSION_WRITE_ERROR = u8"保存扩展失败";
|
||||||
USE_JP_LOCALE = u8"模拟日本区域设置?";
|
USE_JP_LOCALE = u8"模拟日本区域设置?";
|
||||||
|
FAILED_TO_CREATE_CONFIG_FILE = u8"无法创建配置文件 \"%1\"";
|
||||||
HOOK_SEARCH_UNSTABLE_WARNING = u8"搜索钩子的功能是不稳定的! 可能会导致你的游戏崩溃! ";
|
HOOK_SEARCH_UNSTABLE_WARNING = u8"搜索钩子的功能是不稳定的! 可能会导致你的游戏崩溃! ";
|
||||||
|
HOOK_SEARCH_STARTING_VIEW_CONSOLE = u8"正在初始化钩子搜索 - 请查看控制台以获取更多提示";
|
||||||
SEARCH_CJK = u8"搜索中文/日文/韩文";
|
SEARCH_CJK = u8"搜索中文/日文/韩文";
|
||||||
SEARCH_PATTERN = u8"搜索匹配特征 (hex byte array)";
|
SEARCH_PATTERN = u8"搜索匹配特征 (hex byte array)";
|
||||||
SEARCH_DURATION = u8"搜索持续时间 (ms)";
|
SEARCH_DURATION = u8"搜索持续时间 (ms)";
|
||||||
SEARCH_MODULE = u8"搜索指定模块";
|
SEARCH_MODULE = u8"搜索指定模块";
|
||||||
PATTERN_OFFSET = u8"相对于特征地址的偏移值";
|
PATTERN_OFFSET = u8"相对于特征地址的偏移值";
|
||||||
MAX_HOOK_SEARCH_RECORDS = u8"搜索结果上限";
|
MAX_HOOK_SEARCH_RECORDS = u8"搜索结果达到上限";
|
||||||
MIN_ADDRESS = u8"起始地址 (hex)";
|
MIN_ADDRESS = u8"起始地址 (hex)";
|
||||||
MAX_ADDRESS = u8"结束地址 (hex)";
|
MAX_ADDRESS = u8"结束地址 (hex)";
|
||||||
STRING_OFFSET = u8"字符串偏移值 (hex)";
|
STRING_OFFSET = u8"字符串偏移值 (hex)";
|
||||||
@ -369,7 +391,7 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
DOUBLE_CLICK_TO_REMOVE_HOOK = u8"双击移除不需要的钩子";
|
DOUBLE_CLICK_TO_REMOVE_HOOK = u8"双击移除不需要的钩子";
|
||||||
FILTER_REPETITION = u8"过滤重复文本";
|
FILTER_REPETITION = u8"过滤重复文本";
|
||||||
AUTO_ATTACH = u8"自动附加";
|
AUTO_ATTACH = u8"自动附加";
|
||||||
ATTACH_SAVED_ONLY = u8"自动附加 (saved only)";
|
ATTACH_SAVED_ONLY = u8"自动附加 (仅限保存过配置的游戏)";
|
||||||
SHOW_SYSTEM_PROCESSES = u8"显示系统进程";
|
SHOW_SYSTEM_PROCESSES = u8"显示系统进程";
|
||||||
DEFAULT_CODEPAGE = u8"默认代码页";
|
DEFAULT_CODEPAGE = u8"默认代码页";
|
||||||
FLUSH_DELAY = u8"刷新延迟";
|
FLUSH_DELAY = u8"刷新延迟";
|
||||||
@ -380,9 +402,9 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
CLIPBOARD = L"剪贴板";
|
CLIPBOARD = L"剪贴板";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( 作者: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( 作者: Artikash (email: akashmozumdar@gmail.com)
|
||||||
项目主页: https://github.com/Artikash/Textractor
|
项目主页: https://github.com/Artikash/Textractor
|
||||||
教程视频: https://tinyurl.com/textractor-tutorial
|
教程视频: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
如果有任何关于 Textractor 的困难,功能请求或问题,请联系我
|
如果有任何关于 Textractor 的困难,功能请求或问题,请联系我
|
||||||
可以通过项目主页 (问题区) 或通过邮件来联系
|
可以通过项目主页 (Issues 页面) 或通过邮件来联系
|
||||||
项目主页提供基于 GPLv3 协议的源代码)";
|
项目主页提供基于 GPLv3 协议的源代码)";
|
||||||
UPDATE_AVAILABLE = L"有可用的更新: 请从 https://github.com/Artikash/Textractor/releases 下载";
|
UPDATE_AVAILABLE = L"有可用的更新: 请从 https://github.com/Artikash/Textractor/releases 下载";
|
||||||
ALREADY_INJECTED = L"Textractor: 已经注入";
|
ALREADY_INJECTED = L"Textractor: 已经注入";
|
||||||
@ -397,8 +419,9 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
REMOVING_HOOK = u8"Textractor: 移除钩子: %s";
|
REMOVING_HOOK = u8"Textractor: 移除钩子: %s";
|
||||||
HOOK_FAILED = u8"Textractor: 钩子注入失败";
|
HOOK_FAILED = u8"Textractor: 钩子注入失败";
|
||||||
TOO_MANY_HOOKS = u8"Textractor: 钩子数量已达上限: 无法注入";
|
TOO_MANY_HOOKS = u8"Textractor: 钩子数量已达上限: 无法注入";
|
||||||
STARTING_SEARCH = u8"Textractor: 开始搜索钩子";
|
HOOK_SEARCH_STARTING = u8"Textractor: 开始搜索钩子";
|
||||||
NOT_ENOUGH_TEXT = u8"Textractor: 没有足够的文本用来精确搜索";
|
HOOK_SEARCH_INITIALIZING = u8"Textractor: 初始化钩子搜索 (%f%%)";
|
||||||
|
NOT_ENOUGH_TEXT = u8"Textractor: 文本长度不足, 无法精确搜索";
|
||||||
HOOK_SEARCH_INITIALIZED = u8"Textractor: 搜索初始化完成, 创建了 %zd 个钩子";
|
HOOK_SEARCH_INITIALIZED = u8"Textractor: 搜索初始化完成, 创建了 %zd 个钩子";
|
||||||
MAKE_GAME_PROCESS_TEXT = u8"Textractor: 请点击游戏区域, 在接下来的 %d 秒内使游戏强制处理文本";
|
MAKE_GAME_PROCESS_TEXT = u8"Textractor: 请点击游戏区域, 在接下来的 %d 秒内使游戏强制处理文本";
|
||||||
HOOK_SEARCH_FINISHED = u8"Textractor: 钩子搜索完毕, 找到了 %d 条结果";
|
HOOK_SEARCH_FINISHED = u8"Textractor: 钩子搜索完毕, 找到了 %d 条结果";
|
||||||
@ -411,47 +434,65 @@ Clic y arrastra los bordes de la ventana para moverla, o en la esquina inferior
|
|||||||
HIJACK_ERROR = u8"Textractor: Hijack 错误";
|
HIJACK_ERROR = u8"Textractor: Hijack 错误";
|
||||||
COULD_NOT_FIND = u8"Textractor: 无法找到文本";
|
COULD_NOT_FIND = u8"Textractor: 无法找到文本";
|
||||||
TRANSLATE_TO = u8"翻译为";
|
TRANSLATE_TO = u8"翻译为";
|
||||||
|
TRANSLATE_FROM = u8"原文语言";
|
||||||
|
FILTER_GARBAGE = u8"过滤无意义字符";
|
||||||
TRANSLATE_SELECTED_THREAD_ONLY = u8"仅翻译当前选择的文本线程";
|
TRANSLATE_SELECTED_THREAD_ONLY = u8"仅翻译当前选择的文本线程";
|
||||||
RATE_LIMIT_ALL_THREADS = u8"限制所有文本线程的请求频率";
|
RATE_LIMIT_ALL_THREADS = u8"限制请求频率";
|
||||||
RATE_LIMIT_SELECTED_THREAD = u8"限制当前选择线程的请求频率";
|
RATE_LIMIT_SELECTED_THREAD = u8"限制选中线程的请求频率";
|
||||||
USE_TRANS_CACHE = u8"使用缓存的翻译结果";
|
USE_TRANS_CACHE = u8"使用缓存的翻译结果";
|
||||||
RATE_LIMIT_TOKEN_COUNT = u8"限流器令牌数";
|
MAX_TRANSLATIONS_IN_TIMESPAN = u8"单位时间内最大请求次数";
|
||||||
RATE_LIMIT_TOKEN_RESTORE_DELAY = u8"限流器令牌重置时间 (ms)";
|
TIMESPAN = u8"单位时间长度 (ms)";
|
||||||
TOO_MANY_TRANS_REQUESTS = L"太多翻译请求: 拒绝生成更多";
|
TOO_MANY_TRANS_REQUESTS = L"超出频率限制: 拒绝发出翻译请求";
|
||||||
TRANSLATION_ERROR = L"翻译时出错";
|
TRANSLATION_ERROR = L"翻译时出错";
|
||||||
USE_PREV_SENTENCE_CONTEXT = u8"使用之前的句子作为上下文";
|
USE_PREV_SENTENCE_CONTEXT = u8"使用之前的句子作为上下文";
|
||||||
API_KEY = u8"API key";
|
API_KEY = u8"API key";
|
||||||
EXTRA_WINDOW_INFO = u8R"(右键修改设置
|
CHROME_LOCATION = u8"Google Chrome 安装位置";
|
||||||
在窗口边缘点击并拖拽来移动,或在右下角点击并拖拽来调整大小)";
|
START_DEVTOOLS = u8"启动 DevTools";
|
||||||
SENTENCE_TOO_BIG = u8"文本过长无法显示";
|
STOP_DEVTOOLS = u8"停止 DevTools";
|
||||||
|
HIDE_CHROME = u8"隐藏 Chrome 窗口";
|
||||||
|
DEVTOOLS_STATUS = u8"DevTools 状态";
|
||||||
|
AUTO_START = u8"自动启动";
|
||||||
|
ERROR_START_CHROME = L"无法启动或连接到 Chrome";
|
||||||
|
EXTRA_WINDOW_INFO = u8R"(在此点击右键以修改设置
|
||||||
|
在窗口边缘点击并拖拽来移动, 在窗口右下角点击并拖拽来调整大小)";
|
||||||
MAX_SENTENCE_SIZE = u8"最大文本长度";
|
MAX_SENTENCE_SIZE = u8"最大文本长度";
|
||||||
TOPMOST = u8"窗口总是置顶";
|
TOPMOST = u8"窗口总是置顶";
|
||||||
DICTIONARY = u8"字典";
|
DICTIONARY = u8"字典";
|
||||||
SHOW_ORIGINAL = u8"显示原文";
|
SHOW_ORIGINAL = u8"显示原文";
|
||||||
SHOW_ORIGINAL_INFO = u8R"(将不显示原文
|
SHOW_ORIGINAL_INFO = u8R"(原文将被隐藏
|
||||||
仅当此扩展置于翻译扩展之后使用时才有效)";
|
仅当此扩展位于翻译扩展之后使用时才有效)";
|
||||||
OPACITY = u8"透明度";
|
ORIGINAL_AFTER_TRANSLATION = u8"把原文显示在翻译文本的后面";
|
||||||
SIZE_LOCK = u8"锁定窗口大小";
|
SIZE_LOCK = u8"锁定窗口大小";
|
||||||
|
POSITION_LOCK = u8"锁定窗口位置";
|
||||||
|
CENTERED_TEXT = u8"居中显示文本";
|
||||||
|
AUTO_RESIZE_WINDOW_HEIGHT = u8"自动改变窗口高度";
|
||||||
|
CLICK_THROUGH = u8"点击穿透\tAlt+X";
|
||||||
|
HIDE_MOUSEOVER = u8"鼠标经过时隐藏窗口";
|
||||||
|
OPACITY = u8"透明度";
|
||||||
BG_COLOR = u8"背景颜色";
|
BG_COLOR = u8"背景颜色";
|
||||||
TEXT_COLOR = u8"文本颜色";
|
TEXT_COLOR = u8"文本颜色";
|
||||||
TEXT_OUTLINE = u8"文字边框";
|
TEXT_OUTLINE = u8"文字描边";
|
||||||
OUTLINE_COLOR = u8"边框颜色";
|
OUTLINE_COLOR = u8"文字描边颜色";
|
||||||
OUTLINE_SIZE = u8"边框大小";
|
OUTLINE_SIZE = u8"文字描边大小";
|
||||||
OUTLINE_SIZE_INFO = u8"以像素为单位的大小(建议保持在字体大小的20%以下)";
|
OUTLINE_SIZE_INFO = u8"描边的单位为px (建议保持在字体大小的20%以下)";
|
||||||
FONT = u8"字体";
|
FONT = u8"字体";
|
||||||
LUA_INTRO = u8R"(--[[
|
LUA_INTRO = u8R"(--[[
|
||||||
ProcessSentence 是Textractor每接收到一句文本时都会调用的函数。
|
ProcessSentence 是Textractor每接收到一句文本时都会调用的函数.
|
||||||
参数 sentence: Textractor接收到的文本 (UTF-8)。
|
参数 sentence: Textractor接收到的文本 (UTF-8), 数据类型为 string.
|
||||||
参数 sentenceInfo: 用于保存文本相关信息的table。
|
参数 sentenceInfo: 用于保存文本相关信息的表, 数据类型为 tabel.
|
||||||
如果你返回一个字符串,文本将被转换为该字符串。
|
|
||||||
如果返回 nil,该文本将不会被修改。
|
如果你返回一个字符串,文本将被转换为该字符串.
|
||||||
此扩展使用了几个Lua解释器的副本用于保证线程安全。
|
如果返回 nil, 该文本将不会被修改.
|
||||||
在ProcessSentence函数中对全局变量的修改可能不会生效。
|
|
||||||
sentenceInfo有以下成员:
|
此扩展使用了几个Lua解释器的副本用于保证线程安全.
|
||||||
"current select": 除非文本属于用户当前选择的文本线程,否则为0。
|
在 ProcessSentence 函数中对全局变量的修改可能不会生效.
|
||||||
"process id": 这句文本所属的进程id. 0 表示控制台与剪贴板线程。
|
|
||||||
"text number": 当前选择的文本线程的id. 这是在创建文本线程时逐个递增的计数。 0 为控制台, 1 为剪贴板。
|
sentenceInfo 表有以下关键字索引:
|
||||||
|
current select: 除非文本属于用户当前选择的文本线程, 否则为0. 数据类型为 number.
|
||||||
|
process id: 这句文本所属的进程ID. 0 表示控制台与剪贴板线程. 数据类型为 number.
|
||||||
|
text number: 当前选择的文本线程的ID. 这是在创建文本线程时自动递增的计数器. 0 为控制台, 1 为剪贴板. 数据类型为 number.
|
||||||
--]]
|
--]]
|
||||||
|
|
||||||
function ProcessSentence(sentence, sentenceInfo)
|
function ProcessSentence(sentence, sentenceInfo)
|
||||||
--在此处添加你的代码...
|
--在此处添加你的代码...
|
||||||
end)";
|
end)";
|
||||||
@ -463,13 +504,28 @@ end)";
|
|||||||
REPLACER_INSTRUCTIONS = LR"(使用"Replace"扩展时会使用此文件
|
REPLACER_INSTRUCTIONS = LR"(使用"Replace"扩展时会使用此文件
|
||||||
替换指令必须遵循以下格式:
|
替换指令必须遵循以下格式:
|
||||||
|ORIG|原文|BECOMES|替代文本|END|
|
|ORIG|原文|BECOMES|替代文本|END|
|
||||||
替换指令之外的所有文本都会被忽略.
|
此文件中替换指令之外的所有文本都会被忽略.
|
||||||
|
符号 ^ 可以代替任意一个字符.
|
||||||
原文中的空白字符将被忽略,但是替代文本可以包含空白字符、换行符等.
|
原文中的空白字符将被忽略,但是替代文本可以包含空白字符、换行符等.
|
||||||
此文件必须使用Unicode字符集(UTF-16 little endian).)";
|
此文件必须使用 Unicode 编码 (UTF-16 little endian).)";
|
||||||
|
REGEX_REPLACER_INSTRUCTIONS = LR"(使用"Regex Replace"扩展时会使用此文件
|
||||||
|
替换指令必须遵循以下格式:
|
||||||
|
|REGEX|正则表达式|BECOMES|替代文本|MODIFIER|修饰符|END|
|
||||||
|
替代文本可以通过 $[编号] 来引用捕获分组 (例如: $1 引用第1个捕获分组).
|
||||||
|
修饰符可以包含以下选项:
|
||||||
|
g 全局替换.
|
||||||
|
i 忽略大小写差异.
|
||||||
|
如果替代文本为空, 那么只会替换第1个匹配的到结果, 而且强制启用大小写敏感.
|
||||||
|
此文件中替换指令之外的所有文本都会被忽略.
|
||||||
|
此文件必须使用 Unicode 编码 (UTF-16 little endian).
|
||||||
|
你可以通过下列链接来学习编写和在线测试正则表达式:
|
||||||
|
https://www.runoob.com/regexp/regexp-intro.html
|
||||||
|
https://regexr.com/)";
|
||||||
THREAD_LINKER = u8"线程链接器";
|
THREAD_LINKER = u8"线程链接器";
|
||||||
LINK = u8"链接";
|
LINK = u8"链接";
|
||||||
THREAD_LINK_FROM = u8"需要链接的线程id";
|
UNLINK = u8"断开";
|
||||||
THREAD_LINK_TO = u8"链接到的线程id";
|
THREAD_LINK_FROM = u8"需要链接的线程ID";
|
||||||
|
THREAD_LINK_TO = u8"链接到的线程ID";
|
||||||
HEXADECIMAL = u8"十六进制";
|
HEXADECIMAL = u8"十六进制";
|
||||||
#endif // SIMPLIFIED_CHINESE
|
#endif // SIMPLIFIED_CHINESE
|
||||||
|
|
||||||
@ -488,7 +544,7 @@ end)";
|
|||||||
EXTENSIONS = u8"Расширения";
|
EXTENSIONS = u8"Расширения";
|
||||||
SELECT_PROCESS = u8"Выберете процесс";
|
SELECT_PROCESS = u8"Выберете процесс";
|
||||||
ATTACH_INFO = u8R"(Если вы не видите процесс, к которому хотите присоединить, попробуйте запуск с правами администратора
|
ATTACH_INFO = u8R"(Если вы не видите процесс, к которому хотите присоединить, попробуйте запуск с правами администратора
|
||||||
Вы также можете ввести id процесса)";
|
Вы также можете ввести ID процесса)";
|
||||||
SELECT_PROCESS_INFO = u8"При ручном вводе имени файла процесса используйте абсолютный путь";
|
SELECT_PROCESS_INFO = u8"При ручном вводе имени файла процесса используйте абсолютный путь";
|
||||||
FROM_COMPUTER = u8"Найти в проводнике";
|
FROM_COMPUTER = u8"Найти в проводнике";
|
||||||
PROCESSES = u8"Процессы (*.exe)";
|
PROCESSES = u8"Процессы (*.exe)";
|
||||||
@ -553,13 +609,13 @@ padding: длина добавочных данных перед строкой
|
|||||||
CLIPBOARD = L"Буфер обмена";
|
CLIPBOARD = L"Буфер обмена";
|
||||||
ABOUT = L"Textractor " ARCH L" в." VERSION LR"( автор: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" в." VERSION LR"( автор: Artikash (email: akashmozumdar@gmail.com)
|
||||||
Домашняя страница: https://github.com/Artikash/Textractor
|
Домашняя страница: https://github.com/Artikash/Textractor
|
||||||
Обучающее видео: https://tinyurl.com/textractor-tutorial
|
Обучающее видео: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
Сообщайте о любых проблемах, желаемых для добавления функциях, или задавайте вопросы, касающиеся Textractor
|
Сообщайте о любых проблемах, желаемых для добавления функциях, или задавайте вопросы, касающиеся Textractor
|
||||||
Сделать это вы можете на домашней странице (секция issues) или через электронную почту
|
Сделать это вы можете на домашней странице (секция issues) или через электронную почту
|
||||||
Исходный код доступен по лицензии GPLv3 на домашней странице проекта
|
Исходный код доступен по лицензии GPLv3 на домашней странице проекта
|
||||||
Если эта программа вам понравилась, расскажите всем о ней :))";
|
Если эта программа вам понравилась, расскажите всем о ней :))";
|
||||||
CL_OPTIONS = LR"(использование: Textractor [-p{process id|"process name"}]...
|
CL_OPTIONS = LR"(использование: Textractor [-p{process ID|"process name"}]...
|
||||||
пример: Textractor -p4466 -p"My Game.exe" попробует присоединиться к процессу с id 4466 или с именем My Game.exe)";
|
пример: Textractor -p4466 -p"My Game.exe" попробует присоединиться к процессу с ID 4466 или с именем My Game.exe)";
|
||||||
UPDATE_AVAILABLE = L"Доступно обновление: загрузите его на https://github.com/Artikash/Textractor/releases";
|
UPDATE_AVAILABLE = L"Доступно обновление: загрузите его на https://github.com/Artikash/Textractor/releases";
|
||||||
ALREADY_INJECTED = L"Textractor: уже присоединен";
|
ALREADY_INJECTED = L"Textractor: уже присоединен";
|
||||||
NEED_32_BIT = L"Textractor: несоответствие архитектуры: попробуйте Textractor x86 вместо этого";
|
NEED_32_BIT = L"Textractor: несоответствие архитектуры: попробуйте Textractor x86 вместо этого";
|
||||||
@ -573,7 +629,7 @@ padding: длина добавочных данных перед строкой
|
|||||||
REMOVING_HOOK = u8"Textractor: удаление хука: %s";
|
REMOVING_HOOK = u8"Textractor: удаление хука: %s";
|
||||||
HOOK_FAILED = u8"Textractor: не удалось вставить хук";
|
HOOK_FAILED = u8"Textractor: не удалось вставить хук";
|
||||||
TOO_MANY_HOOKS = u8"Textractor: слишком много хуков: невозможно вставить";
|
TOO_MANY_HOOKS = u8"Textractor: слишком много хуков: невозможно вставить";
|
||||||
STARTING_SEARCH = u8"Textractor: начало поиска";
|
HOOK_SEARCH_STARTING = u8"Textractor: начало поиска";
|
||||||
NOT_ENOUGH_TEXT = u8"Textractor: не достаточно текста для точного поиска";
|
NOT_ENOUGH_TEXT = u8"Textractor: не достаточно текста для точного поиска";
|
||||||
HOOK_SEARCH_INITIALIZED = u8"Textractor: поиск инициализирован с %zd хуками";
|
HOOK_SEARCH_INITIALIZED = u8"Textractor: поиск инициализирован с %zd хуками";
|
||||||
MAKE_GAME_PROCESS_TEXT = u8"Textractor: покликайте в игре, чтобы вызвать смену текста в течение %d секунд";
|
MAKE_GAME_PROCESS_TEXT = u8"Textractor: покликайте в игре, чтобы вызвать смену текста в течение %d секунд";
|
||||||
@ -590,15 +646,14 @@ padding: длина добавочных данных перед строкой
|
|||||||
RATE_LIMIT_ALL_THREADS = u8"Ограничение скорости для всех текстовых потоков";
|
RATE_LIMIT_ALL_THREADS = u8"Ограничение скорости для всех текстовых потоков";
|
||||||
RATE_LIMIT_SELECTED_THREAD = u8"Ограничение скорости выделенного текстового потока";
|
RATE_LIMIT_SELECTED_THREAD = u8"Ограничение скорости выделенного текстового потока";
|
||||||
USE_TRANS_CACHE = u8"Использовать кеш перевода";
|
USE_TRANS_CACHE = u8"Использовать кеш перевода";
|
||||||
RATE_LIMIT_TOKEN_COUNT = u8"Ограничение частоты подсчёта токенов";
|
MAX_TRANSLATIONS_IN_TIMESPAN = u8"Ограничение частоты подсчёта токенов";
|
||||||
RATE_LIMIT_TOKEN_RESTORE_DELAY = u8"Ограничение частоты задержки восстановления токенов (мс)";
|
TIMESPAN = u8"Ограничение частоты задержки восстановления токенов (мс)";
|
||||||
TOO_MANY_TRANS_REQUESTS = L"Слишком много запросов для перевода: отклонено";
|
TOO_MANY_TRANS_REQUESTS = L"Слишком много запросов для перевода: отклонено";
|
||||||
TRANSLATION_ERROR = L"Ошибка при переводе";
|
TRANSLATION_ERROR = L"Ошибка при переводе";
|
||||||
USE_PREV_SENTENCE_CONTEXT = u8"Использовать предыдущее предложение как контекст";
|
USE_PREV_SENTENCE_CONTEXT = u8"Использовать предыдущее предложение как контекст";
|
||||||
API_KEY = u8"Ключ API";
|
API_KEY = u8"Ключ API";
|
||||||
EXTRA_WINDOW_INFO = u8R"(Правый клик для изменения настроек
|
EXTRA_WINDOW_INFO = u8R"(Правый клик для изменения настроек
|
||||||
Нажмите и перетащите за края - для перемещения, или за правый-нижний угол - для изменения размера)";
|
Нажмите и перетащите за края - для перемещения, или за правый-нижний угол - для изменения размера)";
|
||||||
SENTENCE_TOO_BIG = u8"Придложение слишком длинное для отображения";
|
|
||||||
MAX_SENTENCE_SIZE = u8"Максимальная длина предложения";
|
MAX_SENTENCE_SIZE = u8"Максимальная длина предложения";
|
||||||
TOPMOST = u8"Поверх всех окон";
|
TOPMOST = u8"Поверх всех окон";
|
||||||
DICTIONARY = u8"Словарь";
|
DICTIONARY = u8"Словарь";
|
||||||
@ -643,7 +698,7 @@ Param sentenceInfo: таблица различной информации о п
|
|||||||
|
|
||||||
Параметры в sentenceInfo:
|
Параметры в sentenceInfo:
|
||||||
"current select": равно 0, если предложение не находится в текстовой нити, выбранной в данный момент пользователем.
|
"current select": равно 0, если предложение не находится в текстовой нити, выбранной в данный момент пользователем.
|
||||||
"process id": id процесса, из которого предложение поступило. Равно 0, когда это консоль или буфер обмена.
|
"process id": ID процесса, из которого предложение поступило. Равно 0, когда это консоль или буфер обмена.
|
||||||
"text number": номер текущей текстовой нити. Растет один за другим по мере создания текстовых нитей. 0 для консоли, 1 для буфера обмена.
|
"text number": номер текущей текстовой нити. Растет один за другим по мере создания текстовых нитей. 0 для консоли, 1 для буфера обмена.
|
||||||
--]]
|
--]]
|
||||||
function ProcessSentence(sentence, sentenceInfo)
|
function ProcessSentence(sentence, sentenceInfo)
|
||||||
@ -677,7 +732,7 @@ end)";
|
|||||||
EXTENSIONS = u8"Ekstensi";
|
EXTENSIONS = u8"Ekstensi";
|
||||||
SELECT_PROCESS = u8"Pilih Proses";
|
SELECT_PROCESS = u8"Pilih Proses";
|
||||||
ATTACH_INFO = u8R"(Jika kamu tidak dapat melihat proses yang akan ditempelkan, coba menjalankan dengan mode administrator
|
ATTACH_INFO = u8R"(Jika kamu tidak dapat melihat proses yang akan ditempelkan, coba menjalankan dengan mode administrator
|
||||||
Kamu juga dapat mengetik process id game yang akan ditempel)";
|
Kamu juga dapat mengetik process ID game yang akan ditempel)";
|
||||||
FROM_COMPUTER = u8"Pilih dari komputer";
|
FROM_COMPUTER = u8"Pilih dari komputer";
|
||||||
PROCESSES = u8"Proses (*.exe)";
|
PROCESSES = u8"Proses (*.exe)";
|
||||||
SAVE_SETTINGS = u8"Simpan pengaturan";
|
SAVE_SETTINGS = u8"Simpan pengaturan";
|
||||||
@ -693,7 +748,7 @@ Tekan delete pada ekstensi yang dipilih untuk menghapus ekstensi)";
|
|||||||
CLIPBOARD = L"Papan clipboard";
|
CLIPBOARD = L"Papan clipboard";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( dibuat oleh saya: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( dibuat oleh saya: Artikash (email: akashmozumdar@gmail.com)
|
||||||
Halaman project: https://github.com/Artikash/Textractor
|
Halaman project: https://github.com/Artikash/Textractor
|
||||||
Video tutorial : https://tinyurl.com/textractor-tutorial
|
Video tutorial : https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
Tolong hubungi saya jika kamu memiliki masalah terkait masalah, permintaan fitur, atau pertanyaan terkait Textractor
|
Tolong hubungi saya jika kamu memiliki masalah terkait masalah, permintaan fitur, atau pertanyaan terkait Textractor
|
||||||
Kamu dapat melakukannya lewat halaman utama project (bagian issues) atau lewat email
|
Kamu dapat melakukannya lewat halaman utama project (bagian issues) atau lewat email
|
||||||
Source code tersedia dibawah lisensi GPLv3 di halaman utama project
|
Source code tersedia dibawah lisensi GPLv3 di halaman utama project
|
||||||
@ -810,13 +865,13 @@ Per rimuovere un estenzione, selezionala e premi rimuovi)";
|
|||||||
CLIPBOARD = L"Appunti";
|
CLIPBOARD = L"Appunti";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( creato da me: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( creato da me: Artikash (email: akashmozumdar@gmail.com)
|
||||||
Pagina principale del progetto: https://github.com/Artikash/Textractor
|
Pagina principale del progetto: https://github.com/Artikash/Textractor
|
||||||
Video tutorial: https://tinyurl.com/textractor-tutorial
|
Video tutorial: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
Contattatemi per ogni problema, richiesta futura, o domande legate a Textractor
|
Contattatemi per ogni problema, richiesta futura, o domande legate a Textractor
|
||||||
Puoi farlo attraverso la pagina principale del progetto (sezione issues) o via email
|
Puoi farlo attraverso la pagina principale del progetto (sezione issues) o via email
|
||||||
Il codice sorgente è disponibile sotto il GPLv3 nella pagina principale
|
Il codice sorgente è disponibile sotto il GPLv3 nella pagina principale
|
||||||
Al momento sono in cerca di un nuovo lavoro: contattatemi per email se conoscete qualcuno che ingaggia periti informatici statunitensi
|
Al momento sono in cerca di un nuovo lavoro: contattatemi per email se conoscete qualcuno che ingaggia periti informatici statunitensi
|
||||||
Se ti piace questo progetto, parlane con tutti per favore :))";
|
Se ti piace questo progetto, parlane con tutti per favore :))";
|
||||||
CL_OPTIONS = LR"(utilizzo: Textractor [-p{process id|"process name"}]...
|
CL_OPTIONS = LR"(utilizzo: Textractor [-p{process ID|"process name"}]...
|
||||||
esempio: Textractor -p4466 -p"My Game.exe" sta tentando di inniettare i processi con l'ID 4466 o con il nome My Game.exe)";
|
esempio: Textractor -p4466 -p"My Game.exe" sta tentando di inniettare i processi con l'ID 4466 o con il nome My Game.exe)";
|
||||||
UPDATE_AVAILABLE = L"Aggiornamento disponibile: scaricala da https://github.com/Artikash/Textractor/releases";
|
UPDATE_AVAILABLE = L"Aggiornamento disponibile: scaricala da https://github.com/Artikash/Textractor/releases";
|
||||||
ALREADY_INJECTED = L"Textractor: già inniettato";
|
ALREADY_INJECTED = L"Textractor: già inniettato";
|
||||||
@ -831,7 +886,7 @@ esempio: Textractor -p4466 -p"My Game.exe" sta tentando di inniettare i processi
|
|||||||
REMOVING_HOOK = u8"Textractor: rimuovi gancio: %s";
|
REMOVING_HOOK = u8"Textractor: rimuovi gancio: %s";
|
||||||
HOOK_FAILED = u8"Textractor: inserimento gancio non riuscito";
|
HOOK_FAILED = u8"Textractor: inserimento gancio non riuscito";
|
||||||
TOO_MANY_HOOKS = u8"Textractor: troppi ganci: impossibile inserirli";
|
TOO_MANY_HOOKS = u8"Textractor: troppi ganci: impossibile inserirli";
|
||||||
STARTING_SEARCH = u8"Textractor: avvia la ricerca";
|
HOOK_SEARCH_STARTING = u8"Textractor: avvia la ricerca";
|
||||||
NOT_ENOUGH_TEXT = u8"Textractor: testo insufficente per la ricerca accurata";
|
NOT_ENOUGH_TEXT = u8"Textractor: testo insufficente per la ricerca accurata";
|
||||||
HOOK_SEARCH_INITIALIZED = u8"Textractor: ricerca inizializzata con %zd ganci";
|
HOOK_SEARCH_INITIALIZED = u8"Textractor: ricerca inizializzata con %zd ganci";
|
||||||
MAKE_GAME_PROCESS_TEXT = u8"Textractor: clicca intorno al gioco per forzarlo nel testo del processo durante i prossimi %d secondi";
|
MAKE_GAME_PROCESS_TEXT = u8"Textractor: clicca intorno al gioco per forzarlo nel testo del processo durante i prossimi %d secondi";
|
||||||
@ -849,15 +904,14 @@ esempio: Textractor -p4466 -p"My Game.exe" sta tentando di inniettare i processi
|
|||||||
RATE_LIMIT_ALL_THREADS = u8"Rate limit tutti i thread del testo";
|
RATE_LIMIT_ALL_THREADS = u8"Rate limit tutti i thread del testo";
|
||||||
RATE_LIMIT_SELECTED_THREAD = u8"Rate limit thread del testo selezionato";
|
RATE_LIMIT_SELECTED_THREAD = u8"Rate limit thread del testo selezionato";
|
||||||
USE_TRANS_CACHE = u8"Utilizza la cache di traduzione";
|
USE_TRANS_CACHE = u8"Utilizza la cache di traduzione";
|
||||||
RATE_LIMIT_TOKEN_COUNT = u8"Numero di token del Rate Limit";
|
MAX_TRANSLATIONS_IN_TIMESPAN = u8"Numero di token del Rate Limit";
|
||||||
RATE_LIMIT_TOKEN_RESTORE_DELAY = u8"Token del rate limit ripristina il ritardo (ms)";
|
TIMESPAN = u8"Token del rate limit ripristina il ritardo (ms)";
|
||||||
TOO_MANY_TRANS_REQUESTS = L"Rate limit superato: rifiuta per fare altre richieste di traduzione";
|
TOO_MANY_TRANS_REQUESTS = L"Rate limit superato: rifiuta per fare altre richieste di traduzione";
|
||||||
TRANSLATION_ERROR = L"Errore durante la traduzione";
|
TRANSLATION_ERROR = L"Errore durante la traduzione";
|
||||||
USE_PREV_SENTENCE_CONTEXT = u8"Utilizza la precedente sentenza come contesto";
|
USE_PREV_SENTENCE_CONTEXT = u8"Utilizza la precedente sentenza come contesto";
|
||||||
API_KEY = u8"Chiave API";
|
API_KEY = u8"Chiave API";
|
||||||
EXTRA_WINDOW_INFO = u8R"(Tasto destro per cambiare le impostazioni
|
EXTRA_WINDOW_INFO = u8R"(Tasto destro per cambiare le impostazioni
|
||||||
Clicca e trascina i bordi della finestra per muoverla, oppure nell'angolo in basso a destra per ridimensionare)";
|
Clicca e trascina i bordi della finestra per muoverla, oppure nell'angolo in basso a destra per ridimensionare)";
|
||||||
SENTENCE_TOO_BIG = u8"Sentenza troppo grande da visualizzare";
|
|
||||||
MAX_SENTENCE_SIZE = u8"Dimensione massima sentenza";
|
MAX_SENTENCE_SIZE = u8"Dimensione massima sentenza";
|
||||||
TOPMOST = u8"Sempre in primo piano";
|
TOPMOST = u8"Sempre in primo piano";
|
||||||
DICTIONARY = u8"Dizionario";
|
DICTIONARY = u8"Dizionario";
|
||||||
@ -880,7 +934,13 @@ Questo file deve essere codificato in UTF-8.)";
|
|||||||
SHOW_ORIGINAL = u8"Testo originale";
|
SHOW_ORIGINAL = u8"Testo originale";
|
||||||
SHOW_ORIGINAL_INFO = u8R"(Testo originale non sarà mostrato
|
SHOW_ORIGINAL_INFO = u8R"(Testo originale non sarà mostrato
|
||||||
Funziona solo se questa estenzione è usata direttamente dopo un'estensione di traduzione)";
|
Funziona solo se questa estenzione è usata direttamente dopo un'estensione di traduzione)";
|
||||||
|
ORIGINAL_AFTER_TRANSLATION = u8"Mostra testo originale dopo traduzione";
|
||||||
SIZE_LOCK = u8"Lock delle dimensione";
|
SIZE_LOCK = u8"Lock delle dimensione";
|
||||||
|
POSITION_LOCK = u8"Lock delle posizione";
|
||||||
|
CENTERED_TEXT = u8"Testo centrato";
|
||||||
|
AUTO_RESIZE_WINDOW_HEIGHT = u8"Auto resize altezza finestra";
|
||||||
|
CLICK_THROUGH = u8"Clicca attraverso\tAlt+X";
|
||||||
|
HIDE_MOUSEOVER = u8"Nascondi testo mouseover";
|
||||||
OPACITY = u8"Opacità";
|
OPACITY = u8"Opacità";
|
||||||
BG_COLOR = u8"Colore dello sfondo";
|
BG_COLOR = u8"Colore dello sfondo";
|
||||||
TEXT_COLOR = u8"Colore del testo";
|
TEXT_COLOR = u8"Colore del testo";
|
||||||
@ -903,7 +963,7 @@ Modifiche alle variabili globali da ProcessSentence non sono garantite di persis
|
|||||||
|
|
||||||
Proprietà in sentenceInfo:
|
Proprietà in sentenceInfo:
|
||||||
"current select": 0 a meno che la sentenza è nel thread di testo attualmente scelto dall'utente.
|
"current select": 0 a meno che la sentenza è nel thread di testo attualmente scelto dall'utente.
|
||||||
"process id": id del processo che da cui proviene la sentenza. 0 per console e per appunti.
|
"process id": ID del processo che da cui proviene la sentenza. 0 per console e per appunti.
|
||||||
"text number": numero dell'attuale thread di testo. Conta uno ad uno quando i thread di testo sono creati. 0 per console, 1 per appunti.
|
"text number": numero dell'attuale thread di testo. Conta uno ad uno quando i thread di testo sono creati. 0 per console, 1 per appunti.
|
||||||
--]]
|
--]]
|
||||||
function ProcessSentence(sentence, sentenceInfo)
|
function ProcessSentence(sentence, sentenceInfo)
|
||||||
@ -920,6 +980,17 @@ I comandi di rimpiazzo devono essere formattati cosi:
|
|||||||
Tutto il testo in questo file all'infuori di un comando di rimpiazzo è ignorato.
|
Tutto il testo in questo file all'infuori di un comando di rimpiazzo è ignorato.
|
||||||
La spaziatura nel testo_originale è ignorato, ma testo_sostituito può contenere spaziature, ritorni a capo, ecc.
|
La spaziatura nel testo_originale è ignorato, ma testo_sostituito può contenere spaziature, ritorni a capo, ecc.
|
||||||
Questo file deve essere codificato in Unicode (UTF-16 Little Endian).)";
|
Questo file deve essere codificato in Unicode (UTF-16 Little Endian).)";
|
||||||
|
REGEX_REPLACER_INSTRUCTIONS = LR"(Questo file fa qualcosa solo quando l'estenzione "Regex Replacer" è utilizzata.
|
||||||
|
I comandi di sostituzione devono essere formattati cosi:
|
||||||
|
|REGEX|espressione_regolare|BECOMES|testo_sostituito|MODIFIER|modificatori|END|
|
||||||
|
Il parametro "MODIFIER" può contenere i seguenti modificatori:
|
||||||
|
"g" la sostituzione è globale.
|
||||||
|
"i" la sostituzione ignora maiuscole/minuscole.
|
||||||
|
Se il modificatore è vuoto, la sostituzione viene applicata alla sola prima corrispondenza
|
||||||
|
e fa distinzione tra maiuscole e minuscole.
|
||||||
|
Tutto il testo in questo file all'infuori di un comando di sostituzione è ignorato.
|
||||||
|
Questo file deve essere codificato in Unicode (UTF-16 Little Endian).
|
||||||
|
Apprendere, creare e testare Espressioni Regolari: https://regexr.com/)";
|
||||||
THREAD_LINKER = u8"Collegatore di thread";
|
THREAD_LINKER = u8"Collegatore di thread";
|
||||||
LINK = u8"Collegamento";
|
LINK = u8"Collegamento";
|
||||||
THREAD_LINK_FROM = u8"Numero di thread da cui collegarsi";
|
THREAD_LINK_FROM = u8"Numero di thread da cui collegarsi";
|
||||||
@ -959,7 +1030,7 @@ Pressione delete com uma extensão selecionada para removê-la.)";
|
|||||||
CLIPBOARD = L"Área de Transferência";
|
CLIPBOARD = L"Área de Transferência";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( Feito por mim: Artikash (e-mail: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( Feito por mim: Artikash (e-mail: akashmozumdar@gmail.com)
|
||||||
Homepage do Projeto: https://github.com/Artikash/Textractor
|
Homepage do Projeto: https://github.com/Artikash/Textractor
|
||||||
Vídeo Tutorial: https://tinyurl.com/textractor-tutorial
|
Vídeo Tutorial: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
Por favor, em caso de problemas, requisição de recurso e/ou funções e de dúvidas, entrar em contato comigo. Use o Inglês para se comunicar.
|
Por favor, em caso de problemas, requisição de recurso e/ou funções e de dúvidas, entrar em contato comigo. Use o Inglês para se comunicar.
|
||||||
Você pode fazê-lo por meio da Homepage do Projeto (na aba "Issues") ou via E-mail.
|
Você pode fazê-lo por meio da Homepage do Projeto (na aba "Issues") ou via E-mail.
|
||||||
O código-fonte se encontra disponível na Homepage do projeto sob a licença GPLv3.
|
O código-fonte se encontra disponível na Homepage do projeto sob a licença GPLv3.
|
||||||
@ -977,7 +1048,7 @@ Se você gostou desse projeto, divulgue a todos :))";
|
|||||||
REMOVING_HOOK = u8"Textractor: removendo hook: %s";
|
REMOVING_HOOK = u8"Textractor: removendo hook: %s";
|
||||||
HOOK_FAILED = u8"Textractor: falha na inserção do hook";
|
HOOK_FAILED = u8"Textractor: falha na inserção do hook";
|
||||||
TOO_MANY_HOOKS = u8"Textractor: há hooks de mais: não é possível inserir mais";
|
TOO_MANY_HOOKS = u8"Textractor: há hooks de mais: não é possível inserir mais";
|
||||||
STARTING_SEARCH = u8"Textractor: iniciando busca ";
|
HOOK_SEARCH_STARTING = u8"Textractor: iniciando busca ";
|
||||||
NOT_ENOUGH_TEXT = u8"Textractor: não há texto suficiente para uma buscar precisa";
|
NOT_ENOUGH_TEXT = u8"Textractor: não há texto suficiente para uma buscar precisa";
|
||||||
HOOK_SEARCH_INITIALIZED = u8"Textractor: busca inicializada com %zd hooks";
|
HOOK_SEARCH_INITIALIZED = u8"Textractor: busca inicializada com %zd hooks";
|
||||||
HOOK_SEARCH_FINISHED = u8"Textractor: busca por hooks finalizada, %d resultados encontrados";
|
HOOK_SEARCH_FINISHED = u8"Textractor: busca por hooks finalizada, %d resultados encontrados";
|
||||||
@ -1036,7 +1107,7 @@ Esse arquívo deve ser codifícado em (UTF-16 little endian).)";
|
|||||||
CLIPBOARD = L"ข้อมูลชั่วคราว";
|
CLIPBOARD = L"ข้อมูลชั่วคราว";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( ได้ถูกพัฒนาโดย: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( ได้ถูกพัฒนาโดย: Artikash (email: akashmozumdar@gmail.com)
|
||||||
หน้าเว็บไซต์หลัก: https://github.com/Artikash/Textractor
|
หน้าเว็บไซต์หลัก: https://github.com/Artikash/Textractor
|
||||||
วีดีโอสอนวิถีใช้: https://tinyurl.com/textractor-tutorial
|
วีดีโอสอนวิถีใช้: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
ถ้าหากพบเจอปัญหาสามารถติดต่อมาได้ รวมไปถึงการแนะนำฟังก์ชั้นที่อยากให้มี หรือ คำถามเกี่ยวกับโปรแกรม Textractor สามารถติดต่อ
|
ถ้าหากพบเจอปัญหาสามารถติดต่อมาได้ รวมไปถึงการแนะนำฟังก์ชั้นที่อยากให้มี หรือ คำถามเกี่ยวกับโปรแกรม Textractor สามารถติดต่อ
|
||||||
ผ่านหน้าเว็บไซต์หลักผ่านทางหน้า Issue หรือทางอีเมลล์
|
ผ่านหน้าเว็บไซต์หลักผ่านทางหน้า Issue หรือทางอีเมลล์
|
||||||
Source code สามารถหาได้จากส่วนของ GPLv3 ที่หน้าหลักของเว็บไซต์)";
|
Source code สามารถหาได้จากส่วนของ GPLv3 ที่หน้าหลักของเว็บไซต์)";
|
||||||
@ -1086,7 +1157,7 @@ Source code สามารถหาได้จากส่วนของ GPLv
|
|||||||
SETTINGS = u8"설정";
|
SETTINGS = u8"설정";
|
||||||
EXTENSIONS = u8"확장기능";
|
EXTENSIONS = u8"확장기능";
|
||||||
SELECT_PROCESS = u8"프로세스 선택";
|
SELECT_PROCESS = u8"프로세스 선택";
|
||||||
ATTACH_INFO = u8R"(부착하려는 게임이 보이지 않는다면, 관리자 권한으로 실행해보세요. 프로세스 id를 입력 할 수도 있습니다.)";
|
ATTACH_INFO = u8R"(부착하려는 게임이 보이지 않는다면, 관리자 권한으로 실행해보세요. 프로세스 ID를 입력 할 수도 있습니다.)";
|
||||||
SELECT_PROCESS_INFO = u8"직접 프로세스파일 이름을 타이핑한다면, 정확한 경로를 입력하세요";
|
SELECT_PROCESS_INFO = u8"직접 프로세스파일 이름을 타이핑한다면, 정확한 경로를 입력하세요";
|
||||||
FROM_COMPUTER = u8"컴퓨터로부터 선택";
|
FROM_COMPUTER = u8"컴퓨터로부터 선택";
|
||||||
PROCESSES = u8"프로세스 (*.exe)";
|
PROCESSES = u8"프로세스 (*.exe)";
|
||||||
@ -1230,14 +1301,14 @@ Pour supprimer une extension, sélectionnez-la et appuyez sur supprimer)";
|
|||||||
CLIPBOARD = L"Presse-papier";
|
CLIPBOARD = L"Presse-papier";
|
||||||
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( Fait par moi: Artikash (email: akashmozumdar@gmail.com)
|
ABOUT = L"Textractor " ARCH L" v" VERSION LR"( Fait par moi: Artikash (email: akashmozumdar@gmail.com)
|
||||||
Page d'acceuil du projet: https://github.com/Artikash/Textractor
|
Page d'acceuil du projet: https://github.com/Artikash/Textractor
|
||||||
Vidéo tuto: https://tinyurl.com/textractor-tutorial
|
Vidéo tuto: https://github.com/Artikash/Textractor/blob/master/docs/TUTORIAL.md
|
||||||
FAQ: https://github.com/Artikash/Textractor/wiki/FAQ
|
FAQ: https://github.com/Artikash/Textractor/wiki/FAQ
|
||||||
Veuillez me contacter pour tout problème, demande de fonctionnalité ou question concernant Textractor
|
Veuillez me contacter pour tout problème, demande de fonctionnalité ou question concernant Textractor
|
||||||
Vous pouvez le faire via la page d'accueil du projet (section problèmes) ou par e-mail
|
Vous pouvez le faire via la page d'accueil du projet (section problèmes) ou par e-mail
|
||||||
Code source disponible sous GPLv3 sur la page d'accueil du projet
|
Code source disponible sous GPLv3 sur la page d'accueil du projet
|
||||||
Si vous aimez ce projet, parlez-en à tout le monde :))";
|
Si vous aimez ce projet, parlez-en à tout le monde :))";
|
||||||
CL_OPTIONS = LR"(usage: Textractor [-p{process id|"process name"}]...
|
CL_OPTIONS = LR"(usage: Textractor [-p{process ID|"process name"}]...
|
||||||
example: Textractor -p4466 -p"My Game.exe" tries to inject processes with id 4466 or with name My Game.exe)";
|
example: Textractor -p4466 -p"My Game.exe" tries to inject processes with ID 4466 or with name My Game.exe)";
|
||||||
UPDATE_AVAILABLE = L"Mise à jour disponible: téléchargez-la depuis https://github.com/Artikash/Textractor/releases";
|
UPDATE_AVAILABLE = L"Mise à jour disponible: téléchargez-la depuis https://github.com/Artikash/Textractor/releases";
|
||||||
ALREADY_INJECTED = L"Textractor: déjà injecté";
|
ALREADY_INJECTED = L"Textractor: déjà injecté";
|
||||||
NEED_32_BIT = L"Textractor: incompatibilité d'architecture: seul Textractor x86 peut injecter ce processus";
|
NEED_32_BIT = L"Textractor: incompatibilité d'architecture: seul Textractor x86 peut injecter ce processus";
|
||||||
@ -1251,7 +1322,7 @@ example: Textractor -p4466 -p"My Game.exe" tries to inject processes with id 446
|
|||||||
REMOVING_HOOK = u8"Textractor: enlève le hook: %s";
|
REMOVING_HOOK = u8"Textractor: enlève le hook: %s";
|
||||||
HOOK_FAILED = u8"Textractor: n'a pas réussi à insérer un hook";
|
HOOK_FAILED = u8"Textractor: n'a pas réussi à insérer un hook";
|
||||||
TOO_MANY_HOOKS = u8"Textractor: trop de hooks: impossible d'insérer";
|
TOO_MANY_HOOKS = u8"Textractor: trop de hooks: impossible d'insérer";
|
||||||
STARTING_SEARCH = u8"Textractor: démarrage de la recherche";
|
HOOK_SEARCH_STARTING = u8"Textractor: démarrage de la recherche";
|
||||||
NOT_ENOUGH_TEXT = u8"Textractor: pas assez de texte pour effectuer une recherche précise";
|
NOT_ENOUGH_TEXT = u8"Textractor: pas assez de texte pour effectuer une recherche précise";
|
||||||
HOOK_SEARCH_INITIALIZED = u8"Textractor: la recherche a été initialisé avec %zd hooks";
|
HOOK_SEARCH_INITIALIZED = u8"Textractor: la recherche a été initialisé avec %zd hooks";
|
||||||
MAKE_GAME_PROCESS_TEXT = u8"Textractor: veuillez cliquer dans le jeu pour le forcer à traiter le texte lors de la prochaine %d seconds";
|
MAKE_GAME_PROCESS_TEXT = u8"Textractor: veuillez cliquer dans le jeu pour le forcer à traiter le texte lors de la prochaine %d seconds";
|
||||||
@ -1269,15 +1340,14 @@ example: Textractor -p4466 -p"My Game.exe" tries to inject processes with id 446
|
|||||||
RATE_LIMIT_ALL_THREADS = u8"Taux limite tout les threads de texte";
|
RATE_LIMIT_ALL_THREADS = u8"Taux limite tout les threads de texte";
|
||||||
RATE_LIMIT_SELECTED_THREAD = u8"Limite de débit du thread de texte sélectionné";
|
RATE_LIMIT_SELECTED_THREAD = u8"Limite de débit du thread de texte sélectionné";
|
||||||
USE_TRANS_CACHE = u8"Utiliser le cache de traduction";
|
USE_TRANS_CACHE = u8"Utiliser le cache de traduction";
|
||||||
RATE_LIMIT_TOKEN_COUNT = u8"Nombre de tokens du limiteur de débit";
|
MAX_TRANSLATIONS_IN_TIMESPAN = u8"Nombre de tokens du limiteur de débit";
|
||||||
RATE_LIMIT_TOKEN_RESTORE_DELAY = u8"Délai de restauration du token du limiteur de débit (ms)";
|
TIMESPAN = u8"Délai de restauration du token du limiteur de débit (ms)";
|
||||||
TOO_MANY_TRANS_REQUESTS = L"Limite de taux dépassée: refus de faire plus de demande de traduction";
|
TOO_MANY_TRANS_REQUESTS = L"Limite de taux dépassée: refus de faire plus de demande de traduction";
|
||||||
TRANSLATION_ERROR = L"Une erreur est survenue pendant la traduction";
|
TRANSLATION_ERROR = L"Une erreur est survenue pendant la traduction";
|
||||||
USE_PREV_SENTENCE_CONTEXT = u8"Utiliser la phrase précédente comme contexte";
|
USE_PREV_SENTENCE_CONTEXT = u8"Utiliser la phrase précédente comme contexte";
|
||||||
API_KEY = u8"API key";
|
API_KEY = u8"API key";
|
||||||
EXTRA_WINDOW_INFO = u8R"(Clic droit pour modifier les paramètres
|
EXTRA_WINDOW_INFO = u8R"(Clic droit pour modifier les paramètres
|
||||||
Cliquez et faites glisser sur les bords de la fenêtre pour vous déplacer ou dans le coin inférieur droit pour redimensionner)";
|
Cliquez et faites glisser sur les bords de la fenêtre pour vous déplacer ou dans le coin inférieur droit pour redimensionner)";
|
||||||
SENTENCE_TOO_BIG = u8"Phrase trop grande pour être affichée";
|
|
||||||
MAX_SENTENCE_SIZE = u8"Taille maximale de la phrase";
|
MAX_SENTENCE_SIZE = u8"Taille maximale de la phrase";
|
||||||
TOPMOST = u8"Toujours au dessus";
|
TOPMOST = u8"Toujours au dessus";
|
||||||
DICTIONARY = u8"Dictionnaire";
|
DICTIONARY = u8"Dictionnaire";
|
||||||
@ -1318,7 +1388,7 @@ Cette extension utilise plusieurs copies de l'interpréteur Lua pour la sécurit
|
|||||||
Les modifications apportées aux variables globales à partir de ProcessSentence ne sont pas garanties de persister.
|
Les modifications apportées aux variables globales à partir de ProcessSentence ne sont pas garanties de persister.
|
||||||
Properties in sentenceInfo:
|
Properties in sentenceInfo:
|
||||||
"current select": 0 unless sentence is in the text thread currently selected by the user.
|
"current select": 0 unless sentence is in the text thread currently selected by the user.
|
||||||
"process id": process id that the sentence is coming from. 0 for console and clipboard.
|
"process id": process ID that the sentence is coming from. 0 for console and clipboard.
|
||||||
"text number": number of the current text thread. Counts up one by one as text threads are created. 0 for console, 1 for clipboard.
|
"text number": number of the current text thread. Counts up one by one as text threads are created. 0 for console, 1 for clipboard.
|
||||||
--]]
|
--]]
|
||||||
function ProcessSentence(sentence, sentenceInfo)
|
function ProcessSentence(sentence, sentenceInfo)
|
||||||
|
@ -1,6 +1,12 @@
|
|||||||
include_directories(. util)
|
include_directories(. util minhook/include)
|
||||||
|
|
||||||
if(${CMAKE_SIZEOF_VOID_P} EQUAL 8)
|
if(${CMAKE_SIZEOF_VOID_P} EQUAL 8)
|
||||||
|
set(minhook_src
|
||||||
|
minhook/src/buffer.c
|
||||||
|
minhook/src/hook.c
|
||||||
|
minhook/src/trampoline.c
|
||||||
|
minhook/src/hde/hde64.c
|
||||||
|
)
|
||||||
set(texthook_src
|
set(texthook_src
|
||||||
main.cc
|
main.cc
|
||||||
texthook.cc
|
texthook.cc
|
||||||
@ -12,6 +18,12 @@ set(texthook_src
|
|||||||
util/util.cc
|
util/util.cc
|
||||||
)
|
)
|
||||||
else()
|
else()
|
||||||
|
set(minhook_src
|
||||||
|
minhook/src/buffer.c
|
||||||
|
minhook/src/hook.c
|
||||||
|
minhook/src/trampoline.c
|
||||||
|
minhook/src/hde/hde32.c
|
||||||
|
)
|
||||||
set(texthook_src
|
set(texthook_src
|
||||||
main.cc
|
main.cc
|
||||||
texthook.cc
|
texthook.cc
|
||||||
@ -27,6 +39,13 @@ set(texthook_src
|
|||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
add_library(minhook ${minhook_src})
|
||||||
add_library(texthook MODULE ${texthook_src})
|
add_library(texthook MODULE ${texthook_src})
|
||||||
target_precompile_headers(texthook REUSE_FROM pch)
|
# isn't there a better way to do this?
|
||||||
|
target_precompile_headers(texthook PRIVATE ../include/common.h)
|
||||||
|
if(NOT CMAKE_BUILD_TYPE MATCHES Debug)
|
||||||
|
target_compile_options(minhook PRIVATE /MT)
|
||||||
|
target_compile_options(texthook PRIVATE /MT)
|
||||||
|
target_link_options(texthook PRIVATE /NODEFAULTLIB:MSVCRT)
|
||||||
|
endif()
|
||||||
target_link_libraries(texthook minhook)
|
target_link_libraries(texthook minhook)
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
//#include <boost/foreach.hpp>
|
//#include <boost/foreach.hpp>
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
// jichi 375/2014: Add offset of pusha/pushad
|
// jichi 375/2014: Add offset of pusha/pushad
|
||||||
// http://faydoc.tripod.com/cpu/pushad.htm
|
// http://faydoc.tripod.com/cpu/pushad.htm
|
||||||
@ -4494,7 +4495,8 @@ bool InsertRUGP1Hook()
|
|||||||
*/
|
*/
|
||||||
bool InsertRUGP2Hook()
|
bool InsertRUGP2Hook()
|
||||||
{
|
{
|
||||||
if (!Util::CheckFile(L"vm60.dll") /*|| !SafeFillRange(L"vm60.dll", &low, &high)*/) {
|
auto module = GetModuleHandleW(L"vm60.dll");
|
||||||
|
if (!module /*|| !SafeFillRange(L"vm60.dll", &low, &high)*/) {
|
||||||
ConsoleOutput("vnreng:rUGP2: vm60.dll does not exist");
|
ConsoleOutput("vnreng:rUGP2: vm60.dll does not exist");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -4508,7 +4510,7 @@ bool InsertRUGP2Hook()
|
|||||||
0x89,0x75, 0x0c // 1001e527 8975 0c mov dword ptr ss:[ebp+0xc],esi
|
0x89,0x75, 0x0c // 1001e527 8975 0c mov dword ptr ss:[ebp+0xc],esi
|
||||||
};
|
};
|
||||||
enum { addr_offset = 0x1001e51d - 0x1001e515 };
|
enum { addr_offset = 0x1001e51d - 0x1001e515 };
|
||||||
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStopAddress);
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), (DWORD)module, Util::QueryModuleLimits(module).second);
|
||||||
//GROWL_DWORD(addr);
|
//GROWL_DWORD(addr);
|
||||||
if (!addr) {
|
if (!addr) {
|
||||||
ConsoleOutput("vnreng:rUGP2: pattern not found");
|
ConsoleOutput("vnreng:rUGP2: pattern not found");
|
||||||
@ -4631,12 +4633,7 @@ static void InsertAliceHook2(DWORD addr)
|
|||||||
// jichi 5/13/2015: Looking for function entries in StoatSpriteEngine.dll
|
// jichi 5/13/2015: Looking for function entries in StoatSpriteEngine.dll
|
||||||
bool InsertAliceHook()
|
bool InsertAliceHook()
|
||||||
{
|
{
|
||||||
DWORD addr;
|
if (auto addr = Util::FindFunction("SP_TextDraw")) {
|
||||||
if (addr = (DWORD)GetProcAddress(GetModuleHandleW(L"SACT2.dll"), "SP_TextDraw")) {
|
|
||||||
InsertAliceHook1(addr);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (addr = (DWORD)GetProcAddress(GetModuleHandleW(L"SACTDX.dll"), "SP_TextDraw")) {
|
|
||||||
InsertAliceHook1(addr);
|
InsertAliceHook1(addr);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -4644,7 +4641,7 @@ bool InsertAliceHook()
|
|||||||
// InsertAliceHook2(addr);
|
// InsertAliceHook2(addr);
|
||||||
// return true;
|
// return true;
|
||||||
//}
|
//}
|
||||||
if (addr = (DWORD)GetProcAddress(GetModuleHandleW(L"StoatSpriteEngine.dll"), "SP_SetTextSprite")) { // Artikash 6/27/2018 not sure if this works
|
if (auto addr = Util::FindFunction("SP_SetTextSprite")) { // Artikash 6/27/2018 not sure if this works
|
||||||
InsertAliceHook2(addr);
|
InsertAliceHook2(addr);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -5734,21 +5731,21 @@ void SpecialHookShina2(DWORD esp_base, HookParam *, BYTE, DWORD *data, DWORD *sp
|
|||||||
// Used to merge correct text thread.
|
// Used to merge correct text thread.
|
||||||
// 1. Only keep threads with 0 and -1 split
|
// 1. Only keep threads with 0 and -1 split
|
||||||
// 2. Skip the thread withb 0 split and with minimum return address
|
// 2. Skip the thread withb 0 split and with minimum return address
|
||||||
void SpecialHookShina1(DWORD esp_base, HookParam *hp, BYTE, DWORD *data, DWORD *split, DWORD *len)
|
//void SpecialHookShina1(DWORD esp_base, HookParam *hp, BYTE, DWORD *data, DWORD *split, DWORD *len)
|
||||||
{
|
//{
|
||||||
static DWORD min_retaddr = -1;
|
// static DWORD min_retaddr = -1;
|
||||||
DWORD s = *(DWORD *)(esp_base + hp->split);
|
// DWORD s = *(DWORD *)(esp_base + hp->split);
|
||||||
if (s == 0 || (s & 0xffff) == 0xffff) { // only keep threads with 0 and -1 split
|
// if (s == 0 || (s & 0xffff) == 0xffff) { // only keep threads with 0 and -1 split
|
||||||
if (s == 0 && retof(esp_base) <= min_retaddr) {
|
// if (s == 0 && retof(esp_base) <= min_retaddr) {
|
||||||
min_retaddr = retof(esp_base);
|
// min_retaddr = retof(esp_base);
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
*split = FIXED_SPLIT_VALUE;
|
// *split = FIXED_SPLIT_VALUE;
|
||||||
// Follow the same logic as the hook.
|
// // Follow the same logic as the hook.
|
||||||
*data = *(DWORD *)*data; // DATA_INDIRECT
|
// *data = *(DWORD *)*data; // DATA_INDIRECT
|
||||||
*len = LeadByteTable[*data & 0xff];
|
// *len = LeadByteTable[*data & 0xff];
|
||||||
}
|
// }
|
||||||
}
|
//}
|
||||||
|
|
||||||
// jichi 8/27/2013
|
// jichi 8/27/2013
|
||||||
// Return ShinaRio version number
|
// Return ShinaRio version number
|
||||||
@ -5807,7 +5804,7 @@ bool InsertShinaHook()
|
|||||||
trigger_fun = [](LPVOID funcAddr, DWORD, DWORD stack)
|
trigger_fun = [](LPVOID funcAddr, DWORD, DWORD stack)
|
||||||
{
|
{
|
||||||
bool ret = false;
|
bool ret = false;
|
||||||
if (funcAddr != GetGlyphOutlineA) return false;
|
if (funcAddr != GetGlyphOutlineA && funcAddr != GetTextExtentPoint32A) return false;
|
||||||
for (int i = 0; i < 100; ++i)
|
for (int i = 0; i < 100; ++i)
|
||||||
{
|
{
|
||||||
// Address of text is somewhere on stack in call to func. Search for it.
|
// Address of text is somewhere on stack in call to func. Search for it.
|
||||||
@ -5821,7 +5818,7 @@ bool InsertShinaHook()
|
|||||||
hp.type = DIRECT_READ;
|
hp.type = DIRECT_READ;
|
||||||
hp.address = addr;
|
hp.address = addr;
|
||||||
ConsoleOutput("Textractor: triggered: adding dynamic reader");
|
ConsoleOutput("Textractor: triggered: adding dynamic reader");
|
||||||
NewHook(hp, "READ");
|
NewHook(hp, "ShinaRio READ");
|
||||||
ret = true;
|
ret = true;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -5829,9 +5826,8 @@ bool InsertShinaHook()
|
|||||||
return ret;
|
return ret;
|
||||||
};
|
};
|
||||||
ConsoleOutput("Textractor: ShinaRio 2.50+: adding trigger");
|
ConsoleOutput("Textractor: ShinaRio 2.50+: adding trigger");
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
else if (ver >= 48) { // v2.48, v2.49
|
if (ver >= 48) { // v2.48, v2.49
|
||||||
HookParam hp = {};
|
HookParam hp = {};
|
||||||
hp.address = (DWORD)::GetTextExtentPoint32A;
|
hp.address = (DWORD)::GetTextExtentPoint32A;
|
||||||
hp.text_fun = SpecialHookShina2;
|
hp.text_fun = SpecialHookShina2;
|
||||||
@ -5988,7 +5984,7 @@ bool InsertWaffleHook()
|
|||||||
{
|
{
|
||||||
bool found = false;
|
bool found = false;
|
||||||
for (DWORD i = processStartAddress + 0x1000; i < processStopAddress - 4; i++)
|
for (DWORD i = processStartAddress + 0x1000; i < processStopAddress - 4; i++)
|
||||||
if (*(DWORD *)i == 0xac68) {
|
if (*(DWORD *)i == 0xac68 && *(BYTE*)(i + 4) == 0) {
|
||||||
HookParam hp = {};
|
HookParam hp = {};
|
||||||
hp.address = i;
|
hp.address = i;
|
||||||
hp.length_offset = 1;
|
hp.length_offset = 1;
|
||||||
@ -6011,9 +6007,9 @@ bool InsertWaffleHook()
|
|||||||
0x50, //50 push eax
|
0x50, //50 push eax
|
||||||
0x8b, 0xce, //8BCE mov ecx,esi
|
0x8b, 0xce, //8BCE mov ecx,esi
|
||||||
0xc6, 0x45, 0xfc, XX, //C645 FC 01 move byte ptr ss:[ebp-4],?
|
0xc6, 0x45, 0xfc, XX, //C645 FC 01 move byte ptr ss:[ebp-4],?
|
||||||
0x89, 0x75, 0xd4, //8975 D4 move dword ptr ss:[ebp-0x2c],esi
|
0x89, 0x75, XX, //8975 D4 move dword ptr ss:[ebp-0x2c],esi
|
||||||
0xe8, XX4, //E8 ?? call ??
|
0xe8, XX4, //E8 ?? call ??
|
||||||
0x8d, 0x45, 0xdc //8D45 DC lea eax,dword ptr ss:[ebp-0x24]
|
0x8d, 0x45, XX //8D45 DC lea eax,dword ptr ss:[ebp-0x24]
|
||||||
};
|
};
|
||||||
if (DWORD addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStopAddress))
|
if (DWORD addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStopAddress))
|
||||||
{
|
{
|
||||||
@ -6356,9 +6352,267 @@ static bool InsertYuris2Hook()
|
|||||||
NewHook(hp, "YU-RIS2");
|
NewHook(hp, "YU-RIS2");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
static bool Yuris3Filter(LPVOID data, DWORD* size, HookParam*, BYTE)
|
||||||
|
{
|
||||||
|
static wchar_t prev_text;
|
||||||
|
wchar_t* pText = reinterpret_cast<wchar_t*>(data);
|
||||||
|
|
||||||
|
if (prev_text == *pText)
|
||||||
|
{
|
||||||
|
prev_text = '\0';
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
prev_text = *pText;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool InsertYuris3Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
HookParam hp = {};
|
||||||
|
wcsncpy_s(hp.module, L"kernel32.dll", MAX_MODULE_SIZE - 1);
|
||||||
|
strncpy_s(hp.function, "MultiByteToWideChar", MAX_MODULE_SIZE - 1);
|
||||||
|
hp.address = 6;
|
||||||
|
hp.offset = 0xC; //arg3
|
||||||
|
hp.index = 0;
|
||||||
|
hp.split = 0x1C;
|
||||||
|
hp.split_index = 0;
|
||||||
|
hp.filter_fun = Yuris3Filter;
|
||||||
|
hp.type = USING_STRING | USING_SPLIT | MODULE_OFFSET | FUNCTION_OFFSET;
|
||||||
|
hp.length_offset = 0x10 / (short)sizeof(void*); // arg4/arg_sz
|
||||||
|
ConsoleOutput("vnreng: INSERT YU-RIS 3");
|
||||||
|
NewHook(hp, "YU-RIS3");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertYuris4Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
/*
|
||||||
|
* Sample games:
|
||||||
|
* https://vndb.org/v6540
|
||||||
|
*/
|
||||||
|
bool found = false;
|
||||||
|
const BYTE pattern[] = {
|
||||||
|
0x52, // 52 push edx
|
||||||
|
0x68, 0x00, 0x42, 0x5C, 0x00, // 68 00425C00 push euphoria.exe+1C4200
|
||||||
|
0xFF, 0x15, 0x90, 0x44, 0x7E, 0x00, // FF 15 90447E00 call dword ptr [euphoria.exe+3E4490]
|
||||||
|
0x83, 0xC4, 0x0C, // 83 C4 0C add esp,0C
|
||||||
|
0xEB, 0x5F, // EB 5F jmp euphoria.exe+4F4C5
|
||||||
|
0xFF, 0x35, 0xA4, 0x19, 0x66, 0x00, // FF 35 A4196600 push [euphoria.exe+2619A4]
|
||||||
|
0x52 // 52 push edx
|
||||||
|
};
|
||||||
|
enum { addr_offset = 12 }; // distance to the beginning of the function, which is 0x83, 0xC4, 0x0C (add esp,0C)
|
||||||
|
|
||||||
|
for (auto addr : Util::SearchMemory(pattern, sizeof(pattern), PAGE_EXECUTE, processStartAddress, processStopAddress))
|
||||||
|
{
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr + addr_offset;
|
||||||
|
hp.offset = pusha_edx_off - 4;
|
||||||
|
hp.type = USING_STRING;
|
||||||
|
ConsoleOutput("Textractor: INSERT YU-RIS 4");
|
||||||
|
NewHook(hp, "YU-RIS4");
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
if (!found) ConsoleOutput("Textractor:YU-RIS 4: pattern not found");
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertYuris5Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
/*
|
||||||
|
* Sample games:
|
||||||
|
* https://vndb.org/v4037
|
||||||
|
*/
|
||||||
|
const BYTE bytes[] = {
|
||||||
|
0x33, 0xD2, // xor edx,edx
|
||||||
|
0x88, 0x14, 0x0F, // mov [edi+ecx],dl
|
||||||
|
0xA1, XX4, // mov eax,[exe+2DE630]
|
||||||
|
0x8B, 0x78, 0x3C, // mov edi,[eax+3C]
|
||||||
|
0x8B, 0x58, 0x5C, // mov ebx,[eax+5C]
|
||||||
|
0x88, 0x14, 0x3B // mov [ebx+edi],dl
|
||||||
|
};
|
||||||
|
|
||||||
|
enum { addr_offset = 0 }; // distance to the beginning of the function, which is 0x55 (push ebp)
|
||||||
|
ULONG range = min(processStopAddress - processStartAddress, MAX_REL_ADDR);
|
||||||
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStartAddress + range);
|
||||||
|
if (!addr)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr + addr_offset;
|
||||||
|
hp.offset = pusha_ecx_off - 4;
|
||||||
|
hp.type = USING_STRING | NO_CONTEXT;
|
||||||
|
|
||||||
|
ConsoleOutput("Textractor: INSERT YU-RIS 5");
|
||||||
|
NewHook(hp, "YU-RIS5");
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool Yuris6Filter(LPVOID data, DWORD* size, HookParam*, BYTE)
|
||||||
|
{
|
||||||
|
auto text = reinterpret_cast<LPSTR>(data);
|
||||||
|
auto len = reinterpret_cast<size_t*>(size);
|
||||||
|
static std::string prevText;
|
||||||
|
|
||||||
|
if (prevText.length() == *len && prevText.find(text, 0, *len) != std::string::npos) // Check if the string is present in the previous one
|
||||||
|
return false;
|
||||||
|
prevText.assign(text, *len);
|
||||||
|
|
||||||
|
// ruby <手水舎/ちょうずや>
|
||||||
|
if (cpp_strnstr(text, "\x81\x83", *len)) { // \x81\x83 -> '<'
|
||||||
|
StringFilterBetween(text, len, "\x81\x5E", 2, "\x81\x84", 2); // \x81\x5E -> '/' , \x81\x84 -> '>'
|
||||||
|
StringFilter(text, len, "\x81\x83", 2); // \x81\x83 -> '<'
|
||||||
|
}
|
||||||
|
// ruby ≪美桜/姉さん≫
|
||||||
|
else if (cpp_strnstr(text, "\x81\xE1", *len)) { // \x81\xE1 -> '≪'
|
||||||
|
StringFilterBetween(text, len, "\x81\x5E", 2, "\x81\xE2", 2); // \x81\x5E -> '/' , \x81\xE2 -> '≫'
|
||||||
|
StringFilter(text, len, "\x81\xE1", 2); // \x81\xE1 -> '≪'
|
||||||
|
}
|
||||||
|
|
||||||
|
CharReplacer(text, len, '=', '-');
|
||||||
|
StringCharReplacer(text, len, "\xEF\xF0", 2, ' ');
|
||||||
|
StringFilter(text, len, "\xEF\xF2", 2);
|
||||||
|
StringFilter(text, len, "\xEF\xF5", 2);
|
||||||
|
StringFilter(text, len, "\x81\x98", 2);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertYuris6Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
/*
|
||||||
|
* Sample games:
|
||||||
|
* https://vndb.org/v40058
|
||||||
|
* https://vndb.org/v42883
|
||||||
|
* https://vndb.org/v44092
|
||||||
|
* https://vndb.org/v21171
|
||||||
|
* https://vndb.org/r46910
|
||||||
|
*/
|
||||||
|
const BYTE bytes[] = {
|
||||||
|
0xE9, XX4, // jmp oshitona01.exe+1B629
|
||||||
|
0xBF, XX4, // mov edi,oshitona01.exe+24EEA0
|
||||||
|
0x8A, 0x17, // mov dl,[edi]
|
||||||
|
0x47, // inc edi
|
||||||
|
0x88, 0x16, // mov [esi],dl
|
||||||
|
0x46, // inc esi
|
||||||
|
0x84, 0xD2 // test dl,dl
|
||||||
|
};
|
||||||
|
|
||||||
|
ULONG range = min(processStopAddress - processStartAddress, MAX_REL_ADDR);
|
||||||
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStartAddress + range);
|
||||||
|
if (!addr)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr;
|
||||||
|
hp.offset = pusha_eax_off - 4;
|
||||||
|
hp.index = 0x38;
|
||||||
|
hp.codepage = 65001;
|
||||||
|
hp.filter_fun = Yuris6Filter;
|
||||||
|
hp.type = USING_STRING | NO_CONTEXT | DATA_INDIRECT;
|
||||||
|
|
||||||
|
ConsoleOutput("Textractor: INSERT YU-RIS 6");
|
||||||
|
NewHook(hp, "YU-RIS6");
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertYuris7Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
/*
|
||||||
|
* Sample games:
|
||||||
|
* https://vndb.org/v45381
|
||||||
|
* https://vndb.org/v47458
|
||||||
|
* https://vndb.org/v21144
|
||||||
|
* https://vndb.org/v18681
|
||||||
|
*/
|
||||||
|
const BYTE bytes[] = {
|
||||||
|
0x03, 0xC1, // add eax,ecx
|
||||||
|
0x99, // cdq
|
||||||
|
0xE9, XX4 // jmp nekonin_spin.exe+55753 << hook here
|
||||||
|
};
|
||||||
|
enum { addr_offset = sizeof(bytes) - 5 };
|
||||||
|
|
||||||
|
ULONG range = min(processStopAddress - processStartAddress, MAX_REL_ADDR);
|
||||||
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStartAddress + range);
|
||||||
|
if (!addr)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr + addr_offset;
|
||||||
|
hp.offset = pusha_eax_off - 4;
|
||||||
|
hp.index = 0;
|
||||||
|
hp.length_offset = 1; // only 1 character
|
||||||
|
hp.text_fun = [](DWORD esp_base, HookParam*, BYTE, DWORD*, DWORD*, DWORD* len)
|
||||||
|
{
|
||||||
|
*len = (retof(esp_base) == 0) ? 2 : 0;
|
||||||
|
};
|
||||||
|
hp.type = BIG_ENDIAN;
|
||||||
|
ConsoleOutput("vnreng: INSERT YU-RIS7");
|
||||||
|
NewHook(hp, "YU-RIS7");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertYuris8Hook()
|
||||||
|
{
|
||||||
|
//by Blu3train
|
||||||
|
/*
|
||||||
|
* Sample games:
|
||||||
|
* https://vndb.org/v47458
|
||||||
|
* https://vndb.org/v45381
|
||||||
|
*/
|
||||||
|
const BYTE bytes[] = {
|
||||||
|
0x57, // push edi << hook here
|
||||||
|
0x56, // push esi
|
||||||
|
0x55, // push ebp
|
||||||
|
0x53, // push ebx
|
||||||
|
0x83, 0xEC, 0x10, // sub esp,10
|
||||||
|
0x8B, 0x5C, 0x24, 0x24, // mov ebx,[esp+24]
|
||||||
|
0x8B, 0x15, XX4, // mov edx,[hajiron.exe+47243C]
|
||||||
|
0x8B, 0x0C, 0x9A, // mov ecx,[edx+ebx*4]
|
||||||
|
0xC6, 0x41, 0x01, 0x03, // mov byte ptr [ecx+01],03
|
||||||
|
0x8B, 0xC3, // mov eax,ebx
|
||||||
|
0xE8,XX4 // call hajiron.exe+54EA4
|
||||||
|
};
|
||||||
|
|
||||||
|
ULONG range = min(processStopAddress - processStartAddress, MAX_REL_ADDR);
|
||||||
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStartAddress + range);
|
||||||
|
if (!addr)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr;
|
||||||
|
hp.offset = pusha_edx_off - 4;
|
||||||
|
hp.index = 0;
|
||||||
|
hp.text_fun = [](DWORD esp_base, HookParam*, BYTE, DWORD*, DWORD*, DWORD* len)
|
||||||
|
{
|
||||||
|
DWORD textLen = regof(eax, esp_base);
|
||||||
|
if (textLen > 2)
|
||||||
|
return;
|
||||||
|
*len = (regof(edi, esp_base) >= 0xFA && regof(edi, esp_base) <= 0xFE || regof(edi, esp_base) >= 0x1A0 && regof(edi, esp_base) <= 0x1C2) ? textLen : 0;
|
||||||
|
};
|
||||||
|
hp.type = USING_STRING;
|
||||||
|
ConsoleOutput("vnreng: INSERT YU-RIS8");
|
||||||
|
NewHook(hp, "YU-RIS8");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
bool InsertYurisHook()
|
bool InsertYurisHook()
|
||||||
{ return InsertYuris1Hook() || InsertYuris2Hook(); }
|
{
|
||||||
|
bool ok = InsertYuris1Hook();
|
||||||
|
ok = InsertYuris2Hook() || ok;
|
||||||
|
ok = InsertYuris3Hook() || ok;
|
||||||
|
ok = InsertYuris4Hook() || ok;
|
||||||
|
ok = InsertYuris5Hook() || ok;
|
||||||
|
ok = InsertYuris6Hook() || ok;
|
||||||
|
ok = InsertYuris7Hook() || ok;
|
||||||
|
ok = InsertYuris8Hook() || ok;
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
bool InsertCotophaHook1()
|
bool InsertCotophaHook1()
|
||||||
{
|
{
|
||||||
@ -10779,8 +11033,59 @@ bool InsertArtemis2Hook()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool InsertArtemis3Hook()
|
||||||
|
{
|
||||||
|
const BYTE bytes[] = {
|
||||||
|
0x55, // 005FD780 | 55 | push ebp |
|
||||||
|
0x8B, 0xEC, // 005FD781 | 8BEC | mov ebp,esp |
|
||||||
|
0x83, 0xE4, 0xF8, // 005FD783 | 83E4 F8 | and esp,FFFFFFF8 |
|
||||||
|
0x83, 0xEC, 0x3C, // 005FD786 | 83EC 3C | sub esp,3C |
|
||||||
|
0xA1, XX4, // 005FD789 | A1 6C908600 | mov eax,dword ptr ds:[86906C] |
|
||||||
|
0x33, 0xC4, // 005FD78E | 33C4 | xor eax,esp |
|
||||||
|
0x89, 0x44, 0x24, 0x38, // 005FD790 | 894424 38 | mov dword ptr ss:[esp+38],eax |
|
||||||
|
0x53, // 005FD794 | 53 | push ebx |
|
||||||
|
0x56, // 005FD795 | 56 | push esi |
|
||||||
|
0x8B, 0xC1, // 005FD796 | 8BC1 | mov eax,ecx |
|
||||||
|
0xC7, 0x44, 0x24, 0x14, 0x00, 0x00, 0x00, 0x00, // 005FD798 | C74424 14 00000000 | mov dword ptr ss:[esp+14],0 |
|
||||||
|
0x8B, 0x4D, 0x0C, // 005FD7A0 | 8B4D 0C | mov ecx,dword ptr ss:[ebp+C] |
|
||||||
|
0x33, 0xF6, // 005FD7A3 | 33F6 | xor esi,esi |
|
||||||
|
0x57, // 005FD7A5 | 57 | push edi |
|
||||||
|
0x8B, 0x7D, 0x08, // 005FD7A6 | 8B7D 08 | mov edi,dword ptr ss:[ebp+8] |
|
||||||
|
0x89, 0x44, 0x24, 0x14, // 005FD7A9 | 894424 14 | mov dword ptr ss:[esp+14],eax |
|
||||||
|
0x89, 0x4C, 0x24, 0x28, // 005FD7AD | 894C24 28 | mov dword ptr ss:[esp+28],ecx |
|
||||||
|
0x80, 0x3F, 0x00, // 005FD7B1 | 803F 00 | cmp byte ptr ds:[edi],0 |
|
||||||
|
0x0F, 0x84, XX4, // 005FD7B4 | 0F84 88040000 | je ヘンタイ・プリズンsplit 1.5FDC42 |
|
||||||
|
0x83, 0xB8, XX4, 0x00, // 005FD7BA | 83B8 74030000 00 | cmp dword ptr ds:[eax+374],0 |
|
||||||
|
0x8B, 0xDF, // 005FD7C1 | 8BDF | mov ebx,edi |
|
||||||
|
};
|
||||||
|
|
||||||
|
enum { addr_offset = 0 }; // distance to the beginning of the function, which is 0x55 (push ebp)
|
||||||
|
ULONG range = min(processStopAddress - processStartAddress, MAX_REL_ADDR);
|
||||||
|
ULONG addr = MemDbg::findBytes(bytes, sizeof(bytes), processStartAddress, processStartAddress + range);
|
||||||
|
if (!addr) {
|
||||||
|
ConsoleOutput("Textractor:Artemis3: pattern not found");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
addr += addr_offset;
|
||||||
|
enum { push_ebp = 0x55 }; // beginning of the function
|
||||||
|
if (*(BYTE *)addr != push_ebp) {
|
||||||
|
ConsoleOutput("Textractor:Artemis3: beginning of the function not found");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = addr;
|
||||||
|
hp.offset = pusha_ebx_off - 4;
|
||||||
|
hp.type = USING_UTF8;
|
||||||
|
|
||||||
|
ConsoleOutput("Textractor: INSERT Artemis3");
|
||||||
|
NewHook(hp, "Artemis3");
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
bool InsertArtemisHook()
|
bool InsertArtemisHook()
|
||||||
{ return InsertArtemis1Hook() || InsertArtemis2Hook(); }
|
{ return InsertArtemis1Hook() || InsertArtemis2Hook() || InsertArtemis3Hook(); }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* jichi 1/2/2014: Taskforce2 Engine
|
* jichi 1/2/2014: Taskforce2 Engine
|
||||||
@ -14015,6 +14320,9 @@ bool InsertHorkEyeHook()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
memcpy(spDefault.pattern, Array<BYTE>{ 0xcc, 0xcc, 0xcc, XX, 0xec }, spDefault.length = 5);
|
||||||
|
spDefault.offset = 3;
|
||||||
|
|
||||||
const BYTE bytes2[] =
|
const BYTE bytes2[] =
|
||||||
{
|
{
|
||||||
0x83, 0xec, XX, // sub esp,??
|
0x83, 0xec, XX, // sub esp,??
|
||||||
@ -16896,13 +17204,16 @@ bool InsertAdobeFlash10Hook()
|
|||||||
*/
|
*/
|
||||||
bool InsertRenpyHook()
|
bool InsertRenpyHook()
|
||||||
{
|
{
|
||||||
|
wchar_t python[] = L"python2X.dll", libpython[] = L"libpython2.X.dll";
|
||||||
|
for (wchar_t* name : { python, libpython })
|
||||||
|
{
|
||||||
|
wchar_t* pos = wcschr(name, L'X');
|
||||||
for (int pythonMinorVersion = 0; pythonMinorVersion <= 8; ++pythonMinorVersion)
|
for (int pythonMinorVersion = 0; pythonMinorVersion <= 8; ++pythonMinorVersion)
|
||||||
{
|
{
|
||||||
wchar_t python[] = L"python2X.dll";
|
*pos = L'0' + pythonMinorVersion;
|
||||||
python[7] = L'0' + pythonMinorVersion;
|
if (HMODULE module = GetModuleHandleW(name))
|
||||||
if (HMODULE module = GetModuleHandleW(python))
|
|
||||||
{
|
{
|
||||||
wcscpy_s(spDefault.exportModule, python);
|
wcscpy_s(spDefault.exportModule, name);
|
||||||
HookParam hp = {};
|
HookParam hp = {};
|
||||||
hp.address = (DWORD)GetProcAddress(module, "PyUnicodeUCS2_Format");
|
hp.address = (DWORD)GetProcAddress(module, "PyUnicodeUCS2_Format");
|
||||||
if (!hp.address)
|
if (!hp.address)
|
||||||
@ -16913,13 +17224,20 @@ bool InsertRenpyHook()
|
|||||||
hp.offset = 4;
|
hp.offset = 4;
|
||||||
hp.index = 0xc;
|
hp.index = 0xc;
|
||||||
hp.length_offset = 0;
|
hp.length_offset = 0;
|
||||||
hp.split = pusha_ebx_off - 4;
|
//hp.split = pusha_ebx_off - 4;
|
||||||
hp.type = USING_STRING | USING_UNICODE | NO_CONTEXT | DATA_INDIRECT | USING_SPLIT;
|
hp.text_fun = [](auto, auto, auto, DWORD* data, DWORD* split, DWORD* count)
|
||||||
|
{
|
||||||
|
*data = *(DWORD*)(*data + 0xc);
|
||||||
|
*count = wcslen((wchar_t*)*data) * sizeof(wchar_t);
|
||||||
|
*split = wcschr((wchar_t*)*data, L'%') == nullptr;
|
||||||
|
};
|
||||||
|
hp.type = USING_STRING | USING_UNICODE | NO_CONTEXT | DATA_INDIRECT/* | USING_SPLIT*/;
|
||||||
//hp.filter_fun = [](void* str, auto, auto, auto) { return *(wchar_t*)str != L'%'; };
|
//hp.filter_fun = [](void* str, auto, auto, auto) { return *(wchar_t*)str != L'%'; };
|
||||||
NewHook(hp, "Ren'py");
|
NewHook(hp, "Ren'py");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
ConsoleOutput("Textractor: Ren'py failed: failed to find python2X.dll");
|
ConsoleOutput("Textractor: Ren'py failed: failed to find python2X.dll");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -16942,7 +17260,7 @@ void InsertMonoHook(HMODULE h)
|
|||||||
if (!getDomain || !getName || !getJitInfo) goto failed;
|
if (!getDomain || !getName || !getJitInfo) goto failed;
|
||||||
static auto domain = getDomain();
|
static auto domain = getDomain();
|
||||||
if (!domain) goto failed;
|
if (!domain) goto failed;
|
||||||
ConsoleOutput("Textractor: Mono Dynamic ENTER (hooks = %s)", loadedConfig ? loadedConfig : "brute force");
|
ConsoleOutput("Textractor: Mono Dynamic ENTER (hooks = %s)", *loadedConfig ? loadedConfig : "brute force");
|
||||||
const BYTE prolog[] = { 0x55, 0x8b, 0xec };
|
const BYTE prolog[] = { 0x55, 0x8b, 0xec };
|
||||||
for (auto addr : Util::SearchMemory(prolog, sizeof(prolog), PAGE_EXECUTE_READWRITE))
|
for (auto addr : Util::SearchMemory(prolog, sizeof(prolog), PAGE_EXECUTE_READWRITE))
|
||||||
{
|
{
|
||||||
@ -16957,7 +17275,7 @@ void InsertMonoHook(HMODULE h)
|
|||||||
HookParam hp = {};
|
HookParam hp = {};
|
||||||
hp.address = addr;
|
hp.address = addr;
|
||||||
hp.type = USING_UNICODE | FULL_STRING;
|
hp.type = USING_UNICODE | FULL_STRING;
|
||||||
if (!loadedConfig) hp.type |= KNOWN_UNSTABLE;
|
if (!*loadedConfig) hp.type |= KNOWN_UNSTABLE;
|
||||||
hp.offset = 4;
|
hp.offset = 4;
|
||||||
char nameForUser[HOOK_NAME_SIZE] = {};
|
char nameForUser[HOOK_NAME_SIZE] = {};
|
||||||
strncpy_s(nameForUser, name + 1, HOOK_NAME_SIZE - 1);
|
strncpy_s(nameForUser, name + 1, HOOK_NAME_SIZE - 1);
|
||||||
@ -16975,7 +17293,7 @@ void InsertMonoHook(HMODULE h)
|
|||||||
__except (EXCEPTION_EXECUTE_HANDLER) {}
|
__except (EXCEPTION_EXECUTE_HANDLER) {}
|
||||||
}(addr);
|
}(addr);
|
||||||
}
|
}
|
||||||
if (!loadedConfig) ConsoleOutput("Textractor: Mono Dynamic used brute force: if performance issues arise, please specify the correct hook in the game configuration");
|
if (!*loadedConfig) ConsoleOutput("Textractor: Mono Dynamic used brute force: if performance issues arise, please specify the correct hook in the game configuration");
|
||||||
return true;
|
return true;
|
||||||
failed:
|
failed:
|
||||||
ConsoleOutput("Textractor: Mono Dynamic failed");
|
ConsoleOutput("Textractor: Mono Dynamic failed");
|
||||||
@ -17032,7 +17350,7 @@ bool InsertMonoHooks()
|
|||||||
if (FARPROC addr = ::GetProcAddress(h, func.functionName)) {
|
if (FARPROC addr = ::GetProcAddress(h, func.functionName)) {
|
||||||
hp.address = (DWORD)addr;
|
hp.address = (DWORD)addr;
|
||||||
hp.type = func.hookType;
|
hp.type = func.hookType;
|
||||||
if (loadedConfig) hp.type |= HOOK_EMPTY;
|
if (*loadedConfig) hp.type |= HOOK_EMPTY;
|
||||||
hp.filter_fun = NoAsciiFilter;
|
hp.filter_fun = NoAsciiFilter;
|
||||||
hp.offset = func.textIndex * 4;
|
hp.offset = func.textIndex * 4;
|
||||||
hp.length_offset = func.lengthIndex * 4;
|
hp.length_offset = func.lengthIndex * 4;
|
||||||
@ -20967,6 +21285,84 @@ bool InsertTecmoPSPHook()
|
|||||||
}
|
}
|
||||||
#endif // 0
|
#endif // 0
|
||||||
|
|
||||||
|
|
||||||
|
//for debug
|
||||||
|
void StringBlockProcessor(char* str, size_t* size, size_t fixlength)
|
||||||
|
{
|
||||||
|
size_t original_len = *size;
|
||||||
|
size_t new_len = 0;
|
||||||
|
char* new_str = new char[original_len + 1];
|
||||||
|
|
||||||
|
size_t i = 0;
|
||||||
|
while (i < original_len) {
|
||||||
|
size_t block_end = i + fixlength;
|
||||||
|
if (block_end > original_len) {
|
||||||
|
block_end = original_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t block_len = block_end - i;
|
||||||
|
char* block = str + i;
|
||||||
|
|
||||||
|
for (size_t j = 0; j < block_len; ++j) {
|
||||||
|
if (block[j] != 0) {
|
||||||
|
new_str[new_len++] = block[j];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
i = block_end;
|
||||||
|
}
|
||||||
|
|
||||||
|
new_str[new_len] = '\0';
|
||||||
|
|
||||||
|
std::memcpy(str, new_str, new_len + 1);
|
||||||
|
*size = new_len;
|
||||||
|
delete[] new_str;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DACFilter(LPVOID data, DWORD* size, HookParam*, BYTE)
|
||||||
|
{
|
||||||
|
auto text = reinterpret_cast<LPSTR>(data);
|
||||||
|
auto len = reinterpret_cast<size_t*>(size);
|
||||||
|
|
||||||
|
StringBlockProcessor(text, len, 0x3f);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InsertDACHook()
|
||||||
|
{
|
||||||
|
const BYTE bytecodes[] = {
|
||||||
|
0x0F, 0xBF, 0x56, 0x34, // movsx edx, word ptr ds:[esi+34]
|
||||||
|
0x83, 0xC2, 0x05, // add edx, 5
|
||||||
|
0x8B, 0x8E, 0x00, 0x01, 0x00, 0x00, // mov ecx, dword ptr ds:[esi+100]
|
||||||
|
0x0F, 0xAF, 0xD3 // imul edx, ebx <--
|
||||||
|
};
|
||||||
|
|
||||||
|
ULONG addr = MemDbg::findBytes(bytecodes, sizeof(bytecodes), processStartAddress, processStopAddress);
|
||||||
|
if (addr == 0) {
|
||||||
|
ConsoleOutput("vnreng:DAC: pattern not found");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
enum { addr_offset = 13 };
|
||||||
|
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.type = USING_STRING; //S
|
||||||
|
hp.address = addr + addr_offset;
|
||||||
|
hp.offset = pusha_ecx_off - 4; //-C
|
||||||
|
hp.index = 0x00;
|
||||||
|
hp.filter_fun = DACFilter;
|
||||||
|
hp.length_fun = [](uintptr_t, uintptr_t data)
|
||||||
|
{
|
||||||
|
int len = 0x3f;
|
||||||
|
return len * 4;
|
||||||
|
};
|
||||||
|
NewHook(hp, "DAC");
|
||||||
|
ConsoleOutput("vnreng: INSERT DAC");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/** jichi 7/19/2014 PCSX2
|
/** jichi 7/19/2014 PCSX2
|
||||||
* Tested wit pcsx2-v1.2.1-328-gef0e3fe-windows-x86, built at http://buildbot.orphis.net/pcsx2
|
* Tested wit pcsx2-v1.2.1-328-gef0e3fe-windows-x86, built at http://buildbot.orphis.net/pcsx2
|
||||||
*/
|
*/
|
||||||
@ -21202,7 +21598,6 @@ bool InsertTypeMoonPS2Hook()
|
|||||||
ConsoleOutput("vnreng: TypeMoon PS2: leave");
|
ConsoleOutput("vnreng: TypeMoon PS2: leave");
|
||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 8/3/2014 jichi
|
/** 8/3/2014 jichi
|
||||||
* Tested game: School Rumble ねる娘<EFBFBD>育つ
|
* Tested game: School Rumble ねる娘<EFBFBD>育つ
|
||||||
*
|
*
|
||||||
|
@ -14,7 +14,7 @@ namespace Engine {
|
|||||||
// Global variables
|
// Global variables
|
||||||
extern wchar_t *processName, // cached
|
extern wchar_t *processName, // cached
|
||||||
processPath[MAX_PATH]; // cached
|
processPath[MAX_PATH]; // cached
|
||||||
inline const char *requestedEngine = nullptr, *loadedConfig = nullptr;
|
inline const char *requestedEngine = "", * loadedConfig = "";
|
||||||
|
|
||||||
bool InsertMonoHooks(); // Mono
|
bool InsertMonoHooks(); // Mono
|
||||||
|
|
||||||
@ -96,6 +96,7 @@ bool InsertCandyHook(); // SystemC@CandySoft: *.fpk
|
|||||||
bool InsertCatSystemHook(); // CatSystem2: *.int
|
bool InsertCatSystemHook(); // CatSystem2: *.int
|
||||||
bool InsertCMVSHook(); // CMVS: data/pack/*.cpz; do not support the latest cmvs32.exe and cmvs64.exe
|
bool InsertCMVSHook(); // CMVS: data/pack/*.cpz; do not support the latest cmvs32.exe and cmvs64.exe
|
||||||
bool InsertCotophaHook(); // Cotopha: *.noa
|
bool InsertCotophaHook(); // Cotopha: *.noa
|
||||||
|
bool InsertDACHook(); // DAC
|
||||||
bool InsertDebonosuHook(); // Debonosu: bmp.bak and dsetup.dll
|
bool InsertDebonosuHook(); // Debonosu: bmp.bak and dsetup.dll
|
||||||
bool InsertEaglsHook(); // E.A.G.L.S: EAGLES.dll
|
bool InsertEaglsHook(); // E.A.G.L.S: EAGLES.dll
|
||||||
bool InsertEMEHook(); // EmonEngine: emecfg.ecf
|
bool InsertEMEHook(); // EmonEngine: emecfg.ecf
|
||||||
|
@ -45,13 +45,13 @@ namespace Engine
|
|||||||
if (AutoHandle<> configFile = CreateFileW(configFilename, GENERIC_READ, FILE_SHARE_READ, nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL))
|
if (AutoHandle<> configFile = CreateFileW(configFilename, GENERIC_READ, FILE_SHARE_READ, nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL))
|
||||||
{
|
{
|
||||||
ReadFile(configFile, configFileData, sizeof(configFileData) - 1, DUMMY, nullptr);
|
ReadFile(configFile, configFileData, sizeof(configFileData) - 1, DUMMY, nullptr);
|
||||||
if (strncmp(configFileData, "Engine:", 7) == 0)
|
if (strnicmp(configFileData, "engine:", 7) == 0)
|
||||||
{
|
{
|
||||||
if (loadedConfig = strchr(configFileData, '\n')) *(char*)loadedConfig++ = 0;
|
if (const char* config = strchr(configFileData, '\n')) *(char*)(loadedConfig = config)++ = 0;
|
||||||
ConsoleOutput("Textractor: Engine = %s", requestedEngine = configFileData + 7);
|
ConsoleOutput("Textractor: Engine = %s", requestedEngine = strlwr(configFileData + 7));
|
||||||
}
|
}
|
||||||
else loadedConfig = configFileData;
|
else loadedConfig = configFileData;
|
||||||
if ((loadedConfig && !*loadedConfig) || strstr(configFileData, "https://")) loadedConfig = nullptr;
|
if (!*loadedConfig || strstr(configFileData, "https://")) loadedConfig = "";
|
||||||
else ConsoleOutput("Textractor: game configuration loaded");
|
else ConsoleOutput("Textractor: game configuration loaded");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,13 +67,14 @@ namespace Engine
|
|||||||
spDefault.maxAddress = processStopAddress;
|
spDefault.maxAddress = processStopAddress;
|
||||||
ConsoleOutput("Textractor: hijacking process located from 0x%p to 0x%p", processStartAddress, processStopAddress);
|
ConsoleOutput("Textractor: hijacking process located from 0x%p to 0x%p", processStartAddress, processStopAddress);
|
||||||
|
|
||||||
DetermineEngineType();
|
if (!strstr(requestedEngine, "none")) DetermineEngineType();
|
||||||
|
if (processStartAddress + 0x40000 > processStopAddress) ConsoleOutput("Textractor: WARNING injected process is very small, possibly a dummy!");
|
||||||
}(), 0);
|
}(), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ShouldMonoHook(const char* name)
|
bool ShouldMonoHook(const char* name)
|
||||||
{
|
{
|
||||||
if (!loadedConfig) return strstr(name, "string:") && !strstr(name, "string:mem");
|
if (!*loadedConfig) return strstr(name, "string:") && !strstr(name, "string:mem");
|
||||||
for (const char* hook = loadedConfig; hook; hook = strchr(hook + 1, '\t'))
|
for (const char* hook = loadedConfig; hook; hook = strchr(hook + 1, '\t'))
|
||||||
for (auto start = name; *start; ++start)
|
for (auto start = name; *start; ++start)
|
||||||
for (int i = 0; ; ++i)
|
for (int i = 0; ; ++i)
|
||||||
|
@ -54,6 +54,11 @@ bool DeterminePCEngine()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (Util::CheckFile(L"script.dpk")) {
|
||||||
|
InsertDACHook();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// jichi 5/14/2015: Skip hijacking BALDRSKY ZEROs
|
// jichi 5/14/2015: Skip hijacking BALDRSKY ZEROs
|
||||||
//if (Util::CheckFile(L"bsz_Data\\Mono\\mono.dll") || Util::CheckFile(L"bsz2_Data\\Mono\\mono.dll")) {
|
//if (Util::CheckFile(L"bsz_Data\\Mono\\mono.dll") || Util::CheckFile(L"bsz2_Data\\Mono\\mono.dll")) {
|
||||||
// ConsoleOutput("vnreng: IGNORE BALDRSKY ZEROs");
|
// ConsoleOutput("vnreng: IGNORE BALDRSKY ZEROs");
|
||||||
@ -629,10 +634,9 @@ bool DetermineEngineAtLast()
|
|||||||
}
|
}
|
||||||
if (Util::CheckFile(L"MovieTexture.dll") && (InsertPensilHook() || Insert2RMHook())) // MovieTexture.dll also exists in 2RM games such as 母子愛2体験版, which is checked first
|
if (Util::CheckFile(L"MovieTexture.dll") && (InsertPensilHook() || Insert2RMHook())) // MovieTexture.dll also exists in 2RM games such as 母子愛2体験版, which is checked first
|
||||||
return true;
|
return true;
|
||||||
if ((Util::CheckFile(L"system") && Util::CheckFile(L"system.dat")) || Util::CheckFile(L"*01")) { // jichi 7/31/2015 & Artikash 6/15/2018
|
if ((Util::CheckFile(L"system") && Util::CheckFile(L"system.dat")) || Util::CheckFile(L"*01")) // jichi 7/31/2015 & Artikash 6/15/2018
|
||||||
InsertAbelHook();
|
if (InsertAbelHook())
|
||||||
return true;
|
return true;
|
||||||
}
|
|
||||||
if (Util::CheckFile(L"data\\*.cpk")) { // jichi 12/2/2014
|
if (Util::CheckFile(L"data\\*.cpk")) { // jichi 12/2/2014
|
||||||
Insert5pbHook();
|
Insert5pbHook();
|
||||||
return true;
|
return true;
|
||||||
|
@ -80,7 +80,7 @@ namespace Engine
|
|||||||
if (!getDomain || !getName || !getJitInfo) goto failed;
|
if (!getDomain || !getName || !getJitInfo) goto failed;
|
||||||
static auto domain = getDomain();
|
static auto domain = getDomain();
|
||||||
if (!domain) goto failed;
|
if (!domain) goto failed;
|
||||||
ConsoleOutput("Textractor: Mono Dynamic ENTER (hooks = %s)", loadedConfig ? loadedConfig : "brute force");
|
ConsoleOutput("Textractor: Mono Dynamic ENTER (hooks = %s)", *loadedConfig ? loadedConfig : "brute force");
|
||||||
const BYTE prolog1[] = { 0x55, 0x48, 0x8b, 0xec };
|
const BYTE prolog1[] = { 0x55, 0x48, 0x8b, 0xec };
|
||||||
const BYTE prolog2[] = { 0x48, 0x83, 0xec };
|
const BYTE prolog2[] = { 0x48, 0x83, 0xec };
|
||||||
for (auto [prolog, size] : Array<const BYTE*, size_t>{ { prolog1, sizeof(prolog1) }, { prolog2, sizeof(prolog2) } })
|
for (auto [prolog, size] : Array<const BYTE*, size_t>{ { prolog1, sizeof(prolog1) }, { prolog2, sizeof(prolog2) } })
|
||||||
@ -97,7 +97,7 @@ namespace Engine
|
|||||||
HookParam hp = {};
|
HookParam hp = {};
|
||||||
hp.address = addr;
|
hp.address = addr;
|
||||||
hp.type = USING_STRING | USING_UNICODE | FULL_STRING;
|
hp.type = USING_STRING | USING_UNICODE | FULL_STRING;
|
||||||
if (!loadedConfig) hp.type |= KNOWN_UNSTABLE;
|
if (!*loadedConfig) hp.type |= KNOWN_UNSTABLE;
|
||||||
hp.offset = -0x20; // rcx
|
hp.offset = -0x20; // rcx
|
||||||
hp.padding = 20;
|
hp.padding = 20;
|
||||||
char nameForUser[HOOK_NAME_SIZE] = {};
|
char nameForUser[HOOK_NAME_SIZE] = {};
|
||||||
@ -119,7 +119,7 @@ namespace Engine
|
|||||||
}(addr);
|
}(addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!loadedConfig) ConsoleOutput("Textractor: Mono Dynamic used brute force: if performance issues arise, please specify the correct hook in the game configuration");
|
if (!*loadedConfig) ConsoleOutput("Textractor: Mono Dynamic used brute force: if performance issues arise, please specify the correct hook in the game configuration");
|
||||||
return true;
|
return true;
|
||||||
failed:
|
failed:
|
||||||
ConsoleOutput("Textractor: Mono Dynamic failed");
|
ConsoleOutput("Textractor: Mono Dynamic failed");
|
||||||
@ -132,25 +132,85 @@ namespace Engine
|
|||||||
// sample game https://www.freem.ne.jp/dl/win/18963
|
// sample game https://www.freem.ne.jp/dl/win/18963
|
||||||
bool InsertV8Hook(HMODULE module)
|
bool InsertV8Hook(HMODULE module)
|
||||||
{
|
{
|
||||||
if (uint64_t addr = (uint64_t)GetProcAddress(module, "?Write@String@v8@@QEBAHPEAGHHH@Z"))
|
auto getV8Length = [](uintptr_t, uintptr_t data)
|
||||||
{
|
|
||||||
std::tie(spDefault.minAddress, spDefault.maxAddress) = Util::QueryModuleLimits(module);
|
|
||||||
spDefault.maxRecords = Util::SearchMemory(spDefault.pattern, spDefault.length, PAGE_EXECUTE, spDefault.minAddress, spDefault.maxAddress).size() * 20;
|
|
||||||
ConsoleOutput("Textractor: JavaScript hook is known to be low quality: try searching for hooks if you don't like it");
|
|
||||||
HookParam hp = {};
|
|
||||||
hp.type = USING_STRING | USING_UNICODE | DATA_INDIRECT;
|
|
||||||
hp.address = addr;
|
|
||||||
hp.offset = -0x20; // rcx
|
|
||||||
hp.index = 0;
|
|
||||||
hp.padding = 23;
|
|
||||||
hp.length_fun = [](uintptr_t, uintptr_t data)
|
|
||||||
{
|
{
|
||||||
int len = *(int*)(data - 4);
|
int len = *(int*)(data - 4);
|
||||||
return len > 0 && len < PIPE_BUFFER_SIZE ? len * 2 : 0;
|
return len > 0 && len < PIPE_BUFFER_SIZE ? len * 2 : 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
uint64_t addr1 = (uint64_t)GetProcAddress(module, "?Write@String@v8@@QEBAHPEAGHHH@Z"),
|
||||||
|
// Artikash 6/7/2021: Add new hook for new version of V8 used by RPG Maker MZ
|
||||||
|
addr2 = (uint64_t)GetProcAddress(module, "??$WriteToFlat@G@String@internal@v8@@SAXV012@PEAGHH@Z");
|
||||||
|
|
||||||
|
if (addr1 || addr2)
|
||||||
|
{
|
||||||
|
std::tie(spDefault.minAddress, spDefault.maxAddress) = Util::QueryModuleLimits(module);
|
||||||
|
spDefault.maxRecords = Util::SearchMemory(spDefault.pattern, spDefault.length, PAGE_EXECUTE, spDefault.minAddress, spDefault.maxAddress).size() * 20;
|
||||||
|
ConsoleOutput("Textractor: JavaScript hook is known to be low quality: try searching for hooks if you don't like it");
|
||||||
|
}
|
||||||
|
if (addr1)
|
||||||
|
{
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.type = USING_STRING | USING_UNICODE | DATA_INDIRECT;
|
||||||
|
hp.address = addr1;
|
||||||
|
hp.offset = -0x20; // rcx
|
||||||
|
hp.index = 0;
|
||||||
|
hp.padding = 23;
|
||||||
|
hp.length_fun = getV8Length;
|
||||||
NewHook(hp, "JavaScript");
|
NewHook(hp, "JavaScript");
|
||||||
|
}
|
||||||
|
if (addr2)
|
||||||
|
{
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.type = USING_STRING | USING_UNICODE;
|
||||||
|
hp.address = addr2;
|
||||||
|
hp.offset = -0x20; // rcx
|
||||||
|
hp.padding = 11;
|
||||||
|
hp.length_fun = getV8Length;
|
||||||
|
NewHook(hp, "JavaScript");
|
||||||
|
}
|
||||||
|
return addr1 || addr2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Artikash 8/10/2018: Ren'py
|
||||||
|
*
|
||||||
|
* Sample games: https://vndb.org/v19843 https://vndb.org/v12038 and many more OELVNs
|
||||||
|
*
|
||||||
|
* Uses CPython, and links to python27.dll. PyUicodeUCS2_Format is the function used to process text.
|
||||||
|
* first argument. offset 0x18 from that is a wchar_t* to the actual string
|
||||||
|
* ebx seems to work well as the split param, not sure why
|
||||||
|
*/
|
||||||
|
bool InsertRenpyHook()
|
||||||
|
{
|
||||||
|
wchar_t python[] = L"python2X.dll", libpython[] = L"libpython2.X.dll";
|
||||||
|
for (wchar_t* name : { python, libpython })
|
||||||
|
{
|
||||||
|
wchar_t* pos = wcschr(name, L'X');
|
||||||
|
for (int pythonMinorVersion = 0; pythonMinorVersion <= 8; ++pythonMinorVersion)
|
||||||
|
{
|
||||||
|
*pos = L'0' + pythonMinorVersion;
|
||||||
|
if (HMODULE module = GetModuleHandleW(name))
|
||||||
|
{
|
||||||
|
wcscpy_s(spDefault.exportModule, name);
|
||||||
|
HookParam hp = {};
|
||||||
|
hp.address = (DWORD)GetProcAddress(module, "PyUnicodeUCS2_Format");
|
||||||
|
if (!hp.address)
|
||||||
|
{
|
||||||
|
ConsoleOutput("Textractor: Ren'py failed: failed to find PyUnicodeUCS2_Format");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
hp.offset = -0x20; // rcx
|
||||||
|
hp.index = 0x18;
|
||||||
|
hp.length_offset = 0;
|
||||||
|
//hp.split = pusha_ebx_off - 4;
|
||||||
|
hp.type = USING_STRING | USING_UNICODE | NO_CONTEXT | DATA_INDIRECT /* | USING_SPLIT*/;
|
||||||
|
//hp.filter_fun = [](void* str, auto, auto, auto) { return *(wchar_t*)str != L'%'; };
|
||||||
|
NewHook(hp, "Ren'py");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ConsoleOutput("Textractor: Ren'py failed: failed to find python2X.dll");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -168,6 +228,8 @@ namespace Engine
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (Util::CheckFile(L"*.py") && InsertRenpyHook()) return true;
|
||||||
|
|
||||||
for (const wchar_t* monoName : { L"mono.dll", L"mono-2.0-bdwgc.dll" }) if (HMODULE module = GetModuleHandleW(monoName)) if (InsertMonoHooks(module)) return true;
|
for (const wchar_t* monoName : { L"mono.dll", L"mono-2.0-bdwgc.dll" }) if (HMODULE module = GetModuleHandleW(monoName)) if (InsertMonoHooks(module)) return true;
|
||||||
|
|
||||||
for (std::wstring DXVersion : { L"d3dx9", L"d3dx10" })
|
for (std::wstring DXVersion : { L"d3dx9", L"d3dx10" })
|
||||||
|
@ -2,8 +2,10 @@
|
|||||||
#include "defs.h"
|
#include "defs.h"
|
||||||
#include "main.h"
|
#include "main.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
#include "MinHook.h"
|
||||||
|
|
||||||
extern const char* STARTING_SEARCH;
|
extern const char* HOOK_SEARCH_STARTING;
|
||||||
|
extern const char* HOOK_SEARCH_INITIALIZING;
|
||||||
extern const char* HOOK_SEARCH_INITIALIZED;
|
extern const char* HOOK_SEARCH_INITIALIZED;
|
||||||
extern const char* MAKE_GAME_PROCESS_TEXT;
|
extern const char* MAKE_GAME_PROCESS_TEXT;
|
||||||
extern const char* HOOK_SEARCH_FINISHED;
|
extern const char* HOOK_SEARCH_FINISHED;
|
||||||
@ -200,12 +202,12 @@ void SearchForHooks(SearchParam spUser)
|
|||||||
|
|
||||||
sp = spUser.length == 0 ? spDefault : spUser;
|
sp = spUser.length == 0 ? spDefault : spUser;
|
||||||
|
|
||||||
|
ConsoleOutput(HOOK_SEARCH_INITIALIZING, 0.);
|
||||||
do
|
do
|
||||||
try { records = std::make_unique<HookRecord[]>(recordsAvailable = sp.maxRecords); }
|
try { records = std::make_unique<HookRecord[]>(recordsAvailable = sp.maxRecords); }
|
||||||
catch (std::bad_alloc) { ConsoleOutput("Textractor: SearchForHooks ERROR: out of memory, retrying to allocate %d", sp.maxRecords /= 2); }
|
catch (std::bad_alloc) { ConsoleOutput("Textractor: SearchForHooks ERROR: out of memory, retrying to allocate %d", sp.maxRecords /= 2); }
|
||||||
while (!records && sp.maxRecords);
|
while (!records && sp.maxRecords);
|
||||||
|
|
||||||
ConsoleOutput(STARTING_SEARCH);
|
|
||||||
std::vector<uint64_t> addresses;
|
std::vector<uint64_t> addresses;
|
||||||
if (*sp.boundaryModule) std::tie(sp.minAddress, sp.maxAddress) = Util::QueryModuleLimits(GetModuleHandleW(sp.boundaryModule));
|
if (*sp.boundaryModule) std::tie(sp.minAddress, sp.maxAddress) = Util::QueryModuleLimits(GetModuleHandleW(sp.boundaryModule));
|
||||||
if (*sp.exportModule) addresses = GetFunctions((uintptr_t)GetModuleHandleW(sp.exportModule));
|
if (*sp.exportModule) addresses = GetFunctions((uintptr_t)GetModuleHandleW(sp.exportModule));
|
||||||
@ -224,9 +226,11 @@ void SearchForHooks(SearchParam spUser)
|
|||||||
memcpy(trampolines[i], trampoline, sizeof(trampoline));
|
memcpy(trampolines[i], trampoline, sizeof(trampoline));
|
||||||
*(uintptr_t*)(trampolines[i] + addr_offset) = addresses[i];
|
*(uintptr_t*)(trampolines[i] + addr_offset) = addresses[i];
|
||||||
*(void**)(trampolines[i] + original_offset) = original;
|
*(void**)(trampolines[i] + original_offset) = original;
|
||||||
|
if (i % 2500 == 0) ConsoleOutput(HOOK_SEARCH_INITIALIZING, 1 + 98. * i / addresses.size());
|
||||||
}
|
}
|
||||||
ConsoleOutput(HOOK_SEARCH_INITIALIZED, addresses.size());
|
ConsoleOutput(HOOK_SEARCH_INITIALIZED, addresses.size());
|
||||||
MH_ApplyQueued();
|
MH_ApplyQueued();
|
||||||
|
ConsoleOutput(HOOK_SEARCH_STARTING);
|
||||||
ConsoleOutput(MAKE_GAME_PROCESS_TEXT, sp.searchTime / 1000);
|
ConsoleOutput(MAKE_GAME_PROCESS_TEXT, sp.searchTime / 1000);
|
||||||
Sleep(sp.searchTime);
|
Sleep(sp.searchTime);
|
||||||
for (auto addr : addresses) MH_QueueDisableHook((void*)addr);
|
for (auto addr : addresses) MH_QueueDisableHook((void*)addr);
|
||||||
@ -261,7 +265,7 @@ void SearchForText(wchar_t* text, UINT codepage)
|
|||||||
char codepageText[PATTERN_SIZE * 4] = {};
|
char codepageText[PATTERN_SIZE * 4] = {};
|
||||||
WideCharToMultiByte(codepage, 0, text, PATTERN_SIZE, codepageText, PATTERN_SIZE * 4, nullptr, nullptr);
|
WideCharToMultiByte(codepage, 0, text, PATTERN_SIZE, codepageText, PATTERN_SIZE * 4, nullptr, nullptr);
|
||||||
if (strlen(utf8Text) < 4 || strlen(codepageText) < 4 || wcslen(text) < 4) return ConsoleOutput(NOT_ENOUGH_TEXT);
|
if (strlen(utf8Text) < 4 || strlen(codepageText) < 4 || wcslen(text) < 4) return ConsoleOutput(NOT_ENOUGH_TEXT);
|
||||||
ConsoleOutput(STARTING_SEARCH);
|
ConsoleOutput(HOOK_SEARCH_STARTING);
|
||||||
auto GenerateHooks = [&](std::vector<uint64_t> addresses, HookParamType type)
|
auto GenerateHooks = [&](std::vector<uint64_t> addresses, HookParamType type)
|
||||||
{
|
{
|
||||||
for (auto addr : addresses)
|
for (auto addr : addresses)
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
#include "texthook.h"
|
#include "texthook.h"
|
||||||
#include "hookfinder.h"
|
#include "hookfinder.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
#include "MinHook.h"
|
||||||
|
|
||||||
extern const char* PIPE_CONNECTED;
|
extern const char* PIPE_CONNECTED;
|
||||||
extern const char* INSERTING_HOOK;
|
extern const char* INSERTING_HOOK;
|
||||||
|
@ -15,49 +15,6 @@ void RemoveHook(uint64_t addr, int maxOffset = 9);
|
|||||||
|
|
||||||
inline SearchParam spDefault;
|
inline SearchParam spDefault;
|
||||||
|
|
||||||
extern "C" // minhook library
|
|
||||||
{
|
|
||||||
enum MH_STATUS
|
|
||||||
{
|
|
||||||
MH_OK,
|
|
||||||
MH_ERROR_ALREADY_INITIALIZED,
|
|
||||||
MH_ERROR_NOT_INITIALIZED,
|
|
||||||
MH_ERROR_ALREADY_CREATED,
|
|
||||||
MH_ERROR_NOT_CREATED,
|
|
||||||
MH_ERROR_ENABLED,
|
|
||||||
MH_ERROR_DISABLED,
|
|
||||||
MH_ERROR_NOT_EXECUTABLE,
|
|
||||||
MH_ERROR_UNSUPPORTED_FUNCTION,
|
|
||||||
MH_ERROR_MEMORY_ALLOC,
|
|
||||||
MH_ERROR_MEMORY_PROTECT,
|
|
||||||
MH_ERROR_MODULE_NOT_FOUND,
|
|
||||||
MH_ERROR_FUNCTION_NOT_FOUND
|
|
||||||
};
|
|
||||||
|
|
||||||
MH_STATUS WINAPI MH_Initialize(VOID);
|
|
||||||
MH_STATUS WINAPI MH_Uninitialize(VOID);
|
|
||||||
|
|
||||||
// Creates a Hook for the specified target function, in disabled state.
|
|
||||||
// Parameters:
|
|
||||||
// pTarget [in] A pointer to the target function, which will be
|
|
||||||
// overridden by the detour function.
|
|
||||||
// pDetour [in] A pointer to the detour function, which will override
|
|
||||||
// the target function.
|
|
||||||
// ppOriginal [out] A pointer to the trampoline function, which will be
|
|
||||||
// used to call the original target function.
|
|
||||||
// This parameter can be NULL.
|
|
||||||
MH_STATUS WINAPI MH_CreateHook(LPVOID pTarget, LPVOID pDetour, LPVOID *ppOriginal);
|
|
||||||
MH_STATUS WINAPI MH_EnableHook(LPVOID pTarget);
|
|
||||||
MH_STATUS WINAPI MH_DisableHook(LPVOID pTarget);
|
|
||||||
MH_STATUS WINAPI MH_RemoveHook(LPVOID pTarget);
|
|
||||||
MH_STATUS WINAPI MH_QueueEnableHook(LPVOID pTarget);
|
|
||||||
MH_STATUS WINAPI MH_QueueDisableHook(LPVOID pTarget);
|
|
||||||
MH_STATUS WINAPI MH_ApplyQueued(VOID);
|
|
||||||
const char* WINAPI MH_StatusToString(MH_STATUS status);
|
|
||||||
}
|
|
||||||
|
|
||||||
#define MH_ALL_HOOKS NULL
|
|
||||||
|
|
||||||
#define ITH_RAISE (*(int*)0 = 0) // raise C000005, for debugging only
|
#define ITH_RAISE (*(int*)0 = 0) // raise C000005, for debugging only
|
||||||
#define ITH_TRY __try
|
#define ITH_TRY __try
|
||||||
#define ITH_EXCEPT __except(EXCEPTION_EXECUTE_HANDLER)
|
#define ITH_EXCEPT __except(EXCEPTION_EXECUTE_HANDLER)
|
||||||
|
1
texthook/minhook
Submodule
1
texthook/minhook
Submodule
@ -0,0 +1 @@
|
|||||||
|
Subproject commit 423d1e45af2ed2719a5c31e990e935ef301ed9c3
|
@ -6,6 +6,7 @@
|
|||||||
#include "texthook.h"
|
#include "texthook.h"
|
||||||
#include "main.h"
|
#include "main.h"
|
||||||
#include "ithsys/ithsys.h"
|
#include "ithsys/ithsys.h"
|
||||||
|
#include "MinHook.h"
|
||||||
|
|
||||||
extern const char* FUNC_MISSING;
|
extern const char* FUNC_MISSING;
|
||||||
extern const char* MODULE_MISSING;
|
extern const char* MODULE_MISSING;
|
||||||
@ -192,6 +193,7 @@ void TextHook::Send(uintptr_t dwDataBase)
|
|||||||
|
|
||||||
TextOutput(tp, buffer, count);
|
TextOutput(tp, buffer, count);
|
||||||
#endif // _WIN64
|
#endif // _WIN64
|
||||||
|
++*pbData;
|
||||||
}
|
}
|
||||||
__except (EXCEPTION_EXECUTE_HANDLER)
|
__except (EXCEPTION_EXECUTE_HANDLER)
|
||||||
{
|
{
|
||||||
@ -307,7 +309,7 @@ int TextHook::GetLength(uintptr_t base, uintptr_t in)
|
|||||||
else {
|
else {
|
||||||
if (hp.type & BIG_ENDIAN)
|
if (hp.type & BIG_ENDIAN)
|
||||||
in >>= 8;
|
in >>= 8;
|
||||||
len = LeadByteTable[in & 0xff]; //Slightly faster than IsDBCSLeadByte
|
len = !!IsDBCSLeadByteEx(hp.codepage, in & 0xff) + 1;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -51,6 +51,6 @@ private:
|
|||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum { MAX_HOOK = 300, HOOK_BUFFER_SIZE = MAX_HOOK * sizeof(TextHook), HOOK_SECTION_SIZE = HOOK_BUFFER_SIZE * 2 };
|
enum { MAX_HOOK = 2500, HOOK_BUFFER_SIZE = MAX_HOOK * sizeof(TextHook), HOOK_SECTION_SIZE = HOOK_BUFFER_SIZE * 2 };
|
||||||
|
|
||||||
// EOF
|
// EOF
|
||||||
|
@ -9,32 +9,6 @@
|
|||||||
#include "ithsys/ithsys.h"
|
#include "ithsys/ithsys.h"
|
||||||
#include "const.h"
|
#include "const.h"
|
||||||
|
|
||||||
// - Global variables -
|
|
||||||
|
|
||||||
// jichi 6/12/2015: https://en.wikipedia.org/wiki/Shift_JIS
|
|
||||||
// Leading table for SHIFT-JIS encoding
|
|
||||||
BYTE LeadByteTable[0x100] = {
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
||||||
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
||||||
2,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
||||||
2,2,2,2,2,2,2,2,2,2,2,2,2,1,1,1
|
|
||||||
};
|
|
||||||
|
|
||||||
// - API functions -
|
|
||||||
|
|
||||||
extern "C" {
|
|
||||||
/**
|
/**
|
||||||
* Return the address of the first matched pattern.
|
* Return the address of the first matched pattern.
|
||||||
* Artikash 7/14/2018: changed implementation, hopefully it behaves the same
|
* Artikash 7/14/2018: changed implementation, hopefully it behaves the same
|
||||||
@ -70,15 +44,25 @@ DWORD IthGetMemoryRange(LPCVOID mem, DWORD *base, DWORD *size)
|
|||||||
return info.Protect > PAGE_NOACCESS;
|
return info.Protect > PAGE_NOACCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline DWORD GetHash(LPSTR str)
|
// jichi 6/12/2015: https://en.wikipedia.org/wiki/Shift_JIS
|
||||||
{
|
// Leading table for SHIFT-JIS encoding
|
||||||
DWORD hash = 0;
|
BYTE LeadByteTable[0x100] = {
|
||||||
//for (; *str; str++)
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
while (*str)
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
hash = ((hash >> 7) | (hash << 25)) + *str++;
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
return hash;
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
}
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
} // extern "C"
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
||||||
|
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
||||||
|
2,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
||||||
|
2,2,2,2,2,2,2,2,2,2,2,2,2,1,1,1
|
||||||
|
};
|
||||||
|
|
||||||
// EOF
|
// EOF
|
@ -10,14 +10,10 @@
|
|||||||
//#include "ntdll/ntdll.h"
|
//#include "ntdll/ntdll.h"
|
||||||
#include <Windows.h>
|
#include <Windows.h>
|
||||||
|
|
||||||
// jichi 8/24/2013: Why extern "C"? Any specific reason to use C instead of C++ naming?
|
|
||||||
extern "C" {
|
|
||||||
|
|
||||||
// jichi 10/1/2013: Return 0 if failed. So, it is ambiguous if the search pattern starts at 0
|
// jichi 10/1/2013: Return 0 if failed. So, it is ambiguous if the search pattern starts at 0
|
||||||
DWORD SearchPattern(DWORD base, DWORD base_length, LPCVOID search, DWORD search_length); // KMP
|
DWORD SearchPattern(DWORD base, DWORD base_length, LPCVOID search, DWORD search_length); // KMP
|
||||||
|
|
||||||
DWORD IthGetMemoryRange(LPCVOID mem, DWORD *base, DWORD *size);
|
DWORD IthGetMemoryRange(LPCVOID mem, DWORD *base, DWORD *size);
|
||||||
} // extern "C"
|
|
||||||
|
|
||||||
extern BYTE LeadByteTable[];
|
extern BYTE LeadByteTable[];
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
#include "util/util.h"
|
#include "util/util.h"
|
||||||
#include "ithsys/ithsys.h"
|
#include "ithsys/ithsys.h"
|
||||||
#include "main.h"
|
#include "main.h"
|
||||||
|
#include <Psapi.h>
|
||||||
|
|
||||||
namespace { // unnamed
|
namespace { // unnamed
|
||||||
|
|
||||||
@ -345,6 +346,15 @@ std::vector<uint64_t> SearchMemory(const void* bytes, short length, DWORD protec
|
|||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uintptr_t FindFunction(const char* function)
|
||||||
|
{
|
||||||
|
static HMODULE modules[300] = {};
|
||||||
|
static auto _ = EnumProcessModules(GetCurrentProcess(), modules, sizeof(modules), DUMMY);
|
||||||
|
for (auto module : modules) if (auto addr = GetProcAddress(module, function)) return (uintptr_t)addr;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// EOF
|
// EOF
|
||||||
|
@ -22,6 +22,7 @@ bool SearchResourceString(LPCWSTR str);
|
|||||||
|
|
||||||
std::pair<uint64_t, uint64_t> QueryModuleLimits(HMODULE module);
|
std::pair<uint64_t, uint64_t> QueryModuleLimits(HMODULE module);
|
||||||
std::vector<uint64_t> SearchMemory(const void* bytes, short length, DWORD protect = PAGE_EXECUTE, uintptr_t minAddr = 0, uintptr_t maxAddr = -1ULL);
|
std::vector<uint64_t> SearchMemory(const void* bytes, short length, DWORD protect = PAGE_EXECUTE, uintptr_t minAddr = 0, uintptr_t maxAddr = -1ULL);
|
||||||
|
uintptr_t FindFunction(const char* function);
|
||||||
|
|
||||||
} // namespace Util
|
} // namespace Util
|
||||||
|
|
||||||
|
Binary file not shown.
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user