// Copyright (c) 2011 The Chromium 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 "webkit/glue/webcursor.h" #include "base/logging.h" #include "base/pickle.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebCursorInfo.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebImage.h" using WebKit::WebCursorInfo; using WebKit::WebImage; static const int kMaxCursorDimension = 1024; WebCursor::WebCursor() : type_(WebCursorInfo::TypePointer) { InitPlatformData(); } WebCursor::WebCursor(const WebCursorInfo& cursor_info) : type_(WebCursorInfo::TypePointer) { InitPlatformData(); InitFromCursorInfo(cursor_info); } WebCursor::~WebCursor() { Clear(); } WebCursor::WebCursor(const WebCursor& other) { InitPlatformData(); Copy(other); } const WebCursor& WebCursor::operator=(const WebCursor& other) { if (this == &other) return *this; Clear(); Copy(other); return *this; } void WebCursor::InitFromCursorInfo(const WebCursorInfo& cursor_info) { Clear(); #if defined(OS_WIN) && !defined(USE_AURA) if (cursor_info.externalHandle) { InitFromExternalCursor(cursor_info.externalHandle); return; } #endif type_ = cursor_info.type; hotspot_ = cursor_info.hotSpot; if (IsCustom()) SetCustomData(cursor_info.customImage); ClampHotspot(); } void WebCursor::GetCursorInfo(WebCursorInfo* cursor_info) const { cursor_info->type = static_cast(type_); cursor_info->hotSpot = hotspot_; ImageFromCustomData(&cursor_info->customImage); #if defined(OS_WIN) && !defined(USE_AURA) cursor_info->externalHandle = external_cursor_; #endif } bool WebCursor::Deserialize(const Pickle* pickle, void** iter) { int type, hotspot_x, hotspot_y, size_x, size_y, data_len; const char* data; // Leave |this| unmodified unless we are going to return success. if (!pickle->ReadInt(iter, &type) || !pickle->ReadInt(iter, &hotspot_x) || !pickle->ReadInt(iter, &hotspot_y) || !pickle->ReadLength(iter, &size_x) || !pickle->ReadLength(iter, &size_y) || !pickle->ReadData(iter, &data, &data_len)) return false; // Ensure the size is sane, and there is enough data. if (size_x > kMaxCursorDimension || size_y > kMaxCursorDimension) return false; type_ = type; if (type == WebCursorInfo::TypeCustom) { if (size_x > 0 && size_y > 0) { // The * 4 is because the expected format is an array of RGBA pixel // values. if (size_x * size_y * 4 > data_len) return false; hotspot_.set_x(hotspot_x); hotspot_.set_y(hotspot_y); custom_size_.set_width(size_x); custom_size_.set_height(size_y); ClampHotspot(); custom_data_.clear(); if (data_len > 0) { custom_data_.resize(data_len); memcpy(&custom_data_[0], data, data_len); } } } return DeserializePlatformData(pickle, iter); } bool WebCursor::Serialize(Pickle* pickle) const { if (!pickle->WriteInt(type_) || !pickle->WriteInt(hotspot_.x()) || !pickle->WriteInt(hotspot_.y()) || !pickle->WriteInt(custom_size_.width()) || !pickle->WriteInt(custom_size_.height())) return false; const char* data = NULL; if (!custom_data_.empty()) data = &custom_data_[0]; if (!pickle->WriteData(data, custom_data_.size())) return false; return SerializePlatformData(pickle); } bool WebCursor::IsCustom() const { return type_ == WebCursorInfo::TypeCustom; } bool WebCursor::IsEqual(const WebCursor& other) const { if (type_ != other.type_) return false; if (!IsPlatformDataEqual(other)) return false; return hotspot_ == other.hotspot_ && custom_size_ == other.custom_size_ && custom_data_ == other.custom_data_; } void WebCursor::Clear() { type_ = WebCursorInfo::TypePointer; hotspot_.set_x(0); hotspot_.set_y(0); custom_size_.set_width(0); custom_size_.set_height(0); custom_data_.clear(); CleanupPlatformData(); } void WebCursor::Copy(const WebCursor& other) { type_ = other.type_; hotspot_ = other.hotspot_; custom_size_ = other.custom_size_; custom_data_ = other.custom_data_; CopyPlatformData(other); } #if WEBKIT_USING_SKIA // The WEBKIT_USING_CG implementation is in webcursor_mac.mm. void WebCursor::SetCustomData(const WebImage& image) { if (image.isNull()) return; // Fill custom_data_ directly with the NativeImage pixels. const SkBitmap& bitmap = image.getSkBitmap(); SkAutoLockPixels bitmap_lock(bitmap); custom_data_.resize(bitmap.getSize()); if (!custom_data_.empty()) memcpy(&custom_data_[0], bitmap.getPixels(), bitmap.getSize()); custom_size_.set_width(bitmap.width()); custom_size_.set_height(bitmap.height()); } void WebCursor::ImageFromCustomData(WebImage* image) const { if (custom_data_.empty()) return; SkBitmap bitmap; bitmap.setConfig(SkBitmap::kARGB_8888_Config, custom_size_.width(), custom_size_.height()); if (!bitmap.allocPixels()) return; memcpy(bitmap.getPixels(), &custom_data_[0], custom_data_.size()); image->assign(bitmap); } #endif void WebCursor::ClampHotspot() { if (!IsCustom()) return; // Clamp the hotspot to the custom image's dimensions. hotspot_.set_x(std::max(0, std::min(custom_size_.width() - 1, hotspot_.x()))); hotspot_.set_y(std::max(0, std::min(custom_size_.height() - 1, hotspot_.y()))); }