/*
* The ManaPlus Client
* Copyright (C) 2004-2009 The Mana World Development Team
* Copyright (C) 2009-2010 The Mana Developers
* Copyright (C) 2011-2016 The ManaPlus Developers
*
* This file is part of The ManaPlus Client.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
/* _______ __ __ __ ______ __ __ _______ __ __
* / _____/\ / /\ / /\ / /\ / ____/\ / /\ / /\ / ___ /\ / |\/ /\
* / /\____\// / // / // / // /\___\// /_// / // /\_/ / // , |/ / /
* / / /__ / / // / // / // / / / ___ / // ___ / // /| ' / /
* / /_// /\ / /_// / // / // /_/_ / / // / // /\_/ / // / | / /
* /______/ //______/ //_/ //_____/\ /_/ //_/ //_/ //_/ //_/ /|_/ /
* \______\/ \______\/ \_\/ \_____\/ \_\/ \_\/ \_\/ \_\/ \_\/ \_\/
*
* Copyright (c) 2004 - 2008 Olof Naessén and Per Larsson
*
*
* Per Larsson a.k.a finalman
* Olof Naessén a.k.a jansem/yakslem
*
* Visit: http://guichan.sourceforge.net
*
* License: (BSD)
* 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.
* 3. Neither the name of Guichan nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "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 THE COPYRIGHT
* OWNER 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 "gui/widgets/scrollarea.h"
#include "settings.h"
#include "gui/gui.h"
#include "gui/skin.h"
#include "utils/delete2.h"
#include "utils/stringutils.h"
#include "render/graphics.h"
#include "render/vertexes/imagecollection.h"
#include "resources/imagerect.h"
#include "resources/image/image.h"
#include "debug.h"
int ScrollArea::instances = 0;
float ScrollArea::mAlpha = 1.0;
bool ScrollArea::mShowButtons = true;
int ScrollArea::mMarkerSize = 0;
int ScrollArea::mScrollbarSize = 12;
ImageRect ScrollArea::background;
ImageRect ScrollArea::vMarker;
ImageRect ScrollArea::vMarkerHi;
ImageRect ScrollArea::vBackground;
ImageRect ScrollArea::hBackground;
Image *ScrollArea::buttons[4][2];
static std::string const buttonFiles[2] =
{
"scrollbuttons.xml",
"scrollbuttons_pressed.xml"
};
ScrollArea::ScrollArea(Widget2 *const widget2,
Widget *const widget,
const Opaque opaque,
const std::string &skin) :
BasicContainer(widget2),
MouseListener(),
WidgetListener(),
mVertexes(new ImageCollection),
mVertexes2(new ImageCollection),
mHPolicy(SHOW_AUTO),
mVPolicy(SHOW_AUTO),
mVScroll(0),
mHScroll(0),
mScrollbarWidth(12),
mUpButtonScrollAmount(10),
mDownButtonScrollAmount(10),
mLeftButtonScrollAmount(10),
mRightButtonScrollAmount(10),
mHorizontalMarkerDragOffset(0),
mVerticalMarkerDragOffset(0),
mX(0),
mY(0),
mClickX(0),
mClickY(0),
mXOffset(0),
mYOffset(0),
mDrawWidth(0),
mDrawHeight(0),
mVBarVisible(false),
mHBarVisible(false),
mUpButtonPressed(false),
mDownButtonPressed(false),
mLeftButtonPressed(false),
mRightButtonPressed(false),
mIsVerticalMarkerDragged(false),
mIsHorizontalMarkerDragged(false),
mOpaque(Opaque_true),
mHasMouse(false)
{
setContent(widget);
addMouseListener(this);
mOpaque = opaque;
init(skin);
}
ScrollArea::~ScrollArea()
{
if (gui)
gui->removeDragged(this);
// Garbage collection
delete getContent();
instances--;
if (instances == 0)
{
Theme::unloadRect(background);
Theme::unloadRect(vMarker);
Theme::unloadRect(vMarkerHi);
Theme::unloadRect(vBackground);
Theme::unloadRect(hBackground);
for (int i = 0; i < 2; i ++)
{
for (int f = UP; f < BUTTONS_DIR; f ++)
{
if (buttons[f][i])
buttons[f][i]->decRef();
}
}
}
delete2(mVertexes);
delete2(mVertexes2);
setContent(nullptr);
}
void ScrollArea::init(std::string skinName)
{
setOpaque(mOpaque);
setUpButtonScrollAmount(2);
setDownButtonScrollAmount(2);
setLeftButtonScrollAmount(2);
setRightButtonScrollAmount(2);
if (instances == 0)
{
for (int f = 0; f < 9; f ++)
{
background.grid[f] = nullptr;
vMarker.grid[f] = nullptr;
vMarkerHi.grid[f] = nullptr;
vBackground.grid[f] = nullptr;
hBackground.grid[f] = nullptr;
}
// +++ here probably need move background from static
if (skinName.empty())
skinName = "scroll_background.xml";
if (theme)
{
theme->loadRect(background, skinName, "scroll_background.xml");
theme->loadRect(vMarker, "scroll.xml", "");
theme->loadRect(vMarkerHi, "scroll_highlighted.xml", "scroll.xml");
theme->loadRect(vBackground, "scroll_vbackground.xml", "");
theme->loadRect(hBackground, "scroll_hbackground.xml", "");
}
for (int i = 0; i < 2; i ++)
{
Skin *skin = nullptr;
if (theme)
skin = theme->load(buttonFiles[i], "scrollbuttons.xml");
if (skin)
{
const ImageRect &rect = skin->getBorder();
for (int f = UP; f < BUTTONS_DIR; f ++)
{
if (rect.grid[f])
rect.grid[f]->incRef();
buttons[f][i] = rect.grid[f];
}
if (i == 0)
{
mShowButtons = (skin->getOption("showbuttons", 1) == 1);
mMarkerSize = skin->getOption("markersize", 0);
mScrollbarSize = skin->getOption("scrollbarsize", 12);
}
}
else
{
for (int f = UP; f < BUTTONS_DIR; f ++)
buttons[f][i] = nullptr;
}
if (theme)
theme->unload(skin);
}
}
mScrollbarWidth = mScrollbarSize;
instances++;
}
void ScrollArea::logic()
{
BLOCK_START("ScrollArea::logic")
if (!isVisible())
{
BLOCK_END("ScrollArea::logic")
return;
}
checkPolicies();
setVerticalScrollAmount(getVerticalScrollAmount());
setHorizontalScrollAmount(getHorizontalScrollAmount());
Widget *const content = getContent();
if (content)
{
unsigned int frameSize = content->getFrameSize();
content->setPosition(-mHScroll + frameSize, -mVScroll + frameSize);
content->logic();
// When no scrollbar in a certain direction,
// adapt content size to match the content dimension exactly.
frameSize = 2 * content->getFrameSize();
if (mHPolicy == ScrollArea::SHOW_NEVER)
{
content->setWidth((mVBarVisible ? (mDimension.width
- mScrollbarWidth) : mDimension.width) - frameSize);
}
if (mVPolicy == ScrollArea::SHOW_NEVER)
{
content->setHeight((mHBarVisible ? (mDimension.height
- mScrollbarWidth) : mDimension.height) - frameSize);
}
}
if (mUpButtonPressed)
setVerticalScrollAmount(mVScroll - mUpButtonScrollAmount);
else if (mDownButtonPressed)
setVerticalScrollAmount(mVScroll + mDownButtonScrollAmount);
else if (mLeftButtonPressed)
setHorizontalScrollAmount(mHScroll - mLeftButtonScrollAmount);
else if (mRightButtonPressed)
setHorizontalScrollAmount(mHScroll + mRightButtonScrollAmount);
BLOCK_END("ScrollArea::logic")
}
void ScrollArea::updateAlpha()
{
const float alpha = std::max(settings.guiAlpha,
theme->getMinimumOpacity());
if (alpha != mAlpha)
{
mAlpha = alpha;
for (int a = 0; a < 9; a++)
{
if (background.grid[a])
background.grid[a]->setAlpha(mAlpha);
if (hBackground.grid[a])
hBackground.grid[a]->setAlpha(mAlpha);
if (vBackground.grid[a])
vBackground.grid[a]->setAlpha(mAlpha);
if (vMarker.grid[a])
vMarker.grid[a]->setAlpha(mAlpha);
if (vMarkerHi.grid[a])
vMarkerHi.grid[a]->setAlpha(mAlpha);
}
}
}
void ScrollArea::draw(Graphics *const graphics)
{
BLOCK_START("ScrollArea::draw")
if (mVBarVisible || mHBarVisible)
{
if (mOpaque == Opaque_false)
updateCalcFlag(graphics);
// need add caching or remove calc calls.
// if (mRedraw)
{
mVertexes->clear();
if (mVBarVisible)
{
if (mShowButtons)
{
calcButton(graphics, UP);
calcButton(graphics, DOWN);
}
calcVBar(graphics);
calcVMarker(graphics);
}
if (mHBarVisible)
{
if (mShowButtons)
{
calcButton(graphics, LEFT);
calcButton(graphics, RIGHT);
}
calcHBar(graphics);
calcHMarker(graphics);
}
graphics->finalize(mVertexes);
}
graphics->drawTileCollection(mVertexes);
}
updateAlpha();
if (mRedraw)
{
const bool redraw = graphics->getRedraw();
graphics->setRedraw(true);
drawChildren(graphics);
graphics->setRedraw(redraw);
}
else
{
drawChildren(graphics);
}
mRedraw = false;
BLOCK_END("ScrollArea::draw")
}
void ScrollArea::safeDraw(Graphics *const graphics)
{
BLOCK_START("ScrollArea::draw")
if (mVBarVisible)
{
if (mShowButtons)
{
drawButton(graphics, UP);
drawButton(graphics, DOWN);
}
drawVBar(graphics);
drawVMarker(graphics);
}
if (mHBarVisible)
{
if (mShowButtons)
{
drawButton(graphics, LEFT);
drawButton(graphics, RIGHT);
}
drawHBar(graphics);
drawHMarker(graphics);
}
updateAlpha();
safeDrawChildren(graphics);
mRedraw = false;
BLOCK_END("ScrollArea::draw")
}
void ScrollArea::updateCalcFlag(const Graphics *const graphics)
{
if (!mRedraw)
{
// because we don't know where parent windows was moved,
// need recalc vertexes
const ClipRect &rect = graphics->getTopClip();
if (rect.xOffset != mXOffset || rect.yOffset != mYOffset)
{
mRedraw = true;
mXOffset = rect.xOffset;
mYOffset = rect.yOffset;
}
else if (rect.width != mDrawWidth || rect.height != mDrawHeight)
{
mRedraw = true;
mDrawWidth = rect.width;
mDrawHeight = rect.height;
}
else if (graphics->getRedraw())
{
mRedraw = true;
}
}
}
void ScrollArea::drawFrame(Graphics *const graphics)
{
BLOCK_START("ScrollArea::drawFrame")
if (mOpaque == Opaque_true)
{
const int bs = mFrameSize * 2;
const int w = mDimension.width + bs;
const int h = mDimension.height + bs;
updateCalcFlag(graphics);
if (mRedraw)
{
mVertexes2->clear();
graphics->calcWindow(mVertexes2,
0, 0,
w, h,
background);
graphics->finalize(mVertexes2);
}
graphics->drawTileCollection(mVertexes2);
}
BLOCK_END("ScrollArea::drawFrame")
}
void ScrollArea::safeDrawFrame(Graphics *const graphics)
{
BLOCK_START("ScrollArea::drawFrame")
if (mOpaque == Opaque_true)
{
const int bs = mFrameSize * 2;
const int w = mDimension.width + bs;
const int h = mDimension.height + bs;
updateCalcFlag(graphics);
graphics->drawImageRect(0, 0,
w, h,
background);
}
BLOCK_END("ScrollArea::drawFrame")
}
void ScrollArea::setOpaque(Opaque opaque)
{
mOpaque = opaque;
setFrameSize(mOpaque == Opaque_true ? 2 : 0);
}
Image *ScrollArea::getImageByState(Rect &dim, const BUTTON_DIR dir)
{
int state = 0;
switch (dir)
{
case UP:
state = mUpButtonPressed ? 1 : 0;
dim = getUpButtonDimension();
break;
case DOWN:
state = mDownButtonPressed ? 1 : 0;
dim = getDownButtonDimension();
break;
case LEFT:
state = mLeftButtonPressed ? 1 : 0;
dim = getLeftButtonDimension();
break;
case RIGHT:
state = mRightButtonPressed ? 1 : 0;
dim = getRightButtonDimension();
break;
case BUTTONS_DIR:
default:
logger->log("ScrollArea::drawButton unknown dir: "
+ toString(CAST_U32(dir)));
return nullptr;
}
return buttons[CAST_SIZE(dir)][state];
}
void ScrollArea::drawButton(Graphics *const graphics,
const BUTTON_DIR dir)
{
Rect dim;
const Image *const image = getImageByState(dim, dir);
if (image)
graphics->drawImage(image, dim.x, dim.y);
}
void ScrollArea::calcButton(Graphics *const graphics,
const BUTTON_DIR dir)
{
Rect dim;
const Image *const image = getImageByState(dim, dir);
if (image)
{
static_cast(graphics)->calcTileCollection(
mVertexes, image, dim.x, dim.y);
}
}
void ScrollArea::drawVBar(Graphics *const graphics) const
{
const Rect &dim = getVerticalBarDimension();
if (vBackground.grid[4])
{
graphics->drawPattern(vBackground.grid[4],
dim.x, dim.y, dim.width, dim.height);
}
if (vBackground.grid[1])
{
graphics->drawPattern(vBackground.grid[1],
dim.x, dim.y,
dim.width, vBackground.grid[1]->getHeight());
}
if (vBackground.grid[7])
{
graphics->drawPattern(vBackground.grid[7],
dim.x, dim.height - vBackground.grid[7]->getHeight() + dim.y,
dim.width, vBackground.grid[7]->getHeight());
}
}
void ScrollArea::calcVBar(const Graphics *const graphics)
{
const Rect &dim = getVerticalBarDimension();
if (vBackground.grid[4])
{
graphics->calcPattern(mVertexes,
vBackground.grid[4],
dim.x, dim.y,
dim.width, dim.height);
}
if (vBackground.grid[1])
{
graphics->calcPattern(mVertexes,
vBackground.grid[1],
dim.x, dim.y,
dim.width, vBackground.grid[1]->getHeight());
}
if (vBackground.grid[7])
{
graphics->calcPattern(mVertexes,
vBackground.grid[7],
dim.x, dim.height - vBackground.grid[7]->getHeight() + dim.y,
dim.width, vBackground.grid[7]->getHeight());
}
}
void ScrollArea::drawHBar(Graphics *const graphics) const
{
const Rect &dim = getHorizontalBarDimension();
if (hBackground.grid[4])
{
graphics->drawPattern(hBackground.grid[4],
dim.x, dim.y,
dim.width, dim.height);
}
if (hBackground.grid[3])
{
graphics->drawPattern(hBackground.grid[3],
dim.x, dim.y,
hBackground.grid[3]->getWidth(), dim.height);
}
if (hBackground.grid[5])
{
graphics->drawPattern(hBackground.grid[5],
dim.x + dim.width - hBackground.grid[5]->getWidth(),
dim.y,
hBackground.grid[5]->getWidth(),
dim.height);
}
}
void ScrollArea::calcHBar(const Graphics *const graphics)
{
const Rect &dim = getHorizontalBarDimension();
if (hBackground.grid[4])
{
graphics->calcPattern(mVertexes,
hBackground.grid[4],
dim.x, dim.y,
dim.width, dim.height);
}
if (hBackground.grid[3])
{
graphics->calcPattern(mVertexes,
hBackground.grid[3],
dim.x, dim.y,
hBackground.grid[3]->getWidth(), dim.height);
}
if (hBackground.grid[5])
{
graphics->calcPattern(mVertexes,
hBackground.grid[5],
dim.x + dim.width - hBackground.grid[5]->getWidth(),
dim.y,
hBackground.grid[5]->getWidth(),
dim.height);
}
}
void ScrollArea::drawVMarker(Graphics *const graphics)
{
const Rect &dim = getVerticalMarkerDimension();
if ((mHasMouse) && (mX > (mDimension.width - mScrollbarWidth)))
{
graphics->drawImageRect(dim.x, dim.y,
dim.width, dim.height,
vMarkerHi);
}
else
{
graphics->drawImageRect(dim.x, dim.y,
dim.width, dim.height,
vMarker);
}
}
void ScrollArea::calcVMarker(Graphics *const graphics)
{
const Rect &dim = getVerticalMarkerDimension();
if ((mHasMouse) && (mX > (mDimension.width - mScrollbarWidth)))
{
graphics->calcWindow(mVertexes,
dim.x, dim.y,
dim.width, dim.height,
vMarkerHi);
}
else
{
graphics->calcWindow(mVertexes,
dim.x, dim.y,
dim.width, dim.height,
vMarker);
}
}
void ScrollArea::drawHMarker(Graphics *const graphics)
{
const Rect dim = getHorizontalMarkerDimension();
if ((mHasMouse) && (mY > (mDimension.height - mScrollbarWidth)))
{
graphics->drawImageRect(dim.x, dim.y,
dim.width, dim.height,
vMarkerHi);
}
else
{
graphics->drawImageRect(
dim.x, dim.y,
dim.width, dim.height,
vMarker);
}
}
void ScrollArea::calcHMarker(Graphics *const graphics)
{
const Rect dim = getHorizontalMarkerDimension();
if ((mHasMouse) && (mY > (mDimension.height - mScrollbarWidth)))
{
graphics->calcWindow(mVertexes,
dim.x, dim.y,
dim.width, dim.height,
vMarkerHi);
}
else
{
graphics->calcWindow(mVertexes,
dim.x, dim.y,
dim.width, dim.height,
vMarker);
}
}
void ScrollArea::mouseMoved(MouseEvent& event)
{
mX = event.getX();
mY = event.getY();
}
void ScrollArea::mouseEntered(MouseEvent& event A_UNUSED)
{
mHasMouse = true;
}
void ScrollArea::mouseExited(MouseEvent& event A_UNUSED)
{
mHasMouse = false;
}
void ScrollArea::widgetResized(const Event &event A_UNUSED)
{
mRedraw = true;
const unsigned int frameSize = 2 * mFrameSize;
Widget *const content = getContent();
if (content)
{
content->setSize(mDimension.width - frameSize,
mDimension.height - frameSize);
}
}
void ScrollArea::widgetMoved(const Event& event A_UNUSED)
{
mRedraw = true;
}
void ScrollArea::mousePressed(MouseEvent& event)
{
const int x = event.getX();
const int y = event.getY();
if (getUpButtonDimension().isPointInRect(x, y))
{
setVerticalScrollAmount(mVScroll
- mUpButtonScrollAmount);
mUpButtonPressed = true;
event.consume();
}
else if (getDownButtonDimension().isPointInRect(x, y))
{
setVerticalScrollAmount(mVScroll
+ mDownButtonScrollAmount);
mDownButtonPressed = true;
event.consume();
}
else if (getLeftButtonDimension().isPointInRect(x, y))
{
setHorizontalScrollAmount(mHScroll
- mLeftButtonScrollAmount);
mLeftButtonPressed = true;
event.consume();
}
else if (getRightButtonDimension().isPointInRect(x, y))
{
setHorizontalScrollAmount(mHScroll
+ mRightButtonScrollAmount);
mRightButtonPressed = true;
event.consume();
}
else if (getVerticalMarkerDimension().isPointInRect(x, y))
{
mIsHorizontalMarkerDragged = false;
mIsVerticalMarkerDragged = true;
mVerticalMarkerDragOffset = y - getVerticalMarkerDimension().y;
event.consume();
}
else if (getVerticalBarDimension().isPointInRect(x, y))
{
if (y < getVerticalMarkerDimension().y)
{
setVerticalScrollAmount(mVScroll
- CAST_S32(getChildrenArea().height * 0.95));
}
else
{
setVerticalScrollAmount(mVScroll
+ CAST_S32(getChildrenArea().height * 0.95));
}
event.consume();
}
else if (getHorizontalMarkerDimension().isPointInRect(x, y))
{
mIsHorizontalMarkerDragged = true;
mIsVerticalMarkerDragged = false;
mHorizontalMarkerDragOffset = x - getHorizontalMarkerDimension().x;
event.consume();
}
else if (getHorizontalBarDimension().isPointInRect(x, y))
{
if (x < getHorizontalMarkerDimension().x)
{
setHorizontalScrollAmount(mHScroll
- CAST_S32(getChildrenArea().width * 0.95));
}
else
{
setHorizontalScrollAmount(mHScroll
+ CAST_S32(getChildrenArea().width * 0.95));
}
event.consume();
}
if (event.getButton() == MouseButton::LEFT)
{
mClickX = event.getX();
mClickY = event.getY();
}
}
void ScrollArea::mouseReleased(MouseEvent& event)
{
if (event.getButton() == MouseButton::LEFT && mClickX && mClickY)
{
if (!event.isConsumed())
{
#ifdef ANDROID
int dx = mClickX - event.getX();
int dy = mClickY - event.getY();
#else // ANDROID
int dx = event.getX() - mClickX;
int dy = event.getY() - mClickY;
#endif // ANDROID
if ((dx < 20 && dx > 0) || (dx > -20 && dx < 0))
dx = 0;
if ((dy < 20 && dy > 0) || (dy > -20 && dy < 0))
dy = 0;
if (abs(dx) > abs(dy))
{
int s = mHScroll + dx;
if (s < 0)
{
s = 0;
}
else
{
const int maxH = getHorizontalMaxScroll();
if (s > maxH)
s = maxH;
}
setHorizontalScrollAmount(s);
}
else if (dy)
{
int s = mVScroll + dy;
if (s < 0)
{
s = 0;
}
else
{
const int maxV = getVerticalMaxScroll();
if (s > maxV)
s = maxV;
}
setVerticalScrollAmount(s);
}
mClickX = 0;
mClickY = 0;
if (mMouseConsume && (dx || dy))
event.consume();
}
}
mUpButtonPressed = false;
mDownButtonPressed = false;
mLeftButtonPressed = false;
mRightButtonPressed = false;
mIsHorizontalMarkerDragged = false;
mIsVerticalMarkerDragged = false;
if (mMouseConsume)
event.consume();
mRedraw = true;
}
void ScrollArea::mouseDragged(MouseEvent &event)
{
if (mIsVerticalMarkerDragged)
{
const Rect barDim = getVerticalBarDimension();
const int pos = event.getY() - barDim.y
- mVerticalMarkerDragOffset;
const int length = getVerticalMarkerDimension().height;
if ((barDim.height - length) > 0)
{
setVerticalScrollAmount((getVerticalMaxScroll() * pos)
/ (barDim.height - length));
}
else
{
setVerticalScrollAmount(0);
}
}
if (mIsHorizontalMarkerDragged)
{
const Rect barDim = getHorizontalBarDimension();
const int pos = event.getX() - barDim.x
- mHorizontalMarkerDragOffset;
const int length = getHorizontalMarkerDimension().width;
if ((barDim.width - length) > 0)
{
setHorizontalScrollAmount((getHorizontalMaxScroll() * pos)
/ (barDim.width - length));
}
else
{
setHorizontalScrollAmount(0);
}
}
event.consume();
mRedraw = true;
}
Rect ScrollArea::getVerticalBarDimension() const
{
if (!mVBarVisible)
return Rect(0, 0, 0, 0);
const int height = (mVBarVisible && mShowButtons) ? mScrollbarWidth : 0;
if (mHBarVisible)
{
return Rect(mDimension.width - mScrollbarWidth,
height,
mScrollbarWidth,
mDimension.height - 2 * height - mScrollbarWidth);
}
return Rect(mDimension.width - mScrollbarWidth,
height,
mScrollbarWidth,
mDimension.height - 2 * height);
}
Rect ScrollArea::getHorizontalBarDimension() const
{
if (!mHBarVisible)
return Rect(0, 0, 0, 0);
const int width = mShowButtons ? mScrollbarWidth : 0;
if (mVBarVisible)
{
return Rect(width,
mDimension.height - mScrollbarWidth,
mDimension.width - 2 * width - mScrollbarWidth,
mScrollbarWidth);
}
return Rect(width,
mDimension.height - mScrollbarWidth,
mDimension.width - 2 * width,
mScrollbarWidth);
}
Rect ScrollArea::getVerticalMarkerDimension()
{
if (!mVBarVisible)
return Rect(0, 0, 0, 0);
int length, pos;
int height;
const int h2 = mShowButtons
? mScrollbarWidth : mMarkerSize / 2;
const Widget *content;
if (!mWidgets.empty())
content = *mWidgets.begin();
else
content = nullptr;
if (mHBarVisible)
height = mDimension.height - 2 * h2 - mScrollbarWidth;
else
height = mDimension.height - 2 * h2;
const int maxV = getVerticalMaxScroll();
if (mMarkerSize && maxV)
{
pos = (mVScroll * height / maxV - mMarkerSize / 2);
length = mMarkerSize;
}
else
{
if (content)
{
const int h3 = content->getHeight();
if (h3)
length = (height * getChildrenArea().height) / h3;
else
length = height;
}
else
{
length = height;
}
if (length < mScrollbarWidth)
length = mScrollbarWidth;
if (length > height)
length = height;
const int maxScroll = getVerticalMaxScroll();
if (maxScroll != 0)
pos = ((height - length) * mVScroll) / maxScroll;
else
pos = 0;
}
return Rect(mDimension.width - mScrollbarWidth, h2 + pos,
mScrollbarWidth, length);
}
Rect ScrollArea::getHorizontalMarkerDimension()
{
if (!mHBarVisible)
return Rect(0, 0, 0, 0);
int length, pos;
int width;
const int w2 = mShowButtons
? mScrollbarWidth : mMarkerSize / 2;
const Widget *content;
if (!mWidgets.empty())
content = *mWidgets.begin();
else
content = nullptr;
if (mVBarVisible)
width = mDimension.width - 2 * w2 - mScrollbarWidth;
else
width = mDimension.width - w2 - mScrollbarWidth;
const int maxH = getHorizontalMaxScroll();
if (mMarkerSize && maxH)
{
pos = (mHScroll * width / maxH - mMarkerSize / 2);
length = mMarkerSize;
}
else
{
if (content)
{
const int w3 = content->getWidth();
if (w3)
length = (width * getChildrenArea().width) / w3;
else
length = width;
}
else
{
length = width;
}
if (length < mScrollbarWidth)
length = mScrollbarWidth;
if (length > width)
length = width;
if (getHorizontalMaxScroll() != 0)
{
pos = ((width - length) * mHScroll)
/ getHorizontalMaxScroll();
}
else
{
pos = 0;
}
}
return Rect(w2 + pos, mDimension.height - mScrollbarWidth,
length, mScrollbarWidth);
}
Rect ScrollArea::getUpButtonDimension() const
{
if (!mVBarVisible || !mShowButtons)
return Rect(0, 0, 0, 0);
return Rect(mDimension.width - mScrollbarWidth, 0,
mScrollbarWidth, mScrollbarWidth);
}
Rect ScrollArea::getDownButtonDimension() const
{
if (!mVBarVisible || !mShowButtons)
return Rect(0, 0, 0, 0);
if (mVBarVisible && mHBarVisible)
{
return Rect(mDimension.width - mScrollbarWidth,
mDimension.height - mScrollbarWidth*2,
mScrollbarWidth,
mScrollbarWidth);
}
return Rect(mDimension.width - mScrollbarWidth,
mDimension.height - mScrollbarWidth,
mScrollbarWidth,
mScrollbarWidth);
}
Rect ScrollArea::getLeftButtonDimension() const
{
if (!mHBarVisible || !mShowButtons)
return Rect(0, 0, 0, 0);
return Rect(0, mDimension.height - mScrollbarWidth,
mScrollbarWidth, mScrollbarWidth);
}
Rect ScrollArea::getRightButtonDimension() const
{
if (!mHBarVisible || !mShowButtons)
return Rect(0, 0, 0, 0);
if (mVBarVisible && mHBarVisible)
{
return Rect(mDimension.width - mScrollbarWidth*2,
mDimension.height - mScrollbarWidth,
mScrollbarWidth,
mScrollbarWidth);
}
return Rect(mDimension.width - mScrollbarWidth,
mDimension.height - mScrollbarWidth,
mScrollbarWidth,
mScrollbarWidth);
}
void ScrollArea::setContent(Widget* widget)
{
if (widget)
{
clear();
add(widget);
widget->setPosition(0, 0);
}
else
{
clear();
}
checkPolicies();
}
Widget* ScrollArea::getContent()
{
if (!mWidgets.empty())
return *mWidgets.begin();
return nullptr;
}
void ScrollArea::setHorizontalScrollPolicy(const ScrollPolicy hPolicy)
{
mHPolicy = hPolicy;
checkPolicies();
}
void ScrollArea::setVerticalScrollPolicy(const ScrollPolicy vPolicy)
{
mVPolicy = vPolicy;
checkPolicies();
}
void ScrollArea::setScrollPolicy(const ScrollPolicy hPolicy,
const ScrollPolicy vPolicy)
{
mHPolicy = hPolicy;
mVPolicy = vPolicy;
checkPolicies();
}
void ScrollArea::setVerticalScrollAmount(const int vScroll)
{
const int max = getVerticalMaxScroll();
mVScroll = vScroll;
if (vScroll > max)
mVScroll = max;
if (vScroll < 0)
mVScroll = 0;
}
void ScrollArea::setHorizontalScrollAmount(int hScroll)
{
const int max = getHorizontalMaxScroll();
mHScroll = hScroll;
if (hScroll > max)
mHScroll = max;
else if (hScroll < 0)
mHScroll = 0;
}
void ScrollArea::setScrollAmount(const int hScroll, const int vScroll)
{
setHorizontalScrollAmount(hScroll);
setVerticalScrollAmount(vScroll);
}
int ScrollArea::getHorizontalMaxScroll()
{
checkPolicies();
const Widget *const content = getContent();
if (!content)
return 0;
const int value = content->getWidth() - getChildrenArea().width +
2 * content->getFrameSize();
if (value < 0)
return 0;
return value;
}
int ScrollArea::getVerticalMaxScroll()
{
checkPolicies();
const Widget *const content = getContent();
if (!content)
return 0;
int value;
value = content->getHeight() - getChildrenArea().height +
2 * content->getFrameSize();
if (value < 0)
return 0;
return value;
}
void ScrollArea::setScrollbarWidth(const int width)
{
if (width > 0)
mScrollbarWidth = width;
}
void ScrollArea::showWidgetPart(Widget *const widget, const Rect &area)
{
const Widget *const content = getContent();
if (widget != content || !content)
return;
BasicContainer::showWidgetPart(widget, area);
setHorizontalScrollAmount(content->getFrameSize()
- content->getX());
setVerticalScrollAmount(content->getFrameSize()
- content->getY());
}
Rect ScrollArea::getChildrenArea()
{
const Rect area = Rect(0, 0,
mVBarVisible ? (getWidth() - mScrollbarWidth) : getWidth(),
mHBarVisible ? (getHeight() - mScrollbarWidth) : getHeight());
if (area.width < 0 || area.height < 0)
return Rect();
return area;
}
Widget *ScrollArea::getWidgetAt(int x, int y)
{
if (getChildrenArea().isPointInRect(x, y))
return getContent();
return nullptr;
}
void ScrollArea::setWidth(int width)
{
Widget::setWidth(width);
checkPolicies();
}
void ScrollArea::setHeight(int height)
{
Widget::setHeight(height);
checkPolicies();
}
void ScrollArea::setDimension(const Rect& dimension)
{
Widget::setDimension(dimension);
checkPolicies();
}
void ScrollArea::mouseWheelMovedUp(MouseEvent& event)
{
if (event.isConsumed())
return;
setVerticalScrollAmount(getVerticalScrollAmount()
- getChildrenArea().height / 8);
event.consume();
}
void ScrollArea::mouseWheelMovedDown(MouseEvent& event)
{
if (event.isConsumed())
return;
setVerticalScrollAmount(getVerticalScrollAmount()
+ getChildrenArea().height / 8);
event.consume();
}
void ScrollArea::checkPolicies()
{
const int w = getWidth();
const int h = getHeight();
mHBarVisible = false;
mVBarVisible = false;
const Widget *const content = getContent();
if (!content)
{
mHBarVisible = (mHPolicy == SHOW_ALWAYS);
mVBarVisible = (mVPolicy == SHOW_ALWAYS);
return;
}
if (mHPolicy == SHOW_AUTO &&
mVPolicy == SHOW_AUTO)
{
if (content->getWidth() <= w
&& content->getHeight() <= h)
{
mHBarVisible = false;
mVBarVisible = false;
}
if (content->getWidth() > w)
{
mHBarVisible = true;
}
if ((content->getHeight() > h)
|| (mHBarVisible && content->getHeight()
> h - mScrollbarWidth))
{
mVBarVisible = true;
}
if (mVBarVisible && content->getWidth() > w - mScrollbarWidth)
mHBarVisible = true;
return;
}
switch (mHPolicy)
{
case SHOW_NEVER:
mHBarVisible = false;
break;
case SHOW_ALWAYS:
mHBarVisible = true;
break;
case SHOW_AUTO:
if (mVPolicy == SHOW_NEVER)
{
mHBarVisible = (content->getWidth() > w);
}
else // (mVPolicy == SHOW_ALWAYS)
{
mHBarVisible = (content->getWidth()
> w - mScrollbarWidth);
}
break;
default:
break;
}
switch (mVPolicy)
{
case SHOW_NEVER:
mVBarVisible = false;
break;
case SHOW_ALWAYS:
mVBarVisible = true;
break;
case SHOW_AUTO:
if (mHPolicy == SHOW_NEVER)
{
mVBarVisible = (content->getHeight() > h);
}
else // (mHPolicy == SHOW_ALWAYS)
{
mVBarVisible = (content->getHeight()
> h - mScrollbarWidth);
}
break;
default:
break;
}
}
bool ScrollArea::isSelectable() const noexcept2
{
if (mVBarVisible || mHBarVisible)
return true;
return Widget::isSelectable();
}