mirror of
https://github.com/HIllya51/LunaTranslator.git
synced 2025-01-13 23:53:54 +08:00
.
This commit is contained in:
parent
566ec24b7b
commit
f4f337474c
@ -1,7 +1,7 @@
|
||||
|
||||
set(VERSION_MAJOR 6)
|
||||
set(VERSION_MINOR 14)
|
||||
set(VERSION_PATCH 7)
|
||||
set(VERSION_PATCH 8)
|
||||
set(VERSION_REVISION 0)
|
||||
set(LUNA_VERSION "{${VERSION_MAJOR},${VERSION_MINOR},${VERSION_PATCH},${VERSION_REVISION}}")
|
||||
add_library(VERSION_DEF ${CMAKE_CURRENT_LIST_DIR}/version_def.cpp)
|
||||
|
@ -16,6 +16,105 @@ MWebBrowser::Create(HWND hwndParent)
|
||||
}
|
||||
return pBrowser;
|
||||
}
|
||||
void JSObject::bindfunction(const std::wstring &funcname, functiontype function)
|
||||
{
|
||||
auto curr = DISPID_VALUE + 1 + funcnames.size();
|
||||
funcnames[funcname] = curr;
|
||||
funcmap[curr] = function;
|
||||
}
|
||||
HRESULT STDMETHODCALLTYPE JSObject::QueryInterface(REFIID riid, void **ppv)
|
||||
{
|
||||
*ppv = NULL;
|
||||
|
||||
if (riid == IID_IUnknown || riid == IID_IDispatch)
|
||||
{
|
||||
*ppv = static_cast<IDispatch *>(this);
|
||||
}
|
||||
|
||||
if (*ppv != NULL)
|
||||
{
|
||||
AddRef();
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
return E_NOINTERFACE;
|
||||
}
|
||||
|
||||
ULONG STDMETHODCALLTYPE JSObject::AddRef()
|
||||
{
|
||||
return InterlockedIncrement(&ref);
|
||||
}
|
||||
|
||||
ULONG STDMETHODCALLTYPE JSObject::Release()
|
||||
{
|
||||
int tmp = InterlockedDecrement(&ref);
|
||||
|
||||
if (tmp == 0)
|
||||
{
|
||||
// OutputDebugString("JSObject::Release(): delete this");
|
||||
delete this;
|
||||
}
|
||||
|
||||
return tmp;
|
||||
}
|
||||
|
||||
HRESULT STDMETHODCALLTYPE JSObject::GetTypeInfoCount(UINT *pctinfo)
|
||||
{
|
||||
*pctinfo = 0;
|
||||
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
HRESULT STDMETHODCALLTYPE JSObject::GetTypeInfo(UINT iTInfo, LCID lcid,
|
||||
ITypeInfo **ppTInfo)
|
||||
{
|
||||
return E_FAIL;
|
||||
}
|
||||
|
||||
HRESULT STDMETHODCALLTYPE JSObject::GetIDsOfNames(REFIID riid,
|
||||
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
||||
{
|
||||
HRESULT hr = S_OK;
|
||||
|
||||
for (UINT i = 0; i < cNames; i++)
|
||||
{
|
||||
auto iter = funcnames.find(rgszNames[i]);
|
||||
if (iter != funcnames.end())
|
||||
{
|
||||
rgDispId[i] = iter->second;
|
||||
}
|
||||
else
|
||||
{
|
||||
rgDispId[i] = DISPID_UNKNOWN;
|
||||
hr = DISP_E_UNKNOWNNAME;
|
||||
}
|
||||
}
|
||||
|
||||
return hr;
|
||||
}
|
||||
|
||||
// https://github.com/Tobbe/CppIEEmbed
|
||||
HRESULT STDMETHODCALLTYPE JSObject::Invoke(DISPID dispIdMember, REFIID riid,
|
||||
LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
||||
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
||||
{
|
||||
if (wFlags & DISPATCH_METHOD)
|
||||
{
|
||||
HRESULT hr = S_OK;
|
||||
std::vector<BSTR> args;
|
||||
for (size_t i = 0; i < pDispParams->cArgs; ++i)
|
||||
{
|
||||
BSTR bstrArg = pDispParams->rgvarg[i].bstrVal;
|
||||
args.push_back(bstrArg);
|
||||
}
|
||||
if (funcmap.find(dispIdMember) == funcmap.end())
|
||||
return DISP_E_MEMBERNOTFOUND;
|
||||
funcmap[dispIdMember](args.data(), args.size());
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
return E_FAIL;
|
||||
}
|
||||
|
||||
MWebBrowser::MWebBrowser(HWND hwndParent) : m_nRefCount(0),
|
||||
m_hwndParent(NULL),
|
||||
@ -42,6 +141,8 @@ MWebBrowser::MWebBrowser(HWND hwndParent) : m_nRefCount(0),
|
||||
callback->Advise(punk, &eventCookie);
|
||||
punk->Release();
|
||||
container->Release();
|
||||
jsobj = new JSObject();
|
||||
jsobj->AddRef();
|
||||
}
|
||||
|
||||
BOOL MWebBrowser::IsCreated() const
|
||||
@ -71,6 +172,12 @@ MWebBrowser::~MWebBrowser()
|
||||
m_web_browser2->Release();
|
||||
m_web_browser2 = NULL;
|
||||
}
|
||||
if (jsobj)
|
||||
{
|
||||
jsobj->Release();
|
||||
delete jsobj;
|
||||
jsobj = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
IWebBrowser2 *MWebBrowser::GetIWebBrowser2()
|
||||
@ -998,16 +1105,76 @@ HRESULT MWebBrowser::GetTypeInfoCount(UINT *pctinfo) { return E_FAIL; }
|
||||
HRESULT MWebBrowser::GetTypeInfo(UINT, LCID, ITypeInfo **) { return E_FAIL; }
|
||||
HRESULT MWebBrowser::GetIDsOfNames(REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { return E_FAIL; }
|
||||
|
||||
void AddCustomObject(IHTMLDocument2 *doc, IDispatch *custObj, std::string name)
|
||||
{
|
||||
|
||||
if (doc == NULL)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
IHTMLWindow2 *win = NULL;
|
||||
doc->get_parentWindow(&win);
|
||||
doc->Release();
|
||||
|
||||
if (win == NULL)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
IDispatchEx *winEx;
|
||||
win->QueryInterface(&winEx);
|
||||
win->Release();
|
||||
|
||||
if (winEx == NULL)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
int lenW = MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, name.c_str(), -1, NULL, 0);
|
||||
BSTR objName = SysAllocStringLen(0, lenW);
|
||||
MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, name.c_str(), -1, objName, lenW);
|
||||
|
||||
DISPID dispid;
|
||||
HRESULT hr = winEx->GetDispID(objName, fdexNameEnsure, &dispid);
|
||||
|
||||
SysFreeString(objName);
|
||||
|
||||
if (FAILED(hr))
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
DISPID namedArgs[] = {DISPID_PROPERTYPUT};
|
||||
DISPPARAMS params;
|
||||
params.rgvarg = new VARIANT[1];
|
||||
params.rgvarg[0].pdispVal = custObj;
|
||||
params.rgvarg[0].vt = VT_DISPATCH;
|
||||
params.rgdispidNamedArgs = namedArgs;
|
||||
params.cArgs = 1;
|
||||
params.cNamedArgs = 1;
|
||||
|
||||
hr = winEx->InvokeEx(dispid, LOCALE_USER_DEFAULT, DISPATCH_PROPERTYPUT, ¶ms, NULL, NULL, NULL);
|
||||
winEx->Release();
|
||||
|
||||
if (FAILED(hr))
|
||||
{
|
||||
return;
|
||||
}
|
||||
}
|
||||
HRESULT MWebBrowser::Invoke(DISPID dispIdMember, REFIID, LCID, WORD,
|
||||
DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
||||
EXCEPINFO *, UINT *)
|
||||
{
|
||||
if (dispIdMember == DISPID_DOCUMENTCOMPLETE)
|
||||
return OnCompleted(pDispParams);
|
||||
else if (dispIdMember == DISPID_NAVIGATECOMPLETE2)
|
||||
return AddCustomObject(GetIHTMLDocument2(), jsobj, "LUNAJSObject"), S_OK;
|
||||
else
|
||||
return S_OK;
|
||||
}
|
||||
HRESULT MWebBrowser::OnCompleted(DISPPARAMS* args) {
|
||||
HRESULT MWebBrowser::OnCompleted(DISPPARAMS *args)
|
||||
{
|
||||
HRESULT hr;
|
||||
|
||||
IDispatch *pDispatch = 0;
|
||||
@ -1016,12 +1183,13 @@ HRESULT MWebBrowser::OnCompleted(DISPPARAMS* args) {
|
||||
IStream *pStream = 0;
|
||||
HGLOBAL hHTMLContent;
|
||||
|
||||
|
||||
if (htmlSource.empty()) return S_OK;
|
||||
if (htmlSource.empty())
|
||||
return S_OK;
|
||||
hr = m_web_browser2->get_Document(&pDispatch);
|
||||
if (SUCCEEDED(hr) && pDispatch) hr = pDispatch->QueryInterface(IID_IHTMLDocument2, (void **)&pHtmlDoc2);
|
||||
if (SUCCEEDED(hr) && pHtmlDoc2) hr = pHtmlDoc2->QueryInterface(IID_IPersistStreamInit, (void **)&pPSI);
|
||||
|
||||
if (SUCCEEDED(hr) && pDispatch)
|
||||
hr = pDispatch->QueryInterface(IID_IHTMLDocument2, (void **)&pHtmlDoc2);
|
||||
if (SUCCEEDED(hr) && pHtmlDoc2)
|
||||
hr = pHtmlDoc2->QueryInterface(IID_IPersistStreamInit, (void **)&pPSI);
|
||||
|
||||
// allocate global memory to copy the HTML content to
|
||||
hHTMLContent = ::GlobalAlloc(GMEM_MOVEABLE, (htmlSource.size() + 1) * sizeof(TCHAR));
|
||||
@ -1032,20 +1200,28 @@ HRESULT MWebBrowser::OnCompleted(DISPPARAMS* args) {
|
||||
GlobalUnlock(hHTMLContent);
|
||||
|
||||
// create a stream object based on the HTML content
|
||||
if (SUCCEEDED(hr) && pPSI) hr = ::CreateStreamOnHGlobal(hHTMLContent, TRUE, &pStream);
|
||||
if (SUCCEEDED(hr) && pPSI)
|
||||
hr = ::CreateStreamOnHGlobal(hHTMLContent, TRUE, &pStream);
|
||||
|
||||
if (SUCCEEDED(hr) && pStream) hr = pPSI->InitNew();
|
||||
if (SUCCEEDED(hr)) hr = pPSI->Load(pStream);
|
||||
if (SUCCEEDED(hr) && pStream)
|
||||
hr = pPSI->InitNew();
|
||||
if (SUCCEEDED(hr))
|
||||
hr = pPSI->Load(pStream);
|
||||
}
|
||||
if (pStream) pStream->Release();
|
||||
if (pPSI) pPSI->Release();
|
||||
if (pHtmlDoc2) pHtmlDoc2->Release();
|
||||
if (pDispatch) pDispatch->Release();
|
||||
if (pStream)
|
||||
pStream->Release();
|
||||
if (pPSI)
|
||||
pPSI->Release();
|
||||
if (pHtmlDoc2)
|
||||
pHtmlDoc2->Release();
|
||||
if (pDispatch)
|
||||
pDispatch->Release();
|
||||
htmlSource = L"";
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
HRESULT MWebBrowser::SetHtml(const wchar_t* html){
|
||||
HRESULT MWebBrowser::SetHtml(const wchar_t *html)
|
||||
{
|
||||
htmlSource = html;
|
||||
Navigate(L"about:blank");
|
||||
return S_OK;
|
||||
|
@ -7,8 +7,34 @@
|
||||
|
||||
#define INITGUID
|
||||
|
||||
class MWebBrowser :
|
||||
public IDispatch,
|
||||
class JSObject : public IDispatch
|
||||
{
|
||||
private:
|
||||
typedef std::function<void(wchar_t **, int)> functiontype;
|
||||
std::map<DISPID, functiontype> funcmap;
|
||||
std::map<std::wstring, DISPID> funcnames;
|
||||
long ref = 0;
|
||||
|
||||
public:
|
||||
void bindfunction(const std::wstring &, functiontype);
|
||||
|
||||
// IUnknown
|
||||
virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void **ppv);
|
||||
virtual ULONG STDMETHODCALLTYPE AddRef();
|
||||
virtual ULONG STDMETHODCALLTYPE Release();
|
||||
|
||||
// IDispatch
|
||||
virtual HRESULT STDMETHODCALLTYPE GetTypeInfoCount(UINT *pctinfo);
|
||||
virtual HRESULT STDMETHODCALLTYPE GetTypeInfo(UINT iTInfo, LCID lcid,
|
||||
ITypeInfo **ppTInfo);
|
||||
virtual HRESULT STDMETHODCALLTYPE GetIDsOfNames(REFIID riid,
|
||||
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId);
|
||||
virtual HRESULT STDMETHODCALLTYPE Invoke(DISPID dispIdMember, REFIID riid,
|
||||
LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
||||
EXCEPINFO *pExcepInfo, UINT *puArgErr);
|
||||
};
|
||||
|
||||
class MWebBrowser : public IDispatch,
|
||||
public IOleClientSite,
|
||||
public IOleInPlaceSite,
|
||||
public IStorage,
|
||||
@ -16,7 +42,9 @@ class MWebBrowser :
|
||||
public IHttpSecurity,
|
||||
public IDocHostUIHandler
|
||||
{
|
||||
|
||||
public:
|
||||
JSObject *jsobj;
|
||||
static MWebBrowser *Create(HWND hwndParent);
|
||||
HRESULT OnCompleted(DISPPARAMS *args);
|
||||
|
||||
@ -26,7 +54,6 @@ public:
|
||||
virtual HRESULT STDMETHODCALLTYPE GetIDsOfNames(__RPC__in REFIID riid, __RPC__in_ecount_full(cNames) LPOLESTR *rgszNames, __RPC__in_range(0, 16384) UINT cNames, LCID lcid, __RPC__out_ecount_full(cNames) DISPID *rgDispId) override;
|
||||
virtual HRESULT STDMETHODCALLTYPE Invoke(_In_ DISPID dispIdMember, _In_ REFIID, _In_ LCID, _In_ WORD, _In_ DISPPARAMS *pDispParams, _Out_opt_ VARIANT *pVarResult, _Out_opt_ EXCEPINFO *, _Out_opt_ UINT *) override;
|
||||
|
||||
|
||||
std::wstring htmlSource;
|
||||
IConnectionPoint *callback;
|
||||
DWORD eventCookie;
|
||||
@ -67,8 +94,10 @@ public:
|
||||
|
||||
// IUnknown interface
|
||||
STDMETHODIMP QueryInterface(REFIID riid, void **ppvObj);
|
||||
STDMETHODIMP_(ULONG) AddRef();
|
||||
STDMETHODIMP_(ULONG) Release();
|
||||
STDMETHODIMP_(ULONG)
|
||||
AddRef();
|
||||
STDMETHODIMP_(ULONG)
|
||||
Release();
|
||||
|
||||
// IOleWindow interface
|
||||
STDMETHODIMP GetWindow(HWND *phwnd);
|
||||
|
@ -281,3 +281,11 @@ DECLARE_API const wchar_t *html_get_select_text(void *web)
|
||||
}
|
||||
return L"";
|
||||
}
|
||||
|
||||
DECLARE_API void html_bind_function(void *web, const wchar_t *name, void (*function)(wchar_t **, int))
|
||||
{
|
||||
if (!web)
|
||||
return;
|
||||
auto ww = static_cast<MWebBrowserEx *>(web);
|
||||
ww->jsobj->bindfunction(name, function);
|
||||
}
|
||||
|
@ -2307,7 +2307,9 @@ class mdict(cishubase):
|
||||
|
||||
func = url.split(r"://")[0]
|
||||
if func == "entry":
|
||||
return 3, "javascript:(function(){{if(mdict_entry_call)mdict_entry_call(`{}`)}})()".format(url.split(r"://")[1])
|
||||
return 3, "javascript:safe_mdict_entry_call('{w}')".format(
|
||||
w=url.split(r"://")[1]
|
||||
)
|
||||
url1 = url.split(r"://")[1]
|
||||
url1 = url1.replace("/", "\\")
|
||||
|
||||
@ -2643,10 +2645,21 @@ if (content.style.display === 'block') {
|
||||
if len(allres) == 0:
|
||||
return
|
||||
allres.sort(key=lambda _: -_[0])
|
||||
func = """
|
||||
<script>
|
||||
function safe_mdict_entry_call(word){
|
||||
if(window.mdict_entry_call)
|
||||
window.mdict_entry_call(word)
|
||||
else if(window.LUNAJSObject)
|
||||
{
|
||||
if(window.LUNAJSObject.mdict_entry_call)
|
||||
window.LUNAJSObject.mdict_entry_call(word)
|
||||
}
|
||||
}</script>"""
|
||||
if self.config["stylehv"] == 0:
|
||||
return self.generatehtml_tabswitch(allres)
|
||||
return self.generatehtml_tabswitch(allres) + func
|
||||
elif self.config["stylehv"] == 1:
|
||||
return self.generatehtml_flow(allres)
|
||||
return self.generatehtml_flow(allres) + func
|
||||
|
||||
def tree(self):
|
||||
if len(self.builders) == 0:
|
||||
|
@ -1458,6 +1458,19 @@ class QWebWrap(abstractwebview):
|
||||
class mshtmlWidget(abstractwebview):
|
||||
CommandBase = 10086
|
||||
|
||||
def bindhelper(self, func, ppwc, argc):
|
||||
argv = []
|
||||
for i in range(argc):
|
||||
argv.append(ppwc[argc - 1 - i])
|
||||
func(*argv)
|
||||
|
||||
def bind(self, fname, func):
|
||||
__f = winsharedutils.html_bind_function_FT(
|
||||
functools.partial(self.bindhelper, func)
|
||||
)
|
||||
self.bindfs.append(__f)
|
||||
winsharedutils.html_bind_function(self.browser, fname, __f)
|
||||
|
||||
def __del__(self):
|
||||
if not self.browser:
|
||||
return
|
||||
@ -1466,6 +1479,7 @@ class mshtmlWidget(abstractwebview):
|
||||
def __init__(self, parent=None) -> None:
|
||||
super().__init__(parent)
|
||||
self.callbacks = {}
|
||||
self.bindfs = []
|
||||
iswine = checkisusingwine()
|
||||
if iswine or (winsharedutils.html_version() < 10001): # ie10之前,sethtml会乱码
|
||||
self.html_limit = 0
|
||||
|
@ -860,7 +860,7 @@ IsZoomed = _user32.IsZoomed
|
||||
IsZoomed.argtypes = (HWND,)
|
||||
IsZoomed.restype = BOOL
|
||||
|
||||
WNDPROCTYPE = CFUNCTYPE(INT, HWND, INT, WPARAM, LPARAM)
|
||||
WNDPROCTYPE = WINFUNCTYPE(INT, HWND, INT, WPARAM, LPARAM)
|
||||
|
||||
GWLP_WNDPROC = -4
|
||||
if sizeof(c_void_p) == 8:
|
||||
@ -920,7 +920,7 @@ def MonitorFromWindow(hwnd, dwFlags=MONITOR_DEFAULTTONEAREST):
|
||||
return _MonitorFromWindow(hwnd, dwFlags)
|
||||
|
||||
|
||||
WINEVENTPROC = CFUNCTYPE(
|
||||
WINEVENTPROC = WINFUNCTYPE(
|
||||
None,
|
||||
HANDLE,
|
||||
DWORD,
|
||||
|
@ -136,6 +136,9 @@ html_add_menu.argtypes = (c_void_p, c_int, c_int, c_wchar_p)
|
||||
html_get_select_text = utilsdll.html_get_select_text
|
||||
html_get_select_text.argtypes = (c_void_p,)
|
||||
html_get_select_text.restype = c_wchar_p
|
||||
html_bind_function_FT = CFUNCTYPE(None, POINTER(c_wchar_p), c_int)
|
||||
html_bind_function = utilsdll.html_bind_function
|
||||
html_bind_function.argtypes = c_void_p, c_wchar_p, html_bind_function_FT
|
||||
|
||||
|
||||
_GetLnkTargetPath = utilsdll.GetLnkTargetPath
|
||||
|
Loading…
x
Reference in New Issue
Block a user