mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-22 15:29:56 +01:00
dea4daffd7
This implementation supports both GPU compositing and software compositing (used when GPU is not supported or when passing `--disable-gpu --disable-gpu-compositing` command-line flags). GPU-accelerated features (WebGL and 3D CSS) that did not work with the previous off-screen rendering implementation do work with this implementation when GPU support is available. Rendering now operates on a per-frame basis. The frame rate is configurable via CefBrowserSettings.windowless_frame_rate up to a maximum of 60fps (potentially limited by how fast the system can generate new frames). CEF generates a bitmap from the compositor backing and passes it to CefRenderHandler::OnPaint. The previous CefRenderHandler/CefBrowserHost API for off-screen rendering has been restored mostly as-is with some minor changes: - CefBrowserHost::Invalidate no longer accepts a CefRect region argument. Instead of invalidating a specific region it now triggers generation of a new frame. - The |dirtyRects| argument to CefRenderHandler::OnPaint will now always be a single CefRect representing the whole view (frame) size. Previously, invalidated regions were listed separately. - Linux: CefBrowserHost::SendKeyEvent now expects X11 event information instead of GTK event information. See cefclient for an example of converting GTK events to the necessary format. - Sizes passed to the CefRenderHandler OnPaint and OnPopupSize methods are now already DPI scaled. Previously, the client had to perform DPI scaling. - Includes drag&drop implementation from issue #1032. - Includes unit test fixes from issue #1245. git-svn-id: https://chromiumembedded.googlecode.com/svn/trunk@1751 5089003a-bbd8-11dd-ad1f-f1f9622dbc98
278 lines
7.1 KiB
C++
278 lines
7.1 KiB
C++
// Copyright (c) 2012 The Chromium Embedded Framework Authors. All rights
|
|
// reserved. Use of this source code is governed by a BSD-style license
|
|
// that can be found in the LICENSE file.
|
|
|
|
#include "cefclient/osrenderer.h"
|
|
|
|
#if defined(OS_WIN)
|
|
#include <gl/gl.h>
|
|
#include <gl/glu.h>
|
|
#elif defined(OS_MACOSX)
|
|
#include <OpenGL/gl.h>
|
|
#elif defined(OS_LINUX)
|
|
#include <GL/gl.h>
|
|
#include <GL/glu.h>
|
|
#else
|
|
#error Platform is not supported.
|
|
#endif
|
|
|
|
#include "cefclient/util.h"
|
|
|
|
#ifndef GL_BGR
|
|
#define GL_BGR 0x80E0
|
|
#endif
|
|
#ifndef GL_BGRA
|
|
#define GL_BGRA 0x80E1
|
|
#endif
|
|
#ifndef GL_UNSIGNED_INT_8_8_8_8_REV
|
|
#define GL_UNSIGNED_INT_8_8_8_8_REV 0x8367
|
|
#endif
|
|
|
|
|
|
ClientOSRenderer::ClientOSRenderer(bool transparent)
|
|
: transparent_(transparent),
|
|
initialized_(false),
|
|
texture_id_(0),
|
|
view_width_(0),
|
|
view_height_(0),
|
|
spin_x_(0),
|
|
spin_y_(0) {
|
|
}
|
|
|
|
ClientOSRenderer::~ClientOSRenderer() {
|
|
Cleanup();
|
|
}
|
|
|
|
void ClientOSRenderer::Initialize() {
|
|
if (initialized_)
|
|
return;
|
|
|
|
glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST);
|
|
|
|
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
|
// Necessary for non-power-of-2 textures to render correctly.
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
|
|
// Create the texture.
|
|
glGenTextures(1, &texture_id_);
|
|
ASSERT(texture_id_ != 0);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, texture_id_);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
|
|
|
|
initialized_ = true;
|
|
}
|
|
|
|
void ClientOSRenderer::Cleanup() {
|
|
if (texture_id_ != 0)
|
|
glDeleteTextures(1, &texture_id_);
|
|
}
|
|
|
|
void ClientOSRenderer::Render() {
|
|
if (view_width_ == 0 || view_height_ == 0)
|
|
return;
|
|
|
|
ASSERT(initialized_);
|
|
|
|
struct {
|
|
float tu, tv;
|
|
float x, y, z;
|
|
} static vertices[] = {
|
|
{0.0f, 1.0f, -1.0f, -1.0f, 0.0f},
|
|
{1.0f, 1.0f, 1.0f, -1.0f, 0.0f},
|
|
{1.0f, 0.0f, 1.0f, 1.0f, 0.0f},
|
|
{0.0f, 0.0f, -1.0f, 1.0f, 0.0f}
|
|
};
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
|
|
|
glMatrixMode(GL_MODELVIEW);
|
|
glLoadIdentity();
|
|
|
|
// Match GL units to screen coordinates.
|
|
glViewport(0, 0, view_width_, view_height_);
|
|
glMatrixMode(GL_PROJECTION);
|
|
glLoadIdentity();
|
|
glOrtho(0, 0, view_width_, view_height_, 0.1, 100.0);
|
|
|
|
// Draw the background gradient.
|
|
glPushAttrib(GL_ALL_ATTRIB_BITS);
|
|
glBegin(GL_QUADS);
|
|
glColor4f(1.0, 0.0, 0.0, 1.0); // red
|
|
glVertex2f(-1.0, -1.0);
|
|
glVertex2f(1.0, -1.0);
|
|
glColor4f(0.0, 0.0, 1.0, 1.0); // blue
|
|
glVertex2f(1.0, 1.0);
|
|
glVertex2f(-1.0, 1.0);
|
|
glEnd();
|
|
glPopAttrib();
|
|
|
|
// Rotate the view based on the mouse spin.
|
|
if (spin_x_ != 0)
|
|
glRotatef(-spin_x_, 1.0f, 0.0f, 0.0f);
|
|
if (spin_y_ != 0)
|
|
glRotatef(-spin_y_, 0.0f, 1.0f, 0.0f);
|
|
|
|
if (transparent_) {
|
|
// Alpha blending style. Texture values have premultiplied alpha.
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
// Enable alpha blending.
|
|
glEnable(GL_BLEND);
|
|
}
|
|
|
|
// Enable 2D textures.
|
|
glEnable(GL_TEXTURE_2D);
|
|
|
|
// Draw the facets with the texture.
|
|
ASSERT(texture_id_ != 0);
|
|
glBindTexture(GL_TEXTURE_2D, texture_id_);
|
|
glInterleavedArrays(GL_T2F_V3F, 0, vertices);
|
|
glDrawArrays(GL_QUADS, 0, 4);
|
|
|
|
// Disable 2D textures.
|
|
glDisable(GL_TEXTURE_2D);
|
|
|
|
if (transparent_) {
|
|
// Disable alpha blending.
|
|
glDisable(GL_BLEND);
|
|
}
|
|
}
|
|
|
|
void ClientOSRenderer::OnPopupShow(CefRefPtr<CefBrowser> browser,
|
|
bool show) {
|
|
if (!show) {
|
|
// Clear the popup rectangle.
|
|
ClearPopupRects();
|
|
}
|
|
}
|
|
|
|
void ClientOSRenderer::OnPopupSize(CefRefPtr<CefBrowser> browser,
|
|
const CefRect& rect) {
|
|
if (rect.width <= 0 || rect.height <= 0)
|
|
return;
|
|
original_popup_rect_ = rect;
|
|
popup_rect_ = GetPopupRectInWebView(original_popup_rect_);
|
|
}
|
|
|
|
CefRect ClientOSRenderer::GetPopupRectInWebView(const CefRect& original_rect) {
|
|
CefRect rc(original_rect);
|
|
// if x or y are negative, move them to 0.
|
|
if (rc.x < 0)
|
|
rc.x = 0;
|
|
if (rc.y < 0)
|
|
rc.y = 0;
|
|
// if popup goes outside the view, try to reposition origin
|
|
if (rc.x + rc.width > view_width_)
|
|
rc.x = view_width_ - rc.width;
|
|
if (rc.y + rc.height > view_height_)
|
|
rc.y = view_height_ - rc.height;
|
|
// if x or y became negative, move them to 0 again.
|
|
if (rc.x < 0)
|
|
rc.x = 0;
|
|
if (rc.y < 0)
|
|
rc.y = 0;
|
|
return rc;
|
|
}
|
|
|
|
void ClientOSRenderer::ClearPopupRects() {
|
|
popup_rect_.Set(0, 0, 0, 0);
|
|
original_popup_rect_.Set(0, 0, 0, 0);
|
|
}
|
|
|
|
void ClientOSRenderer::OnPaint(CefRefPtr<CefBrowser> browser,
|
|
CefRenderHandler::PaintElementType type,
|
|
const CefRenderHandler::RectList& dirtyRects,
|
|
const void* buffer, int width, int height) {
|
|
if (!initialized_)
|
|
Initialize();
|
|
|
|
if (transparent_) {
|
|
// Enable alpha blending.
|
|
glEnable(GL_BLEND);
|
|
}
|
|
|
|
// Enable 2D textures.
|
|
glEnable(GL_TEXTURE_2D);
|
|
|
|
ASSERT(texture_id_ != 0);
|
|
glBindTexture(GL_TEXTURE_2D, texture_id_);
|
|
|
|
if (type == PET_VIEW) {
|
|
int old_width = view_width_;
|
|
int old_height = view_height_;
|
|
|
|
view_width_ = width;
|
|
view_height_ = height;
|
|
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH, view_width_);
|
|
|
|
if (old_width != view_width_ || old_height != view_height_) {
|
|
// Update/resize the whole texture.
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0);
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS, 0);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, view_width_, view_height_, 0,
|
|
GL_BGRA, GL_UNSIGNED_INT_8_8_8_8_REV, buffer);
|
|
} else {
|
|
// Update just the dirty rectangles.
|
|
CefRenderHandler::RectList::const_iterator i = dirtyRects.begin();
|
|
for (; i != dirtyRects.end(); ++i) {
|
|
const CefRect& rect = *i;
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS, rect.x);
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS, rect.y);
|
|
glTexSubImage2D(GL_TEXTURE_2D, 0, rect.x, rect.y, rect.width,
|
|
rect.height, GL_BGRA, GL_UNSIGNED_INT_8_8_8_8_REV,
|
|
buffer);
|
|
}
|
|
}
|
|
} else if (type == PET_POPUP && popup_rect_.width > 0 &&
|
|
popup_rect_.height > 0) {
|
|
int skip_pixels = 0, x = popup_rect_.x;
|
|
int skip_rows = 0, y = popup_rect_.y;
|
|
int w = width;
|
|
int h = height;
|
|
|
|
// Adjust the popup to fit inside the view.
|
|
if (x < 0) {
|
|
skip_pixels = -x;
|
|
x = 0;
|
|
}
|
|
if (y < 0) {
|
|
skip_rows = -y;
|
|
y = 0;
|
|
}
|
|
if (x + w > view_width_)
|
|
w -= x + w - view_width_;
|
|
if (y + h > view_height_)
|
|
h -= y + h - view_height_;
|
|
|
|
// Update the popup rectangle.
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH, width);
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS, skip_pixels);
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS, skip_rows);
|
|
glTexSubImage2D(GL_TEXTURE_2D, 0, x, y, w, h, GL_BGRA,
|
|
GL_UNSIGNED_INT_8_8_8_8_REV, buffer);
|
|
}
|
|
|
|
// Disable 2D textures.
|
|
glDisable(GL_TEXTURE_2D);
|
|
|
|
if (transparent_) {
|
|
// Disable alpha blending.
|
|
glDisable(GL_BLEND);
|
|
}
|
|
}
|
|
|
|
void ClientOSRenderer::SetSpin(float spinX, float spinY) {
|
|
spin_x_ = spinX;
|
|
spin_y_ = spinY;
|
|
}
|
|
|
|
void ClientOSRenderer::IncrementSpin(float spinDX, float spinDY) {
|
|
spin_x_ -= spinDX;
|
|
spin_y_ -= spinDY;
|
|
}
|