Я немного перегружен головой и хотел бы дать несколько советов о том, как это сделать.
По сути, я хочу иметь возможность визуализировать и контролировать silverlight внутри моего приложения C ++.Я хотел бы что-то вроде:
class silverlight_host
{
public:
// Prio 1
silverlight_host(const std::string& filename); // Load a xap file
void draw(void* dest); // Draw with alpha to dest
std::pair<size_t, size_t> get_size(); // Need to know required size of dest
// Prio 2
bool is_dirty() const; // Check for dirty rect since last call to draw
void send_param(const std::string& param); // Send data to silverlight control or call functions. Alternative name maybe, call_function
void set_size(size_t width, size_t height); // Set the size of drawing.
// Prio 3
// Some more nice to have functions, although not as important as those above
double get_desired_frame_rate(); // The desired frame rate which is specified in xap
std::pair<size_t, size_t> get_desired_size(); // The desired size which is specified in xap
void tick(); // Tick a synchronous timeline, disable internal asynchronous timer
void set_param_callback(const std::function<void(const std::string&)>& callback); // Let the xap file call the application
};
Проще сказать, чем сделать.Я нашел следующие статьи Host Silverlight Contorl в C ++ и Связь между C ++ Silverlight Host и приложением Silverlight .Моя проблема с этим заключается в том, что предоставленный код не работает при компиляции в VS2010, и они создают реальное окно вместо элемента управления без окон.Также то, что происходит, не очень хорошо объяснено, и мне немного не хватает знаний COM и ATL.
Я также нашел this , который, кажется, имеет более простой способ реализации xcpcontrolhost, чемстатьи выше.
Я нашел некоторую справочную информацию по msdn .Где ISilverlightViewer кажется довольно интересным для моих нужд.Чтобы разрешить управление без окон, я считаю, что мне, вероятно, нужно реализовать что-то вроде IOleInPlaceSiteWindowless ?.
Тем не менее, я немного над головой и не знаю, с чего начать.Я хотел бы спросить несколько советов относительно того, с чего мне начать, и если у вас есть какие-либо общие советы или опыт работы с чем-то вроде этого?такую реализацию можно сделать независимой от платформы?
EDIT2: я изменил код в "TestProject" из одной из статей выше.Я попытался удалить избыточный код и исправил его так, чтобы он работал на VS2010 (согласно ответу ниже).Вы можете найти его здесь .
EDIT3:
Я попытался реализовать класс XcpControlHost без окон.Я посмотрел на код в CAxHostWindow и попытался воссоздать его.Причина, по которой я не использую CAxHostWindow для создания элемента управления без окон, заключается в том, что он не поддерживает альфа.
Кажется, что он компилируется нормально, однако когда я вызываю DrawControl, я получаю только черную рамку.
XcpContorlHost.h XcpControlHost.cpp
Любые идеи относительно того, что может быть не так?
EDIT4: я берушаг назад.Я использую код из «TestProject». Я хочу изменить CreateXcpControl (HWND hWnd), чтобы иметь возможность принимать hWnd == nullptr (без окон) и использовать OleDraw для рисования элемента управления в памяти.
Итак, я попытался просто обойти вызов «AttachControl» и напрямую вызвать «ActivateXcpControl».И я заменил GetClientRect жестко закодированными значениями.
STDMETHODIMP XcpControlHost::AttachControl(IUnknown* pUnKnown, HWND hWnd)
{
assert(hWnd == nullptr);
ReleaseAll();
// Removed all hWnd related code
return ActivateXcpControl(pUnKnown);
}
HRESULT XcpControlHost::ActivateXcpControl(IUnknown* pUnKnown)
{
// Lots of code
// GetClientRect(&m_rcPos); // Remove this
m_rcPos.top = 0;
m_rcPos.left = 0;
m_rcPos.right = 720;
m_rcPos.bottom = 576;
// Lots of code
}
Я создаю XcpControlHost внутри потока с CoInitialize:
void run()
{
struct co_init
{
co_init(){CoInitialize(nullptr);}
~co_init(){CoUninitialize();}
} co;
if(FAILED(CComObject<XcpControlHost>::CreateInstance(&host_)))
throw std::exception("Failed to create XcpControlHost");
if(FAILED(host_->CreateXcpControl()))
throw std::exception("Failed to create XcpControl");
while(GetMessage(&msg, 0, 0, 0))
{
if(!is_running_)
PostQuitMessage(0);
if(msg.message == WM_USER + 1) // Sent from app
OleDraw(host_->m_pUnKnown, DVASPECT_CONTENT, targetDC, 0);
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
И затем я запускаю обычный цикл обработчика сообщений Windowsс GetMessage, TranslateMessage и DispatchMessage.
Тем не менее, все, что я получаю, это чернота.Что я делаю не так?
Кажется, я получаю E_FAIL из следующего вызова в ActivateXcpControl:
hr = m_spOleObject->DoVerb(OLEIVERB_INPLACEACTIVATE, NULL, spClientSite, 0, NULL, &m_rcPos);