// Copyright (c) 2006-2008 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 "config.h"

#pragma warning(push, 0)
#include "Cursor.h"
#include "FramelessScrollView.h"
#include "FrameView.h"
#include "IntRect.h"
#include "PlatformKeyboardEvent.h"
#include "PlatformMouseEvent.h"
#include "PlatformWheelEvent.h"
#include "SkiaUtils.h"
#pragma warning(pop)

#undef LOG
#include "base/gfx/platform_canvas.h"
#include "base/gfx/rect.h"
#include "base/logging.h"
#include "webkit/glue/event_conversion.h"
#include "webkit/glue/webinputevent.h"
#include "webkit/glue/webwidget_delegate.h"
#include "webkit/glue/webwidget_impl.h"

using namespace WebCore;

// WebWidget ----------------------------------------------------------------

/*static*/
WebWidget* WebWidget::Create(WebWidgetDelegate* delegate) {
  WebWidgetImpl* instance = new WebWidgetImpl(delegate);
  instance->AddRef();
  return instance;
}

WebWidgetImpl::WebWidgetImpl(WebWidgetDelegate* delegate)
    : delegate_(delegate),
      widget_(NULL) {
  // set to impossible point so we always get the first mouse pos
  last_mouse_position_.SetPoint(-1, -1);
}

WebWidgetImpl::~WebWidgetImpl() {
  if (widget_) {
    widget_->setClient(NULL);
  }
}

void WebWidgetImpl::Init(WebCore::Widget* widget, const gfx::Rect& bounds) {
  DCHECK(widget->isFrameView());
  widget_ = static_cast<FramelessScrollView*>(widget);

  widget_->setClient(this);

  if (delegate_) {
    delegate_->SetWindowRect(this, bounds);
    delegate_->Show(this, WindowOpenDisposition());
  }
}

void WebWidgetImpl::MouseMove(const WebMouseEvent& event) {
  // don't send mouse move messages if the mouse hasn't moved.
  if (event.x != last_mouse_position_.x() ||
      event.y != last_mouse_position_.y()) {
    last_mouse_position_.SetPoint(event.x, event.y);

    widget_->handleMouseMoveEvent(MakePlatformMouseEvent(widget_, event));
  }
}

void WebWidgetImpl::MouseLeave(const WebMouseEvent& event) {
  widget_->handleMouseMoveEvent(MakePlatformMouseEvent(widget_, event));
}

void WebWidgetImpl::MouseDown(const WebMouseEvent& event) {
  widget_->handleMouseDownEvent(MakePlatformMouseEvent(widget_, event));
}

void WebWidgetImpl::MouseUp(const WebMouseEvent& event) {
  MouseCaptureLost();
  widget_->handleMouseReleaseEvent(MakePlatformMouseEvent(widget_, event));
}

void WebWidgetImpl::MouseWheel(const WebMouseWheelEvent& event) {
  widget_->handleWheelEvent(MakePlatformWheelEvent(widget_, event));
}

bool WebWidgetImpl::KeyEvent(const WebKeyboardEvent& event) {
  return widget_->handleKeyEvent(MakePlatformKeyboardEvent(event));
}

// WebWidget -------------------------------------------------------------------

void WebWidgetImpl::Close() {
  if (widget_)
    widget_->hide();

  delegate_ = NULL;
}

void WebWidgetImpl::Resize(const gfx::Size& new_size) {
  if (size_ == new_size)
    return;
  size_ = new_size;

  if (widget_) {
    IntRect new_geometry(0, 0, size_.width(), size_.height());
    widget_->setFrameGeometry(new_geometry);
  }

  if (delegate_) {
    gfx::Rect damaged_rect(0, 0, size_.width(), size_.height());
    delegate_->DidInvalidateRect(this, damaged_rect);
  }
}

void WebWidgetImpl::Layout() {
}

void WebWidgetImpl::Paint(gfx::PlatformCanvas* canvas, const gfx::Rect& rect) {
  if (!widget_)
    return;

  if (!rect.IsEmpty()) {
    PlatformContextSkia context(canvas);

    // PlatformGraphicsContext is actually a pointer to PlatformContextSkia.
    GraphicsContext gc(reinterpret_cast<PlatformGraphicsContext*>(&context));
    IntRect dirty_rect(rect.x(), rect.y(), rect.width(), rect.height());

    widget_->paint(&gc, dirty_rect);
  }
}

