2012-04-14 17:47:44 +02:00
# include "tooltips.hpp"
2012-04-15 21:14:14 +02:00
# include <boost/lexical_cast.hpp>
2012-04-14 17:47:44 +02:00
2012-07-03 15:32:38 +02:00
# include <OgreResourceGroupManager.h>
2012-05-28 09:46:05 +02:00
# include <components/settings/settings.hpp>
2012-07-03 12:30:50 +02:00
# include "../mwbase/world.hpp"
# include "../mwbase/environment.hpp"
2012-08-12 18:11:09 +02:00
# include "../mwbase/windowmanager.hpp"
2012-07-03 12:30:50 +02:00
# include "../mwworld/class.hpp"
2012-08-28 17:30:34 +02:00
# include "map_window.hpp"
2012-07-03 12:30:50 +02:00
# include "widgets.hpp"
2012-09-14 14:34:18 +02:00
# include "inventorywindow.hpp"
2012-07-03 12:30:50 +02:00
2012-04-14 17:47:44 +02:00
using namespace MWGui ;
using namespace MyGUI ;
2012-08-12 18:11:09 +02:00
ToolTips : : ToolTips ( MWBase : : WindowManager * windowManager ) :
2012-07-03 11:34:20 +02:00
Layout ( " openmw_tooltips.layout " )
2012-04-14 17:47:44 +02:00
, mGameMode ( true )
2012-04-15 21:14:14 +02:00
, mWindowManager ( windowManager )
2012-04-16 15:00:44 +02:00
, mFullHelp ( false )
2012-05-13 11:25:35 +02:00
, mEnabled ( true )
2012-05-14 21:37:43 +02:00
, mFocusToolTipX ( 0.0 )
, mFocusToolTipY ( 0.0 )
2012-05-28 09:46:05 +02:00
, mDelay ( 0.0 )
, mRemainingDelay ( 0.0 )
, mLastMouseX ( 0 )
, mLastMouseY ( 0 )
2012-04-14 17:47:44 +02:00
{
getWidget ( mDynamicToolTipBox , " DynamicToolTipBox " ) ;
mDynamicToolTipBox - > setVisible ( false ) ;
// turn off mouse focus so that getMouseFocusWidget returns the correct widget,
// even if the mouse is over the tooltip
mDynamicToolTipBox - > setNeedMouseFocus ( false ) ;
mMainWidget - > setNeedMouseFocus ( false ) ;
2012-05-28 09:46:05 +02:00
mDelay = Settings : : Manager : : getFloat ( " tooltip delay " , " GUI " ) ;
mRemainingDelay = mDelay ;
2012-04-14 17:47:44 +02:00
}
2012-05-13 11:25:35 +02:00
void ToolTips : : setEnabled ( bool enabled )
{
mEnabled = enabled ;
}
2012-04-14 17:47:44 +02:00
void ToolTips : : onFrame ( float frameDuration )
{
2012-05-24 15:11:45 +02:00
while ( mDynamicToolTipBox - > getChildCount ( ) )
{
MyGUI : : Gui : : getInstance ( ) . destroyWidget ( mDynamicToolTipBox - > getChildAt ( 0 ) ) ;
}
2012-04-16 00:16:35 +02:00
2012-05-20 22:06:54 +02:00
// start by hiding everything
for ( unsigned int i = 0 ; i < mMainWidget - > getChildCount ( ) ; + + i )
{
mMainWidget - > getChildAt ( i ) - > setVisible ( false ) ;
}
2012-04-14 17:47:44 +02:00
const IntSize & viewSize = RenderManager : : getInstance ( ) . getViewSize ( ) ;
2012-05-13 11:25:35 +02:00
if ( ! mEnabled )
{
return ;
}
2012-04-15 17:10:08 +02:00
if ( ! mGameMode )
{
2012-05-28 09:46:05 +02:00
const MyGUI : : IntPoint & mousePos = InputManager : : getInstance ( ) . getMousePosition ( ) ;
2012-05-29 12:35:03 +02:00
2012-06-02 12:25:24 +02:00
if ( mWindowManager - > getWorldMouseOver ( ) & & ( ( mWindowManager - > getMode ( ) = = GM_Console )
| | ( mWindowManager - > getMode ( ) = = GM_Container )
| | ( mWindowManager - > getMode ( ) = = GM_Inventory ) ) )
2012-05-28 09:46:05 +02:00
{
2012-06-02 12:25:24 +02:00
std : : string handle = MWBase : : Environment : : get ( ) . getWorld ( ) - > getFacedHandle ( ) ;
try
{
mFocusObject = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPtrViaHandle ( handle ) ;
}
2012-07-07 01:14:18 +02:00
catch ( std : : exception /* & e */ )
2012-06-02 12:25:24 +02:00
{
return ;
}
MyGUI : : IntSize tooltipSize = getToolTipViaPtr ( true ) ;
IntPoint tooltipPosition = InputManager : : getInstance ( ) . getMousePosition ( ) + IntPoint ( 0 , 24 ) ;
// make the tooltip stay completely in the viewport
if ( ( tooltipPosition . left + tooltipSize . width ) > viewSize . width )
{
tooltipPosition . left = viewSize . width - tooltipSize . width ;
}
if ( ( tooltipPosition . top + tooltipSize . height ) > viewSize . height )
{
tooltipPosition . top = viewSize . height - tooltipSize . height ;
}
setCoord ( tooltipPosition . left , tooltipPosition . top , tooltipSize . width , tooltipSize . height ) ;
2012-05-28 09:46:05 +02:00
}
2012-06-02 12:25:24 +02:00
2012-05-28 09:46:05 +02:00
else
{
2012-06-02 12:25:24 +02:00
const MyGUI : : IntPoint & lastPressed = InputManager : : getInstance ( ) . getLastPressedPosition ( MyGUI : : MouseButton : : Left ) ;
2012-05-28 09:46:05 +02:00
2012-06-02 12:25:24 +02:00
if ( mousePos = = lastPressed ) // mouseclick makes tooltip disappear
return ;
2012-05-28 09:46:05 +02:00
2012-06-02 12:25:24 +02:00
if ( mousePos . left = = mLastMouseX & & mousePos . top = = mLastMouseY )
{
mRemainingDelay - = frameDuration ;
}
else
{
mRemainingDelay = mDelay ;
}
mLastMouseX = mousePos . left ;
mLastMouseY = mousePos . top ;
2012-04-15 21:14:14 +02:00
2012-06-02 12:25:24 +02:00
if ( mRemainingDelay > 0 )
return ;
2012-04-16 00:16:35 +02:00
2012-06-02 12:25:24 +02:00
Widget * focus = InputManager : : getInstance ( ) . getMouseFocusWidget ( ) ;
if ( focus = = 0 )
{
2012-05-27 06:39:10 +02:00
return ;
2012-06-02 12:25:24 +02:00
}
2012-05-27 06:39:10 +02:00
2012-06-02 12:25:24 +02:00
IntSize tooltipSize ;
2012-04-18 16:53:56 +02:00
2012-06-02 12:25:24 +02:00
// try to go 1 level up until there is a widget that has tooltip
// this is necessary because some skin elements are actually separate widgets
int i = 0 ;
while ( ! focus - > isUserString ( " ToolTipType " ) )
2012-05-27 06:39:10 +02:00
{
2012-06-02 12:25:24 +02:00
focus = focus - > getParent ( ) ;
if ( ! focus )
return ;
+ + i ;
2012-05-27 06:39:10 +02:00
}
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
std : : string type = focus - > getUserString ( " ToolTipType " ) ;
std : : string text = focus - > getUserString ( " ToolTipText " ) ;
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
if ( type = = " " )
{
return ;
2012-05-21 05:14:07 +02:00
}
2012-08-28 17:30:34 +02:00
// special handling for markers on the local map: the tooltip should only be visible
// if the marker is not hidden due to the fog of war.
if ( focus - > getUserString ( " IsMarker " ) = = " true " )
{
LocalMapBase : : MarkerPosition pos = * focus - > getUserData < LocalMapBase : : MarkerPosition > ( ) ;
if ( ! MWBase : : Environment : : get ( ) . getWorld ( ) - > isPositionExplored ( pos . nX , pos . nY , pos . cellX , pos . cellY , pos . interior ) )
return ;
}
if ( type = = " ItemPtr " )
2012-06-02 12:25:24 +02:00
{
mFocusObject = * focus - > getUserData < MWWorld : : Ptr > ( ) ;
tooltipSize = getToolTipViaPtr ( false ) ;
}
2012-09-14 14:34:18 +02:00
else if ( type = = " AvatarItemSelection " )
{
MyGUI : : IntCoord avatarPos = mWindowManager - > getInventoryWindow ( ) - > getAvatarScreenCoord ( ) ;
MyGUI : : IntPoint relMousePos = MyGUI : : InputManager : : getInstance ( ) . getMousePosition ( ) - MyGUI : : IntPoint ( avatarPos . left , avatarPos . top ) ;
int realX = int ( float ( relMousePos . left ) / float ( avatarPos . width ) * 512.f ) ;
int realY = int ( float ( relMousePos . top ) / float ( avatarPos . height ) * 1024.f ) ;
2012-09-15 00:57:29 +02:00
MWWorld : : Ptr item = mWindowManager - > getInventoryWindow ( ) - > getAvatarSelectedItem ( realX , realY ) ;
2012-09-14 14:34:18 +02:00
mFocusObject = item ;
2012-09-14 15:33:24 +02:00
if ( ! mFocusObject . isEmpty ( ) )
tooltipSize = getToolTipViaPtr ( false ) ;
2012-09-14 14:34:18 +02:00
}
2012-06-02 12:25:24 +02:00
else if ( type = = " Spell " )
{
ToolTipInfo info ;
const ESM : : Spell * spell = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . spells . find ( focus - > getUserString ( " Spell " ) ) ;
2012-09-17 11:37:50 +04:00
info . caption = spell - > mName ;
2012-06-02 12:25:24 +02:00
Widgets : : SpellEffectList effects ;
2012-09-17 11:37:50 +04:00
std : : vector < ESM : : ENAMstruct > : : const_iterator end = spell - > mEffects . mList . end ( ) ;
for ( std : : vector < ESM : : ENAMstruct > : : const_iterator it = spell - > mEffects . mList . begin ( ) ; it ! = end ; + + it )
2012-06-02 12:25:24 +02:00
{
Widgets : : SpellEffectParams params ;
2012-09-17 11:37:50 +04:00
params . mEffectID = it - > mEffectID ;
params . mSkill = it - > mSkill ;
params . mAttribute = it - > mAttribute ;
params . mDuration = it - > mDuration ;
params . mMagnMin = it - > mMagnMin ;
params . mMagnMax = it - > mMagnMax ;
params . mRange = it - > mRange ;
params . mIsConstant = ( spell - > mData . mType = = ESM : : Spell : : ST_Ability ) ;
2012-09-09 23:06:24 +02:00
params . mNoTarget = false ;
2012-06-02 12:25:24 +02:00
effects . push_back ( params ) ;
}
info . effects = effects ;
tooltipSize = createToolTip ( info ) ;
}
else if ( type = = " Layout " )
2012-05-20 22:06:54 +02:00
{
2012-06-02 12:25:24 +02:00
// tooltip defined in the layout
MyGUI : : Widget * tooltip ;
getWidget ( tooltip , focus - > getUserString ( " ToolTipLayout " ) ) ;
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
tooltip - > setVisible ( true ) ;
if ( ! tooltip - > isUserString ( " DontResize " ) )
{
tooltip - > setCoord ( 0 , 0 , 450 , 300 ) ; // this is the maximum width of the tooltip before it starts word-wrapping
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
tooltipSize = MyGUI : : IntSize ( 0 , tooltip - > getSize ( ) . height ) ;
}
else
tooltipSize = tooltip - > getSize ( ) ;
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
std : : map < std : : string , std : : string > userStrings = focus - > getUserStrings ( ) ;
for ( std : : map < std : : string , std : : string > : : iterator it = userStrings . begin ( ) ;
it ! = userStrings . end ( ) ; + + it )
{
if ( it - > first = = " ToolTipType "
2012-08-28 17:30:34 +02:00
| | it - > first = = " ToolTipLayout "
| | it - > first = = " IsMarker " )
2012-06-02 12:25:24 +02:00
continue ;
2012-05-20 22:06:54 +02:00
2012-06-02 12:25:24 +02:00
size_t underscorePos = it - > first . find ( " _ " ) ;
std : : string propertyKey = it - > first . substr ( 0 , underscorePos ) ;
std : : string widgetName = it - > first . substr ( underscorePos + 1 , it - > first . size ( ) - ( underscorePos + 1 ) ) ;
MyGUI : : Widget * w ;
getWidget ( w , widgetName ) ;
w - > setProperty ( propertyKey , it - > second ) ;
2012-05-20 22:06:54 +02:00
}
2012-06-02 12:25:24 +02:00
for ( unsigned int i = 0 ; i < tooltip - > getChildCount ( ) ; + + i )
2012-05-20 22:06:54 +02:00
{
2012-06-02 12:25:24 +02:00
MyGUI : : Widget * w = tooltip - > getChildAt ( i ) ;
if ( w - > isUserString ( " AutoResizeHorizontal " ) )
2012-05-21 01:58:11 +02:00
{
2012-06-02 12:25:24 +02:00
MyGUI : : TextBox * text = w - > castType < MyGUI : : TextBox > ( ) ;
tooltipSize . width = std : : max ( tooltipSize . width , w - > getLeft ( ) + text - > getTextSize ( ) . width + 8 ) ;
2012-05-21 01:58:11 +02:00
}
2012-06-02 12:25:24 +02:00
else if ( ! tooltip - > isUserString ( " DontResize " ) )
tooltipSize . width = std : : max ( tooltipSize . width , w - > getLeft ( ) + w - > getWidth ( ) + 8 ) ;
if ( w - > isUserString ( " AutoResizeVertical " ) )
2012-05-21 01:58:11 +02:00
{
2012-06-02 12:25:24 +02:00
MyGUI : : TextBox * text = w - > castType < MyGUI : : TextBox > ( ) ;
int height = text - > getTextSize ( ) . height ;
if ( height > w - > getHeight ( ) )
{
tooltipSize + = MyGUI : : IntSize ( 0 , height - w - > getHeight ( ) ) ;
}
if ( height < w - > getHeight ( ) )
{
tooltipSize - = MyGUI : : IntSize ( 0 , w - > getHeight ( ) - height ) ;
}
2012-05-21 01:58:11 +02:00
}
2012-05-20 22:06:54 +02:00
}
2012-06-02 12:25:24 +02:00
tooltip - > setCoord ( 0 , 0 , tooltipSize . width , tooltipSize . height ) ;
2012-05-20 22:06:54 +02:00
}
2012-06-02 12:25:24 +02:00
else
throw std : : runtime_error ( " unknown tooltip type " ) ;
2012-04-14 17:47:44 +02:00
2012-06-02 12:25:24 +02:00
IntPoint tooltipPosition = InputManager : : getInstance ( ) . getMousePosition ( ) + IntPoint ( 0 , 24 ) ;
2012-04-15 17:10:08 +02:00
2012-06-02 12:25:24 +02:00
// make the tooltip stay completely in the viewport
if ( ( tooltipPosition . left + tooltipSize . width ) > viewSize . width )
{
tooltipPosition . left = viewSize . width - tooltipSize . width ;
}
if ( ( tooltipPosition . top + tooltipSize . height ) > viewSize . height )
{
tooltipPosition . top = viewSize . height - tooltipSize . height ;
}
2012-04-15 17:10:08 +02:00
2012-06-02 12:25:24 +02:00
setCoord ( tooltipPosition . left , tooltipPosition . top , tooltipSize . width , tooltipSize . height ) ;
}
2012-04-14 17:47:44 +02:00
}
2012-04-15 17:10:08 +02:00
else
2012-04-14 17:47:44 +02:00
{
2012-04-15 17:10:08 +02:00
if ( ! mFocusObject . isEmpty ( ) )
{
2012-04-15 21:14:14 +02:00
IntSize tooltipSize = getToolTipViaPtr ( ) ;
2012-04-15 17:10:08 +02:00
2012-05-14 21:37:43 +02:00
setCoord ( viewSize . width / 2 - tooltipSize . width / 2 ,
std : : max ( 0 , int ( mFocusToolTipY * viewSize . height - tooltipSize . height ) ) ,
2012-04-15 21:14:14 +02:00
tooltipSize . width ,
tooltipSize . height ) ;
2012-05-20 22:06:54 +02:00
mDynamicToolTipBox - > setVisible ( true ) ;
2012-04-15 21:14:14 +02:00
}
}
}
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
void ToolTips : : enterGameMode ( )
{
mGameMode = true ;
}
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
void ToolTips : : enterGuiMode ( )
{
mGameMode = false ;
}
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
void ToolTips : : setFocusObject ( const MWWorld : : Ptr & focus )
{
mFocusObject = focus ;
}
2012-04-15 17:10:08 +02:00
2012-05-12 20:28:12 +02:00
IntSize ToolTips : : getToolTipViaPtr ( bool image )
2012-04-15 21:14:14 +02:00
{
// this the maximum width of the tooltip before it starts word-wrapping
setCoord ( 0 , 0 , 300 , 300 ) ;
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
IntSize tooltipSize ;
2012-04-15 17:10:08 +02:00
2012-04-16 22:58:16 +02:00
const MWWorld : : Class & object = MWWorld : : Class : : get ( mFocusObject ) ;
if ( ! object . hasToolTip ( mFocusObject ) )
2012-04-15 21:14:14 +02:00
{
2012-04-16 22:58:16 +02:00
mDynamicToolTipBox - > setVisible ( false ) ;
2012-04-15 21:14:14 +02:00
}
2012-04-16 22:58:16 +02:00
else
2012-04-15 21:14:14 +02:00
{
2012-04-16 22:58:16 +02:00
mDynamicToolTipBox - > setVisible ( true ) ;
2012-04-15 17:10:08 +02:00
2012-04-24 02:02:03 +02:00
ToolTipInfo info = object . getToolTipInfo ( mFocusObject ) ;
2012-05-12 20:28:12 +02:00
if ( ! image )
info . icon = " " ;
2012-04-18 16:53:56 +02:00
tooltipSize = createToolTip ( info ) ;
2012-04-15 21:14:14 +02:00
}
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
return tooltipSize ;
}
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
void ToolTips : : findImageExtension ( std : : string & image )
{
int len = image . size ( ) ;
if ( len < 4 ) return ;
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
if ( ! Ogre : : ResourceGroupManager : : getSingleton ( ) . resourceExists ( Ogre : : ResourceGroupManager : : AUTODETECT_RESOURCE_GROUP_NAME , image ) )
{
// Change texture extension to .dds
image [ len - 3 ] = ' d ' ;
image [ len - 2 ] = ' d ' ;
image [ len - 1 ] = ' s ' ;
}
}
2012-04-15 17:10:08 +02:00
2012-05-11 11:52:07 +02:00
IntSize ToolTips : : createToolTip ( const MWGui : : ToolTipInfo & info )
2012-04-15 21:14:14 +02:00
{
2012-05-20 22:06:54 +02:00
mDynamicToolTipBox - > setVisible ( true ) ;
2012-04-18 16:53:56 +02:00
std : : string caption = info . caption ;
std : : string image = info . icon ;
int imageSize = ( image ! = " " ) ? 32 : 0 ;
std : : string text = info . text ;
2012-04-15 21:14:14 +02:00
// remove the first newline (easier this way)
2012-04-18 16:53:56 +02:00
if ( text . size ( ) > 0 & & text [ 0 ] = = ' \n ' )
text . erase ( 0 , 1 ) ;
2012-04-15 17:10:08 +02:00
2012-09-23 19:36:37 +02:00
const ESM : : Enchantment * enchant = 0 ;
2012-04-30 00:57:41 +02:00
const ESMS : : ESMStore & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
if ( info . enchant ! = " " )
{
enchant = store . enchants . search ( info . enchant ) ;
2012-09-17 11:37:50 +04:00
if ( enchant - > mData . mType = = ESM : : Enchantment : : CastOnce )
2012-09-22 21:35:57 +02:00
text + = " \n #{sItemCastOnce} " ;
2012-09-17 11:37:50 +04:00
else if ( enchant - > mData . mType = = ESM : : Enchantment : : WhenStrikes )
2012-09-22 21:35:57 +02:00
text + = " \n #{sItemCastWhenStrikes} " ;
2012-09-17 11:37:50 +04:00
else if ( enchant - > mData . mType = = ESM : : Enchantment : : WhenUsed )
2012-09-22 21:35:57 +02:00
text + = " \n #{sItemCastWhenUsed} " ;
2012-09-17 11:37:50 +04:00
else if ( enchant - > mData . mType = = ESM : : Enchantment : : ConstantEffect )
2012-09-22 21:35:57 +02:00
text + = " \n #{sItemCastConstant} " ;
2012-04-30 00:57:41 +02:00
}
2012-04-17 16:49:28 +02:00
// this the maximum width of the tooltip before it starts word-wrapping
setCoord ( 0 , 0 , 300 , 300 ) ;
const IntPoint padding ( 8 , 8 ) ;
2012-04-30 22:37:23 +02:00
const int imageCaptionHPadding = ( caption ! = " " ? 8 : 0 ) ;
const int imageCaptionVPadding = ( caption ! = " " ? 4 : 0 ) ;
2012-04-17 16:49:28 +02:00
2012-04-15 21:14:14 +02:00
std : : string realImage = " icons \\ " + image ;
findImageExtension ( realImage ) ;
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
EditBox * captionWidget = mDynamicToolTipBox - > createWidget < EditBox > ( " NormalText " , IntCoord ( 0 , 0 , 300 , 300 ) , Align : : Left | Align : : Top , " ToolTipCaption " ) ;
captionWidget - > setProperty ( " Static " , " true " ) ;
captionWidget - > setCaption ( caption ) ;
2012-04-17 16:49:28 +02:00
IntSize captionSize = captionWidget - > getTextSize ( ) ;
2012-04-30 22:37:23 +02:00
int captionHeight = std : : max ( caption ! = " " ? captionSize . height : 0 , imageSize ) ;
2012-04-17 16:49:28 +02:00
EditBox * textWidget = mDynamicToolTipBox - > createWidget < EditBox > ( " SandText " , IntCoord ( 0 , captionHeight + imageCaptionVPadding , 300 , 300 - captionHeight - imageCaptionVPadding ) , Align : : Stretch , " ToolTipText " ) ;
2012-04-15 21:14:14 +02:00
textWidget - > setProperty ( " Static " , " true " ) ;
textWidget - > setProperty ( " MultiLine " , " true " ) ;
2012-04-16 00:16:35 +02:00
textWidget - > setProperty ( " WordWrap " , " true " ) ;
2012-09-22 21:35:57 +02:00
textWidget - > setCaptionWithReplacing ( text ) ;
2012-04-17 16:49:28 +02:00
textWidget - > setTextAlign ( Align : : HCenter | Align : : Top ) ;
2012-04-15 21:14:14 +02:00
IntSize textSize = textWidget - > getTextSize ( ) ;
2012-04-15 17:10:08 +02:00
2012-04-15 21:14:14 +02:00
captionSize + = IntSize ( imageSize , 0 ) ; // adjust for image
2012-04-17 16:49:28 +02:00
IntSize totalSize = IntSize ( std : : max ( textSize . width , captionSize . width + ( ( image ! = " " ) ? imageCaptionHPadding : 0 ) ) ,
2012-04-18 16:53:56 +02:00
( ( text ! = " " ) ? textSize . height + imageCaptionVPadding : 0 ) + captionHeight ) ;
2012-04-15 17:10:08 +02:00
2012-05-24 14:47:57 +02:00
if ( ! info . effects . empty ( ) )
2012-04-30 01:53:22 +02:00
{
Widget * effectArea = mDynamicToolTipBox - > createWidget < Widget > ( " " ,
IntCoord ( 0 , totalSize . height , 300 , 300 - totalSize . height ) ,
Align : : Stretch , " ToolTipEffectArea " ) ;
IntCoord coord ( 0 , 6 , totalSize . width , 24 ) ;
/**
* \ todo
2012-07-03 12:30:50 +02:00
* the various potion effects should appear in the tooltip depending if the player
2012-04-30 01:53:22 +02:00
* has enough skill in alchemy to know about the effects of this potion .
*/
Widgets : : MWEffectListPtr effectsWidget = effectArea - > createWidget < Widgets : : MWEffectList >
( " MW_StatName " , coord , Align : : Default , " ToolTipEffectsWidget " ) ;
effectsWidget - > setWindowManager ( mWindowManager ) ;
effectsWidget - > setEffectList ( info . effects ) ;
std : : vector < MyGUI : : WidgetPtr > effectItems ;
2012-09-09 23:06:24 +02:00
effectsWidget - > createEffectWidgets ( effectItems , effectArea , coord , true , info . isPotion ? Widgets : : MWEffectList : : EF_NoTarget : 0 ) ;
2012-04-30 01:53:22 +02:00
totalSize . height + = coord . top - 6 ;
totalSize . width = std : : max ( totalSize . width , coord . width ) ;
}
2012-04-30 00:57:41 +02:00
if ( info . enchant ! = " " )
2012-04-17 16:49:28 +02:00
{
2012-09-23 19:36:37 +02:00
assert ( enchant ) ;
2012-04-30 00:57:41 +02:00
Widget * enchantArea = mDynamicToolTipBox - > createWidget < Widget > ( " " ,
IntCoord ( 0 , totalSize . height , 300 , 300 - totalSize . height ) ,
Align : : Stretch , " ToolTipEnchantArea " ) ;
IntCoord coord ( 0 , 6 , totalSize . width , 24 ) ;
2012-04-30 01:08:10 +02:00
Widgets : : MWEffectListPtr enchantWidget = enchantArea - > createWidget < Widgets : : MWEffectList >
2012-04-30 00:57:41 +02:00
( " MW_StatName " , coord , Align : : Default , " ToolTipEnchantWidget " ) ;
enchantWidget - > setWindowManager ( mWindowManager ) ;
2012-09-17 11:37:50 +04:00
enchantWidget - > setEffectList ( Widgets : : MWEffectList : : effectListFromESM ( & enchant - > mEffects ) ) ;
2012-04-30 00:57:41 +02:00
std : : vector < MyGUI : : WidgetPtr > enchantEffectItems ;
2012-09-17 11:37:50 +04:00
int flag = ( enchant - > mData . mType = = ESM : : Enchantment : : ConstantEffect ) ? Widgets : : MWEffectList : : EF_Constant : 0 ;
2012-04-30 01:53:22 +02:00
enchantWidget - > createEffectWidgets ( enchantEffectItems , enchantArea , coord , true , flag ) ;
2012-04-30 00:57:41 +02:00
totalSize . height + = coord . top - 6 ;
totalSize . width = std : : max ( totalSize . width , coord . width ) ;
2012-09-17 11:37:50 +04:00
if ( enchant - > mData . mType = = ESM : : Enchantment : : WhenStrikes
| | enchant - > mData . mType = = ESM : : Enchantment : : WhenUsed )
2012-04-30 00:57:41 +02:00
{
/// \todo store the current enchantment charge somewhere
2012-09-17 11:37:50 +04:00
int charge = enchant - > mData . mCharge ;
2012-04-30 00:57:41 +02:00
const int chargeWidth = 204 ;
TextBox * chargeText = enchantArea - > createWidget < TextBox > ( " SandText " , IntCoord ( 0 , 0 , 10 , 18 ) , Align : : Default , " ToolTipEnchantChargeText " ) ;
2012-09-22 21:35:57 +02:00
chargeText - > setCaptionWithReplacing ( " #{sCharges} " ) ;
2012-10-01 00:23:49 +04:00
2012-04-30 00:57:41 +02:00
const int chargeTextWidth = chargeText - > getTextSize ( ) . width + 5 ;
const int chargeAndTextWidth = chargeWidth + chargeTextWidth ;
2012-05-19 16:28:10 +02:00
totalSize . width = std : : max ( totalSize . width , chargeAndTextWidth ) ;
2012-04-30 00:57:41 +02:00
chargeText - > setCoord ( ( totalSize . width - chargeAndTextWidth ) / 2 , coord . top + 6 , chargeTextWidth , 18 ) ;
IntCoord chargeCoord ;
if ( totalSize . width < chargeWidth )
{
totalSize . width = chargeWidth ;
chargeCoord = IntCoord ( 0 , coord . top + 6 , chargeWidth , 18 ) ;
}
else
{
chargeCoord = IntCoord ( ( totalSize . width - chargeAndTextWidth ) / 2 + chargeTextWidth , coord . top + 6 , chargeWidth , 18 ) ;
}
Widgets : : MWDynamicStatPtr chargeWidget = enchantArea - > createWidget < Widgets : : MWDynamicStat >
( " MW_ChargeBar " , chargeCoord , Align : : Default , " ToolTipEnchantCharge " ) ;
chargeWidget - > setValue ( charge , charge ) ;
totalSize . height + = 24 ;
}
2012-04-17 16:49:28 +02:00
}
2012-04-15 17:10:08 +02:00
2012-04-17 16:49:28 +02:00
captionWidget - > setCoord ( ( totalSize . width - captionSize . width ) / 2 + imageSize ,
( captionHeight - captionSize . height ) / 2 ,
captionSize . width - imageSize ,
captionSize . height ) ;
2012-04-15 17:10:08 +02:00
2012-04-17 16:49:28 +02:00
captionWidget - > setPosition ( captionWidget - > getPosition ( ) + padding ) ;
textWidget - > setPosition ( textWidget - > getPosition ( ) + IntPoint ( 0 , padding . top ) ) ; // only apply vertical padding, the horizontal works automatically due to Align::HCenter
2012-04-15 17:10:08 +02:00
2012-04-30 00:57:41 +02:00
if ( image ! = " " )
{
ImageBox * imageWidget = mDynamicToolTipBox - > createWidget < ImageBox > ( " ImageBox " ,
IntCoord ( ( totalSize . width - captionSize . width - imageCaptionHPadding ) / 2 , 0 , imageSize , imageSize ) ,
Align : : Left | Align : : Top , " ToolTipImage " ) ;
imageWidget - > setImageTexture ( realImage ) ;
imageWidget - > setPosition ( imageWidget - > getPosition ( ) + padding ) ;
}
2012-04-17 16:49:28 +02:00
totalSize + = IntSize ( padding . left * 2 , padding . top * 2 ) ;
2012-04-14 17:47:44 +02:00
2012-04-17 16:49:28 +02:00
return totalSize ;
2012-04-16 00:16:35 +02:00
}
2012-04-15 21:14:14 +02:00
std : : string ToolTips : : toString ( const float value )
2012-04-14 17:47:44 +02:00
{
2012-04-15 21:14:14 +02:00
std : : ostringstream stream ;
stream < < std : : setprecision ( 3 ) < < value ;
return stream . str ( ) ;
2012-04-14 17:47:44 +02:00
}
2012-04-15 21:14:14 +02:00
std : : string ToolTips : : toString ( const int value )
2012-04-14 17:47:44 +02:00
{
2012-04-15 21:14:14 +02:00
std : : ostringstream stream ;
stream < < value ;
return stream . str ( ) ;
2012-04-14 17:47:44 +02:00
}
2012-04-15 17:10:08 +02:00
2012-04-16 22:58:16 +02:00
std : : string ToolTips : : getValueString ( const int value , const std : : string & prefix )
2012-04-15 17:10:08 +02:00
{
2012-04-15 21:14:14 +02:00
if ( value = = 0 )
return " " ;
else
2012-04-16 22:58:16 +02:00
return " \n " + prefix + " : " + toString ( value ) ;
}
std : : string ToolTips : : getMiscString ( const std : : string & text , const std : : string & prefix )
{
if ( text = = " " )
return " " ;
else
return " \n " + prefix + " : " + text ;
2012-04-15 17:10:08 +02:00
}
2012-04-16 15:00:44 +02:00
2012-05-12 13:46:03 +02:00
std : : string ToolTips : : getCountString ( const int value )
{
if ( value = = 1 )
return " " ;
else
return " ( " + boost : : lexical_cast < std : : string > ( value ) + " ) " ;
}
2012-04-16 15:00:44 +02:00
void ToolTips : : toggleFullHelp ( )
{
mFullHelp = ! mFullHelp ;
}
2012-04-16 22:58:16 +02:00
bool ToolTips : : getFullHelp ( ) const
{
return mFullHelp ;
}
2012-05-14 21:37:43 +02:00
void ToolTips : : setFocusObjectScreenCoords ( float min_x , float min_y , float max_x , float max_y )
{
mFocusToolTipX = ( min_x + max_x ) / 2 ;
mFocusToolTipY = min_y ;
}
2012-05-27 06:39:10 +02:00
void ToolTips : : createSkillToolTip ( MyGUI : : Widget * widget , int skillId )
{
if ( skillId = = - 1 )
return ;
const std : : string & skillNameId = ESMS : : Skill : : sSkillNameIds [ skillId ] ;
const ESM : : Skill * skill = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . skills . search ( skillId ) ;
assert ( skill ) ;
2012-09-17 11:37:50 +04:00
const ESM : : Attribute * attr = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . attributes . search ( skill - > mData . mAttribute ) ;
2012-05-27 06:39:10 +02:00
assert ( attr ) ;
std : : string icon = " icons \\ k \\ " + ESM : : Skill : : sIconNames [ skillId ] ;
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " SkillNoProgressToolTip " ) ;
widget - > setUserString ( " Caption_SkillNoProgressName " , " #{ " + skillNameId + " } " ) ;
2012-09-17 11:37:50 +04:00
widget - > setUserString ( " Caption_SkillNoProgressDescription " , skill - > mDescription ) ;
widget - > setUserString ( " Caption_SkillNoProgressAttribute " , " #{sGoverningAttribute}: #{ " + attr - > mName + " } " ) ;
2012-05-27 06:39:10 +02:00
widget - > setUserString ( " ImageTexture_SkillNoProgressImage " , icon ) ;
}
void ToolTips : : createAttributeToolTip ( MyGUI : : Widget * widget , int attributeId )
{
if ( attributeId = = - 1 )
return ;
2012-09-17 11:37:50 +04:00
std : : string icon = ESM : : Attribute : : sAttributeIcons [ attributeId ] ;
std : : string name = ESM : : Attribute : : sGmstAttributeIds [ attributeId ] ;
std : : string desc = ESM : : Attribute : : sGmstAttributeDescIds [ attributeId ] ;
2012-05-27 06:39:10 +02:00
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " AttributeToolTip " ) ;
widget - > setUserString ( " Caption_AttributeName " , " #{ " + name + " } " ) ;
widget - > setUserString ( " Caption_AttributeDescription " , " #{ " + desc + " } " ) ;
widget - > setUserString ( " ImageTexture_AttributeImage " , icon ) ;
}
void ToolTips : : createSpecializationToolTip ( MyGUI : : Widget * widget , const std : : string & name , int specId )
{
widget - > setUserString ( " Caption_CenteredCaption " , name ) ;
std : : string specText ;
// get all skills of this specialisation
std : : map < int , ESM : : Skill > skills = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . skills . list ;
for ( std : : map < int , ESM : : Skill > : : const_iterator it = skills . begin ( ) ;
it ! = skills . end ( ) ; + + it )
{
2012-09-17 11:37:50 +04:00
if ( it - > second . mData . mSpecialization = = specId )
specText + = std : : string ( " \n #{ " ) + ESM : : Skill : : sSkillNameIds [ it - > second . mIndex ] + " } " ;
2012-05-27 06:39:10 +02:00
}
widget - > setUserString ( " Caption_CenteredCaptionText " , specText ) ;
widget - > setUserString ( " ToolTipLayout " , " TextWithCenteredCaptionToolTip " ) ;
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
}
void ToolTips : : createBirthsignToolTip ( MyGUI : : Widget * widget , const std : : string & birthsignId )
{
const ESM : : BirthSign * sign = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . birthSigns . find ( birthsignId ) ;
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " BirthSignToolTip " ) ;
2012-09-17 11:37:50 +04:00
std : : string image = sign - > mTexture ;
2012-05-27 06:39:10 +02:00
image . replace ( image . size ( ) - 3 , 3 , " dds " ) ;
widget - > setUserString ( " ImageTexture_BirthSignImage " , " textures \\ " + image ) ;
std : : string text ;
2012-09-17 11:37:50 +04:00
text + = sign - > mName ;
text + = " \n #BF9959 " + sign - > mDescription ;
2012-05-27 06:39:10 +02:00
std : : vector < std : : string > abilities , powers , spells ;
2012-09-17 11:37:50 +04:00
std : : vector < std : : string > : : const_iterator it = sign - > mPowers . mList . begin ( ) ;
std : : vector < std : : string > : : const_iterator end = sign - > mPowers . mList . end ( ) ;
2012-05-27 06:39:10 +02:00
for ( ; it ! = end ; + + it )
{
const std : : string & spellId = * it ;
const ESM : : Spell * spell = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . spells . search ( spellId ) ;
if ( ! spell )
continue ; // Skip spells which cannot be found
2012-09-17 11:37:50 +04:00
ESM : : Spell : : SpellType type = static_cast < ESM : : Spell : : SpellType > ( spell - > mData . mType ) ;
2012-05-27 06:39:10 +02:00
if ( type ! = ESM : : Spell : : ST_Spell & & type ! = ESM : : Spell : : ST_Ability & & type ! = ESM : : Spell : : ST_Power )
continue ; // We only want spell, ability and powers.
if ( type = = ESM : : Spell : : ST_Ability )
abilities . push_back ( spellId ) ;
else if ( type = = ESM : : Spell : : ST_Power )
powers . push_back ( spellId ) ;
else if ( type = = ESM : : Spell : : ST_Spell )
spells . push_back ( spellId ) ;
}
struct { const std : : vector < std : : string > & spells ; std : : string label ; } categories [ 3 ] = {
{ abilities , " sBirthsignmenu1 " } ,
{ powers , " sPowers " } ,
{ spells , " sBirthsignmenu2 " }
} ;
for ( int category = 0 ; category < 3 ; + + category )
{
for ( std : : vector < std : : string > : : const_iterator it = categories [ category ] . spells . begin ( ) ; it ! = categories [ category ] . spells . end ( ) ; + + it )
{
if ( it = = categories [ category ] . spells . begin ( ) )
{
text + = std : : string ( " \n #DDC79E " ) + std : : string ( " #{ " ) + categories [ category ] . label + " } " ;
}
const std : : string & spellId = * it ;
const ESM : : Spell * spell = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . spells . search ( spellId ) ;
2012-09-17 11:37:50 +04:00
text + = " \n #BF9959 " + spell - > mName ;
2012-05-27 06:39:10 +02:00
}
}
widget - > setUserString ( " Caption_BirthSignText " , text ) ;
}
void ToolTips : : createRaceToolTip ( MyGUI : : Widget * widget , const ESM : : Race * playerRace )
{
2012-09-17 11:37:50 +04:00
widget - > setUserString ( " Caption_CenteredCaption " , playerRace - > mName ) ;
widget - > setUserString ( " Caption_CenteredCaptionText " , playerRace - > mDescription ) ;
2012-05-27 06:39:10 +02:00
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " TextWithCenteredCaptionToolTip " ) ;
}
void ToolTips : : createClassToolTip ( MyGUI : : Widget * widget , const ESM : : Class & playerClass )
{
2012-09-17 11:37:50 +04:00
if ( playerClass . mName = = " " )
2012-05-27 06:39:10 +02:00
return ;
2012-09-17 11:37:50 +04:00
int spec = playerClass . mData . mSpecialization ;
2012-05-27 06:39:10 +02:00
std : : string specStr ;
if ( spec = = 0 )
specStr = " #{sSpecializationCombat} " ;
else if ( spec = = 1 )
specStr = " #{sSpecializationMagic} " ;
else if ( spec = = 2 )
specStr = " #{sSpecializationStealth} " ;
2012-09-17 11:37:50 +04:00
widget - > setUserString ( " Caption_ClassName " , playerClass . mName ) ;
widget - > setUserString ( " Caption_ClassDescription " , playerClass . mDescription ) ;
2012-05-27 06:39:10 +02:00
widget - > setUserString ( " Caption_ClassSpecialisation " , " #{sSpecialization}: " + specStr ) ;
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " ClassToolTip " ) ;
}
2012-05-28 09:46:05 +02:00
2012-09-23 00:36:20 +02:00
void ToolTips : : createMagicEffectToolTip ( MyGUI : : Widget * widget , short id )
{
const ESM : : MagicEffect * effect = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . magicEffects . find ( id ) ;
2012-10-01 23:33:07 +02:00
const std : : string & name = ESM : : MagicEffect : : effectIdToString ( id ) ;
2012-09-23 00:36:20 +02:00
2012-10-01 17:54:45 +02:00
std : : string icon = effect - > mIcon ;
2012-09-23 00:36:20 +02:00
int slashPos = icon . find ( " \\ " ) ;
icon . insert ( slashPos + 1 , " b_ " ) ;
icon [ icon . size ( ) - 3 ] = ' d ' ;
icon [ icon . size ( ) - 2 ] = ' d ' ;
icon [ icon . size ( ) - 1 ] = ' s ' ;
icon = " icons \\ " + icon ;
std : : vector < std : : string > schools ;
schools . push_back ( " #{sSchoolAlteration} " ) ;
schools . push_back ( " #{sSchoolConjuration} " ) ;
schools . push_back ( " #{sSchoolDestruction} " ) ;
schools . push_back ( " #{sSchoolIllusion} " ) ;
schools . push_back ( " #{sSchoolMysticism} " ) ;
schools . push_back ( " #{sSchoolRestoration} " ) ;
widget - > setUserString ( " ToolTipType " , " Layout " ) ;
widget - > setUserString ( " ToolTipLayout " , " MagicEffectToolTip " ) ;
widget - > setUserString ( " Caption_MagicEffectName " , " #{ " + name + " } " ) ;
2012-10-01 17:54:45 +02:00
widget - > setUserString ( " Caption_MagicEffectDescription " , effect - > mDescription ) ;
widget - > setUserString ( " Caption_MagicEffectSchool " , " #{sSchool}: " + schools [ effect - > mData . mSchool ] ) ;
2012-09-23 00:36:20 +02:00
widget - > setUserString ( " ImageTexture_MagicEffectImage " , icon ) ;
}
2012-05-28 09:46:05 +02:00
void ToolTips : : setDelay ( float delay )
{
mDelay = delay ;
mRemainingDelay = mDelay ;
}