| // Copyright 2016 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 "remoting/client/gl_cursor.h" | 
 |  | 
 | #include "remoting/base/util.h" | 
 | #include "remoting/client/gl_canvas.h" | 
 | #include "remoting/client/gl_math.h" | 
 | #include "remoting/client/gl_render_layer.h" | 
 | #include "remoting/client/gl_texture_ids.h" | 
 | #include "remoting/proto/control.pb.h" | 
 | #include "third_party/libyuv/include/libyuv/convert_argb.h" | 
 |  | 
 | namespace remoting { | 
 |  | 
 | namespace { | 
 | const int kDefaultCursorDataSize = 32 * 32 * GlRenderLayer::kBytesPerPixel; | 
 | }  // namespace | 
 |  | 
 | GlCursor::GlCursor() {} | 
 |  | 
 | GlCursor::~GlCursor() {} | 
 |  | 
 | void GlCursor::SetCursorShape(const protocol::CursorShapeInfo& cursor_shape) { | 
 |   int data_size = cursor_shape.width() * cursor_shape.height() * | 
 |       GlRenderLayer::kBytesPerPixel; | 
 |   if (current_cursor_data_size_ < data_size) { | 
 |     current_cursor_data_size_ = | 
 |         kDefaultCursorDataSize > data_size ? kDefaultCursorDataSize : data_size; | 
 |     current_cursor_data_.reset(new uint8_t[current_cursor_data_size_]); | 
 |   } | 
 |   int stride = cursor_shape.width() * GlRenderLayer::kBytesPerPixel; | 
 |   libyuv::ABGRToARGB( | 
 |       reinterpret_cast<const uint8_t*>(cursor_shape.data().data()), stride, | 
 |       current_cursor_data_.get(), stride, cursor_shape.width(), | 
 |       cursor_shape.height()); | 
 |  | 
 |   bool size_changed = current_cursor_width_ != cursor_shape.width() || | 
 |                       current_cursor_height_ != cursor_shape.height(); | 
 |  | 
 |   current_cursor_width_ = cursor_shape.width(); | 
 |   current_cursor_height_ = cursor_shape.height(); | 
 |   current_cursor_hotspot_x_ = cursor_shape.hotspot_x(); | 
 |   current_cursor_hotspot_y_ = cursor_shape.hotspot_y(); | 
 |  | 
 |   SetCurrentCursorShape(size_changed); | 
 |  | 
 |   SetCursorPosition(cursor_x_, cursor_y_); | 
 | } | 
 |  | 
 | void GlCursor::SetCursorPosition(float x, float y) { | 
 |   cursor_x_ = x; | 
 |   cursor_y_ = y; | 
 |   if (!current_cursor_data_) { | 
 |     return; | 
 |   } | 
 |   std::array<float, 8> positions; | 
 |   FillRectangleVertexPositions( | 
 |       x - current_cursor_hotspot_x_, y - current_cursor_hotspot_y_, | 
 |       current_cursor_width_, current_cursor_height_, &positions); | 
 |   if (layer_) { | 
 |     layer_->SetVertexPositions(positions); | 
 |   } | 
 | } | 
 |  | 
 | void GlCursor::SetCursorVisible(bool visible) { | 
 |   visible_ = visible; | 
 | } | 
 |  | 
 | void GlCursor::SetCanvas(GlCanvas* canvas) { | 
 |   if (!canvas) { | 
 |     layer_.reset(); | 
 |     return; | 
 |   } | 
 |   layer_.reset(new GlRenderLayer(kGlCursorTextureId, canvas)); | 
 |   if (current_cursor_data_) { | 
 |     SetCurrentCursorShape(true); | 
 |   } | 
 |   SetCursorPosition(cursor_x_, cursor_y_); | 
 | } | 
 |  | 
 | void GlCursor::Draw() { | 
 |   if (layer_ && current_cursor_data_ && visible_) { | 
 |     layer_->Draw(1.f); | 
 |   } | 
 | } | 
 |  | 
 | void GlCursor::SetCurrentCursorShape(bool size_changed) { | 
 |   if (layer_) { | 
 |     if (size_changed) { | 
 |       layer_->SetTexture(current_cursor_data_.get(), current_cursor_width_, | 
 |                          current_cursor_height_, 0); | 
 |     } else { | 
 |       layer_->UpdateTexture(current_cursor_data_.get(), 0, 0, | 
 |                             current_cursor_width_, current_cursor_width_, 0); | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace remoting |