OLD | NEW |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 // Many of these functions are based on those found in | 5 // Many of these functions are based on those found in |
6 // webkit/port/platform/PasteboardWin.cpp | 6 // webkit/port/platform/PasteboardWin.cpp |
7 | 7 |
8 #include "ui/base/clipboard/clipboard.h" | 8 #include "ui/base/clipboard/clipboard.h" |
9 | 9 |
10 #include <shlobj.h> | 10 #include <shlobj.h> |
(...skipping 420 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
431 } | 431 } |
432 | 432 |
433 SkBitmap Clipboard::ReadImage(Buffer buffer) const { | 433 SkBitmap Clipboard::ReadImage(Buffer buffer) const { |
434 DCHECK_EQ(buffer, BUFFER_STANDARD); | 434 DCHECK_EQ(buffer, BUFFER_STANDARD); |
435 | 435 |
436 // Acquire the clipboard. | 436 // Acquire the clipboard. |
437 ScopedClipboard clipboard; | 437 ScopedClipboard clipboard; |
438 if (!clipboard.Acquire(GetClipboardWindow())) | 438 if (!clipboard.Acquire(GetClipboardWindow())) |
439 return SkBitmap(); | 439 return SkBitmap(); |
440 | 440 |
441 HBITMAP source_bitmap = static_cast<HBITMAP>(::GetClipboardData(CF_BITMAP)); | 441 // We use a DIB rather than a DDB here since ::GetObject() with the |
442 if (!source_bitmap) | 442 // HBITMAP returned from ::GetClipboardData(CF_BITMAP) always reports a color |
443 return SkBitmap(); | 443 // depth of 32bpp. |
| 444 BITMAPINFO* bitmap = static_cast<BITMAPINFO*>(::GetClipboardData(CF_DIB)); |
| 445 int color_table_length = 0; |
| 446 switch (bitmap->bmiHeader.biBitCount) { |
| 447 case 1: |
| 448 case 4: |
| 449 case 8: |
| 450 color_table_length = bitmap->bmiHeader.biClrUsed |
| 451 ? bitmap->bmiHeader.biClrUsed |
| 452 : 1 << bitmap->bmiHeader.biBitCount; |
| 453 break; |
| 454 case 16: |
| 455 case 32: |
| 456 if (bitmap->bmiHeader.biCompression == BI_BITFIELDS) |
| 457 color_table_length = 3; |
| 458 break; |
| 459 case 24: |
| 460 break; |
| 461 default: |
| 462 NOTREACHED(); |
| 463 } |
| 464 const void* bitmap_bits = reinterpret_cast<const char*>(bitmap) |
| 465 + bitmap->bmiHeader.biSize + color_table_length * sizeof(RGBQUAD); |
444 | 466 |
445 base::win::ScopedHDC source_dc(::CreateCompatibleDC(NULL)); | 467 gfx::CanvasSkia canvas(bitmap->bmiHeader.biWidth, bitmap->bmiHeader.biHeight, |
446 if (!source_dc) | 468 false); |
447 return SkBitmap(); | 469 { |
448 ::SelectObject(source_dc, source_bitmap); | 470 skia::ScopedPlatformPaint scoped_platform_paint(&canvas); |
449 | 471 HDC dc = scoped_platform_paint.GetPlatformSurface(); |
450 // Get the dimensions of the bitmap. | 472 ::SetDIBitsToDevice(dc, 0, 0, bitmap->bmiHeader.biWidth, |
451 BITMAPINFO bitmap_info = {}; | 473 bitmap->bmiHeader.biHeight, 0, 0, 0, |
452 bitmap_info.bmiHeader.biSize = sizeof(bitmap_info.bmiHeader); | 474 bitmap->bmiHeader.biHeight, bitmap_bits, bitmap, |
453 ::GetDIBits(source_dc, source_bitmap, 0, 0, 0, &bitmap_info, DIB_RGB_COLORS); | 475 DIB_RGB_COLORS); |
454 int width = bitmap_info.bmiHeader.biWidth; | 476 } |
455 int height = bitmap_info.bmiHeader.biHeight; | 477 // SetDIBitsToDevice doesn't properly set alpha values for bitmaps with |
456 | 478 // depth < 32bpp so manually fix it up. |
457 gfx::CanvasSkia canvas(width, height, false); | 479 if (bitmap->bmiHeader.biBitCount < 32) { |
458 | 480 const SkBitmap& device_bitmap = canvas.getDevice()->accessBitmap(true); |
459 skia::ScopedPlatformPaint scoped_platform_paint(&canvas); | 481 SkAutoLockPixels lock(device_bitmap); |
460 HDC destination_dc = scoped_platform_paint.GetPlatformSurface(); | 482 for (int i = 0; i < device_bitmap.height(); ++i) { |
461 ::BitBlt(destination_dc, 0, 0, width, height, source_dc, 0, 0, SRCCOPY); | 483 for (int j = 0; j < device_bitmap.width(); ++j) { |
| 484 *device_bitmap.getAddr32(i, j) = |
| 485 SkColorSetA(*device_bitmap.getAddr32(i, j), 0xFF); |
| 486 } |
| 487 } |
| 488 } |
462 return canvas.ExtractBitmap(); | 489 return canvas.ExtractBitmap(); |
463 } | 490 } |
464 | 491 |
465 void Clipboard::ReadBookmark(string16* title, std::string* url) const { | 492 void Clipboard::ReadBookmark(string16* title, std::string* url) const { |
466 if (title) | 493 if (title) |
467 title->clear(); | 494 title->clear(); |
468 | 495 |
469 if (url) | 496 if (url) |
470 url->clear(); | 497 url->clear(); |
471 | 498 |
(...skipping 187 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
659 clipboard_owner_ = ::CreateWindow(L"ClipboardOwnerWindowClass", | 686 clipboard_owner_ = ::CreateWindow(L"ClipboardOwnerWindowClass", |
660 L"ClipboardOwnerWindow", | 687 L"ClipboardOwnerWindow", |
661 0, 0, 0, 0, 0, | 688 0, 0, 0, 0, 0, |
662 HWND_MESSAGE, | 689 HWND_MESSAGE, |
663 0, 0, 0); | 690 0, 0, 0); |
664 } | 691 } |
665 return clipboard_owner_; | 692 return clipboard_owner_; |
666 } | 693 } |
667 | 694 |
668 } // namespace ui | 695 } // namespace ui |
OLD | NEW |