CEF = Chromium Embedded Framework
This project shall be considered as proof of concept for this final project the code is not proper, will not be maintained and not all mouse/keyboard callbacks are plugged!
For the GitHub project named Stigmee using Godot Engine, and particularly for its CEF native module (under GPL license, but also forked as MIT license https://github.com/Lecrapouille/gdcef), I needed to make my hand on something simpler and therefore prototype (proof of concept) and modify the C++ CEF example named cefsimple to replace the X11 display by either SDL2 or OpenGL Core (glew, glfw3). I tried the following GitHub repos but without success:
- SDL2: https://github.com/gotnospirit/cef3-sdl2
- SDL2: https://github.com/jamethy/sdl2-cef
- OpenGL Core: https://github.com/if1live/cef-gl-example
- OpenGL Core: https://github.com/andmcgregor/cefgui
They are outdated (more than > 4 years), the CEF API changed, and when I compiled they crashed by forking indefinitely the application until my Linux fell down. So they are not safe to use! When starting, CEF will create several forks and needs to access the command line (main(int argc, char* argv[])
) to give commands to the child process. From memory, issues of these repos come from the fact that the CEF API has evolved, and the command line access has been broken and as a consequence, CEF is forking forever. In this repo is the updated version of these repos. They are inside the following folders:
- cefsimple_sdl: using SDL2.
- cefsimple_opengl: using OpenGL Core (>= 3.3).
- cefsimple_separate: with two separate processes. Two processes are needed when CEF cannot directly access the
main(int argc, char* argv[])
function of your application. You have to know that CEF modifies the content of yourargv
and this may mess up your application if it also parses the command line (you can back it up, meaning using astd::vector
to back upargv
and after CEF init to restore values inargv
).
What is "two separated processes" exactly? Just an extra fork: the main process forks itself and calls the secondary process, which can fully access its own main(int argc, char* argv[])
. The main constraint is the path of the secondary process shall be canonic (and this is a pain to get the real path). Do we really require two separate processes? Yes for applications that access to main(int argc, char* argv[])
since CEF also uses it or for applications such as game engines (Godot, Unreal, Unity ...) where you cannot modify their main(int argc, char* argv[])
code source to add your CEF code. See CEF:
- for Unreal Engine: https://github.com/oivio/BLUI calling https://github.com/ashea-code/BluBrowser.
- for Godot: https://github.com/Lecrapouille/gdcef/tree/master/addons/gdcef/gdcef calling https://github.com/Lecrapouille/gdcef/tree/master/addons/gdcef/subprocess
Fig 1 - The OpenGL version. Click on the image to see the YouTube video of the demo.
In this screenshot, you can see two browsers: one running YouTube, and the second running a PDF. The projection of the mouse to the PDF plan has not been made. You can find here how to make it.
Note 1: The keyboard/mouse event conversion from SDL2/glfw3 to CEF has to be fixed. Update: I have a functional keyboard for Godot/CEF but I have not yet patched this current repo for SDL and glfw3 (request me if you needed it urgently). So you can look at this code.
Note 2: I find the system not very reactive compared to the official cefsimple example.
While this repo shall be considered as a proof of concept for https://github.com/Lecrapouille/gdcef, pull requests and help are welcome to improve these examples :) You can achieve the API by adding more methods such as previous/next page, reload, and zoom ... See https://github.com/oivio/BLUI
Tested with CEF 116.
- Linux: It's working on my Linux 64-bit Debian 11.
- Windows: version can work quite easily but I did not translate the ./install.sh into Python script, but you can adapt this one on the version of this project for Godot here.
- MacOS X: is more complex and is not working yet: #4
- Note: All these codes are quick and dirty since I needed to check how to make CEF run with OpenGL or SDL2 before switching to Godot Engine.
cefsimple_sdl
is a single main file, while cefsimple_opengl has been split into several files.cefsimple_sdl
is using more handlers such asCefLifeSpanHandler
. cefsimple_opengl does not.cefsimple_opengl
is using many viewports. cefsimple_sdl not. Viewport allows you to place the web page where you want on the screen (in the screenshot, you can see two browsers: one running YouTube, the second running a PDF).CefExecuteProcess
shall be called before OpenGL/SDL2 context and using the command line of your application (int argc, char* argv[]
). If not, your system can freeze. Here is the function documentation (caf_app.h
file):
This function should be called from the application entry point function to
execute a secondary process. It can be used to run secondary processes from
the browser client executable (default behavior) or from a separate
executable specified by the CefSettings.browser_subprocess_path value. If
called for the browser process (identified by no "type" command-line value)
it will return immediately with a value of -1. If called for a recognized
secondary process it will block until the process should exit and then return
the process exit code. The |application| parameter may be empty. The
|windows_sandbox_info| parameter is only used on Windows and may be NULL (see
cef_sandbox_win.h for details).
- In
cefsimple_sdl
andcefsimple_opengl
, both programs can access themain(int argc, char* argv[])
function and therefore access the command line. CEF; when forking; modifies the command lines to give behaviors to the forked child. Sometimes, accessing the main function is not possible (or when your application is also using the command line, while possible to save and restore it before and after calling CEF). Therefore, you shall launch a second process that can access its main function and therefore call chromium. As a consequence,cefsimple_separate
has been added when you cannot modify the main application to access theint main(int argc, char* argv[])
function.
Run the bash script for Linux:
./install.sh
Note: If you want to change the CEF version. Take the desired version at https://cef-builds.spotifycdn.com/index.html You will see something similar:
09/01/2023 - 116.0.19+gc6a20bc+chromium-116.0.5845.141 / Chromium 116.0.5845.141
Now, in the install.sh script, search and replace the content of this variable: CEF_VERSION=
with the desired CEF version:
CEF_VERSION="116.0.19+gc6a20bc+chromium-116.0.5845.141"
The install.sh script will:
- Download CEF inside the
thirdparty/
folder. - Use the CEF CMake to compile libraries and assets,
- Copy them inside the
build/
folder, - Compile my applications into the
build/
folder.
Once done with success, you have to go inside the build/
folder and run one of the following applications:
./secondary_process
./primary_process
./cefsimple_opengl
./cefsimple_sdl
Note: A Python version of the script can be used and adapted. This will allow us to use it for Windows. Here.
To compile my applications, some libs and packages are needed: we have to compile them from CEF's example named cefsimple
(OffScreenCEF/thirdparty/cef_binary/tests/cefsimple/ once downloaded). These libs and assets shall be in the same folder as the application binary (some internal configurations can change the path of some of them, but not all of them). We beware, some lib can be > 1 Gigabyte (depending on the operating system).
Here is what the script is doing, but we will explain it manually.
Firstly, let's name some folders. This will shorten the code in this document. Adapt the CEF version to your operating system with the desired version according to https://cef-builds.spotifycdn.com/index.html
TMP=/tmp
CEF_LINK=https://cef-builds.spotifycdn.com/cef_binary_96.0.14%2Bg28ba5c8%2Bchromium-96.0.4664.55_linux64.tar.bz2
CEF=$TMP/cef_binary_96.0.14+g28ba5c8+chromium-96.0.4664.55_linux64
CEF_TARBALL=$CEF.tar.bz2
CEFSIMPLE_SDL=$CEF/tests/cefsimple_sdl
CEFSIMPLE_GL=$CEF/tests/cefsimple_opengl
Download and decompress it inside a temporary folder:
wget -c $CEF_LINK -O- | tar -xj
Git clone this repo inside the CEF folder:
cd $CEF
git clone https://github.com/Lecrapouille/OffScreenCEF.git --depth=1
Compile CEF. This takes some minutes (not hours that would be needed if you had to compile CEF from sources):
mkdir $CEF/build
cd $CEF/build
cmake -DCMAKE_BUILD_TYPE=Debug ..
make -j$(nproc)
$(nproc)
is your number of CPU cores. You can use -DCMAKE_BUILD_TYPE=Release
instead but just avoid mixing debug and release symbols. Now we have to copy needed elements for compiling cefsimple_sdl
and cefsimple_opengl
and making them running:
cp -v $CEF/Debug/libcef.so $CEF/build/libcef_dll_wrapper/libcef_dll_wrapper.a $CEFSIMPLE_SDL
(cd $CEF/build/tests/cefsimple/Debug/
cp -v icudtl.dat resources.pak chrome_100_percent.pak chrome_200_percent.pak v8_context_snapshot.bin $CEFSIMPLE_SDL
)
mkdir -p $CEFSIMPLE_SDL/locales
cp -v $CEF/build/tests/cefsimple/Debug/locales/en-US.pak $CEFSIMPLE_SDL/locales
Do the same thing for $CEFSIMPLE_GL
. See the screenshot for needed assets (note this screenshot is old: in the latest version of
this repo all assets and libs are no longer inside the code source but inside the build/
folder with the binary of applications):
Fig 2 - Needed assets from CEF.
CEF needs all its assets next to it. Some paths can be configured. See OffScreenCEF/thirdparty/cef_binary/include/internal/cef_types.h for more information (once CEF has been downloaded by the install.sh script).
The documentation of CEF is not really beginner-friendly: https://bitbucket.org/chromiumembedded/cef/wiki/GeneralUsage.md
You will have access to int main(int argc, char * argv[])
you will have to deal with two separate processes.
If you can access to int main(int argc, char * argv[])
:
int main(int argc, char * argv[])
{
// Start CEF by forking this current application. Access to argc, argv is mandatory
// to allow passing information to forked CEF processes.
CefMainArgs args(argc, argv);
int result = CefExecuteProcess(args, nullptr, nullptr);
if (result >= 0)
{
// The forked process has ended, so exit this application
exit(result);
}
else if (result == -1)
{
// If called for the browser process (identified by no "type" command-line value)
// it will return immediately with a value of -1
}
// Configure CEF see cef_binary/include/internal/cef_types.h for more information
CefSettings settings;
settings.windowless_rendering_enabled = true;
if (!CefInitialize(args, settings, nullptr, nullptr))
{
return EXIT_FAILURE;
}
// CEF started with success.
// Code for OpenGL/SDL2 init and runtime loop comes here after this line.
...
// End of your application
return EXIT_SUCCESS;
}
For rendering the web document, check out the code with:
class RenderHandler: public CefRenderHandler { ...
Two important methods to override from CefRenderHandler
:
- The
GetViewRect
will get the region of your texture/windows and inform CEF. For a full page:
virtual void GetViewRect(CefRefPtr<CefBrowser> browser, CefRect& rect) override
{
rect = CefRect(0, 0, m_width, m_height);
}
- When CEF has done some content to make you render:
virtual void OnPaint(CefRefPtr<CefBrowser> browser, PaintElementType type,
const RectList &dirtyRects, const void* buffer,
int w, int h) override
The buffer
contains the image of the whole web page in the form BGRA8 (beware this is Blue-Green-Red-Alpha not Red-Green-Blue and 1 byte by chanel!).
The buffer size is w, h:
int const COLOR_CHANELS = 4;
int const SIZEOF_COLOR = COLOR_CHANELS * sizeof(char);
int const TEXTURE_SIZE = SIZEOF_COLOR * width * height;
Color conversion BGRA8 -> RGBA8: swap B and R channels can be made with:
for (int i = 0; i < TEXTURE_SIZE; i += COLOR_CHANELS)
{
std::swap(w[i], w[i + 2]);
}
The dirtyRects
contains a list of dirty regions to refresh. This avoids repainting the entire texture.
In a general way, you can derive some CEF classes to override some of their methods to react to events such as a browser is created, closed, or page loaded ...
class BrowserClient: public CefRenderHandler,
public CefLoadHandler,
public CefClient
{
public:
BrowserClient() { ... }
private: // CefClient::CefBaseRefCounted interfaces
// ---------------------------------------------------------------------
//! \brief CEF reference counting
// ---------------------------------------------------------------------
IMPLEMENT_REFCOUNTING(BrowserClient);
private: // CefClient interfaces
// ---------------------------------------------------------------------
//! \brief Return the handler for off-screen rendering events.
// ---------------------------------------------------------------------
virtual CefRefPtr<CefRenderHandler> GetRenderHandler() override
{
return this;
}
// ---------------------------------------------------------------------
//! \brief Return the handler for browser load status events.
// ---------------------------------------------------------------------
virtual CefRefPtr<CefLoadHandler> GetLoadHandler() override
{
return this;
}
private: // CefRenderHandler interfaces
// ---------------------------------------------------------------------
//! \brief Get the viewport.
// ---------------------------------------------------------------------
virtual void GetViewRect(CefRefPtr<CefBrowser> browser, CefRect& rect) override
{
...
}
// ---------------------------------------------------------------------
//! \brief Called when an element should be painted. Pixel values passed
//! to this method are scaled relative to view coordinates based on the
//! value of CefScreenInfo.device_scale_factor returned from
//! GetScreenInfo. |type| indicates whether the element is the view or
//! the popup widget. |buffer| contains the pixel data for the whole
//! image. |dirtyRects| contains the set of rectangles in pixel
//! coordinates that need to be repainted. |buffer| will be
//! |width|*|height|*4 bytes in size and represents a BGRA image with an
//! upper-left origin. This method is only called when
//! CefWindowInfo::shared_texture_enabled is set to false.
// ---------------------------------------------------------------------
virtual void OnPaint(CefRefPtr<CefBrowser> browser,
CefRenderHandler::PaintElementType type,
const CefRenderHandler::RectList& dirtyRects,
const void* buffer, int width, int height) override
{
...
}
private: // CefLoadHandler interfaces
// ---------------------------------------------------------------------
//! \brief Called when the browser is done loading a frame. The |frame|
//! value will never be empty -- call the IsMain() method to check if
//! this frame is the main frame. Multiple frames may be loading at the
//! same time. Sub-frames may start or continue loading after the main
//! frame load has ended. This method will not be called for the same page
//! navigations (fragments, history state, etc.) or for navigations that
//! fail or are canceled before commit. For notification of overall
//! browser load status use OnLoadingStateChange instead.
// ---------------------------------------------------------------------
virtual void OnLoadEnd(CefRefPtr<CefBrowser> browser,
CefRefPtr<CefFrame> frame,
int httpStatusCode) override
{
...
}
};
The way is always the same: to override methods from CefXXXHandler
you have first to
override methods from CefClient
:
virtual CefRefPtr<CefXXXHandler> GetXXXHandler() override
{
return this;
}
(instead of the default method returning nullptr
) meaning the handler is now used.
Since CefClient
derives from CefBaseRefCounted
you will also have to define
reference counting interfaces by adding IMPLEMENT_REFCOUNTING(BrowserClient);
To create a browser, the following code is needed:
CefBrowserSettings browserSettings;
browserSettings.windowless_frame_rate = 60; // 30 is default
CefWindowInfo window_info;
window_info.SetAsWindowless(0);
CefRefPtr<CefBrowser> browser;
browser = CefBrowserHost::CreateBrowserSync(
window_info,
browserClient.get(),
"https://duckduckgo.com/",
browserSettings,
nullptr, nullptr);
CEF needs to be fed with mouse position, keyboard pressed, window reshaped ... For example, for the mouse:
CefMouseEvent evt;
evt.x = from_sdl2_mouse.x;
evt.y = from_sdl2_mouse.y;
m_browser->GetHost()->SendMouseMoveEvent(evt, mouse_leave);
- CppCon19: Embrace Modern Technology: Using HTML 5 for GUI in C++ - Borislav Stanimirov. GitHub: https://github.com/iboB/html5-gui-demo Youtube video: https://www.youtube.com/watch?v=bbbcZd4cuxg
- For Godot: https://github.com/stigmee/gdnative-cef
- For C++/Qt: https://github.com/CefView/CefViewCore
- For C++/Python: https://github.com/cztomczak/cefpython
- For C#/Unity: https://github.com/Voltstro-Studios/UnityWebBrowser