2015-11-29 02:55:30 +00:00
|
|
|
#include "Specter/ScrollView.hpp"
|
2015-12-30 02:20:24 +00:00
|
|
|
#include "Specter/ViewResources.hpp"
|
|
|
|
#include "Specter/RootView.hpp"
|
2015-11-29 02:55:30 +00:00
|
|
|
|
|
|
|
namespace Specter
|
|
|
|
{
|
2015-12-29 22:58:09 +00:00
|
|
|
#define MAX_SCROLL_SPEED 100
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2015-12-30 02:20:24 +00:00
|
|
|
ScrollView::ScrollView(ViewResources& res, View& parentView, Style style)
|
2016-01-07 00:39:18 +00:00
|
|
|
: View(res, parentView), m_style(style), m_sideButtonBind(*this, rootView().viewManager())
|
2015-11-29 02:55:30 +00:00
|
|
|
{
|
2015-12-30 02:20:24 +00:00
|
|
|
m_vertsBuf = res.m_factory->newDynamicBuffer(boo::BufferUse::Vertex, sizeof(SolidShaderVert), 4);
|
|
|
|
|
|
|
|
if (!res.m_viewRes.m_texVtxFmt)
|
|
|
|
{
|
|
|
|
boo::VertexElementDescriptor vdescs[] =
|
|
|
|
{
|
|
|
|
{m_vertsBuf, nullptr, boo::VertexSemantic::Position4},
|
|
|
|
{m_vertsBuf, nullptr, boo::VertexSemantic::Color}
|
|
|
|
};
|
|
|
|
m_vtxFmt = res.m_factory->newVertexFormat(2, vdescs);
|
|
|
|
boo::IGraphicsBuffer* bufs[] = {m_viewVertBlockBuf};
|
|
|
|
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_solidShader,
|
|
|
|
m_vtxFmt, m_vertsBuf, nullptr,
|
|
|
|
nullptr, 1, bufs, 0, nullptr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
boo::IGraphicsBuffer* bufs[] = {m_viewVertBlockBuf};
|
|
|
|
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_solidShader,
|
|
|
|
res.m_viewRes.m_texVtxFmt,
|
|
|
|
m_vertsBuf, nullptr,
|
|
|
|
nullptr, 1, bufs, 0, nullptr);
|
|
|
|
}
|
2015-12-29 02:02:43 +00:00
|
|
|
commitResources(res);
|
2016-01-07 00:39:18 +00:00
|
|
|
|
|
|
|
if (style == Style::SideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].m_view.reset(new Button(res, *this, &m_sideButtonBind, "<"));
|
|
|
|
m_sideButtons[1].m_view.reset(new Button(res, *this, &m_sideButtonBind, ">"));
|
|
|
|
}
|
2015-12-29 02:02:43 +00:00
|
|
|
}
|
|
|
|
|
2015-12-30 22:04:58 +00:00
|
|
|
bool ScrollView::_scroll(const boo::SScrollDelta& scroll)
|
2015-12-29 02:02:43 +00:00
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_contentView.m_view)
|
2015-12-29 22:58:09 +00:00
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
float ratioX = subRect().size[0] / float(m_contentView.m_view->nominalWidth());
|
|
|
|
float ratioY = subRect().size[1] / float(m_contentView.m_view->nominalHeight());
|
2015-12-30 02:20:24 +00:00
|
|
|
|
|
|
|
float pf = rootView().viewRes().pixelFactor();
|
|
|
|
double mult = 20.0 * pf;
|
2015-12-29 22:58:09 +00:00
|
|
|
if (scroll.isFine)
|
2015-12-30 02:20:24 +00:00
|
|
|
mult = 1.0 * pf;
|
|
|
|
|
2016-01-07 00:39:18 +00:00
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
if (ratioX >= 1.f)
|
|
|
|
{
|
|
|
|
m_scroll[0] = 0;
|
|
|
|
m_targetScroll[0] = 0;
|
|
|
|
m_drawSideButtons = false;
|
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_drawSideButtons = true;
|
|
|
|
m_targetScroll[0] += scroll.delta[0] * mult;
|
|
|
|
m_targetScroll[0] = std::min(m_targetScroll[0], 0);
|
|
|
|
int scrollWidth = m_contentView.m_view->nominalWidth() - scrollAreaWidth();
|
|
|
|
m_targetScroll[0] = std::max(m_targetScroll[0], -scrollWidth);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ratioY >= 1.f)
|
|
|
|
{
|
|
|
|
m_scroll[1] = 0;
|
|
|
|
m_targetScroll[1] = 0;
|
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_targetScroll[1] -= scroll.delta[1] * mult;
|
|
|
|
m_targetScroll[1] = std::max(m_targetScroll[1], 0);
|
|
|
|
int scrollHeight = m_contentView.m_view->nominalHeight() - subRect().size[1];
|
|
|
|
m_targetScroll[1] = std::min(m_targetScroll[1], scrollHeight);
|
|
|
|
}
|
2015-12-30 02:20:24 +00:00
|
|
|
|
2015-12-29 22:58:09 +00:00
|
|
|
if (scroll.isFine)
|
|
|
|
{
|
|
|
|
m_scroll[0] = m_targetScroll[0];
|
|
|
|
m_scroll[1] = m_targetScroll[1];
|
2016-01-07 00:39:18 +00:00
|
|
|
ret = true;
|
2015-12-29 22:58:09 +00:00
|
|
|
}
|
2016-01-07 00:39:18 +00:00
|
|
|
|
|
|
|
return ret;
|
2015-12-29 22:58:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_scroll[0] = 0;
|
|
|
|
m_scroll[1] = 0;
|
|
|
|
m_targetScroll[0] = 0;
|
|
|
|
m_targetScroll[1] = 0;
|
2016-01-07 00:39:18 +00:00
|
|
|
m_drawSideButtons = false;
|
2015-12-30 22:04:58 +00:00
|
|
|
return true;
|
2015-12-29 22:58:09 +00:00
|
|
|
}
|
2015-12-30 22:04:58 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-01-01 00:12:19 +00:00
|
|
|
void ScrollView::mouseDown(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
if (m_sideButtons[0].mouseDown(coord, button, mod) ||
|
|
|
|
m_sideButtons[1].mouseDown(coord, button, mod))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_contentView.mouseDown(coord, button, mod);
|
2016-01-01 00:12:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::mouseUp(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_style == Style::SideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].mouseUp(coord, button, mod);
|
|
|
|
m_sideButtons[1].mouseUp(coord, button, mod);
|
|
|
|
}
|
|
|
|
m_contentView.mouseUp(coord, button, mod);
|
2016-01-01 00:12:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::mouseMove(const boo::SWindowCoord& coord)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].mouseMove(coord);
|
|
|
|
m_sideButtons[1].mouseMove(coord);
|
|
|
|
}
|
|
|
|
m_contentView.mouseMove(coord);
|
2016-01-01 00:12:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::mouseEnter(const boo::SWindowCoord& coord)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].mouseEnter(coord);
|
|
|
|
m_sideButtons[1].mouseEnter(coord);
|
|
|
|
}
|
|
|
|
m_contentView.mouseEnter(coord);
|
2016-01-01 00:12:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::mouseLeave(const boo::SWindowCoord& coord)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_style == Style::SideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].mouseLeave(coord);
|
|
|
|
m_sideButtons[1].mouseLeave(coord);
|
|
|
|
}
|
|
|
|
m_contentView.mouseLeave(coord);
|
2016-01-01 00:12:19 +00:00
|
|
|
}
|
|
|
|
|
2015-12-30 22:04:58 +00:00
|
|
|
void ScrollView::scroll(const boo::SWindowCoord& coord, const boo::SScrollDelta& scroll)
|
|
|
|
{
|
2016-01-01 00:12:19 +00:00
|
|
|
if (!scroll.isAccelerated)
|
|
|
|
{
|
|
|
|
boo::SScrollDelta newScroll = scroll;
|
|
|
|
m_consecutiveScroll[m_consecutiveIdx][0] += scroll.delta[0];
|
|
|
|
m_consecutiveScroll[m_consecutiveIdx][1] += scroll.delta[1];
|
|
|
|
newScroll.delta[0] = 0;
|
|
|
|
newScroll.delta[1] = 0;
|
|
|
|
for (size_t i=0 ; i<16 ; ++i)
|
|
|
|
{
|
|
|
|
newScroll.delta[0] += m_consecutiveScroll[i][0];
|
|
|
|
newScroll.delta[1] += m_consecutiveScroll[i][1];
|
|
|
|
}
|
|
|
|
if (_scroll(newScroll))
|
|
|
|
updateSize();
|
|
|
|
return;
|
|
|
|
}
|
2015-12-30 22:04:58 +00:00
|
|
|
if (_scroll(scroll))
|
|
|
|
updateSize();
|
2015-12-29 22:58:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::think()
|
|
|
|
{
|
2016-01-01 00:12:19 +00:00
|
|
|
m_consecutiveIdx = (m_consecutiveIdx+1) % 16;
|
|
|
|
m_consecutiveScroll[m_consecutiveIdx][0] = 0.0;
|
|
|
|
m_consecutiveScroll[m_consecutiveIdx][1] = 0.0;
|
|
|
|
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_sideButtonState != SideButtonState::None)
|
|
|
|
{
|
|
|
|
if (m_sideButtonState == SideButtonState::ScrollLeft)
|
|
|
|
m_targetScroll[0] -= 3;
|
|
|
|
else if (m_sideButtonState == SideButtonState::ScrollRight)
|
|
|
|
m_targetScroll[0] += 3;
|
|
|
|
m_targetScroll[0] = std::min(m_targetScroll[0], 0);
|
|
|
|
int scrollWidth = m_contentView.m_view->nominalWidth() - scrollAreaWidth();
|
|
|
|
m_targetScroll[0] = std::max(m_targetScroll[0], -scrollWidth);
|
|
|
|
}
|
|
|
|
|
2015-12-29 22:58:09 +00:00
|
|
|
bool update = false;
|
2015-12-30 02:20:24 +00:00
|
|
|
float pf = rootView().viewRes().pixelFactor();
|
2015-12-29 22:58:09 +00:00
|
|
|
|
2015-12-30 02:20:24 +00:00
|
|
|
int xSpeed = std::max(1, std::min(abs(m_targetScroll[0] - m_scroll[0]) / int(5*pf), int(pf*MAX_SCROLL_SPEED)));
|
2015-12-29 22:58:09 +00:00
|
|
|
if (m_scroll[0] < m_targetScroll[0])
|
|
|
|
{
|
|
|
|
m_scroll[0] += xSpeed;
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
else if (m_scroll[0] > m_targetScroll[0])
|
|
|
|
{
|
|
|
|
m_scroll[0] -= xSpeed;
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
|
2015-12-30 02:20:24 +00:00
|
|
|
int ySpeed = std::max(1, std::min(abs(m_targetScroll[1] - m_scroll[1]) / int(5*pf), int(pf*MAX_SCROLL_SPEED)));
|
2015-12-29 22:58:09 +00:00
|
|
|
if (m_scroll[1] < m_targetScroll[1])
|
|
|
|
{
|
|
|
|
m_scroll[1] += ySpeed;
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
else if (m_scroll[1] > m_targetScroll[1])
|
|
|
|
{
|
|
|
|
m_scroll[1] -= ySpeed;
|
|
|
|
update = true;
|
|
|
|
}
|
2015-12-30 02:20:24 +00:00
|
|
|
|
2015-12-29 22:58:09 +00:00
|
|
|
if (update)
|
|
|
|
updateSize();
|
2015-12-29 02:02:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::resized(const boo::SWindowRect& root, const boo::SWindowRect& sub)
|
|
|
|
{
|
|
|
|
View::resized(root, sub);
|
2015-12-30 22:04:58 +00:00
|
|
|
_scroll({});
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_contentView.m_view)
|
2015-12-29 02:02:43 +00:00
|
|
|
{
|
|
|
|
boo::SWindowRect cRect = sub;
|
|
|
|
cRect.location[0] += m_scroll[0];
|
2016-01-07 00:39:18 +00:00
|
|
|
cRect.location[1] += sub.size[1] - m_contentView.m_view->nominalHeight() + m_scroll[1];
|
|
|
|
cRect.size[0] = m_contentView.m_view->nominalWidth();
|
|
|
|
cRect.size[1] = m_contentView.m_view->nominalHeight();
|
|
|
|
m_contentView.m_scissorRect = sub;
|
|
|
|
if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
int width0 = m_sideButtons[0].m_view->nominalWidth() + 2;
|
|
|
|
int width1 = m_sideButtons[1].m_view->nominalWidth();
|
|
|
|
cRect.location[0] += width0;
|
|
|
|
cRect.size[0] -= (width0 + width1);
|
|
|
|
|
|
|
|
m_contentView.m_scissorRect.location[0] += width0;
|
|
|
|
m_contentView.m_scissorRect.size[0] -= (width0 + width1);
|
|
|
|
}
|
|
|
|
m_contentView.m_view->resized(root, cRect, m_contentView.m_scissorRect);
|
|
|
|
|
2015-12-30 02:20:24 +00:00
|
|
|
|
|
|
|
if (m_style == Style::ThinIndicator)
|
|
|
|
{
|
|
|
|
float ratio = sub.size[1] / float(cRect.size[1]);
|
|
|
|
m_drawInd = ratio < 1.f;
|
|
|
|
if (m_drawInd)
|
|
|
|
{
|
|
|
|
float pf = rootView().viewRes().pixelFactor();
|
|
|
|
int barHeight = sub.size[1] * ratio;
|
|
|
|
int scrollHeight = sub.size[1] - barHeight;
|
|
|
|
float prog = m_scroll[1] / float(cRect.size[1] - sub.size[1]);
|
|
|
|
int x = sub.size[0];
|
|
|
|
int y = sub.size[1] - scrollHeight * prog;
|
|
|
|
m_verts[0].m_pos.assign(x, y, 0);
|
|
|
|
m_verts[1].m_pos.assign(x, y-barHeight, 0);
|
|
|
|
m_verts[2].m_pos.assign(x+2*pf, y, 0);
|
|
|
|
m_verts[3].m_pos.assign(x+2*pf, y-barHeight, 0);
|
|
|
|
const Zeus::CColor& color = rootView().themeData().scrollIndicator();
|
|
|
|
for (int i=0 ; i<4 ; ++i)
|
|
|
|
m_verts[i].m_color = color;
|
|
|
|
m_vertsBuf->load(m_verts, sizeof(m_verts));
|
|
|
|
}
|
|
|
|
}
|
2016-01-07 00:39:18 +00:00
|
|
|
else if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
boo::SWindowRect bRect = sub;
|
|
|
|
bRect.size[0] = m_sideButtons[0].m_view->nominalWidth();
|
|
|
|
bRect.size[1] = m_sideButtons[0].m_view->nominalHeight();
|
|
|
|
m_sideButtons[0].m_view->resized(root, bRect);
|
|
|
|
|
|
|
|
bRect.size[0] = m_sideButtons[1].m_view->nominalWidth();
|
|
|
|
bRect.size[1] = m_sideButtons[1].m_view->nominalHeight();
|
|
|
|
bRect.location[0] += sub.size[0] - bRect.size[0];
|
|
|
|
m_sideButtons[1].m_view->resized(root, bRect);
|
|
|
|
}
|
2015-12-29 02:02:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScrollView::draw(boo::IGraphicsCommandQueue* gfxQ)
|
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
if (m_contentView.m_view)
|
2015-12-30 02:20:24 +00:00
|
|
|
{
|
2016-01-07 00:39:18 +00:00
|
|
|
m_contentView.m_view->draw(gfxQ);
|
2015-12-30 02:20:24 +00:00
|
|
|
|
|
|
|
if (m_style == Style::ThinIndicator && m_drawInd)
|
|
|
|
{
|
|
|
|
gfxQ->setShaderDataBinding(m_shaderBinding);
|
|
|
|
gfxQ->setDrawPrimitive(boo::Primitive::TriStrips);
|
|
|
|
gfxQ->draw(0, 4);
|
|
|
|
}
|
2016-01-07 00:39:18 +00:00
|
|
|
else if (m_style == Style::SideButtons && m_drawSideButtons)
|
|
|
|
{
|
|
|
|
m_sideButtons[0].m_view->draw(gfxQ);
|
|
|
|
m_sideButtons[1].m_view->draw(gfxQ);
|
|
|
|
}
|
2015-12-30 02:20:24 +00:00
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|