mfc cef<转>
在mfc单文档程序中加入cef:
.在BOOL CtestCEFApp::InitInstance()中初始化cef
HINSTANCE hInst = GetModuleHandle(NULL);
CefMainArgs main_args(hInst); m_cefApp = new ClientApp(); //cef默认启动四个进程,分别是浏览器主进程,渲染进程,GPU进程,插件进程,如果不是主进程则直接退出
//AfxMessageBox(L"CefExecuteProcess");
int exit_code = CefExecuteProcess(main_args, m_cefApp.get(), NULL);
if (exit_code >= )
return exit_code; //设置cef参数
CString szCEFCache;
CString szPath;
INT nLen = GetTempPath( , NULL ) + ;
GetTempPath( nLen, szPath.GetBuffer( nLen ));
szCEFCache.Format( _T("%scache\0\0"), szPath ); CefSettings settings;
//settings.no_sandbox = TRUE;
//settings.multi_threaded_message_loop = FALSE;
CefString(&settings.cache_path) = szCEFCache; //CefSettingsTraits::init( &cSettings);
//cSettings.multi_threaded_message_loop = false; m_bCEFInitialized = CefInitialize(main_args, settings,m_cefApp.get(), NULL); //初始化自定义协议
scheme_test::InitTest();
.创建浏览器窗口
BOOL ClientApp::CreateBrowser(HWND hWnd, CRect rect, LPCTSTR pszURL)
{
CefBrowserSettings settings;
CefWindowInfo info; info.SetAsChild( hWnd, rect );
std::string url = "test://test/html1.html"; return CefBrowserHost::CreateBrowser( info, m_cefHandler.get(), url, settings, NULL );
}
.资源文件加载
namespace scheme_test { // Implementation of the factory for for creating schema handlers.
class ClientSchemeHandlerFactory : public CefSchemeHandlerFactory {
public:
// Return a new scheme handler instance to handle the request.
virtual CefRefPtr<CefResourceHandler> Create(CefRefPtr<CefBrowser> browser,
CefRefPtr<CefFrame> frame,
const CefString& scheme_name,
CefRefPtr<CefRequest> request)
OVERRIDE {
return NULL;
} IMPLEMENT_REFCOUNTING(ClientSchemeHandlerFactory);
}; //此方法在ClientApp的虚函数中被调用
void RegisterCustomSchemes(CefRefPtr<CefSchemeRegistrar> registrar,
std::vector<CefString>& cookiable_schemes) {
registrar->AddCustomScheme("test", true, false, false);//注册一个新的协议test
} //初始化时调用
void InitTest() {
CefRegisterSchemeHandlerFactory("test", "test", new ClientSchemeHandlerFactory());//注册一个新的域名,并且新建一个可以处理此域名请求的类
} } // namespace scheme_test
.两个关键的类
class CWebClient
: public CefClient
, public CefLifeSpanHandler
, public CefRequestHandler
{ public:
CMyWebBrowser m_wndBrowser;
public:
CWebClient(void){}
virtual ~CWebClient(void){} virtual CefRefPtr<CefLifeSpanHandler> GetLifeSpanHandler() { return this; }
virtual CefRefPtr<CefRequestHandler> GetRequestHandler() { return this; } //virtual bool DoClose( CefRefPtr<CefBrowser> browser );
//virtual void OnBeforeClose( CefRefPtr<CefBrowser> browser );
virtual void OnAfterCreated( CefRefPtr<CefBrowser> browser ); //此处处理接收到的message
virtual bool OnProcessMessageReceived( CefRefPtr<CefBrowser> browser, CefProcessId source_process, CefRefPtr<CefProcessMessage> message );
//此处加载本地html资源
virtual CefRefPtr<CefResourceHandler> GetResourceHandler( CefRefPtr<CefBrowser> browser, CefRefPtr<CefFrame> frame, CefRefPtr<CefRequest> request ); private:
bool ParseTestUrl(const std::string& url, std::string* file_name, std::string* mime_type); public: IMPLEMENT_REFCOUNTING(CWebClient);
IMPLEMENT_LOCKING(CWebClient);
};
class ClientApp: public CefApp
,public CefBrowserProcessHandler
,public CefRenderProcessHandler
{
public:
ClientApp(void);
ClientApp(HWND hWnd);
~ClientApp(void); virtual CefRefPtr<CefBrowserProcessHandler> GetBrowserProcessHandler() OVERRIDE { return this; }
virtual CefRefPtr<CefRenderProcessHandler> GetRenderProcessHandler()OVERRIDE { return this; }
//此处处理js和c++调用
virtual void OnContextCreated( CefRefPtr<CefBrowser> browser, CefRefPtr<CefFrame> frame, CefRefPtr<CefV8Context> context ); virtual void OnContextInitialized() OVERRIDE; //此处要<span style="font-family: Arial, Helvetica, sans-serif;">注册一个新的协议test</span>
virtual void OnRegisterCustomSchemes( CefRefPtr<CefSchemeRegistrar> registrar ); BOOL CreateBrowser(HWND hWnd, CRect rect, LPCTSTR pszURL); public:
CefRefPtr<CWebClient> m_cefHandler; private:
IMPLEMENT_REFCOUNTING(ClientApp); };
.c++和js相互调用
第一种:
class CefV8ExtensionHandler : public CefV8Handler
{
public:
CefV8ExtensionHandler(){}
~CefV8ExtensionHandler(){} virtual bool Execute( const CefString& name, CefRefPtr<CefV8Value> object, const CefV8ValueList& arguments, CefRefPtr<CefV8Value>& retval, CefString& exception ); IMPLEMENT_REFCOUNTING(CefV8ExtensionHandler);
};
void ClientApp::OnContextCreated( CefRefPtr<CefBrowser> browser, CefRefPtr<CefFrame> frame, CefRefPtr<CefV8Context> context )
{
CefRefPtr<CefV8Value> obj = context->GetGlobal(); CefRefPtr<CefV8Handler> handler = new CefV8ExtensionHandler(); obj->SetValue("test", CefV8Value::CreateFunction("test", handler), V8_PROPERTY_ATTRIBUTE_READONLY); }
bool CefV8ExtensionHandler::Execute( const CefString& name, CefRefPtr<CefV8Value> object, const CefV8ValueList& arguments, CefRefPtr<CefV8Value>& retval, CefString& exception )
{
if (name == "test")
{
std::string arg1 = arguments[]->GetStringValue(); //MessageBoxA(NULL, arg1.c_str(), NULL, NULL); retval = CefV8Value::CreateString(arg1); CefRefPtr<CefBrowser> browser = CefV8Context::GetCurrentContext()->GetBrowser();
ASSERT(browser.get()); CefRefPtr<CefProcessMessage> message = CefProcessMessage::Create("test"); browser->SendProcessMessage(PID_BROWSER, message); return true;
} return false;
}
第二种:
std::string app_code =
"var app;"
"if (!app)"
" app = {};"
"(function() {"
" app.sendMessage = function(name, arguments) {"
" native function sendMessage();"
" return sendMessage(name, arguments);"
" };"
" app.setMessageCallback = function(name, callback) {"
" native function setMessageCallback();"
" return setMessageCallback(name, callback);"
" };"
" app.removeMessageCallback = function(name) {"
" native function removeMessageCallback();"
" return removeMessageCallback(name);"
" };"
"})();";
CefRegisterExtension("v8/app", app_code,
new ClientAppExtensionHandler(this));
.自定义标题栏需要子类化浏览器窗口方能接收到消息
class CMyWebBrowser : public CWnd
{
DECLARE_DYNCREATE(CMyWebBrowser) public:
CMyWebBrowser(void);
virtual ~CMyWebBrowser(void); protected:
DECLARE_MESSAGE_MAP() public:
afx_msg LRESULT OnNcHitTest(CPoint point);
};
LRESULT CMyWebBrowser::OnNcHitTest(CPoint point)
{
CRect rc;
GetClientRect(&rc);
rc.top = rc.top;
rc.left = rc.left;
rc.right = rc.right;
rc.bottom = ;
ClientToScreen(&rc); CPoint screenpoint(point);
ScreenToClient(&screenpoint); if (rc.PtInRect(point))
{
AfxGetApp()->GetMainWnd()->PostMessage(WM_NCLBUTTONDOWN, HTCAPTION, MAKELPARAM(screenpoint.x, screenpoint.y)); return ;
}
else
return CWnd::OnNcHitTest(point);
}
void CWebClient::OnAfterCreated( CefRefPtr<CefBrowser> browser )
{
HWND hWnd = browser->GetHost()->GetWindowHandle();
HWND hChildWnd = ::GetNextWindow(hWnd, GW_CHILD );
m_wndBrowser.SubclassWindow(hChildWnd); CefLifeSpanHandler::OnAfterCreated(browser);
}
.消息循环
int CtestCEFApp::ExitInstance()
{
if( m_bCEFInitialized )
{
m_bCEFInitialized = false; m_cefApp = NULL; CefShutdown();
} return CWinAppEx::ExitInstance();
} BOOL CtestCEFApp::PumpMessage()
{
if( m_bCEFInitialized )
CefDoMessageLoopWork(); return CWinAppEx::PumpMessage();
} --------------------- 本文来自 lls2012 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/lls2012/article/details/47976101?utm_source=copy
https://blog.csdn.net/lls2012/article/details/47976101
在mfc单文档程序中加入cef: 1.在BOOL CtestCEFApp::InitInstance()中初始化cef HINSTANCE hInst = GetModuleHandle(NULL); CefMainArgs main_args(hInst); m_cefApp = new ClientApp(); //cef默认启动四个进程,分别是浏览器主进程,渲染进程,GPU进程,插件进程,如果不是主进程则直接退出 //AfxMessageBox(L"CefExecuteProcess"); int exit_code = CefExecuteProcess(main_args, m_cefApp.get(), NULL); if (exit_code >= 0) return exit_code; //设置cef参数 CString szCEFCache; CString szPath; INT nLen = GetTempPath( 0, NULL ) + 1; GetTempPath( nLen, szPath.GetBuffer( nLen )); szCEFCache.Format( _T("%scache\0\0"), szPath ); CefSettings settings; //settings.no_sandbox = TRUE; //settings.multi_threaded_message_loop = FALSE; CefString(&settings.cache_path) = szCEFCache; //CefSettingsTraits::init( &cSettings); //cSettings.multi_threaded_message_loop = false; m_bCEFInitialized = CefInitialize(main_args, settings,m_cefApp.get(), NULL); //初始化自定义协议 scheme_test::InitTest(); 2.创建浏览器窗口 BOOL ClientApp::CreateBrowser(HWND hWnd, CRect rect, LPCTSTR pszURL) { CefBrowserSettings settings; CefWindowInfo info; info.SetAsChild( hWnd, rect ); std::string url = "test://test/html1.html"; return CefBrowserHost::CreateBrowser( info, m_cefHandler.get(), url, settings, NULL ); } 3.资源文件加载 namespace scheme_test { // Implementation of the factory for for creating schema handlers. class ClientSchemeHandlerFactory : public CefSchemeHandlerFactory { public: // Return a new scheme handler instance to handle the request. virtual CefRefPtr Create(CefRefPtr browser, CefRefPtr frame, const CefString& scheme_name, CefRefPtr request) OVERRIDE { return NULL; } IMPLEMENT_REFCOUNTING(ClientSchemeHandlerFactory); }; //此方法在ClientApp的虚函数中被调用 void RegisterCustomSchemes(CefRefPtr registrar, std::vector& cookiable_schemes) { registrar->AddCustomScheme("test", true, false, false);//注册一个新的协议test } //初始化时调用 void InitTest() { CefRegisterSchemeHandlerFactory("test", "test", new ClientSchemeHandlerFactory());//注册一个新的域名,并且新建一个可以处理此域名请求的类 } } // namespace scheme_test 4.两个关键的类 class CWebClient : public CefClient , public CefLifeSpanHandler , public CefRequestHandler { public: CMyWebBrowser m_wndBrowser; public: CWebClient(void){} virtual ~CWebClient(void){} virtual CefRefPtr GetLifeSpanHandler() { return this; } virtual CefRefPtr GetRequestHandler() { return this; } //virtual bool DoClose( CefRefPtr browser ); //virtual void OnBeforeClose( CefRefPtr browser ); virtual void OnAfterCreated( CefRefPtr browser ); //此处处理接收到的message virtual bool OnProcessMessageReceived( CefRefPtr browser, CefProcessId source_process, CefRefPtr message ); //此处加载本地html资源 virtual CefRefPtr GetResourceHandler( CefRefPtr browser, CefRefPtr frame, CefRefPtr request ); private: bool ParseTestUrl(const std::string& url, std::string* file_name, std::string* mime_type); public: IMPLEMENT_REFCOUNTING(CWebClient); IMPLEMENT_LOCKING(CWebClient); }; class ClientApp: public CefApp ,public CefBrowserProcessHandler ,public CefRenderProcessHandler { public: ClientApp(void); ClientApp(HWND hWnd); ~ClientApp(void); virtual CefRefPtr GetBrowserProcessHandler() OVERRIDE { return this; } virtual CefRefPtr GetRenderProcessHandler()OVERRIDE { return this; } //此处处理js和c++调用 virtual void OnContextCreated( CefRefPtr browser, CefRefPtr frame, CefRefPtr context ); virtual void OnContextInitialized() OVERRIDE; //此处要注册一个新的协议test virtual void OnRegisterCustomSchemes( CefRefPtr registrar ); BOOL CreateBrowser(HWND hWnd, CRect rect, LPCTSTR pszURL); public: CefRefPtr m_cefHandler; private: IMPLEMENT_REFCOUNTING(ClientApp); }; 5.c++和js相互调用 第一种: class CefV8ExtensionHandler : public CefV8Handler { public: CefV8ExtensionHandler(){} ~CefV8ExtensionHandler(){} virtual bool Execute( const CefString& name, CefRefPtr object, const CefV8ValueList& arguments, CefRefPtr& retval, CefString& exception ); IMPLEMENT_REFCOUNTING(CefV8ExtensionHandler); }; void ClientApp::OnContextCreated( CefRefPtr browser, CefRefPtr frame, CefRefPtr context ) { CefRefPtr obj = context->GetGlobal(); CefRefPtr handler = new CefV8ExtensionHandler(); obj->SetValue("test", CefV8Value::CreateFunction("test", handler), V8_PROPERTY_ATTRIBUTE_READONLY); } bool CefV8ExtensionHandler::Execute( const CefString& name, CefRefPtr object, const CefV8ValueList& arguments, CefRefPtr& retval, CefString& exception ) { if (name == "test") { std::string arg1 = arguments[0]->GetStringValue(); //MessageBoxA(NULL, arg1.c_str(), NULL, NULL); retval = CefV8Value::CreateString(arg1); CefRefPtr browser = CefV8Context::GetCurrentContext()->GetBrowser(); ASSERT(browser.get()); CefRefPtr message = CefProcessMessage::Create("test"); browser->SendProcessMessage(PID_BROWSER, message); return true; } return false; } 第二种: std::string app_code = "var app;" "if (!app)" " app = {};" "(function() {" " app.sendMessage = function(name, arguments) {" " native function sendMessage();" " return sendMessage(name, arguments);" " };" " app.setMessageCallback = function(name, callback) {" " native function setMessageCallback();" " return setMessageCallback(name, callback);" " };" " app.removeMessageCallback = function(name) {" " native function removeMessageCallback();" " return removeMessageCallback(name);" " };" "})();"; CefRegisterExtension("v8/app", app_code, new ClientAppExtensionHandler(this)); 6.自定义标题栏需要子类化浏览器窗口方能接收到消息 class CMyWebBrowser : public CWnd { DECLARE_DYNCREATE(CMyWebBrowser) public: CMyWebBrowser(void); virtual ~CMyWebBrowser(void); protected: DECLARE_MESSAGE_MAP() public: afx_msg LRESULT OnNcHitTest(CPoint point); }; LRESULT CMyWebBrowser::OnNcHitTest(CPoint point) { CRect rc; GetClientRect(&rc); rc.top = rc.top; rc.left = rc.left; rc.right = rc.right; rc.bottom = 20; ClientToScreen(&rc); CPoint screenpoint(point); ScreenToClient(&screenpoint); if (rc.PtInRect(point)) { AfxGetApp()->GetMainWnd()->PostMessage(WM_NCLBUTTONDOWN, HTCAPTION, MAKELPARAM(screenpoint.x, screenpoint.y)); return 0; } else return CWnd::OnNcHitTest(point); } void CWebClient::OnAfterCreated( CefRefPtr browser ) { HWND hWnd = browser->GetHost()->GetWindowHandle(); HWND hChildWnd = ::GetNextWindow(hWnd, GW_CHILD ); m_wndBrowser.SubclassWindow(hChildWnd); CefLifeSpanHandler::OnAfterCreated(browser); } 7.消息循环 int CtestCEFApp::ExitInstance() { if( m_bCEFInitialized ) { m_bCEFInitialized = false; m_cefApp = NULL; CefShutdown(); } return CWinAppEx::ExitInstance(); } BOOL CtestCEFApp::PumpMessage() { if( m_bCEFInitialized ) CefDoMessageLoopWork(); return CWinAppEx::PumpMessage(); } --------------------- 本文来自 lls2012 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/lls2012/article/details/47976101?utm_source=copy
mfc cef<转>的更多相关文章
- 转:CEF嵌入到单文档mfc
1.下载: http://www.magpcss.net/cef_downloads/下载cef binary 1.1364.1123 windows.zip(可能要FQ,百度goagent教程,最好 ...
- mfc封装cef浏览器 关闭整个窗口程序得时候又重启mfc 应用的程序
最近使用mfc 做了一个cef得浏览器 多标签得.当使用这个封装得浏览器一段时间之后(超过1分钟2分钟) 当关闭封装得浏览器整个窗体 x得时候,整个窗体又重新弹了出来. 大概现象就是一个exe程序你杀 ...
- MFC调用CEF实现单页面单COOKIE管理《转》
cookie简单介绍 cookie存储了网站的一些很重要的信息,如用户身份信息.常用设置.设置地理位置等等各种信息.使用cef访问网站时,如果设置了CefSettings.cache_path参数,则 ...
- CEF源码编译和生产库的使用
CEF版本是Branch 2171 开发环境是VS2012 查看一下libcef_dll_wrapper工程属性,确定Code Generation 选择MTD(Debug) 或者MT(Release ...
- CEF使用的几个注意点
CEF为chrome浏览器的切入其他浏览器中的轻量级框架. 开发的客户端的时候,这是作为界面显示的首先,可以增强客户的易变性,可塑性. 在开发的过程中(侧重于C,C++解决),遇到的几个问题,以及自己 ...
- CEF 相关资料
理解WebKit和Chromium: Content API和CEF3 http://blog.csdn.net/milado_nju/article/details/7455373 如何将Chrom ...
- 【转】MFC内嵌cef3浏览器内核
一.cef3内核的下载 可以从http://opensource.spotify.com/cefbuilds/index.html下载,注意:很多版本编译都可以通过 但是运行的时候会崩溃,以cef_b ...
- Duilib嵌入CEF出现窗口显示不正常
参考资料:https://www.aliyun.com/zixun/wenji/1247250.html 转载:https://www.cnblogs.com/gongxijun/p/4857977. ...
- 笨重的mfc还在基于系统控件,熟练的mfc工程师还比不过学习Qt一个月的学生开发效率高(比较精彩,韦易笑)
作者:韦易笑链接:https://www.zhihu.com/question/29636221/answer/45102191来源:知乎著作权归作者所有,转载请联系作者获得授权. 更新:擦,本来只有 ...
随机推荐
- 让HTMLrunner 报告的子列表都 默认展示出来的 方法(方便发送邮件时可以方便查看)
1.找到生成的测试报告,获取到all元素 2.在HTMLrunner源码,</script> 标签上 加入一个函数 #让所有列表都展示出来window.onload = function ...
- acl的基本知识点
#ACL acl number 3001 rule 1 deny udp destination-port eq 445 rule 2 deny tcp destination-por ...
- Zabbix 告警内容配置
#配置媒介告警类型 #----------------------------------------------------------------------------------------- ...
- HDOJ 2013 蟠桃记
#include<iostream> using namespace std; int main() { int n; while (cin >> n) { ; ;i < ...
- Jmeter(三十四)Jmeter-Question之“Cookie获取”
2018.4.27 还在做性能测试的过程中,唉,只能说坑很多. 无明确需求.无人手协调等问题,什么都需要自己去挖掘. 本次测试的工具选型依然是Jmeter,真实场景中遇到了这么个问题.可能解决办法有点 ...
- golang 常量的用法
1.Golang常量的用法 //常量的用法 var num int num =9 //1.常量声明的时候必须赋值 const tax int = 0 //2.常量值是无法修改的 //tax = 10 ...
- SCCM2012 R2实战系列之七:软件分发(exe)
在上一章节中,我们完成了SCCM 2012客户端代理软件的安装,现在就可以为客户端来部署应用程序了. SCCM2012增加了应用程序分发,同时保留了SCCM 2007里的包分发.应用程序分发可以直接对 ...
- Processing Binary Protocols with Client-Side JavaScript
http://blog.mgechev.com/2015/02/06/parsing-binary-protocol-data-javascript-typedarrays-blobs/ https: ...
- [ERR] Node is not empty. Either the node already knows other nodes (check with C
[root@node00 src]# ./redis-trib.rb add-node --slave --master-id4f6424e47a2275d2b7696bfbf8588e8c4c3a5 ...
- 关于easyUI异步获取数据格式问题
后台向easyUI一般写String类型的数据 成功之后的回调函数中:应将数据转为json格式 $.parsonJSON success:function(result){ ...