Home | History | Annotate | Download | only in aura
      1 // Copyright 2013 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "content/browser/aura/software_output_device_win.h"
      6 
      7 #include "content/public/browser/browser_thread.h"
      8 #include "third_party/skia/include/core/SkBitmap.h"
      9 #include "third_party/skia/include/core/SkDevice.h"
     10 #include "ui/compositor/compositor.h"
     11 #include "ui/gfx/canvas.h"
     12 #include "ui/gfx/canvas_skia_paint.h"
     13 #include "ui/gfx/gdi_util.h"
     14 
     15 namespace content {
     16 
     17 SoftwareOutputDeviceWin::SoftwareOutputDeviceWin(ui::Compositor* compositor)
     18     : hwnd_(compositor->widget()),
     19       is_hwnd_composited_(false) {
     20   // TODO(skaslev) Remove this when crbug.com/180702 is fixed.
     21   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
     22 
     23   LONG style = GetWindowLong(hwnd_, GWL_EXSTYLE);
     24   is_hwnd_composited_ = !!(style & WS_EX_COMPOSITED);
     25 }
     26 
     27 SoftwareOutputDeviceWin::~SoftwareOutputDeviceWin() {
     28   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
     29 }
     30 
     31 void SoftwareOutputDeviceWin::Resize(gfx::Size viewport_size) {
     32   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
     33 
     34   if (viewport_size_ == viewport_size)
     35     return;
     36 
     37   viewport_size_ = viewport_size;
     38   contents_.reset(new gfx::Canvas(viewport_size, ui::SCALE_FACTOR_100P, false));
     39   memset(&bitmap_info_, 0, sizeof(bitmap_info_));
     40   gfx::CreateBitmapHeader(viewport_size_.width(), viewport_size_.height(),
     41                           &bitmap_info_.bmiHeader);
     42 }
     43 
     44 SkCanvas* SoftwareOutputDeviceWin::BeginPaint(gfx::Rect damage_rect) {
     45   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
     46   DCHECK(contents_);
     47 
     48   damage_rect_ = damage_rect;
     49   return contents_ ? contents_->sk_canvas() : NULL;
     50 }
     51 
     52 void SoftwareOutputDeviceWin::EndPaint(cc::SoftwareFrameData* frame_data) {
     53   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
     54   DCHECK(contents_);
     55   DCHECK(frame_data);
     56 
     57   if (!contents_)
     58     return;
     59 
     60   SoftwareOutputDevice::EndPaint(frame_data);
     61 
     62   gfx::Rect rect = damage_rect_;
     63   rect.Intersect(gfx::Rect(viewport_size_));
     64   if (rect.IsEmpty())
     65     return;
     66 
     67   SkCanvas* canvas = contents_->sk_canvas();
     68   DCHECK(canvas);
     69   if (is_hwnd_composited_) {
     70     RECT wr;
     71     GetWindowRect(hwnd_, &wr);
     72     SIZE size = {wr.right - wr.left, wr.bottom - wr.top};
     73     POINT position = {wr.left, wr.top};
     74     POINT zero = {0, 0};
     75     BLENDFUNCTION blend = {AC_SRC_OVER, 0x00, 0xFF, AC_SRC_ALPHA};
     76 
     77     DWORD style = GetWindowLong(hwnd_, GWL_EXSTYLE);
     78     style &= ~WS_EX_COMPOSITED;
     79     style |= WS_EX_LAYERED;
     80     SetWindowLong(hwnd_, GWL_EXSTYLE, style);
     81 
     82     HDC dib_dc = skia::BeginPlatformPaint(canvas);
     83     ::UpdateLayeredWindow(hwnd_, NULL, &position, &size, dib_dc, &zero,
     84                           RGB(0xFF, 0xFF, 0xFF), &blend, ULW_ALPHA);
     85     skia::EndPlatformPaint(canvas);
     86   } else {
     87     SkDevice* device = canvas->getDevice();
     88     const SkBitmap& bitmap = device->accessBitmap(false);
     89     HDC hdc = ::GetDC(hwnd_);
     90     gfx::StretchDIBits(hdc,
     91                        rect.x(), rect.y(),
     92                        rect.width(), rect.height(),
     93                        rect.x(), rect.y(),
     94                        rect.width(), rect.height(),
     95                        bitmap.getPixels(),
     96                        &bitmap_info_);
     97     ::ReleaseDC(hwnd_, hdc);
     98   }
     99 }
    100 
    101 }  // namespace content
    102