#include #include "Specter/Button.hpp" #include "Specter/ViewResources.hpp" #include "Specter/RootView.hpp" namespace Specter { static LogVisor::LogModule Log("Specter::Button"); void Button::Resources::init(boo::IGraphicsDataFactory* factory, const IThemeData& theme) { } Button::Button(ViewResources& res, View& parentView, IButtonBinding* controlBinding, const std::string& text, Style style, RectangleConstraint constraint) : Button(res, parentView, controlBinding, text, res.themeData().uiText(), style, constraint) {} Button::Button(ViewResources& res, View& parentView, IButtonBinding* controlBinding, const std::string& text, const Zeus::CColor& textColor, Style style, RectangleConstraint constraint) : Control(res, parentView, controlBinding), m_style(style), m_textColor(textColor), m_textStr(text), m_constraint(constraint) { m_vertsBinding.initSolid(res, 40, m_viewVertBlockBuf); commitResources(res); m_buttonTarget.m_view.reset(new ButtonTarget(res, *this)); m_menuTarget.m_view.reset(new MenuTarget(res, *this)); if (style == Style::Block) { m_verts[0].m_color = res.themeData().button1Inactive(); m_verts[1].m_color = res.themeData().button2Inactive(); m_verts[2].m_color = res.themeData().button1Inactive(); m_verts[3].m_color = res.themeData().button2Inactive(); m_verts[4].m_color = res.themeData().button2Inactive(); for (int i=5 ; i<28 ; ++i) m_verts[i].m_color = res.themeData().button2Inactive(); m_verts[31].m_color = rootView().themeData().button1Inactive(); m_verts[32].m_color = rootView().themeData().button2Inactive(); m_verts[33].m_color = rootView().themeData().button1Inactive(); m_verts[34].m_color = rootView().themeData().button2Inactive(); for (int i=35 ; i<39 ; ++i) m_verts[i].m_color = res.themeData().button2Inactive(); } else { for (int i=0 ; i<4 ; ++i) m_verts[i].m_color = Zeus::CColor::skClear; for (int i=31 ; i<35 ; ++i) m_verts[i].m_color = Zeus::CColor::skClear; } for (int i=28 ; i<31 ; ++i) m_verts[i].m_color = m_textColor; m_vertsBinding.load(m_verts, sizeof(m_verts)); if (controlBinding) m_menuStyle = controlBinding->menuStyle(this); m_text.reset(new TextView(res, *this, res.m_mainFont, TextView::Alignment::Center)); setText(m_textStr); } void Button::setText(const std::string &text) { setText(text, m_textColor); } void Button::setText(const std::string& text, const Zeus::CColor& textColor) { m_textStr = text; m_textColor = textColor; m_text->typesetGlyphs(text, textColor); float pf = rootView().viewRes().pixelFactor(); int width, height; if (m_style == Style::Block) { std::pair constraint = m_constraint.solve(m_text->nominalWidth() + 12 * pf, 20 * pf); width = constraint.first; height = constraint.second; m_verts[0].m_pos.assign(1, height+1, 0); m_verts[1].m_pos.assign(1, 1, 0); m_verts[2].m_pos.assign(width+1, height+1, 0); m_verts[3].m_pos.assign(width+1, 1, 0); m_verts[4].m_pos.assign(width+1, 1, 0); m_textWidth = width; if (m_menuStyle != IButtonBinding::MenuStyle::None) width += 16*pf; m_verts[5].m_pos.assign(1, height+1, 0); m_verts[6].m_pos.assign(1, height+1, 0); m_verts[7].m_pos.assign(0, height+1, 0); m_verts[8].m_pos.assign(1, 1, 0); m_verts[9].m_pos.assign(0, 1, 0); m_verts[10].m_pos.assign(0, 1, 0); m_verts[11].m_pos.assign(width+2, height+1, 0); m_verts[12].m_pos.assign(width+2, height+1, 0); m_verts[13].m_pos.assign(width+1, height+1, 0); m_verts[14].m_pos.assign(width+2, 1, 0); m_verts[15].m_pos.assign(width+1, 1, 0); m_verts[16].m_pos.assign(width+1, 1, 0); m_verts[17].m_pos.assign(1, height+2, 0); m_verts[18].m_pos.assign(1, height+2, 0); m_verts[19].m_pos.assign(1, height+1, 0); m_verts[20].m_pos.assign(width+1, height+2, 0); m_verts[21].m_pos.assign(width+1, height+1, 0); m_verts[22].m_pos.assign(width+1, height+1, 0); m_verts[23].m_pos.assign(1, 1, 0); m_verts[24].m_pos.assign(1, 1, 0); m_verts[25].m_pos.assign(1, 0, 0); m_verts[26].m_pos.assign(width+1, 1, 0); m_verts[27].m_pos.assign(width+1, 0, 0); int arrowX = m_textWidth + 5*pf; int arrowY = 7*pf; m_verts[28].m_pos.assign(arrowX + 4*pf, arrowY + 1*pf, 0); m_verts[29].m_pos.assign(arrowX, arrowY + 5*pf, 0); m_verts[30].m_pos.assign(arrowX + 8*pf, arrowY + 5*pf, 0); m_verts[31].m_pos.assign(m_textWidth+1, height+1, 0); m_verts[32].m_pos.assign(m_textWidth+1, 1, 0); m_verts[33].m_pos.assign(width+1, height+1, 0); m_verts[34].m_pos.assign(width+1, 1, 0); m_verts[35].m_pos.assign(m_textWidth, height+1, 0); m_verts[36].m_pos.assign(m_textWidth, 1, 0); m_verts[37].m_pos.assign(m_textWidth+1, height+1, 0); m_verts[38].m_pos.assign(m_textWidth+1, 1, 0); m_vertsBinding.load(m_verts, sizeof(m_verts)); } else { width = m_text->nominalWidth(); height = 10 * pf; m_verts[0].m_pos.assign(1*pf, -1*pf, 0); m_verts[1].m_pos.assign(1*pf, -2*pf, 0); m_verts[2].m_pos.assign(width, -1*pf, 0); m_verts[3].m_pos.assign(width, -2*pf, 0); int arrowX = width + 5*pf; m_verts[28].m_pos.assign(arrowX + 4*pf, 1*pf, 0); m_verts[29].m_pos.assign(arrowX, 5*pf, 0); m_verts[30].m_pos.assign(arrowX + 8*pf, 5*pf, 0); m_textWidth = width; int arrowLineWidth = 7*pf; if (m_menuStyle != IButtonBinding::MenuStyle::None) { width += 13*pf; if (m_menuStyle == IButtonBinding::MenuStyle::Primary) { arrowLineWidth = width; arrowX = 1*pf; } } m_verts[31].m_pos.assign(arrowX, -1*pf, 0); m_verts[32].m_pos.assign(arrowX, -2*pf, 0); m_verts[33].m_pos.assign(arrowX + arrowLineWidth, -1*pf, 0); m_verts[34].m_pos.assign(arrowX + arrowLineWidth, -2*pf, 0); m_vertsBinding.load(m_verts, sizeof(m_verts)); } m_nomWidth = width; m_nomHeight = height; } void Button::colorGlyphs(const Zeus::CColor& newColor) { m_textColor = newColor; m_text->colorGlyphs(newColor); for (int i=28 ; i<31 ; ++i) m_verts[i].m_color = newColor; m_vertsBinding.load(m_verts, sizeof(m_verts)); } void Button::ButtonTarget::setInactive() { if (m_button.m_style == Style::Block) { m_button.m_verts[0].m_color = rootView().themeData().button1Inactive(); m_button.m_verts[1].m_color = rootView().themeData().button2Inactive(); m_button.m_verts[2].m_color = rootView().themeData().button1Inactive(); m_button.m_verts[3].m_color = rootView().themeData().button2Inactive(); m_button.m_verts[4].m_color = rootView().themeData().button2Inactive(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=0 ; i<4 ; ++i) m_button.m_verts[i].m_color = Zeus::CColor::skClear; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); m_button.m_text->colorGlyphs(m_button.m_textColor); } } void Button::MenuTarget::setInactive() { if (m_button.m_style == Style::Block) { m_button.m_verts[31].m_color = rootView().themeData().button1Inactive(); m_button.m_verts[32].m_color = rootView().themeData().button2Inactive(); m_button.m_verts[33].m_color = rootView().themeData().button1Inactive(); m_button.m_verts[34].m_color = rootView().themeData().button2Inactive(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=28 ; i<31 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; for (int i=31 ; i<35 ; ++i) m_button.m_verts[i].m_color = Zeus::CColor::skClear; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } } void Button::ButtonTarget::setHover() { if (m_button.m_style == Style::Block) { m_button.m_verts[0].m_color = rootView().themeData().button1Hover(); m_button.m_verts[1].m_color = rootView().themeData().button2Hover(); m_button.m_verts[2].m_color = rootView().themeData().button1Hover(); m_button.m_verts[3].m_color = rootView().themeData().button2Hover(); m_button.m_verts[4].m_color = rootView().themeData().button2Hover(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=0 ; i<4 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); m_button.m_text->colorGlyphs(m_button.m_textColor); } } void Button::MenuTarget::setHover() { if (m_button.m_style == Style::Block) { m_button.m_verts[31].m_color = rootView().themeData().button1Hover(); m_button.m_verts[32].m_color = rootView().themeData().button2Hover(); m_button.m_verts[33].m_color = rootView().themeData().button1Hover(); m_button.m_verts[34].m_color = rootView().themeData().button2Hover(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=28 ; i<31 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; for (int i=31 ; i<35 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } } void Button::ButtonTarget::setPressed() { if (m_button.m_style == Style::Block) { m_button.m_verts[0].m_color = rootView().themeData().button1Press(); m_button.m_verts[1].m_color = rootView().themeData().button2Press(); m_button.m_verts[2].m_color = rootView().themeData().button1Press(); m_button.m_verts[3].m_color = rootView().themeData().button2Press(); m_button.m_verts[4].m_color = rootView().themeData().button2Press(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=0 ; i<4 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); m_button.m_text->colorGlyphs(m_button.m_textColor); } } void Button::MenuTarget::setPressed() { if (m_button.m_style == Style::Block) { m_button.m_verts[31].m_color = rootView().themeData().button1Press(); m_button.m_verts[32].m_color = rootView().themeData().button2Press(); m_button.m_verts[33].m_color = rootView().themeData().button1Press(); m_button.m_verts[34].m_color = rootView().themeData().button2Press(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=28 ; i<31 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; for (int i=31 ; i<35 ; ++i) m_button.m_verts[i].m_color = m_button.m_textColor; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } } void Button::ButtonTarget::setDisabled() { if (m_button.m_style == Style::Block) { m_button.m_verts[0].m_color = rootView().themeData().button1Disabled(); m_button.m_verts[1].m_color = rootView().themeData().button2Disabled(); m_button.m_verts[2].m_color = rootView().themeData().button1Disabled(); m_button.m_verts[3].m_color = rootView().themeData().button2Disabled(); m_button.m_verts[4].m_color = rootView().themeData().button2Disabled(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { for (int i=0 ; i<4 ; ++i) m_button.m_verts[i].m_color = Zeus::CColor::skClear; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); Zeus::CColor dimText = m_button.m_textColor; dimText[3] *= 0.5; m_button.m_text->colorGlyphs(dimText); } } void Button::MenuTarget::setDisabled() { if (m_button.m_style == Style::Block) { m_button.m_verts[31].m_color = rootView().themeData().button1Disabled(); m_button.m_verts[32].m_color = rootView().themeData().button2Disabled(); m_button.m_verts[33].m_color = rootView().themeData().button1Disabled(); m_button.m_verts[34].m_color = rootView().themeData().button2Disabled(); m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } else { Zeus::CColor dimText = m_button.m_textColor; dimText[3] *= 0.5; for (int i=28 ; i<31 ; ++i) m_button.m_verts[i].m_color = dimText; for (int i=31 ; i<35 ; ++i) m_button.m_verts[i].m_color = Zeus::CColor::skClear; m_button.m_vertsBinding.load(m_button.m_verts, sizeof(m_button.m_verts)); } } void Button::mouseDown(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { if (m_menuStyle != IButtonBinding::MenuStyle::Primary) m_buttonTarget.mouseDown(coord, button, mod); m_menuTarget.mouseDown(coord, button, mod); } void Button::ButtonTarget::mouseDown(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { m_pressed = true; setPressed(); if (m_button.m_controlBinding) static_cast(*m_button.m_controlBinding).down(&m_button, coord); } void Button::MenuTarget::mouseDown(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { m_pressed = true; setPressed(); if (m_hovered) { Log.report(LogVisor::Info, "button menu '%s' activated", m_button.m_textStr.c_str()); if (m_button.m_controlBinding) { m_button.m_modalMenu.m_view = static_cast(*m_button.m_controlBinding).buildMenu(&m_button); rootView().setActiveMenuButton(&m_button); m_button.updateSize(); } } } void Button::mouseUp(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { if (m_menuStyle != IButtonBinding::MenuStyle::Primary) m_buttonTarget.mouseUp(coord, button, mod); m_menuTarget.mouseUp(coord, button, mod); } void Button::ButtonTarget::mouseUp(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { if (m_pressed) { if (m_button.m_controlBinding) static_cast(*m_button.m_controlBinding).up(&m_button, coord); if (m_hovered) { Log.report(LogVisor::Info, "button '%s' activated", m_button.m_textStr.c_str()); if (m_button.m_controlBinding) static_cast(*m_button.m_controlBinding).activated(&m_button, coord); } m_pressed = false; } if (m_hovered) setHover(); else setInactive(); } void Button::MenuTarget::mouseUp(const boo::SWindowCoord& coord, boo::EMouseButton button, boo::EModifierKey mod) { m_pressed = false; if (m_hovered) setHover(); else setInactive(); } void Button::mouseMove(const boo::SWindowCoord& coord) { if (m_menuStyle != IButtonBinding::MenuStyle::Primary) m_buttonTarget.mouseMove(coord); m_menuTarget.mouseMove(coord); } void Button::ButtonTarget::mouseEnter(const boo::SWindowCoord& coord) { m_hovered = true; if (m_pressed) setPressed(); else setHover(); } void Button::MenuTarget::mouseEnter(const boo::SWindowCoord& coord) { m_hovered = true; if (m_pressed) setPressed(); else setHover(); } void Button::mouseLeave(const boo::SWindowCoord& coord) { if (m_menuStyle != IButtonBinding::MenuStyle::Primary) m_buttonTarget.mouseLeave(coord); m_menuTarget.mouseLeave(coord); } void Button::ButtonTarget::mouseLeave(const boo::SWindowCoord& coord) { m_hovered = false; setInactive(); } void Button::MenuTarget::mouseLeave(const boo::SWindowCoord& coord) { m_hovered = false; setInactive(); } void Button::closeMenu(const boo::SWindowCoord& coord) { rootView().unsetActiveMenuButton(this); m_modalMenu.m_view.reset(); m_menuTarget.mouseMove(coord); } void Button::think() { if (m_modalMenu.m_view) m_modalMenu.m_view->think(); } void Button::resized(const boo::SWindowRect& root, const boo::SWindowRect& sub) { View::resized(root, sub); boo::SWindowRect textRect = sub; float pf = rootView().viewRes().pixelFactor(); if (m_style == Style::Block) textRect.location[1] += 7 * pf; textRect.location[0] += m_textWidth / 2; textRect.size[0] = m_textWidth; textRect.size[1] = m_nomHeight; m_text->resized(root, textRect); if (m_style == Style::Block) { if (m_menuStyle == IButtonBinding::MenuStyle::None) { m_buttonTarget.m_view->resized(root, sub); } else if (m_menuStyle == IButtonBinding::MenuStyle::Primary) { m_menuTarget.m_view->resized(root, sub); } else { boo::SWindowRect targetRect = sub; targetRect.size[0] = m_textWidth; m_buttonTarget.m_view->resized(root, targetRect); targetRect.location[0] += targetRect.size[0]; targetRect.size[0] = 16*pf; m_menuTarget.m_view->resized(root, targetRect); } } else { if (m_menuStyle == IButtonBinding::MenuStyle::Primary) { boo::SWindowRect targetRect = sub; targetRect.size[0] = m_nomWidth; targetRect.size[1] = m_nomHeight; m_menuTarget.m_view->resized(root, targetRect); } else { boo::SWindowRect targetRect = sub; targetRect.size[0] = m_textWidth + 3*pf; targetRect.size[1] = m_nomHeight; m_buttonTarget.m_view->resized(root, targetRect); targetRect.location[0] += targetRect.size[0]; targetRect.size[0] = 15*pf; m_menuTarget.m_view->resized(root, targetRect); } } if (m_modalMenu.m_view) { boo::SWindowRect menuRect = sub; if (m_style == Style::Text) menuRect.location[1] -= 6*pf; m_modalMenu.m_view->resized(root, menuRect); } } void Button::draw(boo::IGraphicsCommandQueue* gfxQ) { View::draw(gfxQ); gfxQ->setShaderDataBinding(m_vertsBinding); gfxQ->setDrawPrimitive(boo::Primitive::TriStrips); if (m_style == Style::Block) { gfxQ->draw(0, 28); if (m_menuStyle != IButtonBinding::MenuStyle::None) { gfxQ->draw(31, 4); gfxQ->draw(28, 3); gfxQ->draw(35, 4); } } else { gfxQ->draw(0, 4); if (m_menuStyle != IButtonBinding::MenuStyle::None) { gfxQ->draw(28, 3); gfxQ->draw(31, 4); } } m_text->draw(gfxQ); if (m_modalMenu.m_view) m_modalMenu.m_view->draw(gfxQ); } }