1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
|
#include "stdafx.h"
class CREOleCallback : public IRichEditOleCallback
{
private:
unsigned refCount = 1;
IStorage *pictStg = nullptr;
int nextStgId = 0;
public:
CREOleCallback() {}
STDMETHODIMP QueryInterface(REFIID riid, LPVOID *ppvObj) override
{
if (IsEqualIID(riid, IID_IRichEditOleCallback)) {
*ppvObj = this;
AddRef();
return S_OK;
}
*ppvObj = nullptr;
return E_NOINTERFACE;
}
ULONG STDMETHODCALLTYPE AddRef() override
{
if (refCount == 0) {
if (S_OK != StgCreateDocfile(nullptr, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_CREATE | STGM_DELETEONRELEASE, 0, &this->pictStg))
pictStg = nullptr;
nextStgId = 0;
}
return ++refCount;
}
ULONG STDMETHODCALLTYPE Release() override
{
if (--refCount == 0) {
if (pictStg)
pictStg->Release();
}
return refCount;
}
STDMETHODIMP ContextSensitiveHelp(BOOL) override
{
return S_OK;
}
STDMETHODIMP DeleteObject(LPOLEOBJECT) override
{
return S_OK;
}
STDMETHODIMP GetClipboardData(CHARRANGE *, DWORD, LPDATAOBJECT *) override
{
return E_NOTIMPL;
}
STDMETHODIMP GetContextMenu(uint16_t, LPOLEOBJECT, CHARRANGE *, HMENU *) override
{
return E_INVALIDARG;
}
STDMETHODIMP GetDragDropEffect(BOOL, DWORD, LPDWORD) override
{
return S_OK;
}
STDMETHODIMP GetInPlaceContext(LPOLEINPLACEFRAME *, LPOLEINPLACEUIWINDOW *, LPOLEINPLACEFRAMEINFO) override
{
return E_INVALIDARG;
}
STDMETHODIMP GetNewStorage(LPSTORAGE *lplpstg) override
{
wchar_t sztName[64];
mir_snwprintf(sztName, L"s%u", this->nextStgId);
if (this->pictStg == nullptr)
return STG_E_MEDIUMFULL;
return this->pictStg->CreateStorage(sztName, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_CREATE, 0, 0, lplpstg);
}
STDMETHODIMP QueryAcceptData(LPDATAOBJECT, CLIPFORMAT *, DWORD, BOOL, HGLOBAL) override
{
return S_OK;
}
STDMETHODIMP QueryInsertObject(LPCLSID, LPSTORAGE, LONG) override
{
return S_OK;
}
STDMETHODIMP ShowContainerUI(BOOL) override
{
return S_OK;
}
};
static CREOleCallback reOleCallback;
void InitRichEdit(ITextServices *ts)
{
LRESULT lResult;
ts->TxSendMessage(EM_SETOLECALLBACK, 0, (LPARAM)&reOleCallback, &lResult);
}
/////////////////////////////////////////////////////////////////////////////////////////
static ATOM winClass = 0;
static LRESULT CALLBACK RichEditProxyWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
ITextServices *ts = (ITextServices *)GetWindowLongPtr(hwnd, GWLP_USERDATA);
if (ts && (msg != WM_DESTROY)) {
LRESULT lResult;
ts->TxSendMessage(msg, wParam, lParam, &lResult);
return lResult;
}
return 1;
}
HWND CreateProxyWindow(ITextServices *ts)
{
if (winClass == 0) {
WNDCLASSEX wcl = {};
wcl.cbSize = sizeof(wcl);
wcl.lpfnWndProc = RichEditProxyWndProc;
wcl.style = CS_GLOBALCLASS;
wcl.hInstance = g_hInst;
wcl.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcl.hbrBackground = (HBRUSH)GetStockObject(LTGRAY_BRUSH);
wcl.lpszClassName = L"NBRichEditProxyWndClass";
winClass = RegisterClassExW(&wcl);
}
HWND hwnd = CreateWindow(L"NBRichEditProxyWndClass", L"", 0, 0, 0, 0, 0, nullptr, nullptr, g_hInst, nullptr);
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)ts);
return hwnd;
}
void DestroyProxyWindow()
{
if (winClass != 0)
UnregisterClassW(L"NBRichEditProxyWndClass", g_hInst);
}
|