bool WebWidgetImpl::HandleInputEvent(const WebInputEvent* input_event) {
  if (!widget_)
    return false;

  // TODO (jcampan): WebKit seems to always return false on mouse events
  // methods. For now we'll assume it has processed them (as we are only
  // interested in whether keyboard events are processed).
  switch (input_event->type) {
    case WebInputEvent::MOUSE_MOVE:
      MouseMove(*static_cast<const WebMouseEvent*>(input_event));
      return true;

    case WebInputEvent::MOUSE_LEAVE:
      MouseLeave(*static_cast<const WebMouseEvent*>(input_event));
      return true;

    case WebInputEvent::MOUSE_WHEEL:
      MouseWheel(*static_cast<const WebMouseWheelEvent*>(input_event));
      return true;

    case WebInputEvent::MOUSE_DOWN:
    case WebInputEvent::MOUSE_DOUBLE_CLICK:
      MouseDown(*static_cast<const WebMouseEvent*>(input_event));
      return true;

    case WebInputEvent::MOUSE_UP:
      MouseUp(*static_cast<const WebMouseEvent*>(input_event));
      return true;

    case WebInputEvent::KEY_DOWN:
    case WebInputEvent::KEY_UP:
      return KeyEvent(*static_cast<const WebKeyboardEvent*>(input_event));
  }
  return false;
}

void WebWidgetImpl::MouseCaptureLost() {
}

void WebWidgetImpl::SetFocus(bool enable) {
}

void WebWidgetImpl::ImeSetComposition(int string_type, int cursor_position,
                                      int target_start, int target_end,
                                      int string_length,
                                      const wchar_t *string_data) {
}

bool WebWidgetImpl::ImeUpdateStatus(bool* enable_ime, const void** id,
                                    int* x, int* y) {
  return false;
}

const SkBitmap* WebWidgetImpl::getPreloadedResourceBitmap(int resource_id) {
  return NULL;
}

const WTF::Vector<RefPtr<WebCore::Range> >* WebWidgetImpl::getTickmarks(
    WebCore::Frame* frame) {
  return NULL;
}

size_t WebWidgetImpl::getActiveTickmarkIndex(WebCore::Frame* frame) {
  return kNoTickmark;
}

void WebWidgetImpl::onScrollPositionChanged(Widget* widget) {
}

//-----------------------------------------------------------------------------
// WebCore::WidgetClientWin

gfx::ViewHandle WebWidgetImpl::containingWindow() {
  return delegate_ ? delegate_->GetContainingWindow(this) : NULL;
}

void WebWidgetImpl::invalidateRect(const IntRect& damaged_rect) {
  if (delegate_)
    delegate_->DidInvalidateRect(this, gfx::Rect(damaged_rect.x(),
                                                 damaged_rect.y(),
                                                 damaged_rect.width(),
                                                 damaged_rect.height()));
}

void WebWidgetImpl::scrollRect(int dx, int dy, const IntRect& clip_rect) {
  if (delegate_)
    delegate_->DidScrollRect(this, dx, dy, gfx::Rect(clip_rect.x(),
                                                     clip_rect.y(),
                                                     clip_rect.width(),
                                                     clip_rect.height()));
}

void WebWidgetImpl::popupOpened(WebCore::Widget* widget,
                                const WebCore::IntRect& bounds) {
  NOTREACHED() << "popupOpened called on a popup";
}

void WebWidgetImpl::popupClosed(WebCore::Widget* widget) {
  DCHECK(widget == widget_);
  if (widget_) {
    widget_->setClient(NULL);
    widget_ = NULL;
  }
  delegate_->CloseWidgetSoon(this);
}

void WebWidgetImpl::setCursor(const WebCore::Cursor& cursor) {
#if defined(OS_WIN)
  // TODO(pinkerton): re-enable when WebCursor is ported
  if (delegate_)
    delegate_->SetCursor(this, cursor.impl());
#endif
}

void WebWidgetImpl::setFocus() {
  delegate_->Focus(this);
}