/*
* Copyright (C) 2008 Apple Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "ScrollbarThemeWx.h"
#include "HostWindow.h"
#include "NotImplemented.h"
#include "PlatformMouseEvent.h"
#include "Scrollbar.h"
#include "ScrollbarClient.h"
#include "scrollbar_render.h"
#include "ScrollbarThemeComposite.h"
#include "ScrollView.h"
#include <wx/defs.h>
#include <wx/dcgraph.h>
#include <wx/settings.h>
const int cMacButtonOverlap = 4;
namespace WebCore {
ScrollbarTheme* ScrollbarTheme::nativeTheme()
{
static ScrollbarThemeWx theme;
return &theme;
}
ScrollbarThemeWx::~ScrollbarThemeWx()
{
}
int ScrollbarThemeWx::scrollbarThickness(ScrollbarControlSize size)
{
int thickness = wxSystemSettings::GetMetric(wxSYS_VSCROLL_X);
// fallback for when a platform doesn't define this metric
if (thickness <= 0)
thickness = 20;
return thickness;
}
bool ScrollbarThemeWx::hasThumb(Scrollbar* scrollbar)
{
// This method is just called as a paint-time optimization to see if
// painting the thumb can be skipped. We don't have to be exact here.
return thumbLength(scrollbar) > 0;
}
int ScrollbarThemeWx::minimumThumbLength(Scrollbar* scrollbar)
{
return 20;
}
IntSize ScrollbarThemeWx::buttonSize(Scrollbar*)
{
#ifdef __WXMAC__
return IntSize(20,20);
#else
return IntSize(16,16);
#endif
}
void ScrollbarThemeWx::splitTrack(Scrollbar* scrollbar, const IntRect& unconstrainedTrackRect, IntRect& beforeThumbRect, IntRect& thumbRect, IntRect& afterThumbRect)
{
ScrollbarThemeComposite::splitTrack(scrollbar, unconstrainedTrackRect, beforeThumbRect, thumbRect, afterThumbRect);
#ifdef __WXMAC__
// on Mac, there are a few pixels drawn above the actual track and so adjust
// the hit testing rects accordingly
int trackStart = 10;
if (scrollbar->orientation() == HorizontalScrollbar) {
thumbRect.setX(thumbRect.x() + trackStart);
afterThumbRect.setX(afterThumbRect.x() - trackStart);
} else {
thumbRect.setY(thumbRect.y() + trackStart);
afterThumbRect.setY(afterThumbRect.y() - trackStart);
}
#endif
}
IntRect ScrollbarThemeWx::backButtonRect(Scrollbar* scrollbar, ScrollbarPart part, bool)
{
// FIXME: Handling this case is needed when there are two sets of arrow buttons
// on Mac, one at the top and one at the bottom.
if (part == BackButtonEndPart)
return IntRect();
IntSize size = buttonSize(scrollbar);
int x = scrollbar->x();
int y = scrollbar->y();
#if __WXMAC__
if (scrollbar->orientation() == HorizontalScrollbar)
x += scrollbar->width() - (size.width() * 2) + cMacButtonOverlap;
else
y += scrollbar->height() - (size.height() * 2) + cMacButtonOverlap;
#endif
return IntRect(x, y, size.width(), size.height());
}
IntRect ScrollbarThemeWx::forwardButtonRect(Scrollbar* scrollbar, ScrollbarPart part, bool)
{
// FIXME: Handling this case is needed when there are two sets of arrow buttons
// on Mac, one at the top and one at the bottom.
if (part == ForwardButtonStartPart)
return IntRect();
IntSize size = buttonSize(scrollbar);
int x, y;
if (scrollbar->orientation() == HorizontalScrollbar) {
#ifdef __WXMAC__
size.setWidth(size.width() + cMacButtonOverlap);
#endif
x = scrollbar->x() + scrollbar->width() - size.width();
y = scrollbar->y();
} else {
x = scrollbar->x();
#ifdef __WXMAC__
size.setHeight(size.height() + cMacButtonOverlap);
#endif
y = scrollbar->y() + scrollbar->height() - size.height();
}
return IntRect(x, y, size.width(), size.height());
}
IntRect ScrollbarThemeWx::trackRect(Scrollbar* scrollbar, bool)
{
IntSize bs = buttonSize(scrollbar);
int trackStart = 0;
if (scrollbar->orientation() == HorizontalScrollbar)
trackStart = bs.width();
else
trackStart = bs.height();
#if __WXMAC__
trackStart = 0;
#endif
int thickness = scrollbarThickness(scrollbar->controlSize());
if (scrollbar->orientation() == HorizontalScrollbar) {
if (scrollbar->width() < 2 * thickness)
return IntRect();
return IntRect(scrollbar->x() + trackStart, scrollbar->y(), scrollbar->width() - 2 * bs.width(), thickness);
}
if (scrollbar->height() < 2 * thickness)
return IntRect();
return IntRect(scrollbar->x(), scrollbar->y() + trackStart, thickness, scrollbar->height() - 2 * bs.height());
}
void ScrollbarThemeWx::paintScrollCorner(ScrollView* view, GraphicsContext* context, const IntRect& cornerRect)
{
// ScrollbarThemeComposite::paintScrollCorner incorrectly assumes that the
// ScrollView is a FrameView (see FramelessScrollView), so we cannot let
// that code run. For FrameView's this is correct since we don't do custom
// scrollbar corner rendering, which ScrollbarThemeComposite supports.
ScrollbarTheme::paintScrollCorner(view, context, cornerRect);
}
bool ScrollbarThemeWx::paint(Scrollbar* scrollbar, GraphicsContext* context, const IntRect& rect)
{
wxOrientation orientation = (scrollbar->orientation() == HorizontalScrollbar) ? wxHORIZONTAL : wxVERTICAL;
int flags = 0;
if (scrollbar->client()->isActive())
flags |= wxCONTROL_FOCUSED;
if (!scrollbar->enabled())
flags |= wxCONTROL_DISABLED;
wxDC* dc = static_cast<wxDC*>(context->platformContext());
context->save();
ScrollView* root = scrollbar->root();
ASSERT(root);
if (!root)
return false;
wxWindow* webview = root->hostWindow()->platformPageClient();
wxRenderer_DrawScrollbar(webview, *dc, scrollbar->frameRect(), orientation, scrollbar->currentPos(), static_cast<wxScrollbarPart>(scrollbar->pressedPart()),
static_cast<wxScrollbarPart>(scrollbar->hoveredPart()), scrollbar->maximum(), scrollbar->pageStep(), flags);
context->restore();
return true;
}
}