Textractor_test/include/common.h

164 lines
4.7 KiB
C
Raw Normal View History

2018-08-23 00:24:55 +08:00
#pragma once
#define WIN32_LEAN_AND_MEAN
#include <Windows.h>
2021-06-05 16:58:53 +08:00
#include <concrt.h>
2018-08-23 00:24:55 +08:00
#include <string>
#include <vector>
#include <deque>
2019-02-09 13:30:38 +08:00
#include <array>
2018-08-23 00:24:55 +08:00
#include <unordered_map>
2018-11-05 10:19:00 +08:00
#include <unordered_set>
2018-08-23 00:24:55 +08:00
#include <functional>
#include <algorithm>
2018-11-05 10:19:00 +08:00
#include <regex>
#include <memory>
#include <optional>
2018-08-23 00:24:55 +08:00
#include <thread>
2018-08-23 23:53:23 +08:00
#include <mutex>
2019-02-09 13:30:38 +08:00
#include <atomic>
#include <filesystem>
2018-09-21 09:59:07 +08:00
#include <cstdint>
2018-12-27 12:56:42 +08:00
#include <cassert>
2019-02-19 12:12:12 +08:00
2019-06-01 02:48:07 +08:00
#ifdef _WIN64
constexpr bool x64 = true;
#else
constexpr bool x64 = false;
#endif
2020-12-14 21:26:01 +08:00
template <typename T, typename... Xs> struct ArrayImpl { using Type = std::tuple<T, Xs...>[]; };
template <typename T> struct ArrayImpl<T> { using Type = T[]; };
template <typename... Ts> using Array = typename ArrayImpl<Ts...>::Type;
2019-02-19 12:12:12 +08:00
2020-12-14 21:26:01 +08:00
template <auto F> using Functor = std::integral_constant<std::remove_reference_t<decltype(F)>, F>; // shouldn't need remove_reference_t but MSVC is bugged
2019-08-07 17:18:04 +08:00
struct PermissivePointer
{
2020-12-14 21:26:01 +08:00
template <typename T> operator T*() { return (T*)p; }
2019-08-07 17:18:04 +08:00
void* p;
};
template <typename HandleCloser = Functor<CloseHandle>>
class AutoHandle
{
public:
AutoHandle(HANDLE h) : h(h) {}
operator HANDLE() { return h.get(); }
PHANDLE operator&() { static_assert(sizeof(*this) == sizeof(HANDLE)); assert(!h); return (PHANDLE)this; }
operator bool() { return h.get() != NULL && h.get() != INVALID_HANDLE_VALUE; }
private:
struct HandleCleaner { void operator()(void* h) { if (h != INVALID_HANDLE_VALUE) HandleCloser()(PermissivePointer{ h }); } };
std::unique_ptr<void, HandleCleaner> h;
};
2019-06-10 13:49:11 +08:00
template<typename T, typename M = std::mutex>
2019-06-05 11:12:45 +08:00
class Synchronized
2019-02-19 12:12:12 +08:00
{
public:
template <typename... Args>
2019-06-05 11:12:45 +08:00
Synchronized(Args&&... args) : contents(std::forward<Args>(args)...) {}
struct Locker
2019-02-19 12:12:12 +08:00
{
2019-06-10 13:49:11 +08:00
T* operator->() { return &contents; }
2019-06-05 11:12:45 +08:00
std::unique_lock<M> lock;
2019-06-10 13:49:11 +08:00
T& contents;
2019-06-05 11:12:45 +08:00
};
2019-06-10 13:49:11 +08:00
Locker Acquire() { return { std::unique_lock(m), contents }; }
2019-06-05 11:12:45 +08:00
Locker operator->() { return Acquire(); }
2020-12-14 21:26:01 +08:00
T Copy() { return Acquire().contents; }
2019-02-19 12:12:12 +08:00
private:
2019-06-10 13:49:11 +08:00
T contents;
M m;
2019-02-19 12:12:12 +08:00
};
2021-11-14 06:49:07 +08:00
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));
}
2023-04-22 07:43:32 +08:00
inline struct
2019-06-04 05:58:30 +08:00
{
2020-12-14 21:26:01 +08:00
inline static BYTE DUMMY[100];
template <typename T> operator T*() { static_assert(sizeof(T) < sizeof(DUMMY)); return (T*)DUMMY; }
2019-06-04 05:58:30 +08:00
} DUMMY;
inline auto Swallow = [](auto&&...) {};
2020-12-14 21:26:01 +08:00
template <typename T> std::optional<std::remove_cv_t<T>> Copy(T* ptr) { if (ptr) return *ptr; return {}; }
2019-06-13 16:01:29 +08:00
2020-12-14 21:26:01 +08:00
template <typename T> inline auto FormatArg(T arg) { return arg; }
template <typename C> inline auto FormatArg(const std::basic_string<C>& arg) { return arg.c_str(); }
2019-06-13 16:01:29 +08:00
2019-06-02 01:59:37 +08:00
#pragma warning(push)
#pragma warning(disable: 4996)
template <typename... Args>
2019-06-29 18:13:26 +08:00
inline std::string FormatString(const char* format, const Args&... args)
2019-06-02 01:59:37 +08:00
{
2019-06-13 16:01:29 +08:00
std::string buffer(snprintf(nullptr, 0, format, FormatArg(args)...), '\0');
sprintf(buffer.data(), format, FormatArg(args)...);
2019-06-02 01:59:37 +08:00
return buffer;
}
template <typename... Args>
2019-06-29 18:13:26 +08:00
inline std::wstring FormatString(const wchar_t* format, const Args&... args)
2019-06-02 01:59:37 +08:00
{
2019-06-13 16:01:29 +08:00
std::wstring buffer(_snwprintf(nullptr, 0, format, FormatArg(args)...), L'\0');
_swprintf(buffer.data(), format, FormatArg(args)...);
2019-06-02 01:59:37 +08:00
return buffer;
}
#pragma warning(pop)
2023-04-22 07:42:43 +08:00
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)
{
std::vector<wchar_t> buffer(text.size() + 1);
if (int length = MultiByteToWideChar(encoding, 0, text.c_str(), text.size() + 1, buffer.data(), buffer.size()))
return std::wstring(buffer.data(), length - 1);
return {};
}
inline std::wstring StringToWideString(const std::string& text)
{
std::vector<wchar_t> buffer(text.size() + 1);
MultiByteToWideChar(CP_UTF8, 0, text.c_str(), -1, buffer.data(), buffer.size());
return buffer.data();
}
inline std::string WideStringToString(const std::wstring& text)
{
std::vector<char> buffer((text.size() + 1) * 4);
WideCharToMultiByte(CP_UTF8, 0, text.c_str(), -1, buffer.data(), buffer.size(), nullptr, nullptr);
return buffer.data();
}
2020-02-12 14:34:03 +08:00
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); }
2019-09-11 06:53:55 +08:00
2020-09-10 04:54:15 +08:00
template <typename... Args>
2021-11-14 06:49:07 +08:00
inline void TEXTRACTOR_DEBUG(const wchar_t* format, const Args&... args) { SpawnThread([=] { TEXTRACTOR_MESSAGE(format, args...); }); }
2020-09-10 04:54:15 +08:00
void Localize();
2019-02-19 12:12:12 +08:00
#ifdef _DEBUG
2021-03-13 15:51:36 +08:00
#define TEST(...) static auto _ = CreateThread(nullptr, 0, [](auto) { __VA_ARGS__; return 0UL; }, NULL, 0, nullptr)
2019-02-19 12:12:12 +08:00
#else
#define TEST(...)
#endif