2020-04-27 11:27:59 -04:00
// MIT License
// Copyright (c) 2020 Evan Pezent
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
2020-11-10 09:27:28 -05:00
// ImPlot v0.9 WIP
2020-05-11 07:12:22 -04:00
/*
API BREAKING CHANGES
= = = = = = = = = = = = = = = = = = = =
Occasionally introducing changes that are breaking the API . We try to make the breakage minor and easy to fix .
Below is a change - log of API breaking changes only . If you are using one of the functions listed , expect to have to fix some code .
When you are not sure about a old symbol or function name , try using the Search / Find function of your IDE to look for comments or references in all implot files .
You can read releases logs https : //github.com/epezent/implot/releases for more details.
2020-10-19 00:26:34 -04:00
- 2020 / 10 / 16 ( 0.8 ) - ImPlotStyleVar_InfoPadding was changed to ImPlotStyleVar_MousePosPadding
2020-09-10 16:59:08 -04:00
- 2020 / 09 / 10 ( 0.8 ) - The single array versions of PlotLine , PlotScatter , PlotStems , and PlotShaded were given additional arguments for x - scale and x0 .
2020-09-07 17:08:59 -04:00
- 2020 / 09 / 07 ( 0.8 ) - Plotting functions which accept a custom getter function pointer have been post - fixed with a G ( e . g . PlotLineG )
2020-09-06 02:38:18 -04:00
- 2020 / 09 / 06 ( 0.7 ) - Several flags under ImPlotFlags and ImPlotAxisFlags were inverted ( e . g . ImPlotFlags_Legend - > ImPlotFlags_NoLegend ) so that the default flagset
is simply 0. This more closely matches ImGui ' s style and makes it easier to enable non - default but commonly used flags ( e . g . ImPlotAxisFlags_Time ) .
2020-08-28 18:11:36 -04:00
- 2020 / 08 / 28 ( 0.5 ) - ImPlotMarker_ can no longer be combined with bitwise OR , | . This features caused unecessary slow - down , and almost no one used it .
2020-08-25 23:47:03 -04:00
- 2020 / 08 / 25 ( 0.5 ) - ImPlotAxisFlags_Scientific was removed . Logarithmic axes automatically uses scientific notation .
2020-08-17 19:31:30 -04:00
- 2020 / 08 / 17 ( 0.5 ) - PlotText was changed so that text is centered horizontally and vertically about the desired point .
2020-08-16 11:25:06 -04:00
- 2020 / 08 / 16 ( 0.5 ) - An ImPlotContext must be explicitly created and destroyed now with ` CreateContext ` and ` DestroyContext ` . Previously , the context was statically initialized in this source file .
2020-06-14 09:27:58 -04:00
- 2020 / 06 / 13 ( 0.4 ) - The flags ` ImPlotAxisFlag_Adaptive ` and ` ImPlotFlags_Cull ` were removed . Both are now done internally by default .
2020-06-04 10:56:50 -04:00
- 2020 / 06 / 03 ( 0.3 ) - The signature and behavior of PlotPieChart was changed so that data with sum less than 1 can optionally be normalized . The label format can now be specified as well .
2020-06-02 13:34:14 -04:00
- 2020 / 06 / 01 ( 0.3 ) - SetPalette was changed to ` SetColormap ` for consistency with other plotting libraries . ` RestorePalette ` was removed . Use ` SetColormap ( ImPlotColormap_Default ) ` .
2020-05-31 10:07:24 -04:00
- 2020 / 05 / 31 ( 0.3 ) - Plot functions taking custom ImVec2 * getters were removed . Use the ImPlotPoint * getter versions instead .
2020-05-29 13:39:30 -04:00
- 2020 / 05 / 29 ( 0.3 ) - The signature of ImPlotLimits : : Contains was changed to take two doubles instead of ImVec2
2020-05-16 09:39:43 -04:00
- 2020 / 05 / 16 ( 0.2 ) - All plotting functions were reverted to being prefixed with " Plot " to maintain a consistent VerbNoun style . ` Plot ` was split into ` PlotLine `
2020-05-16 10:14:48 -04:00
and ` PlotScatter ` ( however , ` PlotLine ` can still be used to plot scatter points as ` Plot ` did before . ) . ` Bar ` is not ` PlotBars ` , to indicate
that multiple bars will be plotted .
2020-05-16 09:39:43 -04:00
- 2020 / 05 / 13 ( 0.2 ) - ` ImMarker ` was change to ` ImPlotMarker ` and ` ImAxisFlags ` was changed to ` ImPlotAxisFlags ` .
- 2020 / 05 / 11 ( 0.2 ) - ` ImPlotFlags_Selection ` was changed to ` ImPlotFlags_BoxSelect `
2020-05-12 05:19:04 -04:00
- 2020 / 05 / 11 ( 0.2 ) - The namespace ImGui : : was replaced with ImPlot : : . As a result , the following additional changes were made :
2020-05-16 10:14:48 -04:00
- Functions that were prefixed or decorated with the word " Plot " have been truncated . E . g . , ` ImGui : : PlotBars ` is now just ` ImPlot : : Bar ` .
2020-05-12 05:19:04 -04:00
It should be fairly obvious what was what .
2020-05-13 10:11:25 -04:00
- Some functions have been given names that would have otherwise collided with the ImGui namespace . This has been done to maintain a consistent
style with ImGui . E . g . , ' ImGui : : PushPlotStyleVar ` is now ' ImPlot : : PushStyleVar ' .
2020-05-11 07:12:22 -04:00
- 2020 / 05 / 10 ( 0.2 ) - The following function / struct names were changes :
- ImPlotRange - > ImPlotLimits
- GetPlotRange ( ) - > GetPlotLimits ( )
2020-05-13 10:11:25 -04:00
- SetNextPlotRange - > SetNextPlotLimits
2020-05-11 07:12:22 -04:00
- SetNextPlotRangeX - > SetNextPlotLimitsX
- SetNextPlotRangeY - > SetNextPlotLimitsY
- 2020 / 05 / 10 ( 0.2 ) - Plot queries are pixel based by default . Query rects that maintain relative plot position have been removed . This was done to support multi - y - axis .
*/
2020-05-10 22:43:12 -04:00
# include "implot.h"
2020-08-16 16:38:51 -04:00
# include "implot_internal.h"
2020-05-10 22:43:12 -04:00
# ifdef _MSC_VER
# define sprintf sprintf_s
# endif
2020-04-27 11:27:59 -04:00
2020-08-16 16:38:51 -04:00
// Global plot context
ImPlotContext * GImPlot = NULL ;
2020-06-14 09:27:58 -04:00
2020-08-16 16:38:51 -04:00
//-----------------------------------------------------------------------------
// Struct Implementations
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-24 00:45:42 -04:00
ImPlotInputMap : : ImPlotInputMap ( ) {
PanButton = ImGuiMouseButton_Left ;
PanMod = ImGuiKeyModFlags_None ;
FitButton = ImGuiMouseButton_Left ;
ContextMenuButton = ImGuiMouseButton_Right ;
BoxSelectButton = ImGuiMouseButton_Right ;
BoxSelectMod = ImGuiKeyModFlags_None ;
BoxSelectCancelButton = ImGuiMouseButton_Left ;
QueryButton = ImGuiMouseButton_Middle ;
QueryMod = ImGuiKeyModFlags_None ;
QueryToggleMod = ImGuiKeyModFlags_Ctrl ;
HorizontalMod = ImGuiKeyModFlags_Alt ;
VerticalMod = ImGuiKeyModFlags_Shift ;
}
2020-04-27 11:27:59 -04:00
ImPlotStyle : : ImPlotStyle ( ) {
2020-08-16 16:38:51 -04:00
2020-10-19 00:26:34 -04:00
LineWeight = 1 ;
Marker = ImPlotMarker_None ;
MarkerSize = 4 ;
MarkerWeight = 1 ;
FillAlpha = 1 ;
ErrorBarSize = 5 ;
ErrorBarWeight = 1.5f ;
DigitalBitHeight = 8 ;
DigitalBitGap = 4 ;
PlotBorderSize = 1 ;
MinorAlpha = 0.25f ;
MajorTickLen = ImVec2 ( 10 , 10 ) ;
MinorTickLen = ImVec2 ( 5 , 5 ) ;
MajorTickSize = ImVec2 ( 1 , 1 ) ;
MinorTickSize = ImVec2 ( 1 , 1 ) ;
MajorGridSize = ImVec2 ( 1 , 1 ) ;
MinorGridSize = ImVec2 ( 1 , 1 ) ;
PlotPadding = ImVec2 ( 10 , 10 ) ;
LabelPadding = ImVec2 ( 5 , 5 ) ;
LegendPadding = ImVec2 ( 10 , 10 ) ;
LegendInnerPadding = ImVec2 ( 5 , 5 ) ;
LegendSpacing = ImVec2 ( 0 , 0 ) ;
MousePosPadding = ImVec2 ( 10 , 10 ) ;
AnnotationPadding = ImVec2 ( 2 , 2 ) ;
2020-12-04 00:27:38 -05:00
FitPadding = ImVec2 ( 0 , 0 ) ;
2020-10-19 00:26:34 -04:00
PlotDefaultSize = ImVec2 ( 400 , 300 ) ;
PlotMinSize = ImVec2 ( 300 , 225 ) ;
2020-04-27 11:27:59 -04:00
2020-08-24 12:06:29 -04:00
ImPlot : : StyleColorsAuto ( this ) ;
2020-09-06 15:48:16 -04:00
AntiAliasedLines = false ;
UseLocalTime = false ;
2020-09-12 11:25:47 -04:00
Use24HourClock = false ;
2020-10-14 23:07:27 -04:00
UseISO8601 = false ;
2020-04-28 00:57:49 -04:00
}
2020-10-21 11:08:41 -04:00
ImPlotItem * ImPlotPlot : : GetLegendItem ( int i ) {
IM_ASSERT ( Items . GetSize ( ) > 0 ) ;
return Items . GetByIndex ( LegendData . Indices [ i ] ) ;
2020-10-19 00:26:34 -04:00
}
2020-10-21 11:08:41 -04:00
const char * ImPlotPlot : : GetLegendLabel ( int i ) {
ImPlotItem * item = GetLegendItem ( i ) ;
IM_ASSERT ( item ! = NULL ) ;
IM_ASSERT ( item - > NameOffset ! = - 1 & & item - > NameOffset < LegendData . Labels . Buf . Size ) ;
return LegendData . Labels . Buf . Data + item - > NameOffset ;
2020-10-19 00:26:34 -04:00
}
//-----------------------------------------------------------------------------
// Style
//-----------------------------------------------------------------------------
2020-08-24 00:45:42 -04:00
namespace ImPlot {
const char * GetStyleColorName ( ImPlotCol col ) {
static const char * col_names [ ] = {
" Line " ,
" Fill " ,
" MarkerOutline " ,
" MarkerFill " ,
" ErrorBar " ,
" FrameBg " ,
" PlotBg " ,
" PlotBorder " ,
" LegendBg " ,
" LegendBorder " ,
" LegendText " ,
" TitleText " ,
" InlayText " ,
" XAxis " ,
" XAxisGrid " ,
" YAxis " ,
" YAxisGrid " ,
" YAxis2 " ,
" YAxisGrid2 " ,
" YAxis3 " ,
" YAxisGrid3 " ,
" Selection " ,
" Query " ,
" Crosshairs "
} ;
return col_names [ col ] ;
}
2020-09-02 16:55:35 -04:00
const char * GetMarkerName ( ImPlotMarker marker ) {
switch ( marker ) {
case ImPlotMarker_None : return " None " ;
case ImPlotMarker_Circle : return " Circle " ;
case ImPlotMarker_Square : return " Square " ;
2020-09-02 17:02:51 -04:00
case ImPlotMarker_Diamond : return " Diamond " ;
case ImPlotMarker_Up : return " Up " ;
case ImPlotMarker_Down : return " Down " ;
case ImPlotMarker_Left : return " Left " ;
case ImPlotMarker_Right : return " Right " ;
case ImPlotMarker_Cross : return " Cross " ;
case ImPlotMarker_Plus : return " Plus " ;
case ImPlotMarker_Asterisk : return " Asterisk " ;
2020-09-02 16:55:35 -04:00
default : return " " ;
}
}
2020-08-24 00:45:42 -04:00
ImVec4 GetAutoColor ( ImPlotCol idx ) {
ImVec4 col ( 0 , 0 , 0 , 1 ) ;
switch ( idx ) {
case ImPlotCol_Line : return col ; // these are plot dependent!
case ImPlotCol_Fill : return col ; // these are plot dependent!
case ImPlotCol_MarkerOutline : return col ; // these are plot dependent!
case ImPlotCol_MarkerFill : return col ; // these are plot dependent!
case ImPlotCol_ErrorBar : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_FrameBg : return ImGui : : GetStyleColorVec4 ( ImGuiCol_FrameBg ) ;
case ImPlotCol_PlotBg : return ImGui : : GetStyleColorVec4 ( ImGuiCol_WindowBg ) ;
2020-08-24 12:06:29 -04:00
case ImPlotCol_PlotBorder : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Border ) ;
2020-08-24 00:45:42 -04:00
case ImPlotCol_LegendBg : return ImGui : : GetStyleColorVec4 ( ImGuiCol_PopupBg ) ;
case ImPlotCol_LegendBorder : return GetStyleColorVec4 ( ImPlotCol_PlotBorder ) ;
case ImPlotCol_LegendText : return GetStyleColorVec4 ( ImPlotCol_InlayText ) ;
case ImPlotCol_TitleText : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_InlayText : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_XAxis : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_XAxisGrid : return GetStyleColorVec4 ( ImPlotCol_XAxis ) * ImVec4 ( 1 , 1 , 1 , 0.25f ) ;
case ImPlotCol_YAxis : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_YAxisGrid : return GetStyleColorVec4 ( ImPlotCol_YAxis ) * ImVec4 ( 1 , 1 , 1 , 0.25f ) ;
case ImPlotCol_YAxis2 : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_YAxisGrid2 : return GetStyleColorVec4 ( ImPlotCol_YAxis2 ) * ImVec4 ( 1 , 1 , 1 , 0.25f ) ;
case ImPlotCol_YAxis3 : return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
case ImPlotCol_YAxisGrid3 : return GetStyleColorVec4 ( ImPlotCol_YAxis3 ) * ImVec4 ( 1 , 1 , 1 , 0.25f ) ;
case ImPlotCol_Selection : return ImVec4 ( 1 , 1 , 0 , 1 ) ;
case ImPlotCol_Query : return ImVec4 ( 0 , 1 , 0 , 1 ) ;
case ImPlotCol_Crosshairs : return GetStyleColorVec4 ( ImPlotCol_PlotBorder ) ;
default : return col ;
}
2020-06-15 12:55:30 -04:00
}
2020-08-24 00:45:42 -04:00
struct ImPlotStyleVarInfo {
ImGuiDataType Type ;
ImU32 Count ;
ImU32 Offset ;
void * GetVarPtr ( ImPlotStyle * style ) const { return ( void * ) ( ( unsigned char * ) style + Offset ) ; }
} ;
static const ImPlotStyleVarInfo GPlotStyleVarInfo [ ] =
{
2020-10-19 00:26:34 -04:00
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , LineWeight ) } , // ImPlotStyleVar_LineWeight
{ ImGuiDataType_S32 , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , Marker ) } , // ImPlotStyleVar_Marker
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MarkerSize ) } , // ImPlotStyleVar_MarkerSize
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MarkerWeight ) } , // ImPlotStyleVar_MarkerWeight
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , FillAlpha ) } , // ImPlotStyleVar_FillAlpha
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , ErrorBarSize ) } , // ImPlotStyleVar_ErrorBarSize
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , ErrorBarWeight ) } , // ImPlotStyleVar_ErrorBarWeight
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , DigitalBitHeight ) } , // ImPlotStyleVar_DigitalBitHeight
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , DigitalBitGap ) } , // ImPlotStyleVar_DigitalBitGap
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , PlotBorderSize ) } , // ImPlotStyleVar_PlotBorderSize
{ ImGuiDataType_Float , 1 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MinorAlpha ) } , // ImPlotStyleVar_MinorAlpha
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MajorTickLen ) } , // ImPlotStyleVar_MajorTickLen
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MinorTickLen ) } , // ImPlotStyleVar_MinorTickLen
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MajorTickSize ) } , // ImPlotStyleVar_MajorTickSize
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MinorTickSize ) } , // ImPlotStyleVar_MinorTickSize
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MajorGridSize ) } , // ImPlotStyleVar_MajorGridSize
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MinorGridSize ) } , // ImPlotStyleVar_MinorGridSize
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , PlotPadding ) } , // ImPlotStyleVar_PlotPadding
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , LabelPadding ) } , // ImPlotStyleVar_LabelPaddine
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , LegendPadding ) } , // ImPlotStyleVar_LegendPadding
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , LegendInnerPadding ) } , // ImPlotStyleVar_LegendInnerPadding
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , LegendSpacing ) } , // ImPlotStyleVar_LegendSpacing
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , MousePosPadding ) } , // ImPlotStyleVar_MousePosPadding
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , AnnotationPadding ) } , // ImPlotStyleVar_AnnotationPadding
2020-12-04 00:27:38 -05:00
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , FitPadding ) } , // ImPlotStyleVar_FitPadding
2020-10-19 00:26:34 -04:00
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , PlotDefaultSize ) } , // ImPlotStyleVar_PlotDefaultSize
{ ImGuiDataType_Float , 2 , ( ImU32 ) IM_OFFSETOF ( ImPlotStyle , PlotMinSize ) } // ImPlotStyleVar_PlotMinSize
2020-08-24 00:45:42 -04:00
} ;
static const ImPlotStyleVarInfo * GetPlotStyleVarInfo ( ImPlotStyleVar idx ) {
IM_ASSERT ( idx > = 0 & & idx < ImPlotStyleVar_COUNT ) ;
IM_ASSERT ( IM_ARRAYSIZE ( GPlotStyleVarInfo ) = = ImPlotStyleVar_COUNT ) ;
return & GPlotStyleVarInfo [ idx ] ;
}
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-08-16 16:38:51 -04:00
// Generic Helpers
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-24 09:51:03 -04:00
void AddTextVertical ( ImDrawList * DrawList , ImVec2 pos , ImU32 col , const char * text_begin , const char * text_end ) {
if ( ! text_end )
text_end = text_begin + strlen ( text_begin ) ;
ImGuiContext & g = * GImGui ;
ImFont * font = g . Font ;
2020-09-28 17:38:27 -04:00
pos . x = IM_FLOOR ( pos . x ) ; // + font->ConfigData->GlyphOffset.y);
pos . y = IM_FLOOR ( pos . y ) ; // + font->ConfigData->GlyphOffset.x);
2020-08-24 09:51:03 -04:00
const char * s = text_begin ;
const int vtx_count = ( int ) ( text_end - s ) * 4 ;
const int idx_count = ( int ) ( text_end - s ) * 6 ;
DrawList - > PrimReserve ( idx_count , vtx_count ) ;
const float scale = g . FontSize / font - > FontSize ;
while ( s < text_end ) {
unsigned int c = ( unsigned int ) * s ;
if ( c < 0x80 ) {
s + = 1 ;
}
else {
s + = ImTextCharFromUtf8 ( & c , s , text_end ) ;
if ( c = = 0 ) // Malformed UTF-8?
break ;
}
const ImFontGlyph * glyph = font - > FindGlyph ( ( ImWchar ) c ) ;
if ( glyph = = NULL )
2020-04-27 11:27:59 -04:00
continue ;
2020-08-24 09:51:03 -04:00
DrawList - > PrimQuadUV ( pos + ImVec2 ( glyph - > Y0 , - glyph - > X0 ) * scale , pos + ImVec2 ( glyph - > Y0 , - glyph - > X1 ) * scale ,
pos + ImVec2 ( glyph - > Y1 , - glyph - > X1 ) * scale , pos + ImVec2 ( glyph - > Y1 , - glyph - > X0 ) * scale ,
ImVec2 ( glyph - > U0 , glyph - > V0 ) , ImVec2 ( glyph - > U1 , glyph - > V0 ) ,
ImVec2 ( glyph - > U1 , glyph - > V1 ) , ImVec2 ( glyph - > U0 , glyph - > V1 ) ,
col ) ;
pos . y - = glyph - > AdvanceX * scale ;
2020-04-27 11:27:59 -04:00
}
}
2020-08-16 16:38:51 -04:00
double NiceNum ( double x , bool round ) {
double f ; /* fractional part of x */
double nf ; /* nice, rounded fraction */
int expv = ( int ) floor ( ImLog10 ( x ) ) ;
f = x / ImPow ( 10.0 , ( double ) expv ) ; /* between 1 and 10 */
if ( round )
if ( f < 1.5 )
nf = 1 ;
else if ( f < 3 )
nf = 2 ;
else if ( f < 7 )
nf = 5 ;
else
nf = 10 ;
else if ( f < = 1 )
nf = 1 ;
else if ( f < = 2 )
nf = 2 ;
else if ( f < = 5 )
nf = 5 ;
else
nf = 10 ;
return nf * ImPow ( 10.0 , expv ) ;
2020-06-13 13:42:47 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-08-16 16:38:51 -04:00
// Context Utils
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-09-07 21:59:43 -04:00
void SetImGuiContext ( ImGuiContext * ctx ) {
ImGui : : SetCurrentContext ( ctx ) ;
}
2020-08-16 11:25:06 -04:00
ImPlotContext * CreateContext ( ) {
ImPlotContext * ctx = IM_NEW ( ImPlotContext ) ( ) ;
2020-08-16 16:38:51 -04:00
Initialize ( ctx ) ;
2020-08-16 11:25:06 -04:00
if ( GImPlot = = NULL )
SetCurrentContext ( ctx ) ;
return ctx ;
}
void DestroyContext ( ImPlotContext * ctx ) {
if ( ctx = = NULL )
ctx = GImPlot ;
if ( GImPlot = = ctx )
SetCurrentContext ( NULL ) ;
IM_DELETE ( ctx ) ;
2020-08-16 16:38:51 -04:00
}
2020-08-16 11:25:06 -04:00
ImPlotContext * GetCurrentContext ( ) {
return GImPlot ;
}
void SetCurrentContext ( ImPlotContext * ctx ) {
GImPlot = ctx ;
}
2020-04-27 11:27:59 -04:00
2020-08-16 16:38:51 -04:00
void Initialize ( ImPlotContext * ctx ) {
Reset ( ctx ) ;
2020-08-21 23:13:11 -04:00
ctx - > Colormap = GetColormap ( ImPlotColormap_Default , & ctx - > ColormapSize ) ;
2020-08-16 16:38:51 -04:00
}
void Reset ( ImPlotContext * ctx ) {
// end child window if it was made
if ( ctx - > ChildWindowMade )
ImGui : : EndChild ( ) ;
ctx - > ChildWindowMade = false ;
2020-08-30 18:12:36 -04:00
// reset the next plot/item data
2020-11-15 22:47:06 -05:00
ctx - > NextPlotData . Reset ( ) ;
ctx - > NextItemData . Reset ( ) ;
2020-08-16 16:38:51 -04:00
// reset items count
ctx - > VisibleItemCount = 0 ;
// reset ticks/labels
2020-08-25 22:59:43 -04:00
ctx - > XTicks . Reset ( ) ;
2020-10-19 00:26:34 -04:00
for ( int i = 0 ; i < 3 ; + + i )
2020-08-25 22:59:43 -04:00
ctx - > YTicks [ i ] . Reset ( ) ;
2020-09-19 21:54:19 -04:00
// reset labels
ctx - > Annotations . Reset ( ) ;
2020-08-20 00:50:12 -04:00
// reset extents/fit
ctx - > FitThisFrame = false ;
2020-08-16 16:38:51 -04:00
ctx - > FitX = false ;
ctx - > ExtentsX . Min = HUGE_VAL ;
ctx - > ExtentsX . Max = - HUGE_VAL ;
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-08-16 16:38:51 -04:00
ctx - > ExtentsY [ i ] . Min = HUGE_VAL ;
ctx - > ExtentsY [ i ] . Max = - HUGE_VAL ;
ctx - > FitY [ i ] = false ;
}
// reset digital plot items count
ctx - > DigitalPlotItemCnt = 0 ;
ctx - > DigitalPlotOffset = 0 ;
// nullify plot
2020-09-19 21:54:19 -04:00
ctx - > CurrentPlot = NULL ;
ctx - > CurrentItem = NULL ;
ctx - > PreviousItem = NULL ;
2020-08-16 16:38:51 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-08-16 16:38:51 -04:00
// Plot Utils
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-09-21 08:09:14 -04:00
ImPlotPlot * GetPlot ( const char * title ) {
2020-08-16 16:38:51 -04:00
ImGuiWindow * Window = GImGui - > CurrentWindow ;
const ImGuiID ID = Window - > GetID ( title ) ;
return GImPlot - > Plots . GetByKey ( ID ) ;
}
2020-09-21 08:09:14 -04:00
ImPlotPlot * GetCurrentPlot ( ) {
2020-08-16 16:38:51 -04:00
return GImPlot - > CurrentPlot ;
2020-06-15 11:48:22 -04:00
}
2020-08-24 00:45:42 -04:00
void BustPlotCache ( ) {
GImPlot - > Plots . Clear ( ) ;
}
2020-08-16 16:38:51 -04:00
void FitPoint ( const ImPlotPoint & p ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = gp . CurrentPlot - > CurrentYAxis ;
2020-08-22 23:55:37 -04:00
ImPlotRange & ex_x = gp . ExtentsX ;
ImPlotRange & ex_y = gp . ExtentsY [ y_axis ] ;
const bool log_x = ImHasFlag ( gp . CurrentPlot - > XAxis . Flags , ImPlotAxisFlags_LogScale ) ;
const bool log_y = ImHasFlag ( gp . CurrentPlot - > YAxis [ y_axis ] . Flags , ImPlotAxisFlags_LogScale ) ;
2020-09-04 20:33:10 -04:00
if ( ! ImNanOrInf ( p . x ) & & ! ( log_x & & p . x < = 0 ) ) {
2020-08-22 23:55:37 -04:00
ex_x . Min = p . x < ex_x . Min ? p . x : ex_x . Min ;
ex_x . Max = p . x > ex_x . Max ? p . x : ex_x . Max ;
2020-05-03 01:24:10 -04:00
}
2020-09-04 20:33:10 -04:00
if ( ! ImNanOrInf ( p . y ) & & ! ( log_y & & p . y < = 0 ) ) {
2020-08-22 23:55:37 -04:00
ex_y . Min = p . y < ex_y . Min ? p . y : ex_y . Min ;
ex_y . Max = p . y > ex_y . Max ? p . y : ex_y . Max ;
2020-05-03 01:24:10 -04:00
}
}
2020-09-03 10:19:34 -04:00
void PushLinkedAxis ( ImPlotAxis & axis ) {
if ( axis . LinkedMin ) { * axis . LinkedMin = axis . Range . Min ; }
if ( axis . LinkedMax ) { * axis . LinkedMax = axis . Range . Max ; }
}
void PullLinkedAxis ( ImPlotAxis & axis ) {
2020-09-06 17:09:00 -04:00
if ( axis . LinkedMin ) { axis . SetMin ( * axis . LinkedMin ) ; }
2020-09-06 22:08:25 -04:00
if ( axis . LinkedMax ) { axis . SetMax ( * axis . LinkedMax ) ; }
2020-09-03 10:19:34 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-08-16 16:38:51 -04:00
// Coordinate Utils
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-08-16 16:38:51 -04:00
void UpdateTransformCache ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-05-03 01:24:10 -04:00
// get pixels for transforms
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
gp . PixelRange [ i ] = ImRect ( ImHasFlag ( gp . CurrentPlot - > XAxis . Flags , ImPlotAxisFlags_Invert ) ? gp . CurrentPlot - > PlotRect . Max . x : gp . CurrentPlot - > PlotRect . Min . x ,
ImHasFlag ( gp . CurrentPlot - > YAxis [ i ] . Flags , ImPlotAxisFlags_Invert ) ? gp . CurrentPlot - > PlotRect . Min . y : gp . CurrentPlot - > PlotRect . Max . y ,
ImHasFlag ( gp . CurrentPlot - > XAxis . Flags , ImPlotAxisFlags_Invert ) ? gp . CurrentPlot - > PlotRect . Min . x : gp . CurrentPlot - > PlotRect . Max . x ,
ImHasFlag ( gp . CurrentPlot - > YAxis [ i ] . Flags , ImPlotAxisFlags_Invert ) ? gp . CurrentPlot - > PlotRect . Max . y : gp . CurrentPlot - > PlotRect . Min . y ) ;
2020-05-11 07:12:22 -04:00
gp . My [ i ] = ( gp . PixelRange [ i ] . Max . y - gp . PixelRange [ i ] . Min . y ) / gp . CurrentPlot - > YAxis [ i ] . Range . Size ( ) ;
}
2020-05-29 13:39:30 -04:00
gp . LogDenX = ImLog10 ( gp . CurrentPlot - > XAxis . Range . Max / gp . CurrentPlot - > XAxis . Range . Min ) ;
2020-09-01 00:23:48 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + )
gp . LogDenY [ i ] = ImLog10 ( gp . CurrentPlot - > YAxis [ i ] . Range . Max / gp . CurrentPlot - > YAxis [ i ] . Range . Min ) ;
2020-05-11 07:12:22 -04:00
gp . Mx = ( gp . PixelRange [ 0 ] . Max . x - gp . PixelRange [ 0 ] . Min . x ) / gp . CurrentPlot - > XAxis . Range . Size ( ) ;
2020-04-28 01:38:52 -04:00
}
2020-04-27 11:27:59 -04:00
2020-10-19 00:26:34 -04:00
ImPlotPoint PixelsToPlot ( float x , float y , ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " PixelsToPlot() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-05-16 22:09:36 -04:00
ImPlotPoint plt ;
2020-05-11 07:12:22 -04:00
plt . x = ( x - gp . PixelRange [ y_axis ] . Min . x ) / gp . Mx + gp . CurrentPlot - > XAxis . Range . Min ;
plt . y = ( y - gp . PixelRange [ y_axis ] . Min . y ) / gp . My [ y_axis ] + gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min ;
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( gp . CurrentPlot - > XAxis . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-05-29 13:39:30 -04:00
double t = ( plt . x - gp . CurrentPlot - > XAxis . Range . Min ) / gp . CurrentPlot - > XAxis . Range . Size ( ) ;
plt . x = ImPow ( 10 , t * gp . LogDenX ) * gp . CurrentPlot - > XAxis . Range . Min ;
2020-05-03 01:24:10 -04:00
}
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( gp . CurrentPlot - > YAxis [ y_axis ] . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-05-29 13:39:30 -04:00
double t = ( plt . y - gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min ) / gp . CurrentPlot - > YAxis [ y_axis ] . Range . Size ( ) ;
plt . y = ImPow ( 10 , t * gp . LogDenY [ y_axis ] ) * gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min ;
2020-05-03 01:24:10 -04:00
}
return plt ;
}
2020-10-19 00:26:34 -04:00
ImPlotPoint PixelsToPlot ( const ImVec2 & pix , ImPlotYAxis y_axis ) {
2020-05-15 09:05:02 -04:00
return PixelsToPlot ( pix . x , pix . y , y_axis ) ;
}
// This function is convenient but should not be used to process a high volume of points. Use the Transformer structs below instead.
2020-10-19 00:26:34 -04:00
ImVec2 PlotToPixels ( double x , double y , ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " PlotToPixels() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-05-03 01:24:10 -04:00
ImVec2 pix ;
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( gp . CurrentPlot - > XAxis . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-05-29 13:39:30 -04:00
double t = ImLog10 ( x / gp . CurrentPlot - > XAxis . Range . Min ) / gp . LogDenX ;
2020-05-16 22:09:36 -04:00
x = ImLerp ( gp . CurrentPlot - > XAxis . Range . Min , gp . CurrentPlot - > XAxis . Range . Max , ( float ) t ) ;
2020-05-13 10:11:25 -04:00
}
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( gp . CurrentPlot - > YAxis [ y_axis ] . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-05-29 13:39:30 -04:00
double t = ImLog10 ( y / gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min ) / gp . LogDenY [ y_axis ] ;
2020-05-16 22:09:36 -04:00
y = ImLerp ( gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min , gp . CurrentPlot - > YAxis [ y_axis ] . Range . Max , ( float ) t ) ;
2020-05-03 01:24:10 -04:00
}
2020-05-16 22:09:36 -04:00
pix . x = ( float ) ( gp . PixelRange [ y_axis ] . Min . x + gp . Mx * ( x - gp . CurrentPlot - > XAxis . Range . Min ) ) ;
pix . y = ( float ) ( gp . PixelRange [ y_axis ] . Min . y + gp . My [ y_axis ] * ( y - gp . CurrentPlot - > YAxis [ y_axis ] . Range . Min ) ) ;
2020-05-03 01:24:10 -04:00
return pix ;
}
2020-05-15 09:05:02 -04:00
// This function is convenient but should not be used to process a high volume of points. Use the Transformer structs below instead.
2020-10-19 00:26:34 -04:00
ImVec2 PlotToPixels ( const ImPlotPoint & plt , ImPlotYAxis y_axis ) {
2020-05-11 07:12:22 -04:00
return PlotToPixels ( plt . x , plt . y , y_axis ) ;
2020-05-03 01:24:10 -04:00
}
2020-08-16 16:38:51 -04:00
//-----------------------------------------------------------------------------
// Legend Utils
//-----------------------------------------------------------------------------
2020-10-19 00:26:34 -04:00
ImVec2 GetLocationPos ( const ImRect & outer_rect , const ImVec2 & inner_size , ImPlotLocation loc , const ImVec2 & pad ) {
ImVec2 pos ;
if ( ImHasFlag ( loc , ImPlotLocation_West ) & & ! ImHasFlag ( loc , ImPlotLocation_East ) )
pos . x = outer_rect . Min . x + pad . x ;
else if ( ! ImHasFlag ( loc , ImPlotLocation_West ) & & ImHasFlag ( loc , ImPlotLocation_East ) )
pos . x = outer_rect . Max . x - pad . x - inner_size . x ;
else
pos . x = outer_rect . GetCenter ( ) . x - inner_size . x * 0.5f ;
// legend reference point y
if ( ImHasFlag ( loc , ImPlotLocation_North ) & & ! ImHasFlag ( loc , ImPlotLocation_South ) )
pos . y = outer_rect . Min . y + pad . y ;
else if ( ! ImHasFlag ( loc , ImPlotLocation_North ) & & ImHasFlag ( loc , ImPlotLocation_South ) )
pos . y = outer_rect . Max . y - pad . y - inner_size . y ;
else
pos . y = outer_rect . GetCenter ( ) . y - inner_size . y * 0.5f ;
pos . x = IM_ROUND ( pos . x ) ;
pos . y = IM_ROUND ( pos . y ) ;
return pos ;
}
2020-10-21 11:08:41 -04:00
ImVec2 CalcLegendSize ( ImPlotPlot & plot , const ImVec2 & pad , const ImVec2 & spacing , ImPlotOrientation orn ) {
2020-10-19 00:26:34 -04:00
// vars
2020-10-21 11:08:41 -04:00
const int nItems = plot . GetLegendCount ( ) ;
2020-10-19 00:26:34 -04:00
const float txt_ht = ImGui : : GetTextLineHeight ( ) ;
const float icon_size = txt_ht ;
// get label max width
float max_label_width = 0 ;
float sum_label_width = 0 ;
for ( int i = 0 ; i < nItems ; + + i ) {
2020-10-21 11:08:41 -04:00
const char * label = plot . GetLegendLabel ( i ) ;
2020-10-19 00:26:34 -04:00
const float label_width = ImGui : : CalcTextSize ( label , NULL , true ) . x ;
max_label_width = label_width > max_label_width ? label_width : max_label_width ;
sum_label_width + = label_width ;
}
// calc legend size
const ImVec2 legend_size = orn = = ImPlotOrientation_Vertical ?
ImVec2 ( pad . x * 2 + icon_size + max_label_width , pad . y * 2 + nItems * txt_ht + ( nItems - 1 ) * spacing . y ) :
ImVec2 ( pad . x * 2 + icon_size * nItems + sum_label_width + ( nItems - 1 ) * spacing . x , pad . y * 2 + txt_ht ) ;
return legend_size ;
}
2020-10-21 11:08:41 -04:00
void ShowLegendEntries ( ImPlotPlot & plot , const ImRect & legend_bb , bool interactable , const ImVec2 & pad , const ImVec2 & spacing , ImPlotOrientation orn , ImDrawList & DrawList ) {
2020-10-19 00:26:34 -04:00
ImGuiIO & IO = ImGui : : GetIO ( ) ;
// vars
const float txt_ht = ImGui : : GetTextLineHeight ( ) ;
const float icon_size = txt_ht ;
const float icon_shrink = 2 ;
ImVec4 col_txt = GetStyleColorVec4 ( ImPlotCol_LegendText ) ;
ImU32 col_txt_dis = ImGui : : GetColorU32 ( col_txt * ImVec4 ( 1 , 1 , 1 , 0.25f ) ) ;
// render each legend item
float sum_label_width = 0 ;
2020-10-21 11:08:41 -04:00
for ( int i = 0 ; i < plot . GetLegendCount ( ) ; + + i ) {
ImPlotItem * item = plot . GetLegendItem ( i ) ;
const char * label = plot . GetLegendLabel ( i ) ;
2020-10-19 00:26:34 -04:00
const float label_width = ImGui : : CalcTextSize ( label , NULL , true ) . x ;
const ImVec2 top_left = orn = = ImPlotOrientation_Vertical ?
legend_bb . Min + pad + ImVec2 ( 0 , i * ( txt_ht + spacing . y ) ) :
legend_bb . Min + pad + ImVec2 ( i * ( icon_size + spacing . x ) + sum_label_width , 0 ) ;
sum_label_width + = label_width ;
ImRect icon_bb ;
icon_bb . Min = top_left + ImVec2 ( icon_shrink , icon_shrink ) ;
icon_bb . Max = top_left + ImVec2 ( icon_size - icon_shrink , icon_size - icon_shrink ) ;
ImRect label_bb ;
label_bb . Min = top_left ;
label_bb . Max = top_left + ImVec2 ( label_width + icon_size , icon_size ) ;
ImU32 col_hl_txt ;
if ( interactable & & ( icon_bb . Contains ( IO . MousePos ) | | label_bb . Contains ( IO . MousePos ) ) ) {
item - > LegendHovered = true ;
col_hl_txt = ImGui : : GetColorU32 ( ImLerp ( col_txt , item - > Color , 0.25f ) ) ;
}
else {
// item->LegendHovered = false;
col_hl_txt = ImGui : : GetColorU32 ( col_txt ) ;
}
ImU32 iconColor ;
ImVec4 item_color = item - > Color ;
item_color . w = 1 ;
if ( interactable & & icon_bb . Contains ( IO . MousePos ) ) {
ImVec4 colAlpha = item_color ;
colAlpha . w = 0.5f ;
iconColor = item - > Show ? ImGui : : GetColorU32 ( colAlpha ) : ImGui : : GetColorU32 ( ImGuiCol_TextDisabled , 0.5f ) ;
if ( IO . MouseClicked [ 0 ] )
item - > Show = ! item - > Show ;
}
else {
iconColor = item - > Show ? ImGui : : GetColorU32 ( item_color ) : col_txt_dis ;
}
DrawList . AddRectFilled ( icon_bb . Min , icon_bb . Max , iconColor , 1 ) ;
const char * text_display_end = ImGui : : FindRenderedTextEnd ( label , NULL ) ;
if ( label ! = text_display_end )
DrawList . AddText ( top_left + ImVec2 ( icon_size , 0 ) , item - > Show ? col_hl_txt : col_txt_dis , label , text_display_end ) ;
}
2020-05-03 01:24:10 -04:00
}
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// Tick Utils
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-25 22:59:43 -04:00
void LabelTickDefault ( ImPlotTick & tick , ImGuiTextBuffer & buffer ) {
char temp [ 32 ] ;
if ( tick . ShowLabel ) {
2020-09-19 21:54:19 -04:00
tick . TextOffset = buffer . size ( ) ;
2020-09-03 00:30:32 -04:00
snprintf ( temp , 32 , " %.10g " , tick . PlotPos ) ;
2020-08-25 22:59:43 -04:00
buffer . append ( temp , temp + strlen ( temp ) + 1 ) ;
2020-09-19 21:54:19 -04:00
tick . LabelSize = ImGui : : CalcTextSize ( buffer . Buf . Data + tick . TextOffset ) ;
2020-08-25 22:59:43 -04:00
}
}
void LabelTickScientific ( ImPlotTick & tick , ImGuiTextBuffer & buffer ) {
char temp [ 32 ] ;
if ( tick . ShowLabel ) {
2020-09-19 21:54:19 -04:00
tick . TextOffset = buffer . size ( ) ;
2020-09-03 00:30:32 -04:00
snprintf ( temp , 32 , " %.0E " , tick . PlotPos ) ;
buffer . append ( temp , temp + strlen ( temp ) + 1 ) ;
2020-09-19 21:54:19 -04:00
tick . LabelSize = ImGui : : CalcTextSize ( buffer . Buf . Data + tick . TextOffset ) ;
2020-09-03 00:30:32 -04:00
}
}
2020-08-25 22:59:43 -04:00
void AddTicksDefault ( const ImPlotRange & range , int nMajor , int nMinor , ImPlotTickCollection & ticks ) {
2020-08-22 23:55:37 -04:00
const double nice_range = NiceNum ( range . Size ( ) * 0.99 , false ) ;
2020-08-23 00:26:49 -04:00
const double interval = NiceNum ( nice_range / ( nMajor - 1 ) , true ) ;
2020-08-22 23:55:37 -04:00
const double graphmin = floor ( range . Min / interval ) * interval ;
const double graphmax = ceil ( range . Max / interval ) * interval ;
for ( double major = graphmin ; major < graphmax + 0.5 * interval ; major + = interval ) {
2020-09-01 00:23:48 -04:00
if ( range . Contains ( major ) )
2020-09-19 21:54:19 -04:00
ticks . Append ( major , true , true , LabelTickDefault ) ;
2020-08-22 23:55:37 -04:00
for ( int i = 1 ; i < nMinor ; + + i ) {
double minor = major + i * interval / nMinor ;
2020-09-01 00:23:48 -04:00
if ( range . Contains ( minor ) )
2020-09-19 21:54:19 -04:00
ticks . Append ( minor , false , true , LabelTickDefault ) ;
2020-08-22 23:55:37 -04:00
}
2020-08-23 00:26:49 -04:00
}
2020-08-22 23:55:37 -04:00
}
2020-08-25 22:59:43 -04:00
void AddTicksLogarithmic ( const ImPlotRange & range , int nMajor , ImPlotTickCollection & ticks ) {
2020-08-22 23:55:37 -04:00
if ( range . Min < = 0 | | range . Max < = 0 )
return ;
double log_min = ImLog10 ( range . Min ) ;
double log_max = ImLog10 ( range . Max ) ;
int exp_step = ImMax ( 1 , ( int ) ( log_max - log_min ) / nMajor ) ;
int exp_min = ( int ) log_min ;
int exp_max = ( int ) log_max ;
if ( exp_step ! = 1 ) {
while ( exp_step % 3 ! = 0 ) exp_step + + ; // make step size multiple of three
while ( exp_min % exp_step ! = 0 ) exp_min - - ; // decrease exp_min until exp_min + N * exp_step will be 0
}
for ( int e = exp_min - exp_step ; e < ( exp_max + exp_step ) ; e + = exp_step ) {
double major1 = ImPow ( 10 , ( double ) ( e ) ) ;
double major2 = ImPow ( 10 , ( double ) ( e + 1 ) ) ;
double interval = ( major2 - major1 ) / 9 ;
2020-09-01 00:23:48 -04:00
if ( major1 > = ( range . Min - DBL_EPSILON ) & & major1 < = ( range . Max + DBL_EPSILON ) )
2020-09-19 21:54:19 -04:00
ticks . Append ( major1 , true , true , LabelTickScientific ) ;
2020-08-22 23:55:37 -04:00
for ( int j = 0 ; j < exp_step ; + + j ) {
major1 = ImPow ( 10 , ( double ) ( e + j ) ) ;
major2 = ImPow ( 10 , ( double ) ( e + j + 1 ) ) ;
interval = ( major2 - major1 ) / 9 ;
for ( int i = 1 ; i < ( 9 + ( int ) ( j < ( exp_step - 1 ) ) ) ; + + i ) {
2020-04-27 11:27:59 -04:00
double minor = major1 + i * interval ;
2020-09-01 00:23:48 -04:00
if ( minor > = ( range . Min - DBL_EPSILON ) & & minor < = ( range . Max + DBL_EPSILON ) )
2020-09-19 21:54:19 -04:00
ticks . Append ( minor , false , false , LabelTickScientific ) ;
2020-09-01 00:23:48 -04:00
2020-08-23 00:26:49 -04:00
}
2020-04-27 11:27:59 -04:00
}
}
}
2020-09-07 11:17:44 -04:00
void AddTicksCustom ( const double * values , const char * const labels [ ] , int n , ImPlotTickCollection & ticks ) {
2020-09-05 00:30:45 -04:00
for ( int i = 0 ; i < n ; + + i ) {
ImPlotTick tick ( values [ i ] , false , true ) ;
if ( labels ! = NULL ) {
2020-09-19 21:54:19 -04:00
tick . TextOffset = ticks . TextBuffer . size ( ) ;
ticks . TextBuffer . append ( labels [ i ] , labels [ i ] + strlen ( labels [ i ] ) + 1 ) ;
2020-09-05 00:30:45 -04:00
tick . LabelSize = ImGui : : CalcTextSize ( labels [ i ] ) ;
}
else {
2020-09-19 21:54:19 -04:00
LabelTickDefault ( tick , ticks . TextBuffer ) ;
2020-09-05 00:30:45 -04:00
}
2020-09-19 21:54:19 -04:00
ticks . Append ( tick ) ;
2020-09-05 00:30:45 -04:00
}
}
//-----------------------------------------------------------------------------
// Time Ticks and Utils
//-----------------------------------------------------------------------------
// this may not be thread safe?
static const double TimeUnitSpans [ ImPlotTimeUnit_COUNT ] = {
2020-09-06 22:08:25 -04:00
0.000001 ,
0.001 ,
1 ,
60 ,
3600 ,
86400 ,
2629800 ,
2020-09-05 00:30:45 -04:00
31557600
} ;
inline ImPlotTimeUnit GetUnitForRange ( double range ) {
static double cutoffs [ ImPlotTimeUnit_COUNT ] = { 0.001 , 1 , 60 , 3600 , 86400 , 2629800 , 31557600 , IMPLOT_MAX_TIME } ;
for ( int i = 0 ; i < ImPlotTimeUnit_COUNT ; + + i ) {
if ( range < = cutoffs [ i ] )
return ( ImPlotTimeUnit ) i ;
}
return ImPlotTimeUnit_Yr ;
}
2020-09-04 14:13:45 -04:00
inline int LowerBoundStep ( int max_divs , const int * divs , const int * step , int size ) {
if ( max_divs < divs [ 0 ] )
return 0 ;
for ( int i = 1 ; i < size ; + + i ) {
if ( max_divs < divs [ i ] )
return step [ i - 1 ] ;
}
return step [ size - 1 ] ;
}
inline int GetTimeStep ( int max_divs , ImPlotTimeUnit unit ) {
if ( unit = = ImPlotTimeUnit_Ms | | unit = = ImPlotTimeUnit_Us ) {
2020-09-05 13:25:44 -04:00
static const int step [ ] = { 500 , 250 , 200 , 100 , 50 , 25 , 20 , 10 , 5 , 2 , 1 } ;
static const int divs [ ] = { 2 , 4 , 5 , 10 , 20 , 40 , 50 , 100 , 200 , 500 , 1000 } ;
return LowerBoundStep ( max_divs , divs , step , 11 ) ;
2020-09-04 14:13:45 -04:00
}
if ( unit = = ImPlotTimeUnit_S | | unit = = ImPlotTimeUnit_Min ) {
static const int step [ ] = { 30 , 15 , 10 , 5 , 1 } ;
static const int divs [ ] = { 2 , 4 , 6 , 12 , 60 } ;
return LowerBoundStep ( max_divs , divs , step , 5 ) ;
}
else if ( unit = = ImPlotTimeUnit_Hr ) {
static const int step [ ] = { 12 , 6 , 3 , 2 , 1 } ;
static const int divs [ ] = { 2 , 4 , 8 , 12 , 24 } ;
return LowerBoundStep ( max_divs , divs , step , 5 ) ;
}
else if ( unit = = ImPlotTimeUnit_Day ) {
static const int step [ ] = { 14 , 7 , 2 , 1 } ;
static const int divs [ ] = { 2 , 4 , 14 , 28 } ;
return LowerBoundStep ( max_divs , divs , step , 4 ) ;
}
else if ( unit = = ImPlotTimeUnit_Mo ) {
static const int step [ ] = { 6 , 3 , 2 , 1 } ;
static const int divs [ ] = { 2 , 4 , 6 , 12 } ;
return LowerBoundStep ( max_divs , divs , step , 4 ) ;
}
return 0 ;
}
2020-09-06 15:48:16 -04:00
ImPlotTime MkGmtTime ( struct tm * ptm ) {
2020-09-09 00:47:02 -04:00
ImPlotTime t ;
# ifdef _WIN32
t . S = _mkgmtime ( ptm ) ;
# else
t . S = timegm ( ptm ) ;
# endif
2020-09-09 20:17:19 -04:00
if ( t . S < 0 )
t . S = 0 ;
2020-09-09 00:47:02 -04:00
return t ;
2020-09-05 00:30:45 -04:00
}
2020-09-04 14:13:45 -04:00
2020-09-06 15:48:16 -04:00
tm * GetGmtTime ( const ImPlotTime & t , tm * ptm )
2020-09-05 00:30:45 -04:00
{
# ifdef _WIN32
2020-09-06 15:48:16 -04:00
if ( gmtime_s ( ptm , & t . S ) = = 0 )
return ptm ;
else
return NULL ;
# else
return gmtime_r ( & t . S , ptm ) ;
# endif
}
ImPlotTime MkLocTime ( struct tm * ptm ) {
ImPlotTime t ;
t . S = mktime ( ptm ) ;
2020-09-09 20:17:19 -04:00
if ( t . S < 0 )
t . S = 0 ;
2020-09-06 15:48:16 -04:00
return t ;
}
tm * GetLocTime ( const ImPlotTime & t , tm * ptm ) {
# ifdef _WIN32
if ( localtime_s ( ptm , & t . S ) = = 0 )
return ptm ;
2020-09-05 00:30:45 -04:00
else
return NULL ;
# else
2020-09-06 15:48:16 -04:00
return localtime_r ( & t . S , ptm ) ;
2020-09-05 00:30:45 -04:00
# endif
}
2020-09-06 15:48:16 -04:00
inline ImPlotTime MkTime ( struct tm * ptm ) {
if ( GetStyle ( ) . UseLocalTime )
return MkLocTime ( ptm ) ;
else
return MkGmtTime ( ptm ) ;
}
inline tm * GetTime ( const ImPlotTime & t , tm * ptm ) {
if ( GetStyle ( ) . UseLocalTime )
return GetLocTime ( t , ptm ) ;
else
return GetGmtTime ( t , ptm ) ;
}
2020-09-09 00:47:02 -04:00
ImPlotTime MakeTime ( int year , int month , int day , int hour , int min , int sec , int us ) {
tm & Tm = GImPlot - > Tm ;
int yr = year - 1900 ;
if ( yr < 0 )
yr = 0 ;
sec = sec + us / 1000000 ;
us = us % 1000000 ;
Tm . tm_sec = sec ;
Tm . tm_min = min ;
Tm . tm_hour = hour ;
Tm . tm_mday = day ;
Tm . tm_mon = month ;
Tm . tm_year = yr ;
ImPlotTime t = MkTime ( & Tm ) ;
t . Us = us ;
return t ;
}
int GetYear ( const ImPlotTime & t ) {
tm & Tm = GImPlot - > Tm ;
GetTime ( t , & Tm ) ;
return Tm . tm_year + 1900 ;
}
2020-09-05 13:25:44 -04:00
ImPlotTime AddTime ( const ImPlotTime & t , ImPlotTimeUnit unit , int count ) {
2020-09-06 01:06:51 -04:00
tm & Tm = GImPlot - > Tm ;
2020-09-05 13:25:44 -04:00
ImPlotTime t_out = t ;
2020-09-05 00:30:45 -04:00
switch ( unit ) {
2020-09-06 01:06:51 -04:00
case ImPlotTimeUnit_Us : t_out . Us + = count ; break ;
case ImPlotTimeUnit_Ms : t_out . Us + = count * 1000 ; break ;
2020-09-05 13:25:44 -04:00
case ImPlotTimeUnit_S : t_out . S + = count ; break ;
case ImPlotTimeUnit_Min : t_out . S + = count * 60 ; break ;
case ImPlotTimeUnit_Hr : t_out . S + = count * 3600 ; break ;
case ImPlotTimeUnit_Day : t_out . S + = count * 86400 ; break ;
2020-09-08 01:56:00 -04:00
case ImPlotTimeUnit_Mo : for ( int i = 0 ; i < abs ( count ) ; + + i ) {
2020-09-06 15:48:16 -04:00
GetTime ( t_out , & Tm ) ;
2020-09-09 00:47:02 -04:00
if ( count > 0 )
2020-09-08 01:56:00 -04:00
t_out . S + = 86400 * GetDaysInMonth ( Tm . tm_year + 1900 , Tm . tm_mon ) ;
else if ( count < 0 )
2020-09-09 00:47:02 -04:00
t_out . S - = 86400 * GetDaysInMonth ( Tm . tm_year + 1900 - ( Tm . tm_mon = = 0 ? 1 : 0 ) , Tm . tm_mon = = 0 ? 11 : Tm . tm_mon - 1 ) ; // NOT WORKING
2020-09-06 01:06:51 -04:00
}
break ;
2020-09-09 00:47:02 -04:00
case ImPlotTimeUnit_Yr : for ( int i = 0 ; i < abs ( count ) ; + + i ) {
if ( count > 0 )
t_out . S + = 86400 * ( 365 + ( int ) IsLeapYear ( GetYear ( t_out ) ) ) ;
else if ( count < 0 )
t_out . S - = 86400 * ( 365 + ( int ) IsLeapYear ( GetYear ( t_out ) - 1 ) ) ;
// this is incorrect if leap year and we are past Feb 28
2020-09-05 00:30:45 -04:00
}
2020-09-05 13:25:44 -04:00
break ;
default : break ;
2020-09-05 00:30:45 -04:00
}
2020-09-06 01:06:51 -04:00
t_out . RollOver ( ) ;
2020-09-05 13:25:44 -04:00
return t_out ;
2020-09-05 00:30:45 -04:00
}
2020-09-05 13:25:44 -04:00
ImPlotTime FloorTime ( const ImPlotTime & t , ImPlotTimeUnit unit ) {
2020-09-06 15:48:16 -04:00
GetTime ( t , & GImPlot - > Tm ) ;
2020-09-05 00:30:45 -04:00
switch ( unit ) {
2020-09-05 13:25:44 -04:00
case ImPlotTimeUnit_S : return ImPlotTime ( t . S , 0 ) ;
case ImPlotTimeUnit_Ms : return ImPlotTime ( t . S , ( t . Us / 1000 ) * 1000 ) ;
case ImPlotTimeUnit_Us : return t ;
2020-09-05 00:30:45 -04:00
case ImPlotTimeUnit_Yr : GImPlot - > Tm . tm_mon = 0 ; // fall-through
case ImPlotTimeUnit_Mo : GImPlot - > Tm . tm_mday = 1 ; // fall-through
case ImPlotTimeUnit_Day : GImPlot - > Tm . tm_hour = 0 ; // fall-through
case ImPlotTimeUnit_Hr : GImPlot - > Tm . tm_min = 0 ; // fall-through
case ImPlotTimeUnit_Min : GImPlot - > Tm . tm_sec = 0 ; break ;
2020-09-05 13:25:44 -04:00
default : return t ;
2020-09-05 00:30:45 -04:00
}
2020-09-06 15:48:16 -04:00
return MkTime ( & GImPlot - > Tm ) ;
2020-09-05 00:30:45 -04:00
}
2020-09-05 13:25:44 -04:00
ImPlotTime CeilTime ( const ImPlotTime & t , ImPlotTimeUnit unit ) {
2020-09-05 00:30:45 -04:00
return AddTime ( FloorTime ( t , unit ) , unit , 1 ) ;
}
2020-09-05 13:25:44 -04:00
ImPlotTime RoundTime ( const ImPlotTime & t , ImPlotTimeUnit unit ) {
ImPlotTime t1 = FloorTime ( t , unit ) ;
ImPlotTime t2 = AddTime ( t1 , unit , 1 ) ;
if ( t1 . S = = t2 . S )
return t . Us - t1 . Us < t2 . Us - t . Us ? t1 : t2 ;
return t . S - t1 . S < t2 . S - t . S ? t1 : t2 ;
2020-09-05 00:30:45 -04:00
}
2020-09-09 10:00:50 -04:00
ImPlotTime CombineDateTime ( const ImPlotTime & date_part , const ImPlotTime & tod_part ) {
tm & Tm = GImPlot - > Tm ;
GetTime ( date_part , & GImPlot - > Tm ) ;
int y = Tm . tm_year ;
int m = Tm . tm_mon ;
int d = Tm . tm_mday ;
GetTime ( tod_part , & GImPlot - > Tm ) ;
Tm . tm_year = y ;
Tm . tm_mon = m ;
Tm . tm_mday = d ;
ImPlotTime t = MkTime ( & Tm ) ;
t . Us = tod_part . Us ;
return t ;
}
2020-09-12 11:25:47 -04:00
static const char * MONTH_NAMES [ ] = { " January " , " February " , " March " , " April " , " May " , " June " , " July " , " August " , " September " , " October " , " November " , " December " } ;
static const char * WD_ABRVS [ ] = { " Su " , " Mo " , " Tu " , " We " , " Th " , " Fr " , " Sa " } ;
2020-10-14 23:07:27 -04:00
static const char * MONTH_ABRVS [ ] = { " Jan " , " Feb " , " Mar " , " Apr " , " May " , " Jun " , " Jul " , " Aug " , " Sep " , " Oct " , " Nov " , " Dec " } ;
2020-09-12 11:25:47 -04:00
2020-10-14 23:07:27 -04:00
int FormatTime ( const ImPlotTime & t , char * buffer , int size , ImPlotTimeFmt fmt , bool use_24_hr_clk ) {
2020-09-05 00:30:45 -04:00
tm & Tm = GImPlot - > Tm ;
2020-09-06 15:48:16 -04:00
GetTime ( t , & Tm ) ;
2020-09-05 13:25:44 -04:00
const int us = t . Us % 1000 ;
const int ms = t . Us / 1000 ;
2020-09-05 10:38:08 -04:00
const int sec = Tm . tm_sec ;
const int min = Tm . tm_min ;
2020-10-14 23:07:27 -04:00
if ( use_24_hr_clk ) {
const int hr = Tm . tm_hour ;
switch ( fmt ) {
case ImPlotTimeFmt_Us : return snprintf ( buffer , size , " .%03d %03d " , ms , us ) ;
case ImPlotTimeFmt_SUs : return snprintf ( buffer , size , " :%02d.%03d %03d " , sec , ms , us ) ;
case ImPlotTimeFmt_SMs : return snprintf ( buffer , size , " :%02d.%03d " , sec , ms ) ;
case ImPlotTimeFmt_S : return snprintf ( buffer , size , " :%02d " , sec ) ;
case ImPlotTimeFmt_HrMinSMs : return snprintf ( buffer , size , " %02d:%02d:%02d.%03d " , hr , min , sec , ms ) ;
case ImPlotTimeFmt_HrMinS : return snprintf ( buffer , size , " %02d:%02d:%02d " , hr , min , sec ) ;
case ImPlotTimeFmt_HrMin : return snprintf ( buffer , size , " %02d:%02d " , hr , min ) ;
case ImPlotTimeFmt_Hr : return snprintf ( buffer , size , " %02d:00 " , hr ) ;
default : return 0 ;
}
}
else {
const char * ap = Tm . tm_hour < 12 ? " am " : " pm " ;
const int hr = ( Tm . tm_hour = = 0 | | Tm . tm_hour = = 12 ) ? 12 : Tm . tm_hour % 12 ;
switch ( fmt ) {
case ImPlotTimeFmt_Us : return snprintf ( buffer , size , " .%03d %03d " , ms , us ) ;
case ImPlotTimeFmt_SUs : return snprintf ( buffer , size , " :%02d.%03d %03d " , sec , ms , us ) ;
case ImPlotTimeFmt_SMs : return snprintf ( buffer , size , " :%02d.%03d " , sec , ms ) ;
case ImPlotTimeFmt_S : return snprintf ( buffer , size , " :%02d " , sec ) ;
case ImPlotTimeFmt_HrMinSMs : return snprintf ( buffer , size , " %d:%02d:%02d.%03d%s " , hr , min , sec , ms , ap ) ;
case ImPlotTimeFmt_HrMinS : return snprintf ( buffer , size , " %d:%02d:%02d%s " , hr , min , sec , ap ) ;
case ImPlotTimeFmt_HrMin : return snprintf ( buffer , size , " %d:%02d%s " , hr , min , ap ) ;
case ImPlotTimeFmt_Hr : return snprintf ( buffer , size , " %d%s " , hr , ap ) ;
default : return 0 ;
}
}
}
2020-09-05 10:38:08 -04:00
2020-10-14 23:07:27 -04:00
int FormatDate ( const ImPlotTime & t , char * buffer , int size , ImPlotDateFmt fmt , bool use_iso_8601 ) {
2020-09-12 11:25:47 -04:00
tm & Tm = GImPlot - > Tm ;
GetTime ( t , & Tm ) ;
const int day = Tm . tm_mday ;
const int mon = Tm . tm_mon + 1 ;
const int year = Tm . tm_year + 1900 ;
const int yr = year % 100 ;
2020-10-14 23:07:27 -04:00
if ( use_iso_8601 ) {
switch ( fmt ) {
case ImPlotDateFmt_DayMo : return snprintf ( buffer , size , " --%02d-%02d " , mon , day ) ;
case ImPlotDateFmt_DayMoYr : return snprintf ( buffer , size , " %d-%02d-%02d " , year , mon , day ) ;
case ImPlotDateFmt_MoYr : return snprintf ( buffer , size , " %d-%02d " , year , mon ) ;
2020-10-19 00:26:34 -04:00
case ImPlotDateFmt_Mo : return snprintf ( buffer , size , " --%02d " , mon ) ;
2020-10-14 23:07:27 -04:00
case ImPlotDateFmt_Yr : return snprintf ( buffer , size , " %d " , year ) ;
default : return 0 ;
}
}
else {
switch ( fmt ) {
case ImPlotDateFmt_DayMo : return snprintf ( buffer , size , " %d/%d " , mon , day ) ;
case ImPlotDateFmt_DayMoYr : return snprintf ( buffer , size , " %d/%d/%02d " , mon , day , yr ) ;
case ImPlotDateFmt_MoYr : return snprintf ( buffer , size , " %s %d " , MONTH_ABRVS [ Tm . tm_mon ] , year ) ;
case ImPlotDateFmt_Mo : return snprintf ( buffer , size , " %s " , MONTH_ABRVS [ Tm . tm_mon ] ) ;
case ImPlotDateFmt_Yr : return snprintf ( buffer , size , " %d " , year ) ;
default : return 0 ;
}
}
}
int FormatDateTime ( const ImPlotTime & t , char * buffer , int size , ImPlotDateTimeFmt fmt ) {
int written = 0 ;
if ( fmt . Date ! = ImPlotDateFmt_None )
written + = FormatDate ( t , buffer , size , fmt . Date , fmt . UseISO8601 ) ;
if ( fmt . Time ! = ImPlotTimeFmt_None ) {
if ( fmt . Date ! = ImPlotDateFmt_None )
buffer [ written + + ] = ' ' ;
written + = FormatTime ( t , & buffer [ written ] , size - written , fmt . Time , fmt . Use24HourClock ) ;
}
return written ;
}
inline float GetDateTimeWidth ( ImPlotDateTimeFmt fmt ) {
static ImPlotTime t_max_width = MakeTime ( 2888 , 12 , 22 , 12 , 58 , 58 , 888888 ) ; // best guess at time that maximizes pixel width
char buffer [ 32 ] ;
FormatDateTime ( t_max_width , buffer , 32 , fmt ) ;
return ImGui : : CalcTextSize ( buffer ) . x ;
}
2020-09-12 11:25:47 -04:00
2020-12-04 00:47:27 -05:00
void LabelTickTime ( ImPlotTick & tick , ImGuiTextBuffer & buffer , const ImPlotTime & t , ImPlotDateTimeFmt fmt ) {
2020-09-05 00:30:45 -04:00
char temp [ 32 ] ;
if ( tick . ShowLabel ) {
2020-09-19 21:54:19 -04:00
tick . TextOffset = buffer . size ( ) ;
2020-10-14 23:07:27 -04:00
FormatDateTime ( t , temp , 32 , fmt ) ;
2020-09-05 00:30:45 -04:00
buffer . append ( temp , temp + strlen ( temp ) + 1 ) ;
2020-09-19 21:54:19 -04:00
tick . LabelSize = ImGui : : CalcTextSize ( buffer . Buf . Data + tick . TextOffset ) ;
2020-09-05 00:30:45 -04:00
}
}
2020-09-04 00:27:56 -04:00
2020-09-06 01:06:51 -04:00
inline bool TimeLabelSame ( const char * l1 , const char * l2 ) {
size_t len1 = strlen ( l1 ) ;
size_t len2 = strlen ( l2 ) ;
size_t n = len1 < len2 ? len1 : len2 ;
return strcmp ( l1 + len1 - n , l2 + len2 - n ) = = 0 ;
}
2020-10-14 23:07:27 -04:00
static const ImPlotDateTimeFmt TimeFormatLevel0 [ ImPlotTimeUnit_COUNT ] = {
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_Us ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_SMs ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_S ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_Hr ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMo , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Mo , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Yr , ImPlotTimeFmt_None )
2020-09-05 10:38:08 -04:00
} ;
2020-10-14 23:07:27 -04:00
static const ImPlotDateTimeFmt TimeFormatLevel1 [ ImPlotTimeUnit_COUNT ] = {
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMinS ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Yr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Yr , ImPlotTimeFmt_None )
2020-09-05 10:38:08 -04:00
} ;
2020-10-14 23:07:27 -04:00
static const ImPlotDateTimeFmt TimeFormatLevel1First [ ImPlotTimeUnit_COUNT ] = {
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_HrMinS ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_HrMinS ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Yr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_Yr , ImPlotTimeFmt_None )
2020-09-05 10:38:08 -04:00
} ;
2020-10-14 23:07:27 -04:00
static const ImPlotDateTimeFmt TimeFormatMouseCursor [ ImPlotTimeUnit_COUNT ] = {
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_Us ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_SUs ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_SMs ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMinS ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_None , ImPlotTimeFmt_HrMin ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMo , ImPlotTimeFmt_Hr ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_DayMoYr , ImPlotTimeFmt_None ) ,
ImPlotDateTimeFmt ( ImPlotDateFmt_MoYr , ImPlotTimeFmt_None )
2020-09-06 01:06:51 -04:00
} ;
2020-10-14 23:07:27 -04:00
inline ImPlotDateTimeFmt GetDateTimeFmt ( const ImPlotDateTimeFmt * ctx , ImPlotTimeUnit idx ) {
ImPlotStyle & style = GetStyle ( ) ;
ImPlotDateTimeFmt fmt = ctx [ idx ] ;
fmt . UseISO8601 = style . UseISO8601 ;
fmt . Use24HourClock = style . Use24HourClock ;
return fmt ;
}
void AddTicksTime ( const ImPlotRange & range , float plot_width , ImPlotTickCollection & ticks ) {
2020-09-04 14:13:45 -04:00
// get units for level 0 and level 1 labels
2020-09-04 00:27:56 -04:00
const ImPlotTimeUnit unit0 = GetUnitForRange ( range . Size ( ) / ( plot_width / 100 ) ) ; // level = 0 (top)
const ImPlotTimeUnit unit1 = unit0 + 1 ; // level = 1 (bottom)
2020-09-05 10:38:08 -04:00
// get time format specs
2020-10-14 23:07:27 -04:00
const ImPlotDateTimeFmt fmt0 = GetDateTimeFmt ( TimeFormatLevel0 , unit0 ) ;
const ImPlotDateTimeFmt fmt1 = GetDateTimeFmt ( TimeFormatLevel1 , unit1 ) ;
const ImPlotDateTimeFmt fmtf = GetDateTimeFmt ( TimeFormatLevel1First , unit1 ) ;
2020-09-06 01:06:51 -04:00
// min max times
const ImPlotTime t_min = ImPlotTime : : FromDouble ( range . Min ) ;
const ImPlotTime t_max = ImPlotTime : : FromDouble ( range . Max ) ;
2020-09-04 00:27:56 -04:00
// maximum allowable density of labels
2020-09-06 01:06:51 -04:00
const float max_density = 0.5f ;
2020-09-05 13:25:44 -04:00
// book keeping
2020-09-06 01:06:51 -04:00
const char * last_major = NULL ;
2020-09-04 14:13:45 -04:00
if ( unit0 ! = ImPlotTimeUnit_Yr ) {
// pixels per major (level 1) division
const float pix_per_major_div = plot_width / ( float ) ( range . Size ( ) / TimeUnitSpans [ unit1 ] ) ;
2020-09-06 01:06:51 -04:00
// nominal pixels taken up by labels
2020-10-14 23:07:27 -04:00
const float fmt0_width = GetDateTimeWidth ( fmt0 ) ;
const float fmt1_width = GetDateTimeWidth ( fmt1 ) ;
const float fmtf_width = GetDateTimeWidth ( fmtf ) ;
2020-09-04 14:13:45 -04:00
// the maximum number of minor (level 0) labels that can fit between major (level 1) divisions
2020-09-06 01:06:51 -04:00
const int minor_per_major = ( int ) ( max_density * pix_per_major_div / fmt0_width ) ;
2020-09-04 14:13:45 -04:00
// the minor step size (level 0)
const int step = GetTimeStep ( minor_per_major , unit0 ) ;
// generate ticks
2020-09-06 01:06:51 -04:00
ImPlotTime t1 = FloorTime ( ImPlotTime : : FromDouble ( range . Min ) , unit1 ) ;
2020-09-05 13:25:44 -04:00
while ( t1 < t_max ) {
2020-09-06 22:08:25 -04:00
// get next major
const ImPlotTime t2 = AddTime ( t1 , unit1 , 1 ) ;
2020-09-06 01:06:51 -04:00
// add major tick
2020-09-05 13:25:44 -04:00
if ( t1 > = t_min & & t1 < = t_max ) {
2020-09-05 10:38:08 -04:00
// minor level 0 tick
2020-09-05 13:25:44 -04:00
ImPlotTick tick_min ( t1 . ToDouble ( ) , true , true ) ;
2020-09-04 00:27:56 -04:00
tick_min . Level = 0 ;
2020-10-14 23:07:27 -04:00
LabelTickTime ( tick_min , ticks . TextBuffer , t1 , fmt0 ) ;
2020-09-19 21:54:19 -04:00
ticks . Append ( tick_min ) ;
2020-09-05 10:38:08 -04:00
// major level 1 tick
2020-09-05 13:25:44 -04:00
ImPlotTick tick_maj ( t1 . ToDouble ( ) , true , true ) ;
2020-09-05 10:38:08 -04:00
tick_maj . Level = 1 ;
2020-10-14 23:07:27 -04:00
LabelTickTime ( tick_maj , ticks . TextBuffer , t1 , last_major = = NULL ? fmtf : fmt1 ) ;
2020-09-19 21:54:19 -04:00
const char * this_major = ticks . TextBuffer . Buf . Data + tick_maj . TextOffset ;
2020-09-06 22:08:25 -04:00
if ( last_major & & TimeLabelSame ( last_major , this_major ) )
tick_maj . ShowLabel = false ;
last_major = this_major ;
2020-09-19 21:54:19 -04:00
ticks . Append ( tick_maj ) ;
2020-09-04 00:27:56 -04:00
}
// add minor ticks up until next major
2020-09-05 13:25:44 -04:00
if ( minor_per_major > 1 & & ( t_min < = t2 & & t1 < = t_max ) ) {
ImPlotTime t12 = AddTime ( t1 , unit0 , step ) ;
2020-09-04 14:13:45 -04:00
while ( t12 < t2 ) {
2020-09-05 13:25:44 -04:00
float px_to_t2 = ( float ) ( ( t2 - t12 ) . ToDouble ( ) / range . Size ( ) ) * plot_width ;
if ( t12 > = t_min & & t12 < = t_max ) {
2020-09-06 01:06:51 -04:00
ImPlotTick tick ( t12 . ToDouble ( ) , false , px_to_t2 > = fmt0_width ) ;
2020-09-04 00:27:56 -04:00
tick . Level = 0 ;
2020-10-14 23:07:27 -04:00
LabelTickTime ( tick , ticks . TextBuffer , t12 , fmt0 ) ;
2020-09-19 21:54:19 -04:00
ticks . Append ( tick ) ;
2020-09-06 01:06:51 -04:00
if ( last_major = = NULL & & px_to_t2 > = fmt0_width & & px_to_t2 > = ( fmt1_width + fmtf_width ) / 2 ) {
ImPlotTick tick_maj ( t12 . ToDouble ( ) , true , true ) ;
tick_maj . Level = 1 ;
2020-10-14 23:07:27 -04:00
LabelTickTime ( tick_maj , ticks . TextBuffer , t12 , fmtf ) ;
2020-09-19 21:54:19 -04:00
last_major = ticks . TextBuffer . Buf . Data + tick_maj . TextOffset ;
ticks . Append ( tick_maj ) ;
2020-09-06 01:06:51 -04:00
}
2020-09-04 00:27:56 -04:00
}
2020-09-04 14:13:45 -04:00
t12 = AddTime ( t12 , unit0 , step ) ;
2020-09-06 22:08:25 -04:00
}
2020-09-04 00:27:56 -04:00
}
2020-09-05 00:30:45 -04:00
t1 = t2 ;
2020-09-04 00:27:56 -04:00
}
}
else {
2020-10-14 23:07:27 -04:00
const ImPlotDateTimeFmt fmty = GetDateTimeFmt ( TimeFormatLevel0 , ImPlotTimeUnit_Yr ) ;
const float label_width = GetDateTimeWidth ( fmty ) ;
2020-09-04 14:13:45 -04:00
const int max_labels = ( int ) ( max_density * plot_width / label_width ) ;
2020-09-06 01:06:51 -04:00
const int year_min = GetYear ( t_min ) ;
const int year_max = GetYear ( CeilTime ( t_max , ImPlotTimeUnit_Yr ) ) ;
2020-09-04 14:13:45 -04:00
const double nice_range = NiceNum ( ( year_max - year_min ) * 0.99 , false ) ;
const double interval = NiceNum ( nice_range / ( max_labels - 1 ) , true ) ;
const int graphmin = ( int ) ( floor ( year_min / interval ) * interval ) ;
2020-09-06 01:06:51 -04:00
const int graphmax = ( int ) ( ceil ( year_max / interval ) * interval ) ;
2020-09-05 00:30:45 -04:00
const int step = ( int ) interval < = 0 ? 1 : ( int ) interval ;
2020-09-06 01:06:51 -04:00
for ( int y = graphmin ; y < graphmax ; y + = step ) {
2020-09-09 20:17:19 -04:00
ImPlotTime t = MakeTime ( y ) ;
2020-09-06 01:06:51 -04:00
if ( t > = t_min & & t < = t_max ) {
ImPlotTick tick ( t . ToDouble ( ) , true , true ) ;
2020-09-04 14:13:45 -04:00
tick . Level = 0 ;
2020-10-14 23:07:27 -04:00
LabelTickTime ( tick , ticks . TextBuffer , t , fmty ) ;
2020-09-19 21:54:19 -04:00
ticks . Append ( tick ) ;
2020-09-04 14:13:45 -04:00
}
2020-09-06 01:06:51 -04:00
}
2020-09-03 00:30:32 -04:00
}
2020-06-07 14:13:11 -04:00
}
2020-06-04 10:56:50 -04:00
//-----------------------------------------------------------------------------
// Axis Utils
//-----------------------------------------------------------------------------
2020-09-21 08:09:14 -04:00
int LabelAxisValue ( const ImPlotAxis & axis , const ImPlotTickCollection & ticks , double value , char * buff , int size ) {
ImPlotContext & gp = * GImPlot ;
if ( ImHasFlag ( axis . Flags , ImPlotAxisFlags_LogScale ) ) {
return snprintf ( buff , size , " %.3E " , value ) ;
}
else if ( ImHasFlag ( axis . Flags , ImPlotAxisFlags_Time ) ) {
2020-11-15 22:47:06 -05:00
ImPlotTimeUnit unit = ( axis . Orientation = = ImPlotOrientation_Horizontal )
? GetUnitForRange ( axis . Range . Size ( ) / ( gp . CurrentPlot - > PlotRect . GetWidth ( ) / 100 ) )
: GetUnitForRange ( axis . Range . Size ( ) / ( gp . CurrentPlot - > PlotRect . GetHeight ( ) / 100 ) ) ;
2020-10-14 23:07:27 -04:00
return FormatDateTime ( ImPlotTime : : FromDouble ( value ) , buff , size , GetDateTimeFmt ( TimeFormatMouseCursor , unit ) ) ;
2020-09-21 08:09:14 -04:00
}
else {
double range = ticks . Size > 1 ? ( ticks . Ticks [ 1 ] . PlotPos - ticks . Ticks [ 0 ] . PlotPos ) : axis . Range . Size ( ) ;
return snprintf ( buff , size , " %.*f " , Precision ( range ) , value ) ;
}
}
2020-11-15 22:47:06 -05:00
void UpdateAxisColors ( int axis_flag , ImPlotAxis * axis ) {
2020-08-24 00:45:42 -04:00
const ImVec4 col_label = GetStyleColorVec4 ( axis_flag ) ;
const ImVec4 col_grid = GetStyleColorVec4 ( axis_flag + 1 ) ;
2020-11-15 22:47:06 -05:00
axis - > ColorMaj = ImGui : : GetColorU32 ( col_grid ) ;
axis - > ColorMin = ImGui : : GetColorU32 ( col_grid * ImVec4 ( 1 , 1 , 1 , GImPlot - > Style . MinorAlpha ) ) ;
axis - > ColorTxt = ImGui : : GetColorU32 ( col_label ) ;
2020-06-04 10:56:50 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// BeginPlot()
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-09-06 22:08:25 -04:00
bool BeginPlot ( const char * title , const char * x_label , const char * y_label , const ImVec2 & size ,
2020-11-15 22:47:06 -05:00
ImPlotFlags flags , ImPlotAxisFlags x_flags , ImPlotAxisFlags y1_flags , ImPlotAxisFlags y2_flags , ImPlotAxisFlags y3_flags )
2020-09-06 01:06:51 -04:00
{
2020-08-16 16:38:51 -04:00
IM_ASSERT_USER_ERROR ( GImPlot ! = NULL , " No current context. Did you call ImPlot::CreateContext() or ImPlot::SetCurrentContext()? " ) ;
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " Mismatched BeginPlot()/EndPlot()! " ) ;
2020-09-06 01:06:51 -04:00
IM_ASSERT_USER_ERROR ( ! ( ImHasFlag ( x_flags , ImPlotAxisFlags_Time ) & & ImHasFlag ( x_flags , ImPlotAxisFlags_LogScale ) ) , " ImPlotAxisFlags_Time and ImPlotAxisFlags_LogScale cannot be enabled at the same time! " ) ;
2020-11-15 22:47:06 -05:00
IM_ASSERT_USER_ERROR ( ! ImHasFlag ( y1_flags , ImPlotAxisFlags_Time ) , " Y axes cannot display time formatted labels! " ) ;
2020-04-27 11:27:59 -04:00
// FRONT MATTER -----------------------------------------------------------
ImGuiContext & G = * GImGui ;
ImGuiWindow * Window = G . CurrentWindow ;
if ( Window - > SkipItems ) {
2020-08-16 16:38:51 -04:00
Reset ( GImPlot ) ;
2020-04-27 11:27:59 -04:00
return false ;
}
const ImGuiID ID = Window - > GetID ( title ) ;
const ImGuiStyle & Style = G . Style ;
2020-05-13 10:11:25 -04:00
const ImGuiIO & IO = ImGui : : GetIO ( ) ;
2020-04-27 11:27:59 -04:00
2020-10-21 11:08:41 -04:00
bool just_created = gp . Plots . GetByKey ( ID ) = = NULL ;
gp . CurrentPlot = gp . Plots . GetOrAddByKey ( ID ) ;
gp . CurrentPlot - > ID = ID ;
ImPlotPlot & plot = * gp . CurrentPlot ;
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
plot . CurrentYAxis = 0 ;
2020-04-27 11:27:59 -04:00
if ( just_created ) {
2020-05-11 07:12:22 -04:00
plot . Flags = flags ;
plot . XAxis . Flags = x_flags ;
2020-11-15 22:47:06 -05:00
plot . YAxis [ 0 ] . Flags = y1_flags ;
2020-05-11 07:12:22 -04:00
plot . YAxis [ 1 ] . Flags = y2_flags ;
plot . YAxis [ 2 ] . Flags = y3_flags ;
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
else {
2020-05-13 10:11:25 -04:00
// TODO: Check which individual flags changed, and only reset those!
2020-05-11 07:12:22 -04:00
// There's probably an easy bit mask trick I'm not aware of.
2020-05-13 10:11:25 -04:00
if ( flags ! = plot . PreviousFlags )
plot . Flags = flags ;
2020-09-06 01:06:51 -04:00
if ( x_flags ! = plot . XAxis . PreviousFlags )
plot . XAxis . Flags = x_flags ;
2020-11-15 22:47:06 -05:00
if ( y1_flags ! = plot . YAxis [ 0 ] . PreviousFlags )
plot . YAxis [ 0 ] . Flags = y1_flags ;
2020-05-11 07:12:22 -04:00
if ( y2_flags ! = plot . YAxis [ 1 ] . PreviousFlags )
2020-08-16 00:00:51 -04:00
plot . YAxis [ 1 ] . Flags = y2_flags ;
2020-05-11 07:12:22 -04:00
if ( y3_flags ! = plot . YAxis [ 2 ] . PreviousFlags )
2020-08-16 00:00:51 -04:00
plot . YAxis [ 2 ] . Flags = y3_flags ;
2020-05-11 07:12:22 -04:00
}
plot . PreviousFlags = flags ;
plot . XAxis . PreviousFlags = x_flags ;
2020-11-15 22:47:06 -05:00
plot . YAxis [ 0 ] . PreviousFlags = y1_flags ;
2020-05-11 07:12:22 -04:00
plot . YAxis [ 1 ] . PreviousFlags = y2_flags ;
plot . YAxis [ 2 ] . PreviousFlags = y3_flags ;
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
// capture scroll with a child region
2020-08-16 16:38:51 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoChild ) ) {
2020-10-19 00:26:34 -04:00
ImGui : : BeginChild ( title , ImVec2 ( size . x = = 0 ? gp . Style . PlotDefaultSize . x : size . x , size . y = = 0 ? gp . Style . PlotDefaultSize . y : size . y ) , false , ImGuiWindowFlags_NoScrollbar ) ;
2020-05-03 01:24:10 -04:00
Window = ImGui : : GetCurrentWindow ( ) ;
Window - > ScrollMax . y = 1.0f ;
2020-06-03 15:37:01 -04:00
gp . ChildWindowMade = true ;
}
else {
gp . ChildWindowMade = false ;
2020-05-03 01:24:10 -04:00
}
ImDrawList & DrawList = * Window - > DrawList ;
2020-04-27 11:27:59 -04:00
// NextPlotData -----------------------------------------------------------
2020-09-03 10:19:34 -04:00
// linked axes
plot . XAxis . LinkedMin = gp . NextPlotData . LinkedXmin ;
plot . XAxis . LinkedMax = gp . NextPlotData . LinkedXmax ;
PullLinkedAxis ( plot . XAxis ) ;
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
plot . YAxis [ i ] . LinkedMin = gp . NextPlotData . LinkedYmin [ i ] ;
plot . YAxis [ i ] . LinkedMax = gp . NextPlotData . LinkedYmax [ i ] ;
PullLinkedAxis ( plot . YAxis [ i ] ) ;
}
2020-04-27 11:27:59 -04:00
if ( gp . NextPlotData . HasXRange ) {
2020-09-06 22:08:25 -04:00
if ( just_created | | gp . NextPlotData . XRangeCond = = ImGuiCond_Always )
plot . XAxis . SetRange ( gp . NextPlotData . X ) ;
2020-04-27 11:27:59 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-05-11 07:12:22 -04:00
if ( gp . NextPlotData . HasYRange [ i ] ) {
2020-09-06 22:08:25 -04:00
if ( just_created | | gp . NextPlotData . YRangeCond [ i ] = = ImGuiCond_Always )
plot . YAxis [ i ] . SetRange ( gp . NextPlotData . Y [ i ] ) ;
2020-04-27 11:27:59 -04:00
}
}
// AXIS STATES ------------------------------------------------------------
2020-11-15 22:47:06 -05:00
plot . XAxis . HasRange = gp . NextPlotData . HasXRange ; plot . XAxis . RangeCond = gp . NextPlotData . XRangeCond ; plot . XAxis . Present = true ;
plot . YAxis [ 0 ] . HasRange = gp . NextPlotData . HasYRange [ 0 ] ; plot . YAxis [ 0 ] . RangeCond = gp . NextPlotData . YRangeCond [ 0 ] ; plot . YAxis [ 0 ] . Present = true ;
plot . YAxis [ 1 ] . HasRange = gp . NextPlotData . HasYRange [ 1 ] ; plot . YAxis [ 1 ] . RangeCond = gp . NextPlotData . YRangeCond [ 1 ] ; plot . YAxis [ 1 ] . Present = ImHasFlag ( plot . Flags , ImPlotFlags_YAxis2 ) ;
plot . YAxis [ 2 ] . HasRange = gp . NextPlotData . HasYRange [ 2 ] ; plot . YAxis [ 2 ] . RangeCond = gp . NextPlotData . YRangeCond [ 2 ] ; plot . YAxis [ 2 ] . Present = ImHasFlag ( plot . Flags , ImPlotFlags_YAxis3 ) ;
2020-04-27 11:27:59 -04:00
2020-08-30 12:03:25 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) & & ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LogScale ) )
gp . Scales [ i ] = ImPlotScale_LinLin ;
else if ( ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) & & ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LogScale ) )
gp . Scales [ i ] = ImPlotScale_LogLin ;
else if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) & & ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LogScale ) )
2020-09-01 00:23:48 -04:00
gp . Scales [ i ] = ImPlotScale_LinLog ;
2020-08-30 12:03:25 -04:00
else if ( ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) & & ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LogScale ) )
gp . Scales [ i ] = ImPlotScale_LogLog ;
}
2020-09-06 22:04:36 -04:00
// constraints
plot . XAxis . Constrain ( ) ;
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i )
plot . YAxis [ i ] . Constrain ( ) ;
2020-11-15 22:47:06 -05:00
// constain equal axes for primary x and y if not approximately equal
// constains x to y since x pixel size depends on y labels width, and causes feedback loops in opposite case
if ( ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ) {
double xar = plot . XAxis . GetAspect ( ) ;
double yar = plot . YAxis [ 0 ] . GetAspect ( ) ;
if ( ! ImAlmostEqual ( xar , yar ) & & ! plot . YAxis [ 0 ] . IsLocked ( ) )
plot . XAxis . SetAspect ( yar ) ;
}
2020-09-06 22:04:36 -04:00
2020-08-24 00:45:42 -04:00
// AXIS COLORS -----------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-11-15 22:47:06 -05:00
UpdateAxisColors ( ImPlotCol_XAxis , & plot . XAxis ) ;
UpdateAxisColors ( ImPlotCol_YAxis , & plot . YAxis [ 0 ] ) ;
UpdateAxisColors ( ImPlotCol_YAxis2 , & plot . YAxis [ 1 ] ) ;
UpdateAxisColors ( ImPlotCol_YAxis3 , & plot . YAxis [ 2 ] ) ;
2020-04-27 11:27:59 -04:00
2020-09-04 00:27:56 -04:00
// BB, PADDING, HOVER -----------------------------------------------------------
2020-04-27 11:27:59 -04:00
// frame
2020-10-19 00:26:34 -04:00
ImVec2 frame_size = ImGui : : CalcItemSize ( size , gp . Style . PlotDefaultSize . x , gp . Style . PlotDefaultSize . y ) ;
2020-08-24 00:45:42 -04:00
if ( frame_size . x < gp . Style . PlotMinSize . x & & size . x < 0.0f )
frame_size . x = gp . Style . PlotMinSize . x ;
if ( frame_size . y < gp . Style . PlotMinSize . y & & size . y < 0.0f )
frame_size . y = gp . Style . PlotMinSize . y ;
2020-11-15 22:47:06 -05:00
plot . FrameRect = ImRect ( Window - > DC . CursorPos , Window - > DC . CursorPos + frame_size ) ;
ImGui : : ItemSize ( plot . FrameRect ) ;
if ( ! ImGui : : ItemAdd ( plot . FrameRect , ID , & plot . FrameRect ) ) {
2020-08-16 16:38:51 -04:00
Reset ( GImPlot ) ;
2020-04-27 11:27:59 -04:00
return false ;
}
2020-11-15 22:47:06 -05:00
plot . FrameHovered = ImGui : : ItemHoverable ( plot . FrameRect , ID ) ;
2020-09-17 00:46:33 -04:00
if ( G . HoveredIdPreviousFrame ! = 0 & & G . HoveredIdPreviousFrame ! = ID )
2020-11-15 22:47:06 -05:00
plot . FrameHovered = false ;
2020-09-18 23:05:43 -04:00
ImGui : : SetItemAllowOverlap ( ) ;
2020-11-15 22:47:06 -05:00
ImGui : : RenderFrame ( plot . FrameRect . Min , plot . FrameRect . Max , GetStyleColorU32 ( ImPlotCol_FrameBg ) , true , Style . FrameRounding ) ;
2020-04-27 11:27:59 -04:00
2020-10-19 00:26:34 -04:00
// canvas/axes bb
2020-11-15 22:47:06 -05:00
plot . CanvasRect = ImRect ( plot . FrameRect . Min + gp . Style . PlotPadding , plot . FrameRect . Max - gp . Style . PlotPadding ) ;
plot . AxesRect = plot . FrameRect ;
2020-10-19 00:26:34 -04:00
// outside legend adjustments
2020-10-21 11:08:41 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoLegend ) & & plot . GetLegendCount ( ) > 0 & & plot . LegendOutside ) {
const ImVec2 legend_size = CalcLegendSize ( plot , gp . Style . LegendInnerPadding , gp . Style . LegendSpacing , plot . LegendOrientation ) ;
2020-10-19 00:26:34 -04:00
const bool west = ImHasFlag ( plot . LegendLocation , ImPlotLocation_West ) & & ! ImHasFlag ( plot . LegendLocation , ImPlotLocation_East ) ;
const bool east = ImHasFlag ( plot . LegendLocation , ImPlotLocation_East ) & & ! ImHasFlag ( plot . LegendLocation , ImPlotLocation_West ) ;
const bool north = ImHasFlag ( plot . LegendLocation , ImPlotLocation_North ) & & ! ImHasFlag ( plot . LegendLocation , ImPlotLocation_South ) ;
const bool south = ImHasFlag ( plot . LegendLocation , ImPlotLocation_South ) & & ! ImHasFlag ( plot . LegendLocation , ImPlotLocation_North ) ;
const bool horz = plot . LegendOrientation = = ImPlotOrientation_Horizontal ;
if ( ( west & & ! horz ) | | ( west & & horz & & ! north & & ! south ) ) {
2020-11-15 22:47:06 -05:00
plot . CanvasRect . Min . x + = ( legend_size . x + gp . Style . LegendPadding . x ) ;
plot . AxesRect . Min . x + = ( legend_size . x + gp . Style . PlotPadding . x ) ;
2020-10-19 00:26:34 -04:00
}
if ( ( east & & ! horz ) | | ( east & & horz & & ! north & & ! south ) ) {
2020-11-15 22:47:06 -05:00
plot . CanvasRect . Max . x - = ( legend_size . x + gp . Style . LegendPadding . x ) ;
plot . AxesRect . Max . x - = ( legend_size . x + gp . Style . PlotPadding . x ) ;
2020-10-19 00:26:34 -04:00
}
if ( ( north & & horz ) | | ( north & & ! horz & & ! west & & ! east ) ) {
2020-11-15 22:47:06 -05:00
plot . CanvasRect . Min . y + = ( legend_size . y + gp . Style . LegendPadding . y ) ;
plot . AxesRect . Min . y + = ( legend_size . y + gp . Style . PlotPadding . y ) ;
2020-10-19 00:26:34 -04:00
}
if ( ( south & & horz ) | | ( south & & ! horz & & ! west & & ! east ) ) {
2020-11-15 22:47:06 -05:00
plot . CanvasRect . Max . y - = ( legend_size . y + gp . Style . LegendPadding . y ) ;
plot . AxesRect . Max . y - = ( legend_size . y + gp . Style . PlotPadding . y ) ;
2020-10-19 00:26:34 -04:00
}
}
2020-04-27 11:27:59 -04:00
2020-09-06 02:32:15 -04:00
gp . RenderX = ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoGridLines ) | |
! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoTickMarks ) | |
! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoTickLabels ) ) ;
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
gp . RenderY [ i ] = plot . YAxis [ i ] . Present & &
2020-09-06 02:32:15 -04:00
( ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoGridLines ) | |
! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoTickMarks ) | |
! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoTickLabels ) ) ;
2020-08-22 23:55:37 -04:00
}
2020-09-04 00:27:56 -04:00
// plot bb
// (1) calc top/bot padding and plot height
2020-11-10 08:45:25 -05:00
ImVec2 title_size = ImVec2 ( 0.0f , 0.0f ) ;
const float txt_height = ImGui : : GetTextLineHeight ( ) ;
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoTitle ) ) {
title_size = ImGui : : CalcTextSize ( title , NULL , true ) ;
}
2020-09-04 00:27:56 -04:00
const float pad_top = title_size . x > 0.0f ? txt_height + gp . Style . LabelPadding . y : 0 ;
2020-11-15 22:47:06 -05:00
const float pad_bot = ( plot . XAxis . IsLabeled ( ) ? txt_height + gp . Style . LabelPadding . y + ( plot . XAxis . IsTime ( ) ? txt_height + gp . Style . LabelPadding . y : 0 ) : 0 )
2020-09-06 22:08:25 -04:00
+ ( x_label ? txt_height + gp . Style . LabelPadding . y : 0 ) ;
2020-09-04 00:27:56 -04:00
2020-11-15 22:47:06 -05:00
const float plot_height = plot . CanvasRect . GetHeight ( ) - pad_top - pad_bot ;
2020-09-04 00:27:56 -04:00
// (2) get y tick labels (needed for left/right pad)
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-06-03 15:37:01 -04:00
if ( gp . RenderY [ i ] & & gp . NextPlotData . ShowDefaultTicksY [ i ] ) {
2020-08-22 23:55:37 -04:00
if ( ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LogScale ) )
2020-09-04 00:27:56 -04:00
AddTicksLogarithmic ( plot . YAxis [ i ] . Range , ImMax ( 2 , ( int ) IM_ROUND ( plot_height * 0.02f ) ) , gp . YTicks [ i ] ) ;
2020-08-22 23:55:37 -04:00
else
2020-09-04 00:27:56 -04:00
AddTicksDefault ( plot . YAxis [ i ] . Range , ImMax ( 2 , ( int ) IM_ROUND ( 0.0025 * plot_height ) ) , IMPLOT_SUB_DIV , gp . YTicks [ i ] ) ;
2020-05-11 07:12:22 -04:00
}
}
2020-04-27 11:27:59 -04:00
2020-09-04 00:27:56 -04:00
// (3) calc left/right pad
2020-08-24 00:45:42 -04:00
const float pad_left = ( y_label ? txt_height + gp . Style . LabelPadding . x : 0 )
2020-11-15 22:47:06 -05:00
+ ( plot . YAxis [ 0 ] . IsLabeled ( ) ? gp . YTicks [ 0 ] . MaxWidth + gp . Style . LabelPadding . x : 0 ) ;
const float pad_right = ( ( plot . YAxis [ 1 ] . Present & & plot . YAxis [ 1 ] . IsLabeled ( ) ) ? gp . YTicks [ 1 ] . MaxWidth + gp . Style . LabelPadding . x : 0 )
+ ( ( plot . YAxis [ 1 ] . Present & & plot . YAxis [ 2 ] . Present ) ? gp . Style . LabelPadding . x + gp . Style . MinorTickLen . y : 0 )
+ ( ( plot . YAxis [ 2 ] . Present & & plot . YAxis [ 2 ] . IsLabeled ( ) ) ? gp . YTicks [ 2 ] . MaxWidth + gp . Style . LabelPadding . x : 0 ) ;
2020-08-19 12:34:52 -04:00
2020-11-15 22:47:06 -05:00
const float plot_width = plot . CanvasRect . GetWidth ( ) - pad_left - pad_right ;
2020-09-04 00:27:56 -04:00
// (4) get x ticks
if ( gp . RenderX & & gp . NextPlotData . ShowDefaultTicksX ) {
2020-11-15 22:47:06 -05:00
if ( plot . XAxis . IsTime ( ) )
2020-10-14 23:07:27 -04:00
AddTicksTime ( plot . XAxis . Range , plot_width , gp . XTicks ) ;
2020-09-04 14:36:56 -04:00
else if ( ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) )
2020-09-04 00:27:56 -04:00
AddTicksLogarithmic ( plot . XAxis . Range , ( int ) IM_ROUND ( plot_width * 0.01f ) , gp . XTicks ) ;
else
AddTicksDefault ( plot . XAxis . Range , ImMax ( 2 , ( int ) IM_ROUND ( 0.0025 * plot_width ) ) , IMPLOT_SUB_DIV , gp . XTicks ) ;
}
2020-08-19 12:34:52 -04:00
2020-09-04 00:27:56 -04:00
// (5) calc plot bb
2020-11-15 22:47:06 -05:00
plot . PlotRect = ImRect ( plot . CanvasRect . Min + ImVec2 ( pad_left , pad_top ) , plot . CanvasRect . Max - ImVec2 ( pad_right , pad_bot ) ) ;
plot . PlotHovered = plot . PlotRect . Contains ( IO . MousePos ) & & plot . FrameHovered ;
2020-04-27 11:27:59 -04:00
2020-08-19 12:34:52 -04:00
// x axis region bb and hover
2020-11-15 22:47:06 -05:00
plot . XAxis . HoverRect = ImRect ( plot . PlotRect . GetBL ( ) , ImVec2 ( plot . PlotRect . Max . x , plot . AxesRect . Max . y ) ) ;
plot . XAxis . ExtHovered = plot . XAxis . HoverRect . Contains ( IO . MousePos ) ;
plot . XAxis . AllHovered = plot . XAxis . ExtHovered | | plot . PlotHovered ;
2020-05-11 07:12:22 -04:00
2020-08-19 12:34:52 -04:00
// axis label reference
2020-11-15 22:47:06 -05:00
gp . YAxisReference [ 0 ] = plot . PlotRect . Min . x ;
gp . YAxisReference [ 1 ] = plot . PlotRect . Max . x ;
gp . YAxisReference [ 2 ] = ! plot . YAxis [ 1 ] . Present ? plot . PlotRect . Max . x : ( gp . YAxisReference [ 1 ] + ( plot . YAxis [ 1 ] . IsLabeled ( ) ? gp . Style . LabelPadding . x + gp . YTicks [ 1 ] . MaxWidth : 0 ) + gp . Style . LabelPadding . x + gp . Style . MinorTickLen . y ) ;
2020-08-19 12:34:52 -04:00
// y axis regions bb and hover
2020-11-15 22:47:06 -05:00
plot . YAxis [ 0 ] . HoverRect = ImRect ( ImVec2 ( plot . AxesRect . Min . x , plot . PlotRect . Min . y ) , ImVec2 ( plot . PlotRect . Min . x , plot . PlotRect . Max . y ) ) ;
plot . YAxis [ 1 ] . HoverRect = plot . YAxis [ 2 ] . Present
? ImRect ( plot . PlotRect . GetTR ( ) , ImVec2 ( gp . YAxisReference [ 2 ] , plot . PlotRect . Max . y ) )
: ImRect ( plot . PlotRect . GetTR ( ) , ImVec2 ( plot . AxesRect . Max . x , plot . PlotRect . Max . y ) ) ;
2020-08-23 00:26:49 -04:00
2020-11-15 22:47:06 -05:00
plot . YAxis [ 2 ] . HoverRect = ImRect ( ImVec2 ( gp . YAxisReference [ 2 ] , plot . PlotRect . Min . y ) , ImVec2 ( plot . AxesRect . Max . x , plot . PlotRect . Max . y ) ) ;
2020-05-11 07:12:22 -04:00
2020-08-19 12:34:52 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
2020-11-15 22:47:06 -05:00
plot . YAxis [ i ] . ExtHovered = plot . YAxis [ i ] . Present & & plot . YAxis [ i ] . HoverRect . Contains ( IO . MousePos ) ;
plot . YAxis [ i ] . AllHovered = plot . YAxis [ i ] . ExtHovered | | plot . PlotHovered ;
2020-08-19 12:34:52 -04:00
}
2020-05-13 10:11:25 -04:00
2020-11-15 22:47:06 -05:00
const bool any_hov_y_axis_region = plot . YAxis [ 0 ] . AllHovered | | plot . YAxis [ 1 ] . AllHovered | | plot . YAxis [ 2 ] . AllHovered ;
2020-04-27 11:27:59 -04:00
2020-04-30 09:45:03 -04:00
bool hov_query = false ;
2020-11-15 22:47:06 -05:00
if ( plot . FrameHovered & & plot . PlotHovered & & plot . Queried & & ! plot . Querying ) {
2020-05-11 07:12:22 -04:00
ImRect bb_query = plot . QueryRect ;
2020-11-15 22:47:06 -05:00
bb_query . Min + = plot . PlotRect . Min ;
bb_query . Max + = plot . PlotRect . Min ;
2020-04-30 09:45:03 -04:00
hov_query = bb_query . Contains ( IO . MousePos ) ;
}
2020-11-15 22:47:06 -05:00
// AXIS ASPECT RATIOS
plot . XAxis . Pixels = plot . PlotRect . GetWidth ( ) ;
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i )
plot . YAxis [ i ] . Pixels = plot . PlotRect . GetHeight ( ) ;
2020-04-30 09:45:03 -04:00
// QUERY DRAG -------------------------------------------------------------
2020-06-15 11:48:22 -04:00
if ( plot . DraggingQuery & & ( IO . MouseReleased [ gp . InputMap . PanButton ] | | ! IO . MouseDown [ gp . InputMap . PanButton ] ) ) {
2020-04-30 09:45:03 -04:00
plot . DraggingQuery = false ;
}
2020-05-13 10:11:25 -04:00
if ( plot . DraggingQuery ) {
2020-05-12 05:19:04 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeAll ) ;
2020-05-11 07:12:22 -04:00
plot . QueryRect . Min + = IO . MouseDelta ;
plot . QueryRect . Max + = IO . MouseDelta ;
2020-04-30 09:45:03 -04:00
}
2020-11-15 22:47:06 -05:00
if ( plot . FrameHovered & & plot . PlotHovered & & hov_query & & ! plot . DraggingQuery & & ! plot . Selecting & & ! plot . LegendHovered ) {
2020-05-12 05:19:04 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeAll ) ;
2020-05-11 07:12:22 -04:00
const bool any_y_dragging = plot . YAxis [ 0 ] . Dragging | | plot . YAxis [ 1 ] . Dragging | | plot . YAxis [ 2 ] . Dragging ;
2020-06-15 11:48:22 -04:00
if ( IO . MouseDown [ gp . InputMap . PanButton ] & & ! plot . XAxis . Dragging & & ! any_y_dragging ) {
2020-04-30 09:45:03 -04:00
plot . DraggingQuery = true ;
2020-05-13 10:11:25 -04:00
}
}
2020-04-27 11:27:59 -04:00
// DRAG INPUT -------------------------------------------------------------
2020-11-15 22:47:06 -05:00
const bool axis_equal = ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ;
2020-04-27 11:27:59 -04:00
// end drags
2020-06-15 11:48:22 -04:00
if ( plot . XAxis . Dragging & & ( IO . MouseReleased [ gp . InputMap . PanButton ] | | ! IO . MouseDown [ gp . InputMap . PanButton ] ) ) {
2020-04-27 11:27:59 -04:00
plot . XAxis . Dragging = false ;
2020-05-13 10:11:25 -04:00
G . IO . MouseDragMaxDistanceSqr [ 0 ] = 0 ;
2020-04-27 11:27:59 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-06-15 11:48:22 -04:00
if ( plot . YAxis [ i ] . Dragging & & ( IO . MouseReleased [ gp . InputMap . PanButton ] | | ! IO . MouseDown [ gp . InputMap . PanButton ] ) ) {
2020-08-19 01:04:05 -04:00
plot . YAxis [ i ] . Dragging = false ;
2020-05-11 07:12:22 -04:00
G . IO . MouseDragMaxDistanceSqr [ 0 ] = 0 ;
}
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
const bool any_y_dragging = plot . YAxis [ 0 ] . Dragging | | plot . YAxis [ 1 ] . Dragging | | plot . YAxis [ 2 ] . Dragging ;
2020-05-11 09:57:36 -04:00
bool drag_in_progress = plot . XAxis . Dragging | | any_y_dragging ;
2020-04-27 11:27:59 -04:00
// do drag
2020-05-11 07:12:22 -04:00
if ( drag_in_progress ) {
2020-05-03 01:24:10 -04:00
UpdateTransformCache ( ) ;
2020-11-15 22:47:06 -05:00
bool equal_dragged = false ;
// special case for axis equal and both x and y0 hovered
if ( axis_equal & & ! plot . XAxis . IsLocked ( ) & & plot . XAxis . Dragging & & ! plot . YAxis [ 0 ] . IsLocked ( ) & & plot . YAxis [ 0 ] . Dragging ) {
ImPlotPoint plot_tl = PixelsToPlot ( plot . PlotRect . Min - IO . MouseDelta , 0 ) ;
ImPlotPoint plot_br = PixelsToPlot ( plot . PlotRect . Max - IO . MouseDelta , 0 ) ;
if ( ! plot . XAxis . IsLockedMin ( ) )
plot . XAxis . SetMin ( plot . XAxis . IsInverted ( ) ? plot_br . x : plot_tl . x ) ;
if ( ! plot . XAxis . IsLockedMax ( ) )
plot . XAxis . SetMax ( plot . XAxis . IsInverted ( ) ? plot_tl . x : plot_br . x ) ;
if ( ! plot . YAxis [ 0 ] . IsLockedMin ( ) )
plot . YAxis [ 0 ] . SetMin ( plot . YAxis [ 0 ] . IsInverted ( ) ? plot_tl . y : plot_br . y ) ;
if ( ! plot . YAxis [ 0 ] . IsLockedMax ( ) )
plot . YAxis [ 0 ] . SetMax ( plot . YAxis [ 0 ] . IsInverted ( ) ? plot_br . y : plot_tl . y ) ;
double xar = plot . XAxis . GetAspect ( ) ;
double yar = plot . YAxis [ 0 ] . GetAspect ( ) ;
if ( ! ImAlmostEqual ( xar , yar ) & & ! plot . YAxis [ 0 ] . IsLocked ( ) )
plot . XAxis . SetAspect ( yar ) ;
equal_dragged = true ;
}
if ( ! plot . XAxis . IsLocked ( ) & & plot . XAxis . Dragging & & ! equal_dragged ) {
ImPlotPoint plot_tl = PixelsToPlot ( plot . PlotRect . Min - IO . MouseDelta , 0 ) ;
ImPlotPoint plot_br = PixelsToPlot ( plot . PlotRect . Max - IO . MouseDelta , 0 ) ;
if ( ! plot . XAxis . IsLockedMin ( ) )
plot . XAxis . SetMin ( plot . XAxis . IsInverted ( ) ? plot_br . x : plot_tl . x ) ;
if ( ! plot . XAxis . IsLockedMax ( ) )
plot . XAxis . SetMax ( plot . XAxis . IsInverted ( ) ? plot_tl . x : plot_br . x ) ;
if ( axis_equal )
plot . YAxis [ 0 ] . SetAspect ( plot . XAxis . GetAspect ( ) ) ;
2020-05-11 07:12:22 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( ! plot . YAxis [ i ] . IsLocked ( ) & & plot . YAxis [ i ] . Dragging & & ! ( i = = 0 & & equal_dragged ) ) {
ImPlotPoint plot_tl = PixelsToPlot ( plot . PlotRect . Min - IO . MouseDelta , i ) ;
ImPlotPoint plot_br = PixelsToPlot ( plot . PlotRect . Max - IO . MouseDelta , i ) ;
if ( ! plot . YAxis [ i ] . IsLockedMin ( ) )
plot . YAxis [ i ] . SetMin ( plot . YAxis [ i ] . IsInverted ( ) ? plot_tl . y : plot_br . y ) ;
if ( ! plot . YAxis [ i ] . IsLockedMax ( ) )
plot . YAxis [ i ] . SetMax ( plot . YAxis [ i ] . IsInverted ( ) ? plot_br . y : plot_tl . y ) ;
if ( i = = 0 & & axis_equal )
plot . XAxis . SetAspect ( plot . YAxis [ 0 ] . GetAspect ( ) ) ;
2020-05-11 07:12:22 -04:00
}
}
// Set the mouse cursor based on which axes are moving.
int direction = 0 ;
2020-11-15 22:47:06 -05:00
if ( ! plot . XAxis . IsLocked ( ) & & plot . XAxis . Dragging ) {
2020-05-11 07:12:22 -04:00
direction | = ( 1 < < 1 ) ;
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( ! plot . YAxis [ i ] . Present ) { continue ; }
if ( ! plot . YAxis [ i ] . IsLocked ( ) & & plot . YAxis [ i ] . Dragging ) {
2020-05-11 07:12:22 -04:00
direction | = ( 1 < < 2 ) ;
break ;
}
}
2020-06-06 18:54:46 -04:00
if ( IO . MouseDragMaxDistanceSqr [ 0 ] > 5 ) {
2020-08-23 00:26:49 -04:00
if ( direction = = 0 )
ImGui : : SetMouseCursor ( ImGuiMouseCursor_NotAllowed ) ;
else if ( direction = = ( 1 < < 1 ) )
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeEW ) ;
else if ( direction = = ( 1 < < 2 ) )
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeNS ) ;
else
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeAll ) ;
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
// start drag
2020-11-15 22:47:06 -05:00
if ( ! drag_in_progress & & plot . FrameHovered & & IO . MouseClicked [ gp . InputMap . PanButton ] & & ImHasFlag ( IO . KeyMods , gp . InputMap . PanMod ) & & ! plot . Selecting & & ! plot . LegendHovered & & ! hov_query & & ! plot . DraggingQuery ) {
if ( plot . XAxis . AllHovered ) {
2020-05-11 07:12:22 -04:00
plot . XAxis . Dragging = true ;
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( plot . YAxis [ i ] . AllHovered ) {
2020-05-11 07:12:22 -04:00
plot . YAxis [ i ] . Dragging = true ;
}
}
}
2020-04-27 11:27:59 -04:00
// SCROLL INPUT -----------------------------------------------------------
2020-11-15 22:47:06 -05:00
if ( plot . FrameHovered & & ( plot . XAxis . AllHovered | | any_hov_y_axis_region ) & & IO . MouseWheel ! = 0 ) {
2020-05-03 01:24:10 -04:00
UpdateTransformCache ( ) ;
2020-08-19 01:04:05 -04:00
float zoom_rate = IMPLOT_ZOOM_RATE ;
2020-05-13 10:11:25 -04:00
if ( IO . MouseWheel > 0 )
zoom_rate = ( - zoom_rate ) / ( 1.0f + ( 2.0f * zoom_rate ) ) ;
2020-11-15 22:47:06 -05:00
float tx = ImRemap ( IO . MousePos . x , plot . PlotRect . Min . x , plot . PlotRect . Max . x , 0.0f , 1.0f ) ;
float ty = ImRemap ( IO . MousePos . y , plot . PlotRect . Min . y , plot . PlotRect . Max . y , 0.0f , 1.0f ) ;
bool equal_zoomed = false ;
// special case for axis equal and both x and y0 hovered
if ( axis_equal & & plot . XAxis . AllHovered & & ! plot . XAxis . IsLocked ( ) & & plot . YAxis [ 0 ] . AllHovered & & ! plot . YAxis [ 0 ] . IsLocked ( ) ) {
const ImPlotPoint & plot_tl = PixelsToPlot ( plot . PlotRect . Min - plot . PlotRect . GetSize ( ) * ImVec2 ( tx * zoom_rate , ty * zoom_rate ) , 0 ) ;
const ImPlotPoint & plot_br = PixelsToPlot ( plot . PlotRect . Max + plot . PlotRect . GetSize ( ) * ImVec2 ( ( 1 - tx ) * zoom_rate , ( 1 - ty ) * zoom_rate ) , 0 ) ;
if ( ! plot . XAxis . IsLockedMin ( ) )
plot . XAxis . SetMin ( plot . XAxis . IsInverted ( ) ? plot_br . x : plot_tl . x ) ;
if ( ! plot . XAxis . IsLockedMax ( ) )
plot . XAxis . SetMax ( plot . XAxis . IsInverted ( ) ? plot_tl . x : plot_br . x ) ;
if ( ! plot . YAxis [ 0 ] . IsLockedMin ( ) )
plot . YAxis [ 0 ] . SetMin ( plot . YAxis [ 0 ] . IsInverted ( ) ? plot_tl . y : plot_br . y ) ;
if ( ! plot . YAxis [ 0 ] . IsLockedMax ( ) )
plot . YAxis [ 0 ] . SetMax ( plot . YAxis [ 0 ] . IsInverted ( ) ? plot_br . y : plot_tl . y ) ;
double xar = plot . XAxis . GetAspect ( ) ;
double yar = plot . YAxis [ 0 ] . GetAspect ( ) ;
if ( ! ImAlmostEqual ( xar , yar ) & & ! plot . YAxis [ 0 ] . IsLocked ( ) )
plot . XAxis . SetAspect ( yar ) ;
equal_zoomed = true ;
}
if ( plot . XAxis . AllHovered & & ! plot . XAxis . IsLocked ( ) & & ! equal_zoomed ) {
const ImPlotPoint & plot_tl = PixelsToPlot ( plot . PlotRect . Min - plot . PlotRect . GetSize ( ) * ImVec2 ( tx * zoom_rate , ty * zoom_rate ) , 0 ) ;
const ImPlotPoint & plot_br = PixelsToPlot ( plot . PlotRect . Max + plot . PlotRect . GetSize ( ) * ImVec2 ( ( 1 - tx ) * zoom_rate , ( 1 - ty ) * zoom_rate ) , 0 ) ;
if ( ! plot . XAxis . IsLockedMin ( ) )
plot . XAxis . SetMin ( plot . XAxis . IsInverted ( ) ? plot_br . x : plot_tl . x ) ;
if ( ! plot . XAxis . IsLockedMax ( ) )
plot . XAxis . SetMax ( plot . XAxis . IsInverted ( ) ? plot_tl . x : plot_br . x ) ;
if ( axis_equal )
plot . YAxis [ 0 ] . SetAspect ( plot . XAxis . GetAspect ( ) ) ;
2020-05-11 07:12:22 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( plot . YAxis [ i ] . AllHovered & & ! plot . YAxis [ i ] . IsLocked ( ) & & ! ( i = = 0 & & equal_zoomed ) ) {
const ImPlotPoint & plot_tl = PixelsToPlot ( plot . PlotRect . Min - plot . PlotRect . GetSize ( ) * ImVec2 ( tx * zoom_rate , ty * zoom_rate ) , i ) ;
const ImPlotPoint & plot_br = PixelsToPlot ( plot . PlotRect . Max + plot . PlotRect . GetSize ( ) * ImVec2 ( ( 1 - tx ) * zoom_rate , ( 1 - ty ) * zoom_rate ) , i ) ;
if ( ! plot . YAxis [ i ] . IsLockedMin ( ) )
plot . YAxis [ i ] . SetMin ( plot . YAxis [ i ] . IsInverted ( ) ? plot_tl . y : plot_br . y ) ;
if ( ! plot . YAxis [ i ] . IsLockedMax ( ) )
plot . YAxis [ i ] . SetMax ( plot . YAxis [ i ] . IsInverted ( ) ? plot_br . y : plot_tl . y ) ;
if ( i = = 0 & & axis_equal )
plot . XAxis . SetAspect ( plot . YAxis [ 0 ] . GetAspect ( ) ) ;
2020-05-11 07:12:22 -04:00
}
2020-05-13 10:11:25 -04:00
}
2020-04-27 11:27:59 -04:00
}
2020-04-28 21:17:26 -04:00
// BOX-SELECTION AND QUERY ------------------------------------------------
2020-04-27 11:27:59 -04:00
// confirm selection
2020-06-15 11:48:22 -04:00
if ( plot . Selecting & & ( IO . MouseReleased [ gp . InputMap . BoxSelectButton ] | | ! IO . MouseDown [ gp . InputMap . BoxSelectButton ] ) ) {
2020-05-03 01:24:10 -04:00
UpdateTransformCache ( ) ;
2020-04-28 21:17:26 -04:00
ImVec2 select_size = plot . SelectStart - IO . MousePos ;
2020-09-06 01:42:03 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) ) {
2020-05-16 22:09:36 -04:00
ImPlotPoint p1 = PixelsToPlot ( plot . SelectStart ) ;
ImPlotPoint p2 = PixelsToPlot ( IO . MousePos ) ;
2020-08-29 09:31:34 -04:00
const bool x_can_change = ! ImHasFlag ( IO . KeyMods , gp . InputMap . HorizontalMod ) & & ImFabs ( select_size . x ) > 2 ;
const bool y_can_change = ! ImHasFlag ( IO . KeyMods , gp . InputMap . VerticalMod ) & & ImFabs ( select_size . y ) > 2 ;
2020-11-15 22:47:06 -05:00
if ( ! plot . XAxis . IsLockedMin ( ) & & x_can_change )
2020-09-04 20:33:10 -04:00
plot . XAxis . SetMin ( ImMin ( p1 . x , p2 . x ) ) ;
2020-11-15 22:47:06 -05:00
if ( ! plot . XAxis . IsLockedMax ( ) & & x_can_change )
2020-09-04 20:33:10 -04:00
plot . XAxis . SetMax ( ImMax ( p1 . x , p2 . x ) ) ;
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-05-11 07:12:22 -04:00
p1 = PixelsToPlot ( plot . SelectStart , i ) ;
p2 = PixelsToPlot ( IO . MousePos , i ) ;
2020-11-15 22:47:06 -05:00
if ( ! plot . YAxis [ i ] . IsLockedMin ( ) & & y_can_change )
2020-09-04 20:33:10 -04:00
plot . YAxis [ i ] . SetMin ( ImMin ( p1 . y , p2 . y ) ) ;
2020-11-15 22:47:06 -05:00
if ( ! plot . YAxis [ i ] . IsLockedMax ( ) & & y_can_change )
2020-09-04 20:33:10 -04:00
plot . YAxis [ i ] . SetMax ( ImMax ( p1 . y , p2 . y ) ) ;
2020-05-11 07:12:22 -04:00
}
2020-05-13 10:11:25 -04:00
}
2020-04-27 11:27:59 -04:00
plot . Selecting = false ;
}
// bad selection
2020-11-15 22:47:06 -05:00
if ( plot . Selecting & & ( ImHasFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) | | plot . IsLocked ( ) ) & & ImLengthSqr ( plot . SelectStart - IO . MousePos ) > 4 ) {
2020-04-27 11:27:59 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_NotAllowed ) ;
}
// cancel selection
2020-06-15 20:54:58 -04:00
if ( plot . Selecting & & ( IO . MouseClicked [ gp . InputMap . BoxSelectCancelButton ] | | IO . MouseDown [ gp . InputMap . BoxSelectCancelButton ] ) ) {
2020-04-27 11:27:59 -04:00
plot . Selecting = false ;
}
2020-04-28 00:57:49 -04:00
// begin selection or query
2020-11-15 22:47:06 -05:00
if ( plot . FrameHovered & & plot . PlotHovered & & IO . MouseClicked [ gp . InputMap . BoxSelectButton ] & & ImHasFlag ( IO . KeyMods , gp . InputMap . BoxSelectMod ) ) {
2020-04-27 11:27:59 -04:00
plot . SelectStart = IO . MousePos ;
2020-04-28 21:17:26 -04:00
plot . Selecting = true ;
}
// update query
if ( plot . Querying ) {
2020-05-03 01:24:10 -04:00
UpdateTransformCache ( ) ;
2020-11-15 22:47:06 -05:00
plot . QueryRect . Min . x = ImHasFlag ( IO . KeyMods , gp . InputMap . HorizontalMod ) ? plot . PlotRect . Min . x : ImMin ( plot . QueryStart . x , IO . MousePos . x ) ;
plot . QueryRect . Max . x = ImHasFlag ( IO . KeyMods , gp . InputMap . HorizontalMod ) ? plot . PlotRect . Max . x : ImMax ( plot . QueryStart . x , IO . MousePos . x ) ;
plot . QueryRect . Min . y = ImHasFlag ( IO . KeyMods , gp . InputMap . VerticalMod ) ? plot . PlotRect . Min . y : ImMin ( plot . QueryStart . y , IO . MousePos . y ) ;
plot . QueryRect . Max . y = ImHasFlag ( IO . KeyMods , gp . InputMap . VerticalMod ) ? plot . PlotRect . Max . y : ImMax ( plot . QueryStart . y , IO . MousePos . y ) ;
2020-05-11 07:12:22 -04:00
2020-11-15 22:47:06 -05:00
plot . QueryRect . Min - = plot . PlotRect . Min ;
plot . QueryRect . Max - = plot . PlotRect . Min ;
2020-04-28 21:17:26 -04:00
}
// end query
2020-06-15 20:54:58 -04:00
if ( plot . Querying & & ( IO . MouseReleased [ gp . InputMap . QueryButton ] | | IO . MouseReleased [ gp . InputMap . BoxSelectButton ] ) ) {
2020-04-28 21:17:26 -04:00
plot . Querying = false ;
2020-08-23 00:26:49 -04:00
if ( plot . QueryRect . GetWidth ( ) > 2 & & plot . QueryRect . GetHeight ( ) > 2 )
plot . Queried = true ;
else
plot . Queried = false ;
2020-04-28 00:57:49 -04:00
}
2020-06-10 16:50:00 -04:00
2020-04-28 00:57:49 -04:00
// begin query
2020-11-15 22:47:06 -05:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_Query ) & & plot . FrameHovered & & plot . PlotHovered & & IO . MouseClicked [ gp . InputMap . QueryButton ] & & ImHasFlag ( IO . KeyMods , gp . InputMap . QueryMod ) ) {
2020-04-28 21:17:26 -04:00
plot . QueryRect = ImRect ( 0 , 0 , 0 , 0 ) ;
plot . Querying = true ;
plot . Queried = true ;
plot . QueryStart = IO . MousePos ;
}
// toggle between select/query
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_Query ) & & plot . Selecting & & ImHasFlag ( IO . KeyMods , gp . InputMap . QueryToggleMod ) ) {
2020-04-28 21:17:26 -04:00
plot . Selecting = false ;
plot . QueryRect = ImRect ( 0 , 0 , 0 , 0 ) ;
2020-04-28 00:57:49 -04:00
plot . Querying = true ;
2020-04-28 21:17:26 -04:00
plot . Queried = true ;
plot . QueryStart = plot . SelectStart ;
}
2020-09-06 01:42:03 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) & & plot . Querying & & ! ImHasFlag ( IO . KeyMods , gp . InputMap . QueryToggleMod ) & & ! IO . MouseDown [ gp . InputMap . QueryButton ] ) {
2020-04-28 21:17:26 -04:00
plot . Selecting = true ;
plot . Querying = false ;
plot . Queried = false ;
plot . QueryRect = ImRect ( 0 , 0 , 0 , 0 ) ;
2020-04-27 11:27:59 -04:00
}
2020-08-19 12:43:00 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_Query ) ) {
plot . Queried = false ;
plot . Querying = false ;
plot . QueryRect = ImRect ( 0 , 0 , 0 , 0 ) ;
}
2020-05-13 10:11:25 -04:00
2020-08-20 00:50:12 -04:00
// FIT -----------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-20 00:50:12 -04:00
// fit from double click
2020-11-15 22:47:06 -05:00
if ( IO . MouseDoubleClicked [ gp . InputMap . FitButton ] & & plot . FrameHovered & & ( plot . XAxis . AllHovered | | any_hov_y_axis_region ) & & ! plot . LegendHovered & & ! hov_query ) {
2020-04-27 11:27:59 -04:00
gp . FitThisFrame = true ;
2020-11-15 22:47:06 -05:00
gp . FitX = plot . XAxis . AllHovered ;
2020-08-23 00:26:49 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + )
2020-11-15 22:47:06 -05:00
gp . FitY [ i ] = plot . YAxis [ i ] . AllHovered ;
2020-04-30 09:13:44 -04:00
}
2020-08-20 00:50:12 -04:00
// fit from FitNextPlotAxes
if ( gp . NextPlotData . FitX ) {
gp . FitThisFrame = true ;
gp . FitX = true ;
}
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
if ( gp . NextPlotData . FitY [ i ] ) {
gp . FitThisFrame = true ;
gp . FitY [ i ] = true ;
}
2020-04-30 09:13:44 -04:00
}
2020-04-27 11:27:59 -04:00
// FOCUS ------------------------------------------------------------------
// focus window
2020-11-15 22:47:06 -05:00
if ( ( IO . MouseClicked [ 0 ] | | IO . MouseClicked [ 1 ] | | IO . MouseClicked [ 2 ] ) & & plot . FrameHovered )
2020-05-13 10:11:25 -04:00
ImGui : : FocusWindow ( ImGui : : GetCurrentWindow ( ) ) ;
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
UpdateTransformCache ( ) ;
2020-04-27 11:27:59 -04:00
// set mouse position
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-08-22 23:55:37 -04:00
gp . MousePos [ i ] = PixelsToPlot ( IO . MousePos , i ) ;
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
// RENDER -----------------------------------------------------------------
// grid bg
2020-11-15 22:47:06 -05:00
DrawList . AddRectFilled ( plot . PlotRect . Min , plot . PlotRect . Max , GetStyleColorU32 ( ImPlotCol_PlotBg ) ) ;
2020-04-27 11:27:59 -04:00
// render axes
2020-05-03 01:24:10 -04:00
PushPlotClipRect ( ) ;
2020-04-27 11:27:59 -04:00
2020-08-25 22:59:43 -04:00
// transform ticks (TODO: Move this into ImPlotTickCollection)
2020-04-27 11:27:59 -04:00
if ( gp . RenderX ) {
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . XTicks . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick * xt = & gp . XTicks . Ticks [ t ] ;
2020-05-29 13:39:30 -04:00
xt - > PixelPos = PlotToPixels ( xt - > PlotPos , 0 , 0 ) . x ;
2020-05-13 00:45:34 -04:00
}
2020-04-27 11:27:59 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-05-11 07:12:22 -04:00
if ( gp . RenderY [ i ] ) {
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . YTicks [ i ] . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick * yt = & gp . YTicks [ i ] . Ticks [ t ] ;
2020-05-16 22:53:59 -04:00
yt - > PixelPos = PlotToPixels ( 0 , yt - > PlotPos , i ) . y ;
2020-05-13 00:45:34 -04:00
}
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
// render grid
2020-09-06 02:32:15 -04:00
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoGridLines ) ) {
2020-11-15 22:47:06 -05:00
float density = gp . XTicks . Size / plot . PlotRect . GetWidth ( ) ;
ImVec4 col_min = ImGui : : ColorConvertU32ToFloat4 ( plot . XAxis . ColorMin ) ;
2020-08-22 23:55:37 -04:00
col_min . w * = ImClamp ( ImRemap ( density , 0.1f , 0.2f , 1.0f , 0.0f ) , 0.0f , 1.0f ) ;
ImU32 col_min32 = ImGui : : ColorConvertFloat4ToU32 ( col_min ) ;
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . XTicks . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick & xt = gp . XTicks . Ticks [ t ] ;
2020-09-04 00:27:56 -04:00
if ( xt . Level = = 0 ) {
if ( xt . Major )
2020-11-15 22:47:06 -05:00
DrawList . AddLine ( ImVec2 ( xt . PixelPos , plot . PlotRect . Min . y ) , ImVec2 ( xt . PixelPos , plot . PlotRect . Max . y ) , plot . XAxis . ColorMaj , gp . Style . MajorGridSize . x ) ;
2020-09-04 00:27:56 -04:00
else if ( density < 0.2f )
2020-11-15 22:47:06 -05:00
DrawList . AddLine ( ImVec2 ( xt . PixelPos , plot . PlotRect . Min . y ) , ImVec2 ( xt . PixelPos , plot . PlotRect . Max . y ) , col_min32 , gp . Style . MinorGridSize . x ) ;
2020-09-04 00:27:56 -04:00
}
2020-05-13 00:45:34 -04:00
}
2020-04-27 11:27:59 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( plot . YAxis [ i ] . Present & & ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoGridLines ) ) {
float density = gp . YTicks [ i ] . Size / plot . PlotRect . GetHeight ( ) ;
ImVec4 col_min = ImGui : : ColorConvertU32ToFloat4 ( plot . YAxis [ i ] . ColorMin ) ;
2020-08-22 23:55:37 -04:00
col_min . w * = ImClamp ( ImRemap ( density , 0.1f , 0.2f , 1.0f , 0.0f ) , 0.0f , 1.0f ) ;
ImU32 col_min32 = ImGui : : ColorConvertFloat4ToU32 ( col_min ) ;
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . YTicks [ i ] . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick & yt = gp . YTicks [ i ] . Ticks [ t ] ;
2020-08-22 23:55:37 -04:00
if ( yt . Major )
2020-11-15 22:47:06 -05:00
DrawList . AddLine ( ImVec2 ( plot . PlotRect . Min . x , yt . PixelPos ) , ImVec2 ( plot . PlotRect . Max . x , yt . PixelPos ) , plot . YAxis [ i ] . ColorMaj , gp . Style . MajorGridSize . y ) ;
2020-08-22 23:55:37 -04:00
else if ( density < 0.2f )
2020-11-15 22:47:06 -05:00
DrawList . AddLine ( ImVec2 ( plot . PlotRect . Min . x , yt . PixelPos ) , ImVec2 ( plot . PlotRect . Max . x , yt . PixelPos ) , col_min32 , gp . Style . MinorGridSize . y ) ;
2020-05-13 00:45:34 -04:00
}
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect ( ) ;
2020-04-27 11:27:59 -04:00
// render title
2020-11-10 08:45:25 -05:00
if ( title_size . x > 0.0f & & ! ImHasFlag ( plot . Flags , ImPlotFlags_NoTitle ) ) {
2020-08-24 00:45:42 -04:00
ImU32 col = GetStyleColorU32 ( ImPlotCol_TitleText ) ;
const char * title_end = ImGui : : FindRenderedTextEnd ( title , NULL ) ;
2020-11-15 22:47:06 -05:00
DrawList . AddText ( ImVec2 ( plot . CanvasRect . GetCenter ( ) . x - title_size . x * 0.5f , plot . CanvasRect . Min . y ) , col , title , title_end ) ;
2020-04-27 11:27:59 -04:00
}
2020-08-22 23:55:37 -04:00
// render axis labels
if ( x_label ) {
const ImVec2 xLabel_size = ImGui : : CalcTextSize ( x_label ) ;
2020-11-15 22:47:06 -05:00
const ImVec2 xLabel_pos ( plot . PlotRect . GetCenter ( ) . x - xLabel_size . x * 0.5f , plot . CanvasRect . Max . y - txt_height ) ;
DrawList . AddText ( xLabel_pos , plot . XAxis . ColorTxt , x_label ) ;
2020-08-22 23:55:37 -04:00
}
if ( y_label ) {
const ImVec2 yLabel_size = CalcTextSizeVertical ( y_label ) ;
2020-11-15 22:47:06 -05:00
const ImVec2 yLabel_pos ( plot . CanvasRect . Min . x , plot . PlotRect . GetCenter ( ) . y + yLabel_size . y * 0.5f ) ;
AddTextVertical ( & DrawList , yLabel_pos , plot . YAxis [ 0 ] . ColorTxt , y_label ) ;
2020-08-24 09:51:03 -04:00
}
2020-08-22 23:55:37 -04:00
// render tick labels
2020-11-15 22:47:06 -05:00
ImGui : : PushClipRect ( plot . FrameRect . Min , plot . FrameRect . Max , true ) ;
2020-09-06 02:32:15 -04:00
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoTickLabels ) ) {
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . XTicks . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick * xt = & gp . XTicks . Ticks [ t ] ;
2020-11-15 22:47:06 -05:00
if ( xt - > ShowLabel & & xt - > PixelPos > = plot . PlotRect . Min . x - 1 & & xt - > PixelPos < = plot . PlotRect . Max . x + 1 )
DrawList . AddText ( ImVec2 ( xt - > PixelPos - xt - > LabelSize . x * 0.5f , plot . PlotRect . Max . y + gp . Style . LabelPadding . y + xt - > Level * ( txt_height + gp . Style . LabelPadding . y ) ) ,
xt - > Major ? plot . XAxis . ColorTxt : plot . XAxis . ColorTxt , gp . XTicks . GetText ( t ) ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( plot . YAxis [ i ] . Present & & ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoTickLabels ) ) {
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . YTicks [ i ] . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
const float x_start = gp . YAxisReference [ i ] + ( i = = 0 ? ( - gp . Style . LabelPadding . x - gp . YTicks [ i ] . Ticks [ t ] . LabelSize . x ) : gp . Style . LabelPadding . x ) ;
ImPlotTick * yt = & gp . YTicks [ i ] . Ticks [ t ] ;
2020-11-15 22:47:06 -05:00
if ( yt - > ShowLabel & & yt - > PixelPos > = plot . PlotRect . Min . y - 1 & & yt - > PixelPos < = plot . PlotRect . Max . y + 1 ) {
2020-08-19 01:04:05 -04:00
ImVec2 start ( x_start , yt - > PixelPos - 0.5f * yt - > LabelSize . y ) ;
2020-11-15 22:47:06 -05:00
DrawList . AddText ( start , yt - > Major ? plot . YAxis [ i ] . ColorTxt : plot . YAxis [ i ] . ColorTxt , gp . YTicks [ i ] . GetText ( t ) ) ;
2020-05-11 07:12:22 -04:00
}
}
2020-04-27 11:27:59 -04:00
}
}
2020-05-12 05:19:04 -04:00
ImGui : : PopClipRect ( ) ;
2020-10-19 00:26:34 -04:00
// clear legend
2020-10-21 11:08:41 -04:00
plot . LegendData . Reset ( ) ;
2020-04-27 11:27:59 -04:00
// push plot ID into stack
2020-05-12 05:19:04 -04:00
ImGui : : PushID ( ID ) ;
2020-04-27 11:27:59 -04:00
return true ;
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// Context Menu
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-05-17 00:25:15 -04:00
template < typename F >
2020-09-07 22:30:57 -04:00
bool DragFloat ( const char * , F * , float , F , F ) {
2020-05-17 00:25:15 -04:00
return false ;
}
template < >
2020-06-02 23:07:27 -04:00
bool DragFloat < double > ( const char * label , double * v , float v_speed , double v_min , double v_max ) {
2020-05-17 00:25:15 -04:00
return ImGui : : DragScalar ( label , ImGuiDataType_Double , v , v_speed , & v_min , & v_max , " %.3f " , 1 ) ;
}
2020-05-16 22:09:36 -04:00
2020-05-17 00:25:15 -04:00
template < >
2020-06-02 23:07:27 -04:00
bool DragFloat < float > ( const char * label , float * v , float v_speed , float v_min , float v_max ) {
2020-05-17 00:25:15 -04:00
return ImGui : : DragScalar ( label , ImGuiDataType_Float , v , v_speed , & v_min , & v_max , " %.3f " , 1 ) ;
2020-05-16 22:09:36 -04:00
}
2020-06-04 01:11:43 -04:00
inline void BeginDisabledControls ( bool cond ) {
if ( cond ) {
2020-04-27 11:27:59 -04:00
ImGui : : PushItemFlag ( ImGuiItemFlags_Disabled , true ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_Alpha , ImGui : : GetStyle ( ) . Alpha * 0.25f ) ;
}
2020-06-04 01:11:43 -04:00
}
2020-04-27 11:27:59 -04:00
2020-06-04 01:11:43 -04:00
inline void EndDisabledControls ( bool cond ) {
if ( cond ) {
2020-04-27 11:27:59 -04:00
ImGui : : PopItemFlag ( ) ;
ImGui : : PopStyleVar ( ) ;
}
2020-06-04 01:11:43 -04:00
}
2020-11-15 22:47:06 -05:00
void ShowAxisContextMenu ( ImPlotAxis & axis , ImPlotAxis * equal_axis , bool time_allowed ) {
2020-06-04 01:11:43 -04:00
2020-09-09 00:47:02 -04:00
ImGui : : PushItemWidth ( 75 ) ;
2020-11-15 22:47:06 -05:00
bool always_locked = axis . IsAlwaysLocked ( ) ;
bool grid = ! ImHasFlag ( axis . Flags , ImPlotAxisFlags_NoGridLines ) ;
bool ticks = ! ImHasFlag ( axis . Flags , ImPlotAxisFlags_NoTickMarks ) ;
bool labels = ! ImHasFlag ( axis . Flags , ImPlotAxisFlags_NoTickLabels ) ;
double drag_speed = ( axis . Range . Size ( ) < = DBL_EPSILON ) ? DBL_EPSILON * 1.0e+13 : 0.01 * axis . Range . Size ( ) ; // recover from almost equal axis limits.
if ( axis . IsTime ( ) ) {
2020-09-09 00:47:02 -04:00
ImPlotTime tmin = ImPlotTime : : FromDouble ( axis . Range . Min ) ;
ImPlotTime tmax = ImPlotTime : : FromDouble ( axis . Range . Max ) ;
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( always_locked ) ;
ImGui : : CheckboxFlags ( " ##LockMin " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_LockMin ) ;
EndDisabledControls ( always_locked ) ;
2020-09-09 00:47:02 -04:00
ImGui : : SameLine ( ) ;
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( axis . IsLockedMin ( ) ) ;
2020-09-09 00:47:02 -04:00
if ( ImGui : : BeginMenu ( " Min Time " ) ) {
2020-10-14 23:07:27 -04:00
if ( ShowTimePicker ( " mintime " , & tmin ) ) {
2020-09-09 10:00:50 -04:00
if ( tmin > = tmax )
tmax = AddTime ( tmin , ImPlotTimeUnit_S , 1 ) ;
axis . SetRange ( tmin . ToDouble ( ) , tmax . ToDouble ( ) ) ;
}
ImGui : : Separator ( ) ;
if ( ShowDatePicker ( " mindate " , & axis . PickerLevel , & axis . PickerTimeMin , & tmin , & tmax ) ) {
tmin = CombineDateTime ( axis . PickerTimeMin , tmin ) ;
if ( tmin > = tmax )
tmax = AddTime ( tmin , ImPlotTimeUnit_S , 1 ) ;
axis . SetRange ( tmin . ToDouble ( ) , tmax . ToDouble ( ) ) ;
2020-09-09 00:47:02 -04:00
}
ImGui : : EndMenu ( ) ;
}
2020-11-15 22:47:06 -05:00
EndDisabledControls ( axis . IsLockedMin ( ) ) ;
2020-09-09 00:47:02 -04:00
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( always_locked ) ;
ImGui : : CheckboxFlags ( " ##LockMax " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_LockMax ) ;
EndDisabledControls ( always_locked ) ;
2020-09-09 00:47:02 -04:00
ImGui : : SameLine ( ) ;
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( axis . IsLockedMax ( ) ) ;
2020-09-09 00:47:02 -04:00
if ( ImGui : : BeginMenu ( " Max Time " ) ) {
2020-10-14 23:07:27 -04:00
if ( ShowTimePicker ( " maxtime " , & tmax ) ) {
2020-09-09 10:00:50 -04:00
if ( tmax < = tmin )
tmin = AddTime ( tmax , ImPlotTimeUnit_S , - 1 ) ;
axis . SetRange ( tmin . ToDouble ( ) , tmax . ToDouble ( ) ) ;
}
ImGui : : Separator ( ) ;
if ( ShowDatePicker ( " maxdate " , & axis . PickerLevel , & axis . PickerTimeMax , & tmin , & tmax ) ) {
tmax = CombineDateTime ( axis . PickerTimeMax , tmax ) ;
if ( tmax < = tmin )
tmin = AddTime ( tmax , ImPlotTimeUnit_S , - 1 ) ;
axis . SetRange ( tmin . ToDouble ( ) , tmax . ToDouble ( ) ) ;
2020-09-09 00:47:02 -04:00
}
ImGui : : EndMenu ( ) ;
}
2020-11-15 22:47:06 -05:00
EndDisabledControls ( axis . IsLockedMax ( ) ) ;
2020-09-09 00:47:02 -04:00
}
else {
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( always_locked ) ;
ImGui : : CheckboxFlags ( " ##LockMin " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_LockMin ) ;
EndDisabledControls ( always_locked ) ;
2020-09-09 00:47:02 -04:00
ImGui : : SameLine ( ) ;
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( axis . IsLockedMin ( ) ) ;
2020-09-09 00:47:02 -04:00
double temp_min = axis . Range . Min ;
2020-11-15 22:47:06 -05:00
if ( DragFloat ( " Min " , & temp_min , ( float ) drag_speed , - HUGE_VAL , axis . Range . Max - DBL_EPSILON ) ) {
2020-09-09 00:47:02 -04:00
axis . SetMin ( temp_min ) ;
2020-11-15 22:47:06 -05:00
if ( equal_axis ! = NULL )
equal_axis - > SetAspect ( axis . GetAspect ( ) ) ;
}
EndDisabledControls ( axis . IsLockedMin ( ) ) ;
2020-09-09 00:47:02 -04:00
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( always_locked ) ;
ImGui : : CheckboxFlags ( " ##LockMax " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_LockMax ) ;
EndDisabledControls ( always_locked ) ;
2020-09-09 00:47:02 -04:00
ImGui : : SameLine ( ) ;
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( axis . IsLockedMax ( ) ) ;
2020-09-09 00:47:02 -04:00
double temp_max = axis . Range . Max ;
2020-11-15 22:47:06 -05:00
if ( DragFloat ( " Max " , & temp_max , ( float ) drag_speed , axis . Range . Min + DBL_EPSILON , HUGE_VAL ) ) {
2020-09-09 00:47:02 -04:00
axis . SetMax ( temp_max ) ;
2020-11-15 22:47:06 -05:00
if ( equal_axis ! = NULL )
equal_axis - > SetAspect ( axis . GetAspect ( ) ) ;
}
EndDisabledControls ( axis . IsLockedMax ( ) ) ;
2020-09-09 00:47:02 -04:00
}
2020-06-04 01:11:43 -04:00
2020-04-27 11:27:59 -04:00
ImGui : : Separator ( ) ;
2020-06-04 01:11:43 -04:00
2020-11-15 22:47:06 -05:00
ImGui : : CheckboxFlags ( " Invert " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_Invert ) ;
BeginDisabledControls ( axis . IsTime ( ) & & time_allowed ) ;
ImGui : : CheckboxFlags ( " Log Scale " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_LogScale ) ;
EndDisabledControls ( axis . IsTime ( ) & & time_allowed ) ;
2020-09-06 01:06:51 -04:00
if ( time_allowed ) {
2020-11-15 22:47:06 -05:00
BeginDisabledControls ( axis . IsLog ( ) ) ;
ImGui : : CheckboxFlags ( " Time " , ( unsigned int * ) & axis . Flags , ImPlotAxisFlags_Time ) ;
EndDisabledControls ( axis . IsLog ( ) ) ;
2020-09-06 01:06:51 -04:00
}
2020-04-27 11:27:59 -04:00
ImGui : : Separator ( ) ;
if ( ImGui : : Checkbox ( " Grid Lines " , & grid ) )
2020-09-09 00:47:02 -04:00
ImFlipFlag ( axis . Flags , ImPlotAxisFlags_NoGridLines ) ;
2020-04-27 11:27:59 -04:00
if ( ImGui : : Checkbox ( " Tick Marks " , & ticks ) )
2020-09-09 00:47:02 -04:00
ImFlipFlag ( axis . Flags , ImPlotAxisFlags_NoTickMarks ) ;
2020-04-27 11:27:59 -04:00
if ( ImGui : : Checkbox ( " Labels " , & labels ) )
2020-09-09 00:47:02 -04:00
ImFlipFlag ( axis . Flags , ImPlotAxisFlags_NoTickLabels ) ;
2020-04-27 11:27:59 -04:00
}
2020-09-21 08:09:14 -04:00
void ShowPlotContextMenu ( ImPlotPlot & plot ) {
2020-11-15 22:47:06 -05:00
const bool equal = ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ;
2020-05-13 10:11:25 -04:00
if ( ImGui : : BeginMenu ( " X-Axis " ) ) {
ImGui : : PushID ( " X " ) ;
2020-11-15 22:47:06 -05:00
ShowAxisContextMenu ( plot . XAxis , equal ? & plot . YAxis [ 0 ] : NULL , true ) ;
2020-05-12 05:19:04 -04:00
ImGui : : PopID ( ) ;
2020-04-27 11:27:59 -04:00
ImGui : : EndMenu ( ) ;
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-08-16 16:38:51 -04:00
if ( i = = 1 & & ! ImHasFlag ( plot . Flags , ImPlotFlags_YAxis2 ) ) {
2020-05-11 07:12:22 -04:00
continue ;
}
2020-08-16 16:38:51 -04:00
if ( i = = 2 & & ! ImHasFlag ( plot . Flags , ImPlotFlags_YAxis3 ) ) {
2020-05-11 07:12:22 -04:00
continue ;
}
char buf [ 10 ] = { } ;
if ( i = = 0 ) {
snprintf ( buf , sizeof ( buf ) - 1 , " Y-Axis " ) ;
} else {
snprintf ( buf , sizeof ( buf ) - 1 , " Y-Axis %d " , i + 1 ) ;
}
if ( ImGui : : BeginMenu ( buf ) ) {
2020-05-13 10:11:25 -04:00
ImGui : : PushID ( i ) ;
2020-11-15 22:47:06 -05:00
ShowAxisContextMenu ( plot . YAxis [ i ] , ( equal & & i = = 0 ) ? & plot . XAxis : NULL , false ) ;
2020-05-12 05:19:04 -04:00
ImGui : : PopID ( ) ;
2020-05-11 07:12:22 -04:00
ImGui : : EndMenu ( ) ;
}
2020-04-27 11:27:59 -04:00
}
2020-05-13 10:11:25 -04:00
2020-04-27 11:27:59 -04:00
ImGui : : Separator ( ) ;
if ( ( ImGui : : BeginMenu ( " Settings " ) ) ) {
2020-11-15 22:47:06 -05:00
if ( ImGui : : MenuItem ( " Anti-Aliased Lines " , NULL , ImHasFlag ( plot . Flags , ImPlotFlags_AntiAliased ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_AntiAliased ) ;
if ( ImGui : : MenuItem ( " Equal " , NULL , ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_Equal ) ;
if ( ImGui : : MenuItem ( " Box Select " , NULL , ! ImHasFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) ;
if ( ImGui : : MenuItem ( " Query " , NULL , ImHasFlag ( plot . Flags , ImPlotFlags_Query ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_Query ) ;
2020-11-10 09:27:28 -05:00
if ( ImGui : : MenuItem ( " Title " , NULL , ! ImHasFlag ( plot . Flags , ImPlotFlags_NoTitle ) ) )
2020-11-15 22:47:06 -05:00
ImFlipFlag ( plot . Flags , ImPlotFlags_NoTitle ) ;
if ( ImGui : : MenuItem ( " Mouse Position " , NULL , ! ImHasFlag ( plot . Flags , ImPlotFlags_NoMousePos ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_NoMousePos ) ;
if ( ImGui : : MenuItem ( " Crosshairs " , NULL , ImHasFlag ( plot . Flags , ImPlotFlags_Crosshairs ) ) )
ImFlipFlag ( plot . Flags , ImPlotFlags_Crosshairs ) ;
2020-11-10 09:27:28 -05:00
if ( ( ImGui : : BeginMenu ( " Legend " ) ) ) {
const float s = ImGui : : GetFrameHeight ( ) ;
if ( ImGui : : RadioButton ( " H " , plot . LegendOrientation = = ImPlotOrientation_Horizontal ) )
plot . LegendOrientation = ImPlotOrientation_Horizontal ;
ImGui : : SameLine ( ) ;
if ( ImGui : : RadioButton ( " V " , plot . LegendOrientation = = ImPlotOrientation_Vertical ) )
plot . LegendOrientation = ImPlotOrientation_Vertical ;
ImGui : : Checkbox ( " Outside " , & plot . LegendOutside ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_ItemSpacing , ImVec2 ( 1 , 1 ) ) ;
2020-12-04 07:46:34 -05:00
if ( ImGui : : Button ( " ##NW " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_NorthWest ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##N " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_North ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##NE " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_NorthEast ; }
if ( ImGui : : Button ( " ##W " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_West ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##C " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_Center ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##E " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_East ; }
if ( ImGui : : Button ( " ##SW " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_SouthWest ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##S " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_South ; } ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " ##SE " , ImVec2 ( 1.5f * s , s ) ) ) { plot . LegendLocation = ImPlotLocation_SouthEast ; }
2020-11-10 09:27:28 -05:00
ImGui : : PopStyleVar ( ) ;
ImGui : : EndMenu ( ) ;
2020-04-27 11:27:59 -04:00
}
ImGui : : EndMenu ( ) ;
}
2020-09-06 01:42:03 -04:00
if ( ImGui : : MenuItem ( " Legend " , NULL , ! ImHasFlag ( plot . Flags , ImPlotFlags_NoLegend ) ) ) {
ImFlipFlag ( plot . Flags , ImPlotFlags_NoLegend ) ;
2020-04-27 11:27:59 -04:00
}
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// EndPlot()
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
void EndPlot ( ) {
2020-08-16 16:38:51 -04:00
IM_ASSERT_USER_ERROR ( GImPlot ! = NULL , " No current context. Did you call ImPlot::CreateContext() or ImPlot::SetCurrentContext()? " ) ;
2020-08-19 01:04:05 -04:00
ImPlotContext & gp = * GImPlot ;
2020-05-13 10:11:25 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " Mismatched BeginPlot()/EndPlot()! " ) ;
2020-08-19 01:04:05 -04:00
ImGuiContext & G = * GImGui ;
2020-09-21 08:09:14 -04:00
ImPlotPlot & plot = * gp . CurrentPlot ;
2020-08-19 01:04:05 -04:00
ImGuiWindow * Window = G . CurrentWindow ;
2020-04-27 11:27:59 -04:00
ImDrawList & DrawList = * Window - > DrawList ;
2020-08-19 01:04:05 -04:00
const ImGuiIO & IO = ImGui : : GetIO ( ) ;
2020-04-27 11:27:59 -04:00
// AXIS STATES ------------------------------------------------------------
2020-11-15 22:47:06 -05:00
const bool any_y_locked = plot . YAxis [ 0 ] . IsLocked ( ) | | plot . YAxis [ 1 ] . Present ? plot . YAxis [ 1 ] . IsLocked ( ) : false | | plot . YAxis [ 2 ] . Present ? plot . YAxis [ 2 ] . IsLocked ( ) : false ;
2020-05-11 07:12:22 -04:00
const bool any_y_dragging = plot . YAxis [ 0 ] . Dragging | | plot . YAxis [ 1 ] . Dragging | | plot . YAxis [ 2 ] . Dragging ;
2020-04-27 11:27:59 -04:00
2020-06-03 15:37:01 -04:00
2020-04-27 11:27:59 -04:00
// FINAL RENDER -----------------------------------------------------------
2020-04-28 21:17:26 -04:00
// render ticks
2020-05-11 07:12:22 -04:00
PushPlotClipRect ( ) ;
2020-09-06 02:32:15 -04:00
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_NoTickMarks ) ) {
2020-05-13 00:45:34 -04:00
for ( int t = 0 ; t < gp . XTicks . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick * xt = & gp . XTicks . Ticks [ t ] ;
2020-09-06 22:08:25 -04:00
if ( xt - > Level = = 0 )
2020-11-15 22:47:06 -05:00
DrawList . AddLine ( ImVec2 ( xt - > PixelPos , plot . PlotRect . Max . y ) ,
ImVec2 ( xt - > PixelPos , plot . PlotRect . Max . y - ( xt - > Major ? gp . Style . MajorTickLen . x : gp . Style . MinorTickLen . x ) ) ,
plot . XAxis . ColorMaj ,
2020-09-06 01:06:51 -04:00
xt - > Major ? gp . Style . MajorTickSize . x : gp . Style . MinorTickSize . x ) ;
2020-05-13 00:45:34 -04:00
}
2020-04-28 21:17:26 -04:00
}
2020-05-11 07:12:22 -04:00
PopPlotClipRect ( ) ;
2020-11-15 22:47:06 -05:00
ImGui : : PushClipRect ( plot . PlotRect . Min , ImVec2 ( plot . FrameRect . Max . x , plot . PlotRect . Max . y ) , true ) ;
2020-05-11 07:12:22 -04:00
int axis_count = 0 ;
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( ! plot . YAxis [ i ] . Present ) { continue ; }
2020-05-11 07:12:22 -04:00
axis_count + + ;
2020-08-19 12:34:52 -04:00
float x_start = gp . YAxisReference [ i ] ;
2020-09-06 02:32:15 -04:00
if ( ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_NoTickMarks ) ) {
2020-07-26 22:28:22 -04:00
float direction = ( i = = 0 ) ? 1.0f : - 1.0f ;
bool no_major = axis_count > = 3 ;
for ( int t = 0 ; t < gp . YTicks [ i ] . Size ; t + + ) {
2020-08-25 22:59:43 -04:00
ImPlotTick * yt = & gp . YTicks [ i ] . Ticks [ t ] ;
2020-07-26 22:28:22 -04:00
ImVec2 start = ImVec2 ( x_start , yt - > PixelPos ) ;
2020-08-24 00:45:42 -04:00
DrawList . AddLine ( start ,
start + ImVec2 ( direction * ( ( ! no_major & & yt - > Major ) ? gp . Style . MajorTickLen . y : gp . Style . MinorTickLen . y ) , 0 ) ,
2020-11-15 22:47:06 -05:00
plot . YAxis [ i ] . ColorMaj ,
2020-08-24 00:45:42 -04:00
( ! no_major & & yt - > Major ) ? gp . Style . MajorTickSize . y : gp . Style . MinorTickSize . y ) ;
2020-07-26 22:28:22 -04:00
}
2020-05-11 07:12:22 -04:00
}
if ( axis_count > = 3 ) {
// Draw a bar next to the ticks to act as a visual separator.
DrawList . AddLine (
2020-11-15 22:47:06 -05:00
ImVec2 ( x_start , plot . PlotRect . Min . y ) ,
ImVec2 ( x_start , plot . PlotRect . Max . y ) ,
2020-08-24 14:59:57 -04:00
GetStyleColorU32 ( ImPlotCol_YAxisGrid3 ) , 1 ) ;
2020-05-11 07:12:22 -04:00
}
2020-04-28 21:17:26 -04:00
}
2020-05-12 05:19:04 -04:00
ImGui : : PopClipRect ( ) ;
2020-05-11 07:12:22 -04:00
2020-09-19 21:54:19 -04:00
// render annotations
PushPlotClipRect ( ) ;
for ( int i = 0 ; i < gp . Annotations . Size ; + + i ) {
const char * txt = gp . Annotations . GetText ( i ) ;
ImPlotAnnotation & an = gp . Annotations . Annotations [ i ] ;
const ImVec2 txt_size = ImGui : : CalcTextSize ( txt ) ;
const ImVec2 size = txt_size + gp . Style . AnnotationPadding * 2 ;
ImVec2 pos = an . Pos ;
if ( an . Offset . x = = 0 )
pos . x - = size . x / 2 ;
else if ( an . Offset . x > 0 )
pos . x + = an . Offset . x ;
else
pos . x - = size . x - an . Offset . x ;
if ( an . Offset . y = = 0 )
pos . y - = size . y / 2 ;
else if ( an . Offset . y > 0 )
pos . y + = an . Offset . y ;
else
pos . y - = size . y - an . Offset . y ;
if ( an . Clamp )
2020-11-15 22:47:06 -05:00
pos = ClampLabelPos ( pos , size , plot . PlotRect . Min , plot . PlotRect . Max ) ;
2020-09-19 21:54:19 -04:00
ImRect rect ( pos , pos + size ) ;
if ( an . Offset . x ! = 0 | | an . Offset . y ! = 0 ) {
ImVec2 corners [ 4 ] = { rect . GetTL ( ) , rect . GetTR ( ) , rect . GetBR ( ) , rect . GetBL ( ) } ;
int min_corner = 0 ;
float min_len = FLT_MAX ;
for ( int c = 0 ; c < 4 ; + + c ) {
float len = ImLengthSqr ( an . Pos - corners [ c ] ) ;
if ( len < min_len ) {
min_corner = c ;
min_len = len ;
}
}
DrawList . AddLine ( an . Pos , corners [ min_corner ] , an . ColorBg ) ;
}
DrawList . AddRectFilled ( rect . Min , rect . Max , an . ColorBg ) ;
DrawList . AddText ( pos + gp . Style . AnnotationPadding , an . ColorFg , txt ) ;
}
PopPlotClipRect ( ) ;
2020-08-16 16:38:51 -04:00
// render y-axis drag/drop hover
2020-11-15 22:47:06 -05:00
if ( ( plot . YAxis [ 1 ] . Present | | plot . YAxis [ 2 ] . Present ) & & ImGui : : IsDragDropPayloadBeingAccepted ( ) ) {
2020-08-19 12:34:52 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
2020-11-15 22:47:06 -05:00
if ( plot . YAxis [ i ] . ExtHovered ) {
2020-08-19 12:34:52 -04:00
float x_loc = gp . YAxisReference [ i ] ;
2020-11-15 22:47:06 -05:00
ImVec2 p1 ( x_loc - 5 , plot . PlotRect . Min . y - 5 ) ;
ImVec2 p2 ( x_loc + 5 , plot . PlotRect . Max . y + 5 ) ;
2020-08-19 12:34:52 -04:00
DrawList . AddRect ( p1 , p2 , ImGui : : GetColorU32 ( ImGuiCol_DragDropTarget ) , 0.0f , ImDrawCornerFlags_All , 2.0f ) ;
2020-07-26 22:28:22 -04:00
}
2020-08-23 00:26:49 -04:00
}
2020-07-26 22:28:22 -04:00
}
2020-05-11 07:12:22 -04:00
PushPlotClipRect ( ) ;
2020-04-28 21:17:26 -04:00
// render selection/query
if ( plot . Selecting ) {
2020-08-29 09:31:34 -04:00
const ImRect select_bb ( ImMin ( IO . MousePos , plot . SelectStart ) , ImMax ( IO . MousePos , plot . SelectStart ) ) ;
const bool wide_enough = ImFabs ( select_bb . GetWidth ( ) ) > 2 ;
const bool tall_enough = ImFabs ( select_bb . GetHeight ( ) ) > 2 ;
const bool big_enough = wide_enough & & tall_enough ;
2020-11-15 22:47:06 -05:00
if ( plot . Selecting & & ! plot . IsLocked ( ) & & ! ImHasFlag ( plot . Flags , ImPlotFlags_NoBoxSelect ) ) {
2020-08-24 00:45:42 -04:00
const ImVec4 col = GetStyleColorVec4 ( ImPlotCol_Selection ) ;
const ImU32 col_bg = ImGui : : GetColorU32 ( col * ImVec4 ( 1 , 1 , 1 , 0.25f ) ) ;
const ImU32 col_bd = ImGui : : GetColorU32 ( col ) ;
2020-08-28 18:11:36 -04:00
if ( IO . KeyMods = = ( gp . InputMap . HorizontalMod | gp . InputMap . VerticalMod ) & & big_enough ) {
2020-11-15 22:47:06 -05:00
DrawList . AddRectFilled ( plot . PlotRect . Min , plot . PlotRect . Max , col_bg ) ;
DrawList . AddRect ( plot . PlotRect . Min , plot . PlotRect . Max , col_bd ) ;
2020-04-28 21:17:26 -04:00
}
2020-11-15 22:47:06 -05:00
else if ( ( plot . XAxis . IsLocked ( ) | | IO . KeyMods = = gp . InputMap . HorizontalMod ) & & tall_enough ) {
DrawList . AddRectFilled ( ImVec2 ( plot . PlotRect . Min . x , select_bb . Min . y ) , ImVec2 ( plot . PlotRect . Max . x , select_bb . Max . y ) , col_bg ) ;
DrawList . AddRect ( ImVec2 ( plot . PlotRect . Min . x , select_bb . Min . y ) , ImVec2 ( plot . PlotRect . Max . x , select_bb . Max . y ) , col_bd ) ;
2020-04-28 21:17:26 -04:00
}
2020-08-29 09:31:34 -04:00
else if ( ( any_y_locked | | IO . KeyMods = = gp . InputMap . VerticalMod ) & & wide_enough ) {
2020-11-15 22:47:06 -05:00
DrawList . AddRectFilled ( ImVec2 ( select_bb . Min . x , plot . PlotRect . Min . y ) , ImVec2 ( select_bb . Max . x , plot . PlotRect . Max . y ) , col_bg ) ;
DrawList . AddRect ( ImVec2 ( select_bb . Min . x , plot . PlotRect . Min . y ) , ImVec2 ( select_bb . Max . x , plot . PlotRect . Max . y ) , col_bd ) ;
2020-04-28 21:17:26 -04:00
}
2020-08-28 18:11:36 -04:00
else if ( big_enough ) {
2020-08-24 00:45:42 -04:00
DrawList . AddRectFilled ( select_bb . Min , select_bb . Max , col_bg ) ;
DrawList . AddRect ( select_bb . Min , select_bb . Max , col_bd ) ;
2020-04-28 21:17:26 -04:00
}
}
}
2020-08-19 08:31:26 -04:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_Query ) ) // draw query rect only when query enabled.
{
2020-08-24 00:45:42 -04:00
const ImVec4 col = GetStyleColorVec4 ( ImPlotCol_Query ) ;
const ImU32 col_bd = ImGui : : GetColorU32 ( col * ImVec4 ( 1 , 1 , 1 , 0.25f ) ) ;
const ImU32 col_bg = ImGui : : GetColorU32 ( col ) ;
2020-08-19 08:31:26 -04:00
if ( plot . Querying | | plot . Queried ) {
if ( plot . QueryRect . GetWidth ( ) > 2 & & plot . QueryRect . GetHeight ( ) > 2 ) {
2020-11-15 22:47:06 -05:00
DrawList . AddRectFilled ( plot . QueryRect . Min + plot . PlotRect . Min , plot . QueryRect . Max + plot . PlotRect . Min , col_bd ) ;
DrawList . AddRect ( plot . QueryRect . Min + plot . PlotRect . Min , plot . QueryRect . Max + plot . PlotRect . Min , col_bg ) ;
2020-08-19 08:31:26 -04:00
}
2020-05-13 10:11:25 -04:00
}
2020-08-19 08:31:26 -04:00
else if ( plot . Queried ) {
ImRect bb_query = plot . QueryRect ;
2020-11-15 22:47:06 -05:00
bb_query . Min + = plot . PlotRect . Min ;
bb_query . Max + = plot . PlotRect . Min ;
2020-08-24 00:45:42 -04:00
DrawList . AddRectFilled ( bb_query . Min , bb_query . Max , col_bd ) ;
DrawList . AddRect ( bb_query . Min , bb_query . Max , col_bg ) ;
2020-08-19 08:31:26 -04:00
}
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
// render crosshairs
2020-11-15 22:47:06 -05:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_Crosshairs ) & & plot . PlotHovered & & plot . FrameHovered & &
2020-10-19 00:26:34 -04:00
! ( plot . XAxis . Dragging | | any_y_dragging ) & & ! plot . Selecting & & ! plot . Querying & & ! plot . LegendHovered ) {
2020-04-27 11:27:59 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_None ) ;
ImVec2 xy = IO . MousePos ;
2020-11-15 22:47:06 -05:00
ImVec2 h1 ( plot . PlotRect . Min . x , xy . y ) ;
2020-04-27 11:27:59 -04:00
ImVec2 h2 ( xy . x - 5 , xy . y ) ;
ImVec2 h3 ( xy . x + 5 , xy . y ) ;
2020-11-15 22:47:06 -05:00
ImVec2 h4 ( plot . PlotRect . Max . x , xy . y ) ;
ImVec2 v1 ( xy . x , plot . PlotRect . Min . y ) ;
2020-04-27 11:27:59 -04:00
ImVec2 v2 ( xy . x , xy . y - 5 ) ;
ImVec2 v3 ( xy . x , xy . y + 5 ) ;
2020-11-15 22:47:06 -05:00
ImVec2 v4 ( xy . x , plot . PlotRect . Max . y ) ;
2020-08-24 00:45:42 -04:00
ImU32 col = GetStyleColorU32 ( ImPlotCol_Crosshairs ) ;
DrawList . AddLine ( h1 , h2 , col ) ;
DrawList . AddLine ( h3 , h4 , col ) ;
DrawList . AddLine ( v1 , v2 , col ) ;
DrawList . AddLine ( v3 , v4 , col ) ;
2020-04-27 11:27:59 -04:00
}
2020-09-21 08:09:14 -04:00
// render mouse pos (TODO: use LabelAxisValue)
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoMousePos ) & & plot . PlotHovered ) {
2020-05-10 22:43:12 -04:00
char buffer [ 128 ] = { } ;
2020-08-16 16:38:51 -04:00
ImBufferWriter writer ( buffer , sizeof ( buffer ) ) ;
2020-05-11 07:12:22 -04:00
2020-08-22 23:55:37 -04:00
// x
2020-08-23 00:26:49 -04:00
if ( ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-08-22 23:55:37 -04:00
writer . Write ( " %.3E " , gp . MousePos [ 0 ] . x ) ;
}
2020-09-06 01:06:51 -04:00
else if ( ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_Time ) ) {
2020-11-15 22:47:06 -05:00
ImPlotTimeUnit unit = GetUnitForRange ( plot . XAxis . Range . Size ( ) / ( plot . PlotRect . GetWidth ( ) / 100 ) ) ;
2020-10-14 23:07:27 -04:00
const int written = FormatDateTime ( ImPlotTime : : FromDouble ( gp . MousePos [ 0 ] . x ) , & writer . Buffer [ writer . Pos ] , writer . Size - writer . Pos - 1 , GetDateTimeFmt ( TimeFormatMouseCursor , unit ) ) ;
2020-09-06 01:06:51 -04:00
if ( written > 0 )
writer . Pos + = ImMin ( written , writer . Size - writer . Pos - 1 ) ;
}
2020-08-22 23:55:37 -04:00
else {
2020-08-25 22:59:43 -04:00
double range_x = gp . XTicks . Size > 1 ? ( gp . XTicks . Ticks [ 1 ] . PlotPos - gp . XTicks . Ticks [ 0 ] . PlotPos ) : plot . XAxis . Range . Size ( ) ;
2020-08-22 23:55:37 -04:00
writer . Write ( " %.*f " , Precision ( range_x ) , gp . MousePos [ 0 ] . x ) ;
}
// y1
2020-08-23 00:26:49 -04:00
if ( ImHasFlag ( plot . YAxis [ 0 ] . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-08-22 23:55:37 -04:00
writer . Write ( " ,%.3E " , gp . MousePos [ 0 ] . y ) ;
}
else {
2020-08-25 22:59:43 -04:00
double range_y = gp . YTicks [ 0 ] . Size > 1 ? ( gp . YTicks [ 0 ] . Ticks [ 1 ] . PlotPos - gp . YTicks [ 0 ] . Ticks [ 0 ] . PlotPos ) : plot . YAxis [ 0 ] . Range . Size ( ) ;
2020-08-22 23:55:37 -04:00
writer . Write ( " ,%.*f " , Precision ( range_y ) , gp . MousePos [ 0 ] . y ) ;
}
// y2
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_YAxis2 ) ) {
2020-08-23 00:26:49 -04:00
if ( ImHasFlag ( plot . YAxis [ 1 ] . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-08-22 23:55:37 -04:00
writer . Write ( " ,(%.3E) " , gp . MousePos [ 1 ] . y ) ;
}
else {
2020-08-25 22:59:43 -04:00
double range_y = gp . YTicks [ 1 ] . Size > 1 ? ( gp . YTicks [ 1 ] . Ticks [ 1 ] . PlotPos - gp . YTicks [ 1 ] . Ticks [ 0 ] . PlotPos ) : plot . YAxis [ 1 ] . Range . Size ( ) ;
2020-08-22 23:55:37 -04:00
writer . Write ( " ,(%.*f) " , Precision ( range_y ) , gp . MousePos [ 1 ] . y ) ;
}
2020-05-11 07:12:22 -04:00
}
2020-08-22 23:55:37 -04:00
// y3
2020-08-16 16:38:51 -04:00
if ( ImHasFlag ( plot . Flags , ImPlotFlags_YAxis3 ) ) {
2020-08-23 00:26:49 -04:00
if ( ImHasFlag ( plot . YAxis [ 2 ] . Flags , ImPlotAxisFlags_LogScale ) ) {
2020-08-22 23:55:37 -04:00
writer . Write ( " ,(%.3E) " , gp . MousePos [ 2 ] . y ) ;
}
else {
2020-08-25 22:59:43 -04:00
double range_y = gp . YTicks [ 2 ] . Size > 1 ? ( gp . YTicks [ 2 ] . Ticks [ 1 ] . PlotPos - gp . YTicks [ 2 ] . Ticks [ 0 ] . PlotPos ) : plot . YAxis [ 2 ] . Range . Size ( ) ;
2020-08-22 23:55:37 -04:00
writer . Write ( " ,(%.*f) " , Precision ( range_y ) , gp . MousePos [ 2 ] . y ) ;
}
2020-05-11 07:12:22 -04:00
}
2020-10-19 00:26:34 -04:00
const ImVec2 size = ImGui : : CalcTextSize ( buffer ) ;
2020-11-15 22:47:06 -05:00
const ImVec2 pos = GetLocationPos ( plot . PlotRect , size , plot . MousePosLocation , gp . Style . MousePosPadding ) ;
2020-08-24 00:45:42 -04:00
DrawList . AddText ( pos , GetStyleColorU32 ( ImPlotCol_InlayText ) , buffer ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect ( ) ;
2020-04-27 11:27:59 -04:00
2020-10-19 00:26:34 -04:00
// reset legend hovers
plot . LegendHovered = false ;
for ( int i = 0 ; i < plot . Items . GetSize ( ) ; + + i )
plot . Items . GetByIndex ( i ) - > LegendHovered = false ;
// render legend
2020-10-21 11:08:41 -04:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoLegend ) & & plot . GetLegendCount ( ) > 0 ) {
const ImVec2 legend_size = CalcLegendSize ( plot , gp . Style . LegendInnerPadding , gp . Style . LegendSpacing , plot . LegendOrientation ) ;
2020-11-15 22:47:06 -05:00
const ImVec2 legend_pos = GetLocationPos ( plot . LegendOutside ? plot . FrameRect : plot . PlotRect ,
2020-10-19 00:26:34 -04:00
legend_size ,
plot . LegendLocation ,
plot . LegendOutside ? gp . Style . PlotPadding : gp . Style . LegendPadding ) ;
const ImRect legend_bb ( legend_pos , legend_pos + legend_size ) ;
// test hover
2020-11-15 22:47:06 -05:00
plot . LegendHovered = plot . FrameHovered & & legend_bb . Contains ( IO . MousePos ) ;
2020-10-19 00:26:34 -04:00
if ( plot . LegendOutside )
2020-11-15 22:47:06 -05:00
ImGui : : PushClipRect ( plot . FrameRect . Min , plot . FrameRect . Max , true ) ;
2020-10-19 00:26:34 -04:00
else
PushPlotClipRect ( ) ;
ImU32 col_bg = GetStyleColorU32 ( ImPlotCol_LegendBg ) ;
ImU32 col_bd = GetStyleColorU32 ( ImPlotCol_LegendBorder ) ;
DrawList . AddRectFilled ( legend_bb . Min , legend_bb . Max , col_bg ) ;
DrawList . AddRect ( legend_bb . Min , legend_bb . Max , col_bd ) ;
2020-10-21 11:08:41 -04:00
ShowLegendEntries ( plot , legend_bb , plot . LegendHovered , gp . Style . LegendInnerPadding , gp . Style . LegendSpacing , plot . LegendOrientation , DrawList ) ;
2020-10-19 00:26:34 -04:00
ImGui : : PopClipRect ( ) ;
}
2020-11-15 22:47:06 -05:00
if ( plot . LegendFlipSideNextFrame ) {
2020-10-19 00:26:34 -04:00
plot . LegendOutside = ! plot . LegendOutside ;
2020-11-15 22:47:06 -05:00
plot . LegendFlipSideNextFrame = false ;
2020-10-19 00:26:34 -04:00
}
2020-04-27 11:27:59 -04:00
// render border
2020-08-24 00:45:42 -04:00
if ( gp . Style . PlotBorderSize > 0 )
2020-11-15 22:47:06 -05:00
DrawList . AddRect ( plot . PlotRect . Min , plot . PlotRect . Max , GetStyleColorU32 ( ImPlotCol_PlotBorder ) , 0 , ImDrawCornerFlags_All , gp . Style . PlotBorderSize ) ;
2020-04-27 11:27:59 -04:00
// FIT DATA --------------------------------------------------------------
2020-11-15 22:47:06 -05:00
const bool axis_equal = ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ;
2020-04-28 21:17:26 -04:00
if ( gp . FitThisFrame & & ( gp . VisibleItemCount > 0 | | plot . Queried ) ) {
2020-11-15 22:47:06 -05:00
if ( gp . FitX ) {
2020-12-04 00:27:38 -05:00
const double ext_size = gp . ExtentsX . Size ( ) * 0.5 ;
gp . ExtentsX . Min - = ext_size * gp . Style . FitPadding . x ;
gp . ExtentsX . Max + = ext_size * gp . Style . FitPadding . x ;
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LockMin ) & & ! ImNanOrInf ( gp . ExtentsX . Min ) )
plot . XAxis . Range . Min = ( gp . ExtentsX . Min ) ;
if ( ! ImHasFlag ( plot . XAxis . Flags , ImPlotAxisFlags_LockMax ) & & ! ImNanOrInf ( gp . ExtentsX . Max ) )
plot . XAxis . Range . Max = ( gp . ExtentsX . Max ) ;
if ( ImAlmostEqual ( plot . XAxis . Range . Max , plot . XAxis . Range . Min ) ) {
2020-12-04 14:33:44 -05:00
plot . XAxis . Range . Max + = 0.5 ;
plot . XAxis . Range . Min - = 0.5 ;
2020-11-15 22:47:06 -05:00
}
plot . XAxis . Constrain ( ) ;
if ( axis_equal & & ! gp . FitY [ 0 ] )
plot . YAxis [ 0 ] . SetAspect ( plot . XAxis . GetAspect ( ) ) ;
2020-09-02 17:09:57 -04:00
}
2020-08-19 01:04:05 -04:00
for ( int i = 0 ; i < IMPLOT_Y_AXES ; i + + ) {
2020-11-15 22:47:06 -05:00
if ( gp . FitY [ i ] ) {
2020-12-04 00:27:38 -05:00
const double ext_size = gp . ExtentsY [ i ] . Size ( ) * 0.5 ;
gp . ExtentsY [ i ] . Min - = ext_size * gp . Style . FitPadding . y ;
gp . ExtentsY [ i ] . Max + = ext_size * gp . Style . FitPadding . y ;
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LockMin ) & & ! ImNanOrInf ( gp . ExtentsY [ i ] . Min ) )
plot . YAxis [ i ] . Range . Min = ( gp . ExtentsY [ i ] . Min ) ;
if ( ! ImHasFlag ( plot . YAxis [ i ] . Flags , ImPlotAxisFlags_LockMax ) & & ! ImNanOrInf ( gp . ExtentsY [ i ] . Max ) )
plot . YAxis [ i ] . Range . Max = ( gp . ExtentsY [ i ] . Max ) ;
if ( ImAlmostEqual ( plot . YAxis [ i ] . Range . Max , plot . YAxis [ i ] . Range . Min ) ) {
2020-12-04 14:33:44 -05:00
plot . YAxis [ i ] . Range . Max + = 0.5 ;
plot . YAxis [ i ] . Range . Min - = 0.5 ;
2020-11-15 22:47:06 -05:00
}
plot . YAxis [ i ] . Constrain ( ) ;
if ( i = = 0 & & axis_equal & & ! gp . FitX )
plot . XAxis . SetAspect ( plot . YAxis [ 0 ] . GetAspect ( ) ) ;
2020-09-02 17:02:51 -04:00
}
2020-11-15 22:47:06 -05:00
}
if ( axis_equal & & gp . FitX & & gp . FitY [ 0 ] ) {
double aspect = ImMax ( plot . XAxis . GetAspect ( ) , plot . YAxis [ 0 ] . GetAspect ( ) ) ;
plot . XAxis . SetAspect ( aspect ) ;
plot . YAxis [ 0 ] . SetAspect ( aspect ) ;
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
2020-08-19 02:18:05 -04:00
// CONTEXT MENUS -----------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoMenus ) & & plot . FrameHovered & & plot . PlotHovered & & IO . MouseDoubleClicked [ gp . InputMap . ContextMenuButton ] & & ! plot . LegendHovered )
2020-08-19 02:18:05 -04:00
ImGui : : OpenPopup ( " ##PlotContext " ) ;
if ( ImGui : : BeginPopup ( " ##PlotContext " ) ) {
ShowPlotContextMenu ( plot ) ;
2020-04-27 11:27:59 -04:00
ImGui : : EndPopup ( ) ;
}
2020-08-19 02:18:05 -04:00
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoMenus ) & & plot . FrameHovered & & plot . XAxis . ExtHovered & & IO . MouseDoubleClicked [ gp . InputMap . ContextMenuButton ] & & ! plot . LegendHovered )
2020-08-23 00:26:49 -04:00
ImGui : : OpenPopup ( " ##XContext " ) ;
2020-08-19 02:18:05 -04:00
if ( ImGui : : BeginPopup ( " ##XContext " ) ) {
ImGui : : Text ( " X-Axis " ) ; ImGui : : Separator ( ) ;
2020-11-15 22:47:06 -05:00
ShowAxisContextMenu ( plot . XAxis , ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ? & plot . YAxis [ 0 ] : NULL , true ) ;
2020-08-19 02:18:05 -04:00
ImGui : : EndPopup ( ) ;
}
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i ) {
ImGui : : PushID ( i ) ;
2020-11-15 22:47:06 -05:00
if ( ! ImHasFlag ( plot . Flags , ImPlotFlags_NoMenus ) & & plot . FrameHovered & & plot . YAxis [ i ] . ExtHovered & & IO . MouseDoubleClicked [ gp . InputMap . ContextMenuButton ] & & ! plot . LegendHovered )
2020-08-23 00:26:49 -04:00
ImGui : : OpenPopup ( " ##YContext " ) ;
2020-08-19 02:18:05 -04:00
if ( ImGui : : BeginPopup ( " ##YContext " ) ) {
if ( i = = 0 ) {
ImGui : : Text ( " Y-Axis " ) ; ImGui : : Separator ( ) ;
}
else {
ImGui : : Text ( " Y-Axis %d " , i + 1 ) ; ImGui : : Separator ( ) ;
}
2020-11-15 22:47:06 -05:00
ShowAxisContextMenu ( plot . YAxis [ i ] , ( i = = 0 & & ImHasFlag ( plot . Flags , ImPlotFlags_Equal ) ) ? & plot . XAxis : NULL , false ) ;
2020-08-19 02:18:05 -04:00
ImGui : : EndPopup ( ) ;
}
ImGui : : PopID ( ) ;
}
2020-09-03 10:19:34 -04:00
// LINKED AXES ------------------------------------------------------------
PushLinkedAxis ( plot . XAxis ) ;
for ( int i = 0 ; i < IMPLOT_Y_AXES ; + + i )
PushLinkedAxis ( plot . YAxis [ i ] ) ;
2020-04-27 11:27:59 -04:00
// CLEANUP ----------------------------------------------------------------
2020-06-12 23:02:43 -04:00
// reset the plot items for the next frame
for ( int i = 0 ; i < gp . CurrentPlot - > Items . GetSize ( ) ; + + i ) {
gp . CurrentPlot - > Items . GetByIndex ( i ) - > SeenThisFrame = false ;
}
2020-05-12 05:19:04 -04:00
// Pop ImGui::PushID at the end of BeginPlot
2020-05-13 10:11:25 -04:00
ImGui : : PopID ( ) ;
2020-06-03 15:37:01 -04:00
// Reset context for next plot
2020-08-16 16:38:51 -04:00
Reset ( GImPlot ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// MISC API
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-16 16:38:51 -04:00
ImPlotInputMap & GetInputMap ( ) {
return GImPlot - > InputMap ;
}
2020-05-29 13:39:30 -04:00
void SetNextPlotLimits ( double x_min , double x_max , double y_min , double y_max , ImGuiCond cond ) {
2020-08-16 19:48:09 -04:00
IM_ASSERT_USER_ERROR ( GImPlot - > CurrentPlot = = NULL , " SetNextPlotLimits() needs to be called before BeginPlot()! " ) ;
2020-05-11 07:12:22 -04:00
SetNextPlotLimitsX ( x_min , x_max , cond ) ;
SetNextPlotLimitsY ( y_min , y_max , cond ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-29 13:39:30 -04:00
void SetNextPlotLimitsX ( double x_min , double x_max , ImGuiCond cond ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " SetNextPlotLSetNextPlotLimitsXimitsY() needs to be called before BeginPlot()! " ) ;
2020-04-27 11:27:59 -04:00
IM_ASSERT ( cond = = 0 | | ImIsPowerOfTwo ( cond ) ) ; // Make sure the user doesn't attempt to combine multiple condition flags.
gp . NextPlotData . HasXRange = true ;
gp . NextPlotData . XRangeCond = cond ;
2020-05-11 07:12:22 -04:00
gp . NextPlotData . X . Min = x_min ;
gp . NextPlotData . X . Max = x_max ;
2020-04-27 11:27:59 -04:00
}
2020-10-19 00:26:34 -04:00
void SetNextPlotLimitsY ( double y_min , double y_max , ImGuiCond cond , ImPlotYAxis y_axis ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " SetNextPlotLimitsY() needs to be called before BeginPlot()! " ) ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis > = 0 & & y_axis < IMPLOT_Y_AXES , " y_axis needs to be between 0 and IMPLOT_Y_AXES " ) ;
2020-04-27 11:27:59 -04:00
IM_ASSERT ( cond = = 0 | | ImIsPowerOfTwo ( cond ) ) ; // Make sure the user doesn't attempt to combine multiple condition flags.
2020-05-11 07:12:22 -04:00
gp . NextPlotData . HasYRange [ y_axis ] = true ;
gp . NextPlotData . YRangeCond [ y_axis ] = cond ;
gp . NextPlotData . Y [ y_axis ] . Min = y_min ;
gp . NextPlotData . Y [ y_axis ] . Max = y_max ;
}
2020-09-03 10:19:34 -04:00
void LinkNextPlotLimits ( double * xmin , double * xmax , double * ymin , double * ymax , double * ymin2 , double * ymax2 , double * ymin3 , double * ymax3 ) {
ImPlotContext & gp = * GImPlot ;
gp . NextPlotData . LinkedXmin = xmin ;
gp . NextPlotData . LinkedXmax = xmax ;
gp . NextPlotData . LinkedYmin [ 0 ] = ymin ;
gp . NextPlotData . LinkedYmax [ 0 ] = ymax ;
gp . NextPlotData . LinkedYmin [ 1 ] = ymin2 ;
gp . NextPlotData . LinkedYmax [ 1 ] = ymax2 ;
gp . NextPlotData . LinkedYmin [ 2 ] = ymin3 ;
gp . NextPlotData . LinkedYmax [ 2 ] = ymax3 ;
}
2020-08-20 00:50:12 -04:00
void FitNextPlotAxes ( bool x , bool y , bool y2 , bool y3 ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " FitNextPlotAxes() needs to be called before BeginPlot()! " ) ;
gp . NextPlotData . FitX = x ;
gp . NextPlotData . FitY [ 0 ] = y ;
gp . NextPlotData . FitY [ 1 ] = y2 ;
gp . NextPlotData . FitY [ 2 ] = y3 ;
}
2020-09-07 11:17:44 -04:00
void SetNextPlotTicksX ( const double * values , int n_ticks , const char * const labels [ ] , bool show_default ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " SetNextPlotTicksX() needs to be called before BeginPlot()! " ) ;
gp . NextPlotData . ShowDefaultTicksX = show_default ;
2020-08-25 22:59:43 -04:00
AddTicksCustom ( values , labels , n_ticks , gp . XTicks ) ;
2020-06-03 15:37:01 -04:00
}
2020-09-07 11:17:44 -04:00
void SetNextPlotTicksX ( double x_min , double x_max , int n_ticks , const char * const labels [ ] , bool show_default ) {
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( n_ticks > 1 , " The number of ticks must be greater than 1 " ) ;
static ImVector < double > buffer ;
FillRange ( buffer , n_ticks , x_min , x_max ) ;
SetNextPlotTicksX ( & buffer [ 0 ] , n_ticks , labels , show_default ) ;
}
2020-10-19 00:26:34 -04:00
void SetNextPlotTicksY ( const double * values , int n_ticks , const char * const labels [ ] , bool show_default , ImPlotYAxis y_axis ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot = = NULL , " SetNextPlotTicksY() needs to be called before BeginPlot()! " ) ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis > = 0 & & y_axis < IMPLOT_Y_AXES , " y_axis needs to be between 0 and IMPLOT_Y_AXES " ) ;
2020-06-03 15:37:01 -04:00
gp . NextPlotData . ShowDefaultTicksY [ y_axis ] = show_default ;
2020-08-25 22:59:43 -04:00
AddTicksCustom ( values , labels , n_ticks , gp . YTicks [ y_axis ] ) ;
2020-06-03 15:37:01 -04:00
}
2020-10-19 00:26:34 -04:00
void SetNextPlotTicksY ( double y_min , double y_max , int n_ticks , const char * const labels [ ] , bool show_default , ImPlotYAxis y_axis ) {
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( n_ticks > 1 , " The number of ticks must be greater than 1 " ) ;
static ImVector < double > buffer ;
FillRange ( buffer , n_ticks , y_min , y_max ) ;
SetNextPlotTicksY ( & buffer [ 0 ] , n_ticks , labels , show_default , y_axis ) ;
}
2020-10-19 00:26:34 -04:00
void SetPlotYAxis ( ImPlotYAxis y_axis ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " SetPlotYAxis() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis > = 0 & & y_axis < IMPLOT_Y_AXES , " y_axis needs to be between 0 and IMPLOT_Y_AXES " ) ;
2020-05-11 07:12:22 -04:00
gp . CurrentPlot - > CurrentYAxis = y_axis ;
2020-04-27 11:27:59 -04:00
}
2020-04-30 09:45:03 -04:00
ImVec2 GetPlotPos ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " GetPlotPos() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-11-15 22:47:06 -05:00
return gp . CurrentPlot - > PlotRect . Min ;
2020-04-30 09:45:03 -04:00
}
ImVec2 GetPlotSize ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " GetPlotSize() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-11-15 22:47:06 -05:00
return gp . CurrentPlot - > PlotRect . GetSize ( ) ;
2020-04-30 09:45:03 -04:00
}
2020-08-30 18:12:36 -04:00
ImDrawList * GetPlotDrawList ( ) {
return ImGui : : GetWindowDrawList ( ) ;
}
2020-04-30 09:45:03 -04:00
void PushPlotClipRect ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " PushPlotClipRect() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-11-15 22:47:06 -05:00
ImGui : : PushClipRect ( gp . CurrentPlot - > PlotRect . Min , gp . CurrentPlot - > PlotRect . Max , true ) ;
2020-04-30 09:45:03 -04:00
}
void PopPlotClipRect ( ) {
2020-05-12 05:19:04 -04:00
ImGui : : PopClipRect ( ) ;
2020-04-30 09:45:03 -04:00
}
2020-05-13 10:11:25 -04:00
bool IsPlotHovered ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " IsPlotHovered() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-11-15 22:47:06 -05:00
return gp . CurrentPlot - > FrameHovered & & gp . CurrentPlot - > PlotHovered ;
2020-04-28 21:17:26 -04:00
}
2020-07-14 17:43:57 -04:00
bool IsPlotXAxisHovered ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-07-14 17:43:57 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " IsPlotXAxisHovered() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-11-15 22:47:06 -05:00
return gp . CurrentPlot - > XAxis . ExtHovered ;
2020-07-14 17:43:57 -04:00
}
2020-10-19 00:26:34 -04:00
bool IsPlotYAxisHovered ( ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis_in > = - 1 & & y_axis_in < IMPLOT_Y_AXES , " y_axis needs to between -1 and IMPLOT_Y_AXES " ) ;
2020-07-14 17:43:57 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " IsPlotYAxisHovered() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-11-15 22:47:06 -05:00
return gp . CurrentPlot - > YAxis [ y_axis ] . ExtHovered ;
2020-07-14 17:43:57 -04:00
}
2020-10-19 00:26:34 -04:00
ImPlotPoint GetPlotMousePos ( ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis_in > = - 1 & & y_axis_in < IMPLOT_Y_AXES , " y_axis needs to between -1 and IMPLOT_Y_AXES " ) ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " GetPlotMousePos() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-08-22 23:55:37 -04:00
return gp . MousePos [ y_axis ] ;
2020-04-28 21:17:26 -04:00
}
2020-04-27 11:27:59 -04:00
2020-10-19 00:26:34 -04:00
ImPlotLimits GetPlotLimits ( ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis_in > = - 1 & & y_axis_in < IMPLOT_Y_AXES , " y_axis needs to between -1 and IMPLOT_Y_AXES " ) ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " GetPlotLimits() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-05-11 07:12:22 -04:00
2020-09-21 08:09:14 -04:00
ImPlotPlot & plot = * gp . CurrentPlot ;
2020-05-11 07:12:22 -04:00
ImPlotLimits limits ;
limits . X = plot . XAxis . Range ;
limits . Y = plot . YAxis [ y_axis ] . Range ;
return limits ;
2020-04-28 00:57:49 -04:00
}
bool IsPlotQueried ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " IsPlotQueried() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-04-28 21:17:26 -04:00
return gp . CurrentPlot - > Queried ;
2020-04-28 00:57:49 -04:00
}
2020-10-19 00:26:34 -04:00
ImPlotLimits GetPlotQuery ( ImPlotYAxis y_axis_in ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-19 01:04:05 -04:00
IM_ASSERT_USER_ERROR ( y_axis_in > = - 1 & & y_axis_in < IMPLOT_Y_AXES , " y_axis needs to between -1 and IMPLOT_Y_AXES " ) ;
2020-06-03 15:37:01 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " GetPlotQuery() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-09-21 08:09:14 -04:00
ImPlotPlot & plot = * gp . CurrentPlot ;
2020-10-19 00:26:34 -04:00
const ImPlotYAxis y_axis = y_axis_in > = 0 ? y_axis_in : gp . CurrentPlot - > CurrentYAxis ;
2020-05-11 07:12:22 -04:00
UpdateTransformCache ( ) ;
2020-11-15 22:47:06 -05:00
ImPlotPoint p1 = PixelsToPlot ( plot . QueryRect . Min + plot . PlotRect . Min , y_axis ) ;
ImPlotPoint p2 = PixelsToPlot ( plot . QueryRect . Max + plot . PlotRect . Min , y_axis ) ;
2020-05-11 07:12:22 -04:00
ImPlotLimits result ;
result . X . Min = ImMin ( p1 . x , p2 . x ) ;
result . X . Max = ImMax ( p1 . x , p2 . x ) ;
result . Y . Min = ImMin ( p1 . y , p2 . y ) ;
result . Y . Max = ImMax ( p1 . y , p2 . y ) ;
return result ;
2020-04-28 00:57:49 -04:00
}
2020-09-19 21:54:19 -04:00
void AnnotateEx ( double x , double y , bool clamp , const ImVec4 & col , const ImVec2 & off , const char * fmt , va_list args ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " Annotate() needs to be called between BeginPlot() and EndPlot()! " ) ;
ImVec2 pos = PlotToPixels ( x , y ) ;
ImU32 bg = ImGui : : GetColorU32 ( col ) ;
ImU32 fg = col . w = = 0 ? GetStyleColorU32 ( ImPlotCol_InlayText ) : CalcTextColor ( col ) ;
gp . Annotations . AppendV ( pos , off , bg , fg , clamp , fmt , args ) ;
}
2020-10-11 03:53:28 -04:00
void AnnotateV ( double x , double y , const ImVec2 & offset , const char * fmt , va_list args ) {
AnnotateEx ( x , y , false , ImVec4 ( 0 , 0 , 0 , 0 ) , offset , fmt , args ) ;
}
2020-09-19 22:25:44 -04:00
void Annotate ( double x , double y , const ImVec2 & offset , const char * fmt , . . . ) {
2020-09-19 21:54:19 -04:00
va_list args ;
va_start ( args , fmt ) ;
2020-10-11 03:53:28 -04:00
AnnotateV ( x , y , offset , fmt , args ) ;
2020-09-19 21:54:19 -04:00
va_end ( args ) ;
}
2020-10-11 03:53:28 -04:00
void AnnotateV ( double x , double y , const ImVec2 & offset , const ImVec4 & col , const char * fmt , va_list args ) {
AnnotateEx ( x , y , false , col , offset , fmt , args ) ;
}
2020-09-19 22:25:44 -04:00
void Annotate ( double x , double y , const ImVec2 & offset , const ImVec4 & col , const char * fmt , . . . ) {
2020-09-19 21:54:19 -04:00
va_list args ;
va_start ( args , fmt ) ;
2020-10-11 03:53:28 -04:00
AnnotateV ( x , y , offset , col , fmt , args ) ;
2020-09-19 21:54:19 -04:00
va_end ( args ) ;
}
2020-10-11 03:53:28 -04:00
void AnnotateClampedV ( double x , double y , const ImVec2 & offset , const char * fmt , va_list args ) {
AnnotateEx ( x , y , true , ImVec4 ( 0 , 0 , 0 , 0 ) , offset , fmt , args ) ;
}
2020-09-19 22:25:44 -04:00
void AnnotateClamped ( double x , double y , const ImVec2 & offset , const char * fmt , . . . ) {
2020-09-19 21:54:19 -04:00
va_list args ;
va_start ( args , fmt ) ;
2020-10-11 03:53:28 -04:00
AnnotateClampedV ( x , y , offset , fmt , args ) ;
2020-09-19 21:54:19 -04:00
va_end ( args ) ;
}
2020-10-11 03:53:28 -04:00
void AnnotateClampedV ( double x , double y , const ImVec2 & offset , const ImVec4 & col , const char * fmt , va_list args ) {
AnnotateEx ( x , y , true , col , offset , fmt , args ) ;
}
2020-09-19 22:25:44 -04:00
void AnnotateClamped ( double x , double y , const ImVec2 & offset , const ImVec4 & col , const char * fmt , . . . ) {
2020-09-19 21:54:19 -04:00
va_list args ;
va_start ( args , fmt ) ;
2020-10-11 03:53:28 -04:00
AnnotateClampedV ( x , y , offset , col , fmt , args ) ;
2020-09-19 21:54:19 -04:00
va_end ( args ) ;
}
2020-09-19 13:33:33 -04:00
bool DragLineX ( const char * id , double * value , bool show_label , const ImVec4 & col , float thickness ) {
2020-09-17 10:58:33 -04:00
ImPlotContext & gp = * GImPlot ;
2020-09-19 13:33:33 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " DragLineX() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-09-18 14:37:55 -04:00
const float grab_size = ImMax ( 5.0f , thickness ) ;
2020-11-15 22:47:06 -05:00
float yt = gp . CurrentPlot - > PlotRect . Min . y ;
float yb = gp . CurrentPlot - > PlotRect . Max . y ;
2020-09-19 13:33:33 -04:00
float x = IM_ROUND ( PlotToPixels ( * value , 0 ) . x ) ;
2020-11-15 22:47:06 -05:00
const bool outside = x < ( gp . CurrentPlot - > PlotRect . Min . x - grab_size / 2 ) | | x > ( gp . CurrentPlot - > PlotRect . Max . x + grab_size / 2 ) ;
2020-09-17 10:58:33 -04:00
if ( outside )
return false ;
2020-09-19 13:33:33 -04:00
float len = gp . Style . MajorTickLen . x ;
2020-09-18 00:31:16 -04:00
ImVec4 color = IsColorAuto ( col ) ? ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) : col ;
ImU32 col32 = ImGui : : ColorConvertFloat4ToU32 ( color ) ;
ImDrawList & DrawList = * GetPlotDrawList ( ) ;
PushPlotClipRect ( ) ;
2020-09-19 13:33:33 -04:00
DrawList . AddLine ( ImVec2 ( x , yt ) , ImVec2 ( x , yb ) , col32 , thickness ) ;
DrawList . AddLine ( ImVec2 ( x , yt ) , ImVec2 ( x , yt + len ) , col32 , 3 * thickness ) ;
DrawList . AddLine ( ImVec2 ( x , yb ) , ImVec2 ( x , yb - len ) , col32 , 3 * thickness ) ;
2020-09-18 09:22:06 -04:00
PopPlotClipRect ( ) ;
if ( gp . CurrentPlot - > Selecting | | gp . CurrentPlot - > Querying )
return false ;
2020-09-17 10:58:33 -04:00
ImVec2 old_cursor_pos = ImGui : : GetCursorScreenPos ( ) ;
2020-09-19 13:33:33 -04:00
ImVec2 new_cursor_pos = ImVec2 ( x - grab_size / 2.0f , yt ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = new_cursor_pos ;
2020-09-19 13:33:33 -04:00
ImGui : : InvisibleButton ( id , ImVec2 ( grab_size , yb - yt ) ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = old_cursor_pos ;
2020-09-19 21:54:19 -04:00
if ( ImGui : : IsItemHovered ( ) | | ImGui : : IsItemActive ( ) ) {
2020-11-15 22:47:06 -05:00
gp . CurrentPlot - > PlotHovered = false ;
2020-09-19 13:33:33 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeEW ) ;
if ( show_label ) {
2020-09-21 08:09:14 -04:00
char buff [ 32 ] ;
LabelAxisValue ( gp . CurrentPlot - > XAxis , gp . XTicks , * value , buff , 32 ) ;
gp . Annotations . Append ( ImVec2 ( x , yb ) , ImVec2 ( 0 , 0 ) , col32 , CalcTextColor ( color ) , true , " %s = %s " , id , buff ) ;
2020-09-18 00:31:16 -04:00
}
2020-09-17 10:58:33 -04:00
}
bool dragging = false ;
if ( ImGui : : IsItemActive ( ) & & ImGui : : IsMouseDragging ( 0 ) ) {
2020-09-19 13:33:33 -04:00
* value = ImPlot : : GetPlotMousePos ( ) . x ;
2020-11-15 22:47:06 -05:00
* value = ImClamp ( * value , gp . CurrentPlot - > XAxis . Range . Min , gp . CurrentPlot - > XAxis . Range . Max ) ;
2020-09-17 10:58:33 -04:00
dragging = true ;
}
return dragging ;
}
2020-09-19 13:33:33 -04:00
bool DragLineY ( const char * id , double * value , bool show_label , const ImVec4 & col , float thickness ) {
2020-09-17 10:58:33 -04:00
ImPlotContext & gp = * GImPlot ;
2020-09-19 13:33:33 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " DragLineY() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-09-18 14:37:55 -04:00
const float grab_size = ImMax ( 5.0f , thickness ) ;
2020-11-15 22:47:06 -05:00
float xl = gp . CurrentPlot - > PlotRect . Min . x ;
float xr = gp . CurrentPlot - > PlotRect . Max . x ;
2020-09-19 13:33:33 -04:00
float y = IM_ROUND ( PlotToPixels ( 0 , * value ) . y ) ;
2020-11-15 22:47:06 -05:00
const bool outside = y < ( gp . CurrentPlot - > PlotRect . Min . y - grab_size / 2 ) | | y > ( gp . CurrentPlot - > PlotRect . Max . y + grab_size / 2 ) ;
2020-09-17 10:58:33 -04:00
if ( outside )
return false ;
2020-09-19 13:33:33 -04:00
float len = gp . Style . MajorTickLen . y ;
2020-09-18 00:31:16 -04:00
ImVec4 color = IsColorAuto ( col ) ? ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) : col ;
ImU32 col32 = ImGui : : ColorConvertFloat4ToU32 ( color ) ;
ImDrawList & DrawList = * GetPlotDrawList ( ) ;
PushPlotClipRect ( ) ;
2020-09-19 13:33:33 -04:00
DrawList . AddLine ( ImVec2 ( xl , y ) , ImVec2 ( xr , y ) , col32 , thickness ) ;
DrawList . AddLine ( ImVec2 ( xl , y ) , ImVec2 ( xl + len , y ) , col32 , 3 * thickness ) ;
DrawList . AddLine ( ImVec2 ( xr , y ) , ImVec2 ( xr - len , y ) , col32 , 3 * thickness ) ;
2020-09-18 09:22:06 -04:00
PopPlotClipRect ( ) ;
if ( gp . CurrentPlot - > Selecting | | gp . CurrentPlot - > Querying )
return false ;
2020-09-17 10:58:33 -04:00
ImVec2 old_cursor_pos = ImGui : : GetCursorScreenPos ( ) ;
2020-09-19 13:33:33 -04:00
ImVec2 new_cursor_pos = ImVec2 ( xl , y - grab_size / 2.0f ) ;
ImGui : : SetItemAllowOverlap ( ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = new_cursor_pos ;
2020-09-19 13:33:33 -04:00
ImGui : : InvisibleButton ( id , ImVec2 ( xr - xl , grab_size ) ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = old_cursor_pos ;
2020-09-19 13:33:33 -04:00
int yax = GetCurrentYAxis ( ) ;
2020-09-17 10:58:33 -04:00
if ( ImGui : : IsItemHovered ( ) | | ImGui : : IsItemActive ( ) ) {
2020-11-15 22:47:06 -05:00
gp . CurrentPlot - > PlotHovered = false ;
2020-09-19 13:33:33 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeNS ) ;
if ( show_label ) {
2020-09-21 08:09:14 -04:00
char buff [ 32 ] ;
LabelAxisValue ( gp . CurrentPlot - > YAxis [ yax ] , gp . YTicks [ yax ] , * value , buff , 32 ) ;
gp . Annotations . Append ( ImVec2 ( yax = = 0 ? xl : xr , y ) , ImVec2 ( 0 , 0 ) , col32 , CalcTextColor ( color ) , true , " %s = %s " , id , buff ) ;
2020-09-19 13:33:33 -04:00
}
2020-09-17 10:58:33 -04:00
}
bool dragging = false ;
if ( ImGui : : IsItemActive ( ) & & ImGui : : IsMouseDragging ( 0 ) ) {
2020-09-19 13:33:33 -04:00
* value = ImPlot : : GetPlotMousePos ( ) . y ;
2020-11-15 22:47:06 -05:00
* value = ImClamp ( * value , gp . CurrentPlot - > YAxis [ yax ] . Range . Min , gp . CurrentPlot - > YAxis [ yax ] . Range . Max ) ;
2020-09-17 10:58:33 -04:00
dragging = true ;
}
return dragging ;
}
2020-09-19 13:33:33 -04:00
bool DragPoint ( const char * id , double * x , double * y , bool show_label , const ImVec4 & col , float radius ) {
2020-09-18 14:37:55 -04:00
ImPlotContext & gp = * GImPlot ;
2020-09-19 13:33:33 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " DragPoint() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-09-18 14:37:55 -04:00
const float grab_size = ImMax ( 5.0f , 2 * radius ) ;
const bool outside = ! GetPlotLimits ( ) . Contains ( * x , * y ) ;
if ( outside )
return false ;
ImVec4 color = IsColorAuto ( col ) ? ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) : col ;
ImU32 col32 = ImGui : : ColorConvertFloat4ToU32 ( color ) ;
ImDrawList & DrawList = * GetPlotDrawList ( ) ;
ImVec2 pos = PlotToPixels ( * x , * y ) ;
PushPlotClipRect ( ) ;
DrawList . AddCircleFilled ( pos , radius , col32 ) ;
PopPlotClipRect ( ) ;
int yax = GetCurrentYAxis ( ) ;
ImVec2 old_cursor_pos = ImGui : : GetCursorScreenPos ( ) ;
ImVec2 new_cursor_pos = ImVec2 ( pos - ImVec2 ( grab_size , grab_size ) * 0.5f ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = new_cursor_pos ;
2020-09-18 14:37:55 -04:00
ImGui : : InvisibleButton ( id , ImVec2 ( grab_size , grab_size ) ) ;
2020-09-18 23:05:43 -04:00
ImGui : : GetCurrentWindow ( ) - > DC . CursorPos = old_cursor_pos ;
2020-09-18 14:37:55 -04:00
if ( ImGui : : IsItemHovered ( ) | | ImGui : : IsItemActive ( ) ) {
2020-11-15 22:47:06 -05:00
gp . CurrentPlot - > PlotHovered = false ;
2020-09-18 14:37:55 -04:00
ImGui : : SetMouseCursor ( ImGuiMouseCursor_ResizeAll ) ;
2020-09-19 13:33:33 -04:00
if ( show_label ) {
ImVec2 label_pos = pos + ImVec2 ( 16 * GImGui - > Style . MouseCursorScale , 8 * GImGui - > Style . MouseCursorScale ) ;
2020-09-21 08:09:14 -04:00
char buff1 [ 32 ] ;
char buff2 [ 32 ] ;
LabelAxisValue ( gp . CurrentPlot - > XAxis , gp . XTicks , * x , buff1 , 32 ) ;
LabelAxisValue ( gp . CurrentPlot - > YAxis [ yax ] , gp . YTicks [ yax ] , * y , buff2 , 32 ) ;
gp . Annotations . Append ( label_pos , ImVec2 ( 0.0001f , 0.00001f ) , col32 , CalcTextColor ( color ) , true , " %s = %s,%s " , id , buff1 , buff2 ) ;
2020-09-19 13:33:33 -04:00
}
2020-09-18 14:37:55 -04:00
}
bool dragging = false ;
if ( ImGui : : IsItemActive ( ) & & ImGui : : IsMouseDragging ( 0 ) ) {
* x = ImPlot : : GetPlotMousePos ( ) . x ;
* y = ImPlot : : GetPlotMousePos ( ) . y ;
2020-11-15 22:47:06 -05:00
* x = ImClamp ( * x , gp . CurrentPlot - > XAxis . Range . Min , gp . CurrentPlot - > XAxis . Range . Max ) ;
* y = ImClamp ( * y , gp . CurrentPlot - > YAxis [ yax ] . Range . Min , gp . CurrentPlot - > YAxis [ yax ] . Range . Max ) ;
2020-09-18 14:37:55 -04:00
dragging = true ;
}
return dragging ;
}
2020-10-19 00:26:34 -04:00
void SetLegendLocation ( ImPlotLocation location , ImPlotOrientation orientation , bool outside ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " SetLegendLocation() needs to be called between BeginPlot() and EndPlot()! " ) ;
gp . CurrentPlot - > LegendLocation = location ;
gp . CurrentPlot - > LegendOrientation = orientation ;
if ( gp . CurrentPlot - > LegendOutside ! = outside )
2020-11-15 22:47:06 -05:00
gp . CurrentPlot - > LegendFlipSideNextFrame = true ;
2020-10-19 00:26:34 -04:00
}
void SetMousePosLocation ( ImPlotLocation location ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " SetMousePosLocation() needs to be called between BeginPlot() and EndPlot()! " ) ;
gp . CurrentPlot - > MousePosLocation = location ;
}
2020-07-26 22:49:31 -04:00
bool IsLegendEntryHovered ( const char * label_id ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-07-13 00:44:51 -04:00
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " IsPlotItemHighlight() needs to be called between BeginPlot() and EndPlot()! " ) ;
2020-07-14 17:43:57 -04:00
ImGuiID id = ImGui : : GetID ( label_id ) ;
2020-07-26 22:28:22 -04:00
ImPlotItem * item = gp . CurrentPlot - > Items . GetByKey ( id ) ;
2020-09-01 00:58:15 -04:00
return item & & item - > LegendHovered ;
2020-07-13 00:44:51 -04:00
}
2020-08-31 23:38:14 -04:00
bool BeginLegendDragDropSource ( const char * label_id , ImGuiDragDropFlags flags ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " BeginLegendDragDropSource() needs to be called between BeginPlot() and EndPlot()! " ) ;
ImGuiID source_id = ImGui : : GetID ( label_id ) ;
ImPlotItem * item = gp . CurrentPlot - > Items . GetByKey ( source_id ) ;
2020-09-01 00:58:15 -04:00
bool is_hovered = item & & item - > LegendHovered ;
2020-08-31 23:38:14 -04:00
ImGuiContext & g = * GImGui ;
ImGuiWindow * window = g . CurrentWindow ;
ImGuiMouseButton mouse_button = ImGuiMouseButton_Left ;
if ( g . IO . MouseDown [ mouse_button ] = = false ) {
if ( g . ActiveId = = source_id )
ImGui : : ClearActiveID ( ) ;
return false ;
}
if ( is_hovered & & g . IO . MouseClicked [ mouse_button ] ) {
ImGui : : SetActiveID ( source_id , window ) ;
ImGui : : FocusWindow ( window ) ;
}
if ( g . ActiveId ! = source_id )
return false ;
// Allow the underlying widget to display/return hovered during the mouse
// release frame, else we would get a flicker.
g . ActiveIdAllowOverlap = is_hovered ;
// Disable navigation and key inputs while dragging
g . ActiveIdUsingNavDirMask = ~ ( ImU32 ) 0 ;
g . ActiveIdUsingNavInputMask = ~ ( ImU32 ) 0 ;
g . ActiveIdUsingKeyInputMask = ~ ( ImU64 ) 0 ;
if ( ImGui : : IsMouseDragging ( mouse_button ) ) {
if ( ! g . DragDropActive ) {
ImGui : : ClearDragDrop ( ) ;
ImGuiPayload & payload = g . DragDropPayload ;
payload . SourceId = source_id ;
payload . SourceParentId = 0 ;
g . DragDropActive = true ;
g . DragDropSourceFlags = 0 ;
g . DragDropMouseButton = mouse_button ;
}
g . DragDropSourceFrameCount = g . FrameCount ;
g . DragDropWithinSource = true ;
if ( ! ( flags & ImGuiDragDropFlags_SourceNoPreviewTooltip ) ) {
// Target can request the Source to not display its tooltip (we use a
// dedicated flag to make this request explicit) We unfortunately can't
// just modify the source flags and skip the call to BeginTooltip, as
// caller may be emitting contents.
ImGui : : BeginTooltip ( ) ;
if ( g . DragDropAcceptIdPrev & & ( g . DragDropAcceptFlags & ImGuiDragDropFlags_AcceptNoPreviewTooltip ) ) {
ImGuiWindow * tooltip_window = g . CurrentWindow ;
tooltip_window - > SkipItems = true ;
tooltip_window - > HiddenFramesCanSkipItems = 1 ;
}
}
return true ;
}
return false ;
}
2020-09-01 00:58:15 -04:00
void EndLegendDragDropSource ( ) {
2020-08-31 23:38:14 -04:00
ImGui : : EndDragDropSource ( ) ;
}
2020-09-01 02:14:09 -04:00
bool BeginLegendPopup ( const char * label_id , ImGuiMouseButton mouse_button ) {
2020-08-31 23:08:53 -04:00
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " BeginLegendPopup() needs to be called between BeginPlot() and EndPlot()! " ) ;
ImGuiWindow * window = GImGui - > CurrentWindow ;
if ( window - > SkipItems )
return false ;
ImGuiID id = ImGui : : GetID ( label_id ) ;
if ( ImGui : : IsMouseReleased ( mouse_button ) ) {
ImPlotItem * item = gp . CurrentPlot - > Items . GetByKey ( id ) ;
2020-09-01 02:14:09 -04:00
if ( item & & item - > LegendHovered )
2020-08-31 23:08:53 -04:00
ImGui : : OpenPopupEx ( id ) ;
}
return ImGui : : BeginPopupEx ( id , ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoSavedSettings ) ;
}
2020-09-01 02:14:09 -04:00
void EndLegendPopup ( ) {
2020-08-31 23:08:53 -04:00
ImGui : : EndPopup ( ) ;
}
2020-10-19 00:26:34 -04:00
void ShowAltLegend ( const char * title_id , ImPlotOrientation orientation , const ImVec2 size , bool interactable ) {
ImPlotContext & gp = * GImPlot ;
ImGuiContext & G = * GImGui ;
ImGuiWindow * Window = G . CurrentWindow ;
if ( Window - > SkipItems )
return ;
ImDrawList & DrawList = * Window - > DrawList ;
ImPlotPlot * plot = GetPlot ( title_id ) ;
ImVec2 legend_size ;
ImVec2 default_size = gp . Style . LegendPadding * 2 ;
if ( plot ! = NULL ) {
2020-10-21 11:08:41 -04:00
legend_size = CalcLegendSize ( * plot , gp . Style . LegendInnerPadding , gp . Style . LegendSpacing , orientation ) ;
2020-10-19 00:26:34 -04:00
default_size = legend_size + gp . Style . LegendPadding * 2 ;
}
ImVec2 frame_size = ImGui : : CalcItemSize ( size , default_size . x , default_size . y ) ;
ImRect bb_frame = ImRect ( Window - > DC . CursorPos , Window - > DC . CursorPos + frame_size ) ;
ImGui : : ItemSize ( bb_frame ) ;
if ( ! ImGui : : ItemAdd ( bb_frame , 0 , & bb_frame ) )
return ;
ImGui : : RenderFrame ( bb_frame . Min , bb_frame . Max , GetStyleColorU32 ( ImPlotCol_FrameBg ) , true , G . Style . FrameRounding ) ;
DrawList . PushClipRect ( bb_frame . Min , bb_frame . Max , true ) ;
if ( plot ! = NULL ) {
const ImVec2 legend_pos = GetLocationPos ( bb_frame , legend_size , 0 , gp . Style . LegendPadding ) ;
const ImRect legend_bb ( legend_pos , legend_pos + legend_size ) ;
interactable = interactable & & bb_frame . Contains ( ImGui : : GetIO ( ) . MousePos ) ;
// render legend box
ImU32 col_bg = GetStyleColorU32 ( ImPlotCol_LegendBg ) ;
ImU32 col_bd = GetStyleColorU32 ( ImPlotCol_LegendBorder ) ;
DrawList . AddRectFilled ( legend_bb . Min , legend_bb . Max , col_bg ) ;
DrawList . AddRect ( legend_bb . Min , legend_bb . Max , col_bd ) ;
// render entries
2020-10-21 11:08:41 -04:00
ShowLegendEntries ( * plot , legend_bb , interactable , gp . Style . LegendInnerPadding , gp . Style . LegendSpacing , orientation , DrawList ) ;
2020-10-19 00:26:34 -04:00
}
DrawList . PopClipRect ( ) ;
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// STYLING
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
ImPlotStyle & GetStyle ( ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
return gp . Style ;
}
2020-05-12 05:19:04 -04:00
void PushStyleColor ( ImPlotCol idx , ImU32 col ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
ImGuiColorMod backup ;
backup . Col = idx ;
backup . BackupValue = gp . Style . Colors [ idx ] ;
gp . ColorModifiers . push_back ( backup ) ;
2020-05-12 05:19:04 -04:00
gp . Style . Colors [ idx ] = ImGui : : ColorConvertU32ToFloat4 ( col ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
void PushStyleColor ( ImPlotCol idx , const ImVec4 & col ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
ImGuiColorMod backup ;
backup . Col = idx ;
backup . BackupValue = gp . Style . Colors [ idx ] ;
gp . ColorModifiers . push_back ( backup ) ;
gp . Style . Colors [ idx ] = col ;
}
2020-05-12 05:19:04 -04:00
void PopStyleColor ( int count ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
while ( count > 0 )
{
ImGuiColorMod & backup = gp . ColorModifiers . back ( ) ;
gp . Style . Colors [ backup . Col ] = backup . BackupValue ;
gp . ColorModifiers . pop_back ( ) ;
count - - ;
}
}
2020-05-12 05:19:04 -04:00
void PushStyleVar ( ImPlotStyleVar idx , float val ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
const ImPlotStyleVarInfo * var_info = GetPlotStyleVarInfo ( idx ) ;
if ( var_info - > Type = = ImGuiDataType_Float & & var_info - > Count = = 1 ) {
float * pvar = ( float * ) var_info - > GetVarPtr ( & gp . Style ) ;
gp . StyleModifiers . push_back ( ImGuiStyleMod ( idx , * pvar ) ) ;
* pvar = val ;
return ;
}
2020-05-12 05:19:04 -04:00
IM_ASSERT ( 0 & & " Called PushStyleVar() float variant but variable is not a float! " ) ;
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
void PushStyleVar ( ImPlotStyleVar idx , int val ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
const ImPlotStyleVarInfo * var_info = GetPlotStyleVarInfo ( idx ) ;
if ( var_info - > Type = = ImGuiDataType_S32 & & var_info - > Count = = 1 ) {
int * pvar = ( int * ) var_info - > GetVarPtr ( & gp . Style ) ;
gp . StyleModifiers . push_back ( ImGuiStyleMod ( idx , * pvar ) ) ;
* pvar = val ;
return ;
}
else if ( var_info - > Type = = ImGuiDataType_Float & & var_info - > Count = = 1 ) {
float * pvar = ( float * ) var_info - > GetVarPtr ( & gp . Style ) ;
gp . StyleModifiers . push_back ( ImGuiStyleMod ( idx , * pvar ) ) ;
* pvar = ( float ) val ;
return ;
}
2020-05-12 05:19:04 -04:00
IM_ASSERT ( 0 & & " Called PushStyleVar() int variant but variable is not a int! " ) ;
2020-04-27 11:27:59 -04:00
}
2020-08-21 00:01:21 -04:00
void PushStyleVar ( ImGuiStyleVar idx , const ImVec2 & val )
{
ImPlotContext & gp = * GImPlot ;
const ImPlotStyleVarInfo * var_info = GetPlotStyleVarInfo ( idx ) ;
if ( var_info - > Type = = ImGuiDataType_Float & & var_info - > Count = = 2 )
{
ImVec2 * pvar = ( ImVec2 * ) var_info - > GetVarPtr ( & gp . Style ) ;
gp . StyleModifiers . push_back ( ImGuiStyleMod ( idx , * pvar ) ) ;
* pvar = val ;
return ;
}
IM_ASSERT ( 0 & & " Called PushStyleVar() ImVec2 variant but variable is not a ImVec2! " ) ;
}
2020-05-12 05:19:04 -04:00
void PopStyleVar ( int count ) {
2020-08-16 11:25:06 -04:00
ImPlotContext & gp = * GImPlot ;
2020-04-27 11:27:59 -04:00
while ( count > 0 ) {
ImGuiStyleMod & backup = gp . StyleModifiers . back ( ) ;
const ImPlotStyleVarInfo * info = GetPlotStyleVarInfo ( backup . VarIdx ) ;
void * data = info - > GetVarPtr ( & gp . Style ) ;
if ( info - > Type = = ImGuiDataType_Float & & info - > Count = = 1 ) {
( ( float * ) data ) [ 0 ] = backup . BackupFloat [ 0 ] ;
}
else if ( info - > Type = = ImGuiDataType_Float & & info - > Count = = 2 ) {
2020-05-13 10:11:25 -04:00
( ( float * ) data ) [ 0 ] = backup . BackupFloat [ 0 ] ;
( ( float * ) data ) [ 1 ] = backup . BackupFloat [ 1 ] ;
2020-04-27 11:27:59 -04:00
}
else if ( info - > Type = = ImGuiDataType_S32 & & info - > Count = = 1 ) {
( ( int * ) data ) [ 0 ] = backup . BackupInt [ 0 ] ;
}
gp . StyleModifiers . pop_back ( ) ;
count - - ;
}
}
2020-08-16 16:38:51 -04:00
//------------------------------------------------------------------------------
// COLORMAPS
//------------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-08-21 23:13:11 -04:00
void PushColormap ( ImPlotColormap colormap ) {
2020-08-16 16:38:51 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-21 23:13:11 -04:00
gp . ColormapModifiers . push_back ( ImPlotColormapMod ( gp . Colormap , gp . ColormapSize ) ) ;
gp . Colormap = GetColormap ( colormap , & gp . ColormapSize ) ;
2020-08-16 16:38:51 -04:00
}
2020-04-27 11:27:59 -04:00
2020-08-21 23:13:11 -04:00
void PushColormap ( const ImVec4 * colormap , int size ) {
2020-08-16 16:38:51 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-21 23:13:11 -04:00
gp . ColormapModifiers . push_back ( ImPlotColormapMod ( gp . Colormap , gp . ColormapSize ) ) ;
gp . Colormap = colormap ;
gp . ColormapSize = size ;
2020-08-16 16:38:51 -04:00
}
2020-04-27 11:27:59 -04:00
2020-08-21 23:13:11 -04:00
void PopColormap ( int count ) {
2020-08-16 16:38:51 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-21 23:13:11 -04:00
while ( count > 0 ) {
const ImPlotColormapMod & backup = gp . ColormapModifiers . back ( ) ;
gp . Colormap = backup . Colormap ;
gp . ColormapSize = backup . ColormapSize ;
gp . ColormapModifiers . pop_back ( ) ;
count - - ;
}
2020-08-16 16:38:51 -04:00
}
2020-04-27 11:27:59 -04:00
2020-08-21 23:13:11 -04:00
void SetColormap ( ImPlotColormap colormap , int samples ) {
2020-08-16 16:38:51 -04:00
ImPlotContext & gp = * GImPlot ;
2020-08-21 23:13:11 -04:00
gp . Colormap = GetColormap ( colormap , & gp . ColormapSize ) ;
if ( samples > 1 ) {
static ImVector < ImVec4 > resampled ;
resampled . resize ( samples ) ;
ResampleColormap ( gp . Colormap , gp . ColormapSize , & resampled [ 0 ] , samples ) ;
SetColormap ( & resampled [ 0 ] , samples ) ;
2020-06-02 23:07:27 -04:00
}
2020-08-21 23:13:11 -04:00
}
2020-06-02 23:07:27 -04:00
2020-08-21 23:13:11 -04:00
void SetColormap ( const ImVec4 * colors , int size ) {
ImPlotContext & gp = * GImPlot ;
2020-08-24 00:45:42 -04:00
IM_ASSERT_USER_ERROR ( colors ! = NULL , " You can't set the colors to NULL! " ) ;
IM_ASSERT_USER_ERROR ( size > 0 , " The number of colors must be greater than 0! " ) ;
2020-08-21 23:13:11 -04:00
static ImVector < ImVec4 > user_colormap ;
user_colormap . shrink ( 0 ) ;
user_colormap . reserve ( size ) ;
for ( int i = 0 ; i < size ; + + i )
user_colormap . push_back ( colors [ i ] ) ;
gp . Colormap = & user_colormap [ 0 ] ;
gp . ColormapSize = size ;
2020-06-02 13:34:14 -04:00
}
2020-08-21 23:13:11 -04:00
const ImVec4 * GetColormap ( ImPlotColormap colormap , int * size_out ) {
2020-08-24 12:06:29 -04:00
static const int csizes [ ImPlotColormap_COUNT ] = { 10 , 10 , 9 , 9 , 12 , 11 , 11 , 11 , 11 , 11 , 11 } ;
2020-08-21 23:13:11 -04:00
static const ImOffsetCalculator < ImPlotColormap_COUNT > coffs ( csizes ) ;
2020-08-24 00:45:42 -04:00
static ImVec4 cdata [ ] = {
2020-06-02 13:34:14 -04:00
// ImPlotColormap_Default // X11 Named Colors
ImVec4 ( 0.0f , 0.7490196228f , 1.0f , 1.0f ) , // Blues::DeepSkyBlue,
ImVec4 ( 1.0f , 0.0f , 0.0f , 1.0f ) , // Reds::Red,
ImVec4 ( 0.4980392158f , 1.0f , 0.0f , 1.0f ) , // Greens::Chartreuse,
ImVec4 ( 1.0f , 1.0f , 0.0f , 1.0f ) , // Yellows::Yellow,
ImVec4 ( 0.0f , 1.0f , 1.0f , 1.0f ) , // Cyans::Cyan,
ImVec4 ( 1.0f , 0.6470588446f , 0.0f , 1.0f ) , // Oranges::Orange,
ImVec4 ( 1.0f , 0.0f , 1.0f , 1.0f ) , // Purples::Magenta,
ImVec4 ( 0.5411764979f , 0.1686274558f , 0.8862745166f , 1.0f ) , // Purples::BlueViolet,
ImVec4 ( 0.5f , 0.5f , 0.5f , 1.0f ) , // Grays::Gray50,
ImVec4 ( 0.8235294223f , 0.7058823705f , 0.5490196347f , 1.0f ) , // Browns::Tan
2020-08-24 12:06:29 -04:00
// ImPlotColormap_Deep
ImVec4 ( 0.298f , 0.447f , 0.690f , 1.000f ) ,
ImVec4 ( 0.867f , 0.518f , 0.322f , 1.000f ) ,
ImVec4 ( 0.333f , 0.659f , 0.408f , 1.000f ) ,
ImVec4 ( 0.769f , 0.306f , 0.322f , 1.000f ) ,
ImVec4 ( 0.506f , 0.446f , 0.702f , 1.000f ) ,
ImVec4 ( 0.576f , 0.471f , 0.376f , 1.000f ) ,
ImVec4 ( 0.855f , 0.545f , 0.765f , 1.000f ) ,
ImVec4 ( 0.549f , 0.549f , 0.549f , 1.000f ) ,
ImVec4 ( 0.800f , 0.725f , 0.455f , 1.000f ) ,
ImVec4 ( 0.392f , 0.710f , 0.804f , 1.000f ) ,
2020-06-02 13:34:14 -04:00
// ImPlotColormap_Dark
ImVec4 ( 0.894118f , 0.101961f , 0.109804f , 1.0f ) ,
ImVec4 ( 0.215686f , 0.494118f , 0.721569f , 1.0f ) ,
ImVec4 ( 0.301961f , 0.686275f , 0.290196f , 1.0f ) ,
ImVec4 ( 0.596078f , 0.305882f , 0.639216f , 1.0f ) ,
ImVec4 ( 1.000000f , 0.498039f , 0.000000f , 1.0f ) ,
ImVec4 ( 1.000000f , 1.000000f , 0.200000f , 1.0f ) ,
ImVec4 ( 0.650980f , 0.337255f , 0.156863f , 1.0f ) ,
ImVec4 ( 0.968627f , 0.505882f , 0.749020f , 1.0f ) ,
ImVec4 ( 0.600000f , 0.600000f , 0.600000f , 1.0f ) ,
// ImPlotColormap_Pastel
ImVec4 ( 0.984314f , 0.705882f , 0.682353f , 1.0f ) ,
ImVec4 ( 0.701961f , 0.803922f , 0.890196f , 1.0f ) ,
ImVec4 ( 0.800000f , 0.921569f , 0.772549f , 1.0f ) ,
ImVec4 ( 0.870588f , 0.796078f , 0.894118f , 1.0f ) ,
ImVec4 ( 0.996078f , 0.850980f , 0.650980f , 1.0f ) ,
ImVec4 ( 1.000000f , 1.000000f , 0.800000f , 1.0f ) ,
ImVec4 ( 0.898039f , 0.847059f , 0.741176f , 1.0f ) ,
ImVec4 ( 0.992157f , 0.854902f , 0.925490f , 1.0f ) ,
ImVec4 ( 0.949020f , 0.949020f , 0.949020f , 1.0f ) ,
// ImPlotColormap_Paired
ImVec4 ( 0.258824f , 0.807843f , 0.890196f , 1.0f ) ,
ImVec4 ( 0.121569f , 0.470588f , 0.705882f , 1.0f ) ,
ImVec4 ( 0.698039f , 0.874510f , 0.541176f , 1.0f ) ,
ImVec4 ( 0.200000f , 0.627451f , 0.172549f , 1.0f ) ,
ImVec4 ( 0.984314f , 0.603922f , 0.600000f , 1.0f ) ,
ImVec4 ( 0.890196f , 0.101961f , 0.109804f , 1.0f ) ,
ImVec4 ( 0.992157f , 0.749020f , 0.435294f , 1.0f ) ,
ImVec4 ( 1.000000f , 0.498039f , 0.000000f , 1.0f ) ,
ImVec4 ( 0.792157f , 0.698039f , 0.839216f , 1.0f ) ,
ImVec4 ( 0.415686f , 0.239216f , 0.603922f , 1.0f ) ,
ImVec4 ( 1.000000f , 1.000000f , 0.600000f , 1.0f ) ,
ImVec4 ( 0.694118f , 0.349020f , 0.156863f , 1.0f ) ,
// ImPlotColormap_Viridis
ImVec4 ( 0.267004f , 0.004874f , 0.329415f , 1.0f ) ,
ImVec4 ( 0.282623f , 0.140926f , 0.457517f , 1.0f ) ,
ImVec4 ( 0.253935f , 0.265254f , 0.529983f , 1.0f ) ,
ImVec4 ( 0.206756f , 0.371758f , 0.553117f , 1.0f ) ,
ImVec4 ( 0.163625f , 0.471133f , 0.558148f , 1.0f ) ,
ImVec4 ( 0.127568f , 0.566949f , 0.550556f , 1.0f ) ,
ImVec4 ( 0.134692f , 0.658636f , 0.517649f , 1.0f ) ,
ImVec4 ( 0.266941f , 0.748751f , 0.440573f , 1.0f ) ,
ImVec4 ( 0.477504f , 0.821444f , 0.318195f , 1.0f ) ,
ImVec4 ( 0.741388f , 0.873449f , 0.149561f , 1.0f ) ,
ImVec4 ( 0.993248f , 0.906157f , 0.143936f , 1.0f ) ,
// ImPlotColormap_Plasma
ImVec4 ( 5.03830e-02 f , 2.98030e-02 f , 5.27975e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 2.54627e-01 f , 1.38820e-02 f , 6.15419e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 4.17642e-01 f , 5.64000e-04 f , 6.58390e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 5.62738e-01 f , 5.15450e-02 f , 6.41509e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 6.92840e-01 f , 1.65141e-01 f , 5.64522e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 7.98216e-01 f , 2.80197e-01 f , 4.69538e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 8.81443e-01 f , 3.92529e-01 f , 3.83229e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 9.49217e-01 f , 5.17763e-01 f , 2.95662e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 9.88260e-01 f , 6.52325e-01 f , 2.11364e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 9.88648e-01 f , 8.09579e-01 f , 1.45357e-01 f , 1.00000e+00 f ) ,
ImVec4 ( 9.40015e-01 f , 9.75158e-01 f , 1.31326e-01 f , 1.00000e+00 f ) ,
// ImPlotColormap_Hot
ImVec4 ( 0.2500f , 0.f , 0.f , 1.0f ) ,
ImVec4 ( 0.5000f , 0.f , 0.f , 1.0f ) ,
ImVec4 ( 0.7500f , 0.f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.2500f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.5000f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.7500f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 0.3333f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 0.6667f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 1.0000f , 1.0f ) ,
// ImPlotColormap_Cool
ImVec4 ( 0.f , 1.0000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.1000f , 0.9000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.2000f , 0.8000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.3000f , 0.7000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.4000f , 0.6000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.5000f , 0.5000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.6000f , 0.4000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.7000f , 0.3000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.8000f , 0.2000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.9000f , 0.1000f , 1.0000f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.f , 1.0000f , 1.0f ) ,
// ImPlotColormap_Pink
ImVec4 ( 0.2887f , 0.f , 0.f , 1.0f ) ,
ImVec4 ( 0.4830f , 0.2582f , 0.2582f , 1.0f ) ,
ImVec4 ( 0.6191f , 0.3651f , 0.3651f , 1.0f ) ,
ImVec4 ( 0.7303f , 0.4472f , 0.4472f , 1.0f ) ,
ImVec4 ( 0.7746f , 0.5916f , 0.5164f , 1.0f ) ,
ImVec4 ( 0.8165f , 0.7071f , 0.5774f , 1.0f ) ,
ImVec4 ( 0.8563f , 0.8062f , 0.6325f , 1.0f ) ,
ImVec4 ( 0.8944f , 0.8944f , 0.6831f , 1.0f ) ,
ImVec4 ( 0.9309f , 0.9309f , 0.8028f , 1.0f ) ,
ImVec4 ( 0.9661f , 0.9661f , 0.9068f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 1.0000f , 1.0f ) ,
// ImPlotColormap_Jet
ImVec4 ( 0.f , 0.f , 0.6667f , 1.0f ) ,
ImVec4 ( 0.f , 0.f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.f , 0.3333f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.f , 0.6667f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.f , 1.0000f , 1.0000f , 1.0f ) ,
ImVec4 ( 0.3333f , 1.0000f , 0.6667f , 1.0f ) ,
ImVec4 ( 0.6667f , 1.0000f , 0.3333f , 1.0f ) ,
ImVec4 ( 1.0000f , 1.0000f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.6667f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.3333f , 0.f , 1.0f ) ,
ImVec4 ( 1.0000f , 0.f , 0.f , 1.0f )
} ;
2020-08-21 23:13:11 -04:00
* size_out = csizes [ colormap ] ;
return & cdata [ coffs . Offsets [ colormap ] ] ;
}
2020-08-22 23:55:37 -04:00
const char * GetColormapName ( ImPlotColormap colormap ) {
2020-08-24 12:06:29 -04:00
static const char * cmap_names [ ] = { " Default " , " Deep " , " Dark " , " Pastel " , " Paired " , " Viridis " , " Plasma " , " Hot " , " Cool " , " Pink " , " Jet " } ;
2020-08-22 23:55:37 -04:00
return cmap_names [ colormap ] ;
}
2020-08-21 23:13:11 -04:00
void ResampleColormap ( const ImVec4 * colormap_in , int size_in , ImVec4 * colormap_out , int size_out ) {
for ( int i = 0 ; i < size_out ; + + i ) {
float t = i * 1.0f / ( size_out - 1 ) ;
colormap_out [ i ] = LerpColormap ( colormap_in , size_in , t ) ;
2020-06-03 23:18:47 -04:00
}
2020-06-02 13:34:14 -04:00
}
2020-08-21 23:13:11 -04:00
int GetColormapSize ( ) {
ImPlotContext & gp = * GImPlot ;
return gp . ColormapSize ;
2020-08-16 16:38:51 -04:00
}
2020-08-21 23:13:11 -04:00
ImVec4 GetColormapColor ( int index ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( index > = 0 , " The Colormap index must be greater than zero! " ) ;
return gp . Colormap [ index % gp . ColormapSize ] ;
}
ImVec4 LerpColormap ( const ImVec4 * colormap , int size , float t ) {
float tc = ImClamp ( t , 0.0f , 1.0f ) ;
int i1 = ( int ) ( ( size - 1 ) * tc ) ;
int i2 = i1 + 1 ;
2020-08-24 00:45:42 -04:00
if ( i2 = = size | | size = = 1 )
2020-08-21 23:13:11 -04:00
return colormap [ i1 ] ;
float t1 = ( float ) i1 / ( float ) ( size - 1 ) ;
float t2 = ( float ) i2 / ( float ) ( size - 1 ) ;
float tr = ImRemap ( t , t1 , t2 , 0.0f , 1.0f ) ;
return ImLerp ( colormap [ i1 ] , colormap [ i2 ] , tr ) ;
2020-08-16 11:25:06 -04:00
}
2020-08-21 23:13:11 -04:00
ImVec4 LerpColormap ( float t ) {
ImPlotContext & gp = * GImPlot ;
return LerpColormap ( gp . Colormap , gp . ColormapSize , t ) ;
2020-08-16 16:38:51 -04:00
}
2020-08-21 23:13:11 -04:00
ImVec4 NextColormapColor ( ) {
ImPlotContext & gp = * GImPlot ;
IM_ASSERT_USER_ERROR ( gp . CurrentPlot ! = NULL , " NextColormapColor() needs to be called between BeginPlot() and EndPlot()! " ) ;
ImVec4 col = gp . Colormap [ gp . CurrentPlot - > ColormapIdx % gp . ColormapSize ] ;
gp . CurrentPlot - > ColormapIdx + + ;
return col ;
}
void ShowColormapScale ( double scale_min , double scale_max , float height ) {
ImPlotContext & gp = * GImPlot ;
2020-08-25 22:59:43 -04:00
static ImPlotTickCollection ticks ;
ticks . Reset ( ) ;
2020-08-21 23:13:11 -04:00
ImPlotRange range ;
range . Min = scale_min ;
range . Max = scale_max ;
2020-09-01 00:23:48 -04:00
2020-08-22 23:55:37 -04:00
AddTicksDefault ( range , 10 , 0 , ticks ) ;
2020-08-25 22:59:43 -04:00
2020-08-21 23:13:11 -04:00
ImGuiContext & G = * GImGui ;
ImGuiWindow * Window = G . CurrentWindow ;
if ( Window - > SkipItems )
return ;
const float txt_off = 5 ;
const float bar_w = 20 ;
ImDrawList & DrawList = * Window - > DrawList ;
2020-08-25 22:59:43 -04:00
ImVec2 size ( bar_w + txt_off + ticks . MaxWidth + 2 * gp . Style . PlotPadding . x , height ) ;
2020-08-21 23:13:11 -04:00
ImRect bb_frame = ImRect ( Window - > DC . CursorPos , Window - > DC . CursorPos + size ) ;
ImGui : : ItemSize ( bb_frame ) ;
if ( ! ImGui : : ItemAdd ( bb_frame , 0 , & bb_frame ) )
return ;
2020-10-19 00:26:34 -04:00
ImGui : : RenderFrame ( bb_frame . Min , bb_frame . Max , GetStyleColorU32 ( ImPlotCol_FrameBg ) , true , G . Style . FrameRounding ) ;
2020-08-21 23:13:11 -04:00
ImRect bb_grad ( bb_frame . Min + gp . Style . PlotPadding , bb_frame . Min + ImVec2 ( bar_w + gp . Style . PlotPadding . x , height - gp . Style . PlotPadding . y ) ) ;
int num_cols = GetColormapSize ( ) ;
float h_step = ( height - 2 * gp . Style . PlotPadding . y ) / ( num_cols - 1 ) ;
for ( int i = 0 ; i < num_cols - 1 ; + + i ) {
ImRect rect ( bb_grad . Min . x , bb_grad . Min . y + h_step * i , bb_grad . Max . x , bb_grad . Min . y + h_step * ( i + 1 ) ) ;
ImU32 col1 = ImGui : : GetColorU32 ( GetColormapColor ( num_cols - 1 - i ) ) ;
ImU32 col2 = ImGui : : GetColorU32 ( GetColormapColor ( num_cols - 1 - ( i + 1 ) ) ) ;
DrawList . AddRectFilledMultiColor ( rect . Min , rect . Max , col1 , col1 , col2 , col2 ) ;
}
2020-09-01 00:23:48 -04:00
ImVec4 col_tik4 = ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
col_tik4 . w * = 0.25f ;
const ImU32 col_tick = ImGui : : GetColorU32 ( col_tik4 ) ;
2020-08-21 23:13:11 -04:00
ImGui : : PushClipRect ( bb_frame . Min , bb_frame . Max , true ) ;
for ( int i = 0 ; i < ticks . Size ; + + i ) {
2020-08-25 22:59:43 -04:00
float ypos = ImRemap ( ( float ) ticks . Ticks [ i ] . PlotPos , ( float ) range . Max , ( float ) range . Min , bb_grad . Min . y , bb_grad . Max . y ) ;
2020-08-21 23:13:11 -04:00
if ( ypos < bb_grad . Max . y - 2 & & ypos > bb_grad . Min . y + 2 )
2020-09-01 00:23:48 -04:00
DrawList . AddLine ( ImVec2 ( bb_grad . Max . x - 1 , ypos ) , ImVec2 ( bb_grad . Max . x - ( ticks . Ticks [ i ] . Major ? 10.0f : 5.0f ) , ypos ) , col_tick , 1.0f ) ;
2020-09-19 21:54:19 -04:00
DrawList . AddText ( ImVec2 ( bb_grad . Max . x - 1 , ypos ) + ImVec2 ( txt_off , - ticks . Ticks [ i ] . LabelSize . y * 0.5f ) , GetStyleColorU32 ( ImPlotCol_TitleText ) , ticks . GetText ( i ) ) ;
2020-08-21 23:13:11 -04:00
}
ImGui : : PopClipRect ( ) ;
2020-09-01 00:23:48 -04:00
DrawList . AddRect ( bb_grad . Min , bb_grad . Max , GetStyleColorU32 ( ImPlotCol_PlotBorder ) ) ;
2020-08-21 23:13:11 -04:00
}
2020-08-24 00:45:42 -04:00
//-----------------------------------------------------------------------------
// Style Editor etc.
//-----------------------------------------------------------------------------
static void HelpMarker ( const char * desc ) {
ImGui : : TextDisabled ( " (?) " ) ;
if ( ImGui : : IsItemHovered ( ) ) {
ImGui : : BeginTooltip ( ) ;
ImGui : : PushTextWrapPos ( ImGui : : GetFontSize ( ) * 35.0f ) ;
ImGui : : TextUnformatted ( desc ) ;
ImGui : : PopTextWrapPos ( ) ;
ImGui : : EndTooltip ( ) ;
}
}
2020-08-24 12:06:29 -04:00
bool ShowStyleSelector ( const char * label )
{
static int style_idx = - 1 ;
if ( ImGui : : Combo ( label , & style_idx , " Auto \0 Classic \0 Dark \0 Light \0 " ) )
{
switch ( style_idx )
{
case 0 : StyleColorsAuto ( ) ; break ;
case 1 : StyleColorsClassic ( ) ; break ;
case 2 : StyleColorsDark ( ) ; break ;
case 3 : StyleColorsLight ( ) ; break ;
}
return true ;
}
return false ;
}
2020-10-11 01:38:18 -04:00
bool ShowColormapSelector ( const char * label ) {
bool set = false ;
static const char * map = ImPlot : : GetColormapName ( ImPlotColormap_Default ) ;
if ( ImGui : : BeginCombo ( label , map ) ) {
for ( int i = 0 ; i < ImPlotColormap_COUNT ; + + i ) {
const char * name = GetColormapName ( i ) ;
if ( ImGui : : Selectable ( name , map = = name ) ) {
map = name ;
ImPlot : : SetColormap ( i ) ;
ImPlot : : BustItemCache ( ) ;
set = true ;
}
}
ImGui : : EndCombo ( ) ;
}
return set ;
}
2020-08-24 00:45:42 -04:00
void ShowStyleEditor ( ImPlotStyle * ref ) {
ImPlotContext & gp = * GImPlot ;
ImPlotStyle & style = GetStyle ( ) ;
static ImPlotStyle ref_saved_style ;
// Default to using internal storage as reference
static bool init = true ;
if ( init & & ref = = NULL )
ref_saved_style = style ;
init = false ;
if ( ref = = NULL )
ref = & ref_saved_style ;
2020-08-24 12:06:29 -04:00
if ( ImPlot : : ShowStyleSelector ( " Colors##Selector " ) )
ref_saved_style = style ;
2020-08-24 00:45:42 -04:00
// Save/Revert button
if ( ImGui : : Button ( " Save Ref " ) )
* ref = ref_saved_style = style ;
ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " Revert Ref " ) )
style = * ref ;
ImGui : : SameLine ( ) ;
HelpMarker ( " Save/Revert in local non-persistent storage. Default Colors definition are not affected. "
" Use \" Export \" below to save them somewhere. " ) ;
if ( ImGui : : BeginTabBar ( " ##StyleEditor " ) ) {
if ( ImGui : : BeginTabItem ( " Variables " ) ) {
ImGui : : Text ( " Item Styling " ) ;
ImGui : : SliderFloat ( " LineWeight " , & style . LineWeight , 0.0f , 5.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " MarkerSize " , & style . MarkerSize , 2.0f , 10.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " MarkerWeight " , & style . MarkerWeight , 0.0f , 5.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " FillAlpha " , & style . FillAlpha , 0.0f , 1.0f , " %.2f " ) ;
ImGui : : SliderFloat ( " ErrorBarSize " , & style . ErrorBarSize , 0.0f , 10.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " ErrorBarWeight " , & style . ErrorBarWeight , 0.0f , 5.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " DigitalBitHeight " , & style . DigitalBitHeight , 0.0f , 20.0f , " %.1f " ) ;
ImGui : : SliderFloat ( " DigitalBitGap " , & style . DigitalBitGap , 0.0f , 20.0f , " %.1f " ) ;
2020-08-25 23:47:03 -04:00
float indent = ImGui : : CalcItemWidth ( ) - ImGui : : GetFrameHeight ( ) ;
ImGui : : Indent ( ImGui : : CalcItemWidth ( ) - ImGui : : GetFrameHeight ( ) ) ;
ImGui : : Checkbox ( " AntiAliasedLines " , & style . AntiAliasedLines ) ;
ImGui : : Unindent ( indent ) ;
2020-08-24 00:45:42 -04:00
ImGui : : Text ( " Plot Styling " ) ;
ImGui : : SliderFloat ( " PlotBorderSize " , & style . PlotBorderSize , 0.0f , 2.0f , " %.0f " ) ;
ImGui : : SliderFloat ( " MinorAlpha " , & style . MinorAlpha , 0.0f , 1.0f , " %.2f " ) ;
ImGui : : SliderFloat2 ( " MajorTickLen " , ( float * ) & style . MajorTickLen , 0.0f , 20.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " MinorTickLen " , ( float * ) & style . MinorTickLen , 0.0f , 20.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " MajorTickSize " , ( float * ) & style . MajorTickSize , 0.0f , 2.0f , " %.1f " ) ;
ImGui : : SliderFloat2 ( " MinorTickSize " , ( float * ) & style . MinorTickSize , 0.0f , 2.0f , " %.1f " ) ;
ImGui : : SliderFloat2 ( " MajorGridSize " , ( float * ) & style . MajorGridSize , 0.0f , 2.0f , " %.1f " ) ;
ImGui : : SliderFloat2 ( " MinorGridSize " , ( float * ) & style . MinorGridSize , 0.0f , 2.0f , " %.1f " ) ;
2020-10-19 00:26:34 -04:00
ImGui : : SliderFloat2 ( " PlotDefaultSize " , ( float * ) & style . PlotDefaultSize , 0.0f , 1000 , " %.0f " ) ;
2020-09-19 21:54:19 -04:00
ImGui : : SliderFloat2 ( " PlotMinSize " , ( float * ) & style . PlotMinSize , 0.0f , 300 , " %.0f " ) ;
2020-08-24 00:45:42 -04:00
ImGui : : Text ( " Plot Padding " ) ;
ImGui : : SliderFloat2 ( " PlotPadding " , ( float * ) & style . PlotPadding , 0.0f , 20.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " LabelPadding " , ( float * ) & style . LabelPadding , 0.0f , 20.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " LegendPadding " , ( float * ) & style . LegendPadding , 0.0f , 20.0f , " %.0f " ) ;
2020-10-19 00:26:34 -04:00
ImGui : : SliderFloat2 ( " LegendInnerPadding " , ( float * ) & style . LegendInnerPadding , 0.0f , 10.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " LegendSpacing " , ( float * ) & style . LegendSpacing , 0.0f , 5.0f , " %.0f " ) ;
ImGui : : SliderFloat2 ( " MousePosPadding " , ( float * ) & style . MousePosPadding , 0.0f , 20.0f , " %.0f " ) ;
2020-09-19 21:54:19 -04:00
ImGui : : SliderFloat2 ( " AnnotationPadding " , ( float * ) & style . AnnotationPadding , 0.0f , 5.0f , " %.0f " ) ;
2020-12-04 00:27:38 -05:00
ImGui : : SliderFloat2 ( " FitPadding " , ( float * ) & style . FitPadding , 0 , 0.2f , " %.2f " ) ;
2020-08-24 00:45:42 -04:00
ImGui : : EndTabItem ( ) ;
}
if ( ImGui : : BeginTabItem ( " Colors " ) ) {
static int output_dest = 0 ;
static bool output_only_modified = false ;
if ( ImGui : : Button ( " Export " , ImVec2 ( 75 , 0 ) ) ) {
if ( output_dest = = 0 )
ImGui : : LogToClipboard ( ) ;
else
ImGui : : LogToTTY ( ) ;
ImGui : : LogText ( " ImVec4* colors = ImPlot::GetStyle().Colors; \n " ) ;
for ( int i = 0 ; i < ImPlotCol_COUNT ; i + + ) {
const ImVec4 & col = style . Colors [ i ] ;
const char * name = ImPlot : : GetStyleColorName ( i ) ;
if ( ! output_only_modified | | memcmp ( & col , & ref - > Colors [ i ] , sizeof ( ImVec4 ) ) ! = 0 ) {
if ( IsColorAuto ( i ) )
2020-08-30 18:12:36 -04:00
ImGui : : LogText ( " colors[ImPlotCol_%s]%*s= IMPLOT_AUTO_COL; \n " , name , 14 - ( int ) strlen ( name ) , " " ) ;
2020-08-24 00:45:42 -04:00
else
ImGui : : LogText ( " colors[ImPlotCol_%s]%*s= ImVec4(%.2ff, %.2ff, %.2ff, %.2ff); \n " ,
name , 14 - ( int ) strlen ( name ) , " " , col . x , col . y , col . z , col . w ) ;
}
}
ImGui : : LogFinish ( ) ;
}
ImGui : : SameLine ( ) ; ImGui : : SetNextItemWidth ( 120 ) ; ImGui : : Combo ( " ##output_type " , & output_dest , " To Clipboard \0 To TTY \0 " ) ;
ImGui : : SameLine ( ) ; ImGui : : Checkbox ( " Only Modified Colors " , & output_only_modified ) ;
static ImGuiTextFilter filter ;
filter . Draw ( " Filter colors " , ImGui : : GetFontSize ( ) * 16 ) ;
static ImGuiColorEditFlags alpha_flags = ImGuiColorEditFlags_AlphaPreviewHalf ;
if ( ImGui : : RadioButton ( " Opaque " , alpha_flags = = ImGuiColorEditFlags_None ) ) { alpha_flags = ImGuiColorEditFlags_None ; } ImGui : : SameLine ( ) ;
if ( ImGui : : RadioButton ( " Alpha " , alpha_flags = = ImGuiColorEditFlags_AlphaPreview ) ) { alpha_flags = ImGuiColorEditFlags_AlphaPreview ; } ImGui : : SameLine ( ) ;
if ( ImGui : : RadioButton ( " Both " , alpha_flags = = ImGuiColorEditFlags_AlphaPreviewHalf ) ) { alpha_flags = ImGuiColorEditFlags_AlphaPreviewHalf ; } ImGui : : SameLine ( ) ;
HelpMarker (
" In the color list: \n "
" Left-click on colored square to open color picker, \n "
" Right-click to open edit options menu. " ) ;
ImGui : : Separator ( ) ;
ImGui : : PushItemWidth ( - 160 ) ;
for ( int i = 0 ; i < ImPlotCol_COUNT ; i + + ) {
const char * name = ImPlot : : GetStyleColorName ( i ) ;
if ( ! filter . PassFilter ( name ) )
continue ;
ImGui : : PushID ( i ) ;
ImVec4 temp = GetStyleColorVec4 ( i ) ;
const bool is_auto = IsColorAuto ( i ) ;
if ( ! is_auto )
ImGui : : PushStyleVar ( ImGuiStyleVar_Alpha , 0.25f ) ;
if ( ImGui : : Button ( " Auto " ) ) {
if ( is_auto )
style . Colors [ i ] = temp ;
else
2020-08-30 18:12:36 -04:00
style . Colors [ i ] = IMPLOT_AUTO_COL ;
2020-08-24 00:45:42 -04:00
BustItemCache ( ) ;
}
if ( ! is_auto )
ImGui : : PopStyleVar ( ) ;
ImGui : : SameLine ( ) ;
if ( ImGui : : ColorEdit4 ( name , & temp . x , ImGuiColorEditFlags_NoInputs | alpha_flags ) ) {
style . Colors [ i ] = temp ;
BustItemCache ( ) ;
}
if ( memcmp ( & style . Colors [ i ] , & ref - > Colors [ i ] , sizeof ( ImVec4 ) ) ! = 0 ) {
ImGui : : SameLine ( 175 ) ; if ( ImGui : : Button ( " Save " ) ) { ref - > Colors [ i ] = style . Colors [ i ] ; }
ImGui : : SameLine ( ) ; if ( ImGui : : Button ( " Revert " ) ) {
style . Colors [ i ] = ref - > Colors [ i ] ;
BustItemCache ( ) ;
}
}
ImGui : : PopID ( ) ;
}
ImGui : : PopItemWidth ( ) ;
ImGui : : Separator ( ) ;
2020-08-30 18:12:36 -04:00
ImGui : : Text ( " Colors that are set to Auto (i.e. IMPLOT_AUTO_COL) will \n "
2020-08-24 00:45:42 -04:00
" be automatically deduced from your ImGui style or the \n "
" current ImPlot Colormap. If you want to style individual \n "
" plot items, use Push/PopStyleColor around its function. " ) ;
ImGui : : EndTabItem ( ) ;
}
if ( ImGui : : BeginTabItem ( " Colormaps " ) ) {
static int output_dest = 0 ;
if ( ImGui : : Button ( " Export " , ImVec2 ( 75 , 0 ) ) ) {
if ( output_dest = = 0 )
ImGui : : LogToClipboard ( ) ;
else
ImGui : : LogToTTY ( ) ;
ImGui : : LogText ( " static const ImVec4 colormap[%d] = { \n " , gp . ColormapSize ) ;
for ( int i = 0 ; i < gp . ColormapSize ; + + i ) {
const ImVec4 & col = gp . Colormap [ i ] ;
ImGui : : LogText ( " ImVec4(%.2ff, %.2ff, %.2ff, %.2ff)%s \n " , col . x , col . y , col . z , col . w , i = = gp . ColormapSize - 1 ? " " : " , " ) ;
}
ImGui : : LogText ( " }; " ) ;
ImGui : : LogFinish ( ) ;
}
ImGui : : SameLine ( ) ; ImGui : : SetNextItemWidth ( 120 ) ; ImGui : : Combo ( " ##output_type " , & output_dest , " To Clipboard \0 To TTY \0 " ) ;
ImGui : : SameLine ( ) ; HelpMarker ( " Export code for selected Colormap \n (built in or custom). " ) ;
ImGui : : Separator ( ) ;
static ImVector < ImVec4 > custom ;
static bool custom_set = false ;
for ( int i = 0 ; i < ImPlotColormap_COUNT ; + + i ) {
ImGui : : PushID ( i ) ;
int size ;
const ImVec4 * cmap = GetColormap ( i , & size ) ;
bool selected = cmap = = gp . Colormap ;
2020-08-24 12:48:00 -04:00
if ( selected ) {
custom_set = false ;
}
2020-08-24 00:45:42 -04:00
if ( ! selected )
ImGui : : PushStyleVar ( ImGuiStyleVar_Alpha , 0.25f ) ;
if ( ImGui : : Button ( GetColormapName ( i ) , ImVec2 ( 75 , 0 ) ) ) {
SetColormap ( i ) ;
2020-10-11 01:38:18 -04:00
BustItemCache ( ) ;
2020-08-24 00:45:42 -04:00
custom_set = false ;
}
if ( ! selected )
ImGui : : PopStyleVar ( ) ;
ImGui : : SameLine ( ) ;
for ( int c = 0 ; c < size ; + + c ) {
ImGui : : PushID ( c ) ;
ImGui : : ColorButton ( " " , cmap [ c ] ) ;
if ( c ! = size - 1 )
ImGui : : SameLine ( ) ;
ImGui : : PopID ( ) ;
}
ImGui : : PopID ( ) ;
}
if ( custom . Size = = 0 ) {
custom . push_back ( ImVec4 ( 1 , 1 , 1 , 1 ) ) ;
custom . push_back ( ImVec4 ( 0.5f , 0.5f , 0.5f , 1 ) ) ;
}
ImGui : : Separator ( ) ;
ImGui : : BeginGroup ( ) ;
bool custom_set_now = custom_set ;
if ( ! custom_set_now )
ImGui : : PushStyleVar ( ImGuiStyleVar_Alpha , 0.25f ) ;
if ( ImGui : : Button ( " Custom " , ImVec2 ( 75 , 0 ) ) ) {
SetColormap ( & custom [ 0 ] , custom . Size ) ;
2020-10-11 01:38:18 -04:00
BustItemCache ( ) ;
2020-08-24 00:45:42 -04:00
custom_set = true ;
}
if ( ! custom_set_now )
ImGui : : PopStyleVar ( ) ;
if ( ImGui : : Button ( " + " , ImVec2 ( ( 75 - ImGui : : GetStyle ( ) . ItemSpacing . x ) / 2 , 0 ) ) ) {
custom . push_back ( ImVec4 ( 0 , 0 , 0 , 1 ) ) ;
if ( custom_set ) {
SetColormap ( & custom [ 0 ] , custom . Size ) ;
2020-10-11 01:38:18 -04:00
BustItemCache ( ) ;
2020-08-24 00:45:42 -04:00
}
}
ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( " - " , ImVec2 ( ( 75 - ImGui : : GetStyle ( ) . ItemSpacing . x ) / 2 , 0 ) ) & & custom . Size > 1 ) {
custom . pop_back ( ) ;
if ( custom_set ) {
SetColormap ( & custom [ 0 ] , custom . Size ) ;
2020-10-11 01:38:18 -04:00
BustItemCache ( ) ;
2020-08-24 00:45:42 -04:00
}
}
ImGui : : EndGroup ( ) ;
ImGui : : SameLine ( ) ;
ImGui : : BeginGroup ( ) ;
for ( int c = 0 ; c < custom . Size ; + + c ) {
ImGui : : PushID ( c ) ;
if ( ImGui : : ColorEdit4 ( " ##Col1 " , & custom [ c ] . x , ImGuiColorEditFlags_NoInputs ) & & custom_set ) {
SetColormap ( & custom [ 0 ] , custom . Size ) ;
2020-10-11 01:38:18 -04:00
BustItemCache ( ) ;
2020-08-24 00:45:42 -04:00
}
if ( ( c + 1 ) % 12 ! = 0 )
ImGui : : SameLine ( ) ;
ImGui : : PopID ( ) ;
}
ImGui : : EndGroup ( ) ;
ImGui : : EndTabItem ( ) ;
}
ImGui : : EndTabBar ( ) ;
}
}
void ShowUserGuide ( ) {
ImGui : : BulletText ( " Left click and drag within the plot area to pan X and Y axes. " ) ;
ImGui : : Indent ( ) ;
ImGui : : BulletText ( " Left click and drag on an axis to pan an individual axis. " ) ;
ImGui : : Unindent ( ) ;
ImGui : : BulletText ( " Scroll in the plot area to zoom both X any Y axes. " ) ;
ImGui : : Indent ( ) ;
ImGui : : BulletText ( " Scroll on an axis to zoom an individual axis. " ) ;
ImGui : : Unindent ( ) ;
ImGui : : BulletText ( " Right click and drag to box select data. " ) ;
ImGui : : Indent ( ) ;
ImGui : : BulletText ( " Hold Alt to expand box selection horizontally. " ) ;
ImGui : : BulletText ( " Hold Shift to expand box selection vertically. " ) ;
ImGui : : BulletText ( " Left click while box selecting to cancel the selection. " ) ;
ImGui : : Unindent ( ) ;
ImGui : : BulletText ( " Double left click to fit all visible data. " ) ;
ImGui : : Indent ( ) ;
ImGui : : BulletText ( " Double left click on an axis to fit the individual axis. " ) ;
ImGui : : Unindent ( ) ;
ImGui : : BulletText ( " Double right click to open the full plot context menu. " ) ;
ImGui : : Indent ( ) ;
ImGui : : BulletText ( " Double right click on an axis to open the axis context menu. " ) ;
ImGui : : Unindent ( ) ;
ImGui : : BulletText ( " Click legend label icons to show/hide plot items. " ) ;
}
2020-11-15 22:47:06 -05:00
void ShowAxisMetrics ( ImPlotAxis * axis , bool show_axis_rects ) {
ImGui : : Bullet ( ) ; ImGui : : Text ( " Flags: %d " , axis - > Flags ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Range: [%f,%f] " , axis - > Range . Min , axis - > Range . Max ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Pixels: %f " , axis - > Pixels ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Aspect: %f " , axis - > GetAspect ( ) ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Dragging: %s " , axis - > Dragging ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " ExtHovered: %s " , axis - > ExtHovered ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " AllHovered: %s " , axis - > AllHovered ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Present: %s " , axis - > Present ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " HasRange: %s " , axis - > HasRange ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " LinkedMin: %p " , axis - > LinkedMin ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " LinkedMax: %p " , axis - > LinkedMax ) ;
if ( show_axis_rects ) {
ImDrawList & fg = * ImGui : : GetForegroundDrawList ( ) ;
fg . AddRect ( axis - > HoverRect . Min , axis - > HoverRect . Max , IM_COL32 ( 0 , 255 , 0 , 255 ) ) ;
}
}
2020-10-21 11:08:41 -04:00
void ShowMetricsWindow ( bool * p_popen ) {
2020-11-15 22:47:06 -05:00
static bool show_plot_rects = false ;
static bool show_axes_rects = false ;
ImDrawList & fg = * ImGui : : GetForegroundDrawList ( ) ;
2020-10-21 11:08:41 -04:00
ImPlotContext & gp = * GImPlot ;
// ImGuiContext& g = *GImGui;
ImGuiIO & io = ImGui : : GetIO ( ) ;
ImGui : : Begin ( " ImPlot Metrics " , p_popen ) ;
ImGui : : Text ( " ImPlot " IMPLOT_VERSION ) ;
ImGui : : Text ( " Application average %.3f ms/frame (%.1f FPS) " , 1000.0f / io . Framerate , io . Framerate ) ;
ImGui : : Separator ( ) ;
int n_plots = gp . Plots . GetSize ( ) ;
2020-11-15 22:47:06 -05:00
if ( ImGui : : TreeNode ( " Tools " ) ) {
ImGui : : Checkbox ( " Show Plot Rects " , & show_plot_rects ) ;
ImGui : : Checkbox ( " Show Axes Rects " , & show_axes_rects ) ;
ImGui : : TreePop ( ) ;
}
2020-10-21 11:08:41 -04:00
if ( ImGui : : TreeNode ( " Plots " , " Plots (%d) " , n_plots ) ) {
for ( int p = 0 ; p < n_plots ; + + p ) {
2020-11-15 22:47:06 -05:00
// plot
2020-10-21 11:08:41 -04:00
ImPlotPlot * plot = gp . Plots . GetByIndex ( p ) ;
ImGui : : PushID ( p ) ;
if ( ImGui : : TreeNode ( " Plot " , " Plot [ID=%u] " , plot - > ID ) ) {
int n_items = plot - > Items . GetSize ( ) ;
if ( ImGui : : TreeNode ( " Items " , " Items (%d) " , n_items ) ) {
for ( int i = 0 ; i < n_items ; + + i ) {
ImPlotItem * item = plot - > Items . GetByIndex ( i ) ;
ImGui : : PushID ( i ) ;
if ( ImGui : : TreeNode ( " Item " , " Item [ID=%u] " , item - > ID ) ) {
ImGui : : Bullet ( ) ; ImGui : : Checkbox ( " Show " , & item - > Show ) ;
ImGui : : Bullet ( ) ; ImGui : : ColorEdit4 ( " Color " , & item - > Color . x , ImGuiColorEditFlags_NoInputs ) ;
2020-11-15 22:47:06 -05:00
ImGui : : Bullet ( ) ; ImGui : : Text ( " NameOffset: %d " , item - > NameOffset ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Name: %s " , item - > NameOffset ! = - 1 ? plot - > LegendData . Labels . Buf . Data + item - > NameOffset : " N/A " ) ;
ImGui : : Bullet ( ) ; ImGui : : Value ( " Hovered: %s " , item - > LegendHovered ? " true " : " false " ) ;
2020-10-21 11:08:41 -04:00
ImGui : : TreePop ( ) ;
}
ImGui : : PopID ( ) ;
}
ImGui : : TreePop ( ) ;
}
2020-11-15 22:47:06 -05:00
if ( ImGui : : TreeNode ( " X-Axis " ) ) {
ShowAxisMetrics ( & plot - > XAxis , show_axes_rects ) ;
ImGui : : TreePop ( ) ;
}
if ( ImGui : : TreeNode ( " Y-Axis " ) ) {
ShowAxisMetrics ( & plot - > YAxis [ 0 ] , show_axes_rects ) ;
ImGui : : TreePop ( ) ;
}
if ( ImHasFlag ( plot - > Flags , ImPlotFlags_YAxis2 ) & & ImGui : : TreeNode ( " Y-Axis 2 " ) ) {
ShowAxisMetrics ( & plot - > YAxis [ 1 ] , show_axes_rects ) ;
ImGui : : TreePop ( ) ;
}
if ( ImHasFlag ( plot - > Flags , ImPlotFlags_YAxis3 ) & & ImGui : : TreeNode ( " Y-Axis 3 " ) ) {
ShowAxisMetrics ( & plot - > YAxis [ 2 ] , show_axes_rects ) ;
ImGui : : TreePop ( ) ;
}
ImGui : : Bullet ( ) ; ImGui : : Text ( " Flags: %d " , plot - > Flags ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Selecting: %s " , plot - > Selecting ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Querying: %s " , plot - > Querying ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " Queried: %s " , plot - > Queried ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " FrameHovered: %s " , plot - > FrameHovered ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " PlotHovered: %s " , plot - > PlotHovered ? " true " : " false " ) ;
ImGui : : Bullet ( ) ; ImGui : : Text ( " LegendHovered: %s " , plot - > LegendHovered ? " true " : " false " ) ;
2020-10-21 11:08:41 -04:00
ImGui : : TreePop ( ) ;
2020-11-15 22:47:06 -05:00
if ( show_plot_rects )
fg . AddRect ( plot - > PlotRect . Min , plot - > PlotRect . Max , IM_COL32 ( 255 , 255 , 0 , 255 ) ) ;
2020-10-21 11:08:41 -04:00
}
ImGui : : PopID ( ) ;
}
ImGui : : TreePop ( ) ;
}
ImGui : : End ( ) ;
}
2020-09-09 00:47:02 -04:00
bool ShowDatePicker ( const char * id , int * level , ImPlotTime * t , const ImPlotTime * t1 , const ImPlotTime * t2 ) {
ImGui : : PushID ( id ) ;
ImGui : : BeginGroup ( ) ;
ImGui : : PushStyleColor ( ImGuiCol_Button , ImVec4 ( 0 , 0 , 0 , 0 ) ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_ItemSpacing , ImVec2 ( 0 , 0 ) ) ;
2020-09-08 01:56:00 -04:00
ImGuiStyle & style = ImGui : : GetStyle ( ) ;
2020-09-09 00:47:02 -04:00
ImVec4 col_txt = style . Colors [ ImGuiCol_Text ] ;
ImVec4 col_dis = style . Colors [ ImGuiCol_TextDisabled ] ;
const float ht = ImGui : : GetFrameHeight ( ) ;
ImVec2 cell_size ( ht * 1.25f , ht ) ;
char buff [ 32 ] ;
bool clk = false ;
2020-09-08 01:56:00 -04:00
tm & Tm = GImPlot - > Tm ;
2020-09-09 00:47:02 -04:00
const int min_yr = 1970 ;
const int max_yr = 2999 ;
// t1 parts
int t1_mo = 0 ; int t1_md = 0 ; int t1_yr = 0 ;
if ( t1 ! = NULL ) {
GetTime ( * t1 , & Tm ) ;
t1_mo = Tm . tm_mon ;
t1_md = Tm . tm_mday ;
t1_yr = Tm . tm_year + 1900 ;
}
// t2 parts
int t2_mo = 0 ; int t2_md = 0 ; int t2_yr = 0 ;
if ( t2 ! = NULL ) {
GetTime ( * t2 , & Tm ) ;
t2_mo = Tm . tm_mon ;
t2_md = Tm . tm_mday ;
t2_yr = Tm . tm_year + 1900 ;
}
// day widget
if ( * level = = 0 ) {
* t = FloorTime ( * t , ImPlotTimeUnit_Day ) ;
GetTime ( * t , & Tm ) ;
const int this_year = Tm . tm_year + 1900 ;
const int last_year = this_year - 1 ;
const int next_year = this_year + 1 ;
const int this_mon = Tm . tm_mon ;
const int last_mon = this_mon = = 0 ? 11 : this_mon - 1 ;
const int next_mon = this_mon = = 11 ? 0 : this_mon + 1 ;
const int days_this_mo = GetDaysInMonth ( this_year , this_mon ) ;
const int days_last_mo = GetDaysInMonth ( this_mon = = 0 ? last_year : this_year , last_mon ) ;
ImPlotTime t_first_mo = FloorTime ( * t , ImPlotTimeUnit_Mo ) ;
GetTime ( t_first_mo , & Tm ) ;
const int first_wd = Tm . tm_wday ;
// month year
2020-09-12 11:25:47 -04:00
snprintf ( buff , 32 , " %s %d " , MONTH_NAMES [ this_mon ] , this_year ) ;
2020-09-09 00:47:02 -04:00
if ( ImGui : : Button ( buff ) )
* level = 1 ;
ImGui : : SameLine ( 5 * cell_size . x ) ;
BeginDisabledControls ( this_year < = min_yr & & this_mon = = 0 ) ;
if ( ImGui : : ArrowButtonEx ( " ##Up " , ImGuiDir_Up , cell_size ) )
* t = AddTime ( * t , ImPlotTimeUnit_Mo , - 1 ) ;
EndDisabledControls ( this_year < = min_yr & & this_mon = = 0 ) ;
ImGui : : SameLine ( ) ;
BeginDisabledControls ( this_year > = max_yr & & this_mon = = 11 ) ;
if ( ImGui : : ArrowButtonEx ( " ##Down " , ImGuiDir_Down , cell_size ) )
* t = AddTime ( * t , ImPlotTimeUnit_Mo , 1 ) ;
EndDisabledControls ( this_year > = max_yr & & this_mon = = 11 ) ;
// render weekday abbreviations
ImGui : : PushItemFlag ( ImGuiItemFlags_Disabled , true ) ;
for ( int i = 0 ; i < 7 ; + + i ) {
2020-09-12 11:25:47 -04:00
ImGui : : Button ( WD_ABRVS [ i ] , cell_size ) ;
2020-09-09 00:47:02 -04:00
if ( i ! = 6 ) { ImGui : : SameLine ( ) ; }
}
ImGui : : PopItemFlag ( ) ;
// 0 = last mo, 1 = this mo, 2 = next mo
int mo = first_wd > 0 ? 0 : 1 ;
int day = mo = = 1 ? 1 : days_last_mo - first_wd + 1 ;
for ( int i = 0 ; i < 6 ; + + i ) {
for ( int j = 0 ; j < 7 ; + + j ) {
if ( mo = = 0 & & day > days_last_mo ) {
mo = 1 ; day = 1 ;
}
else if ( mo = = 1 & & day > days_this_mo ) {
mo = 2 ; day = 1 ;
}
const int now_yr = ( mo = = 0 & & this_mon = = 0 ) ? last_year : ( ( mo = = 2 & & this_mon = = 11 ) ? next_year : this_year ) ;
const int now_mo = mo = = 0 ? last_mon : ( mo = = 1 ? this_mon : next_mon ) ;
const int now_md = day ;
const bool off_mo = mo = = 0 | | mo = = 2 ;
const bool t1_or_t2 = ( t1 ! = NULL & & t1_mo = = now_mo & & t1_yr = = now_yr & & t1_md = = now_md ) | |
( t2 ! = NULL & & t2_mo = = now_mo & & t2_yr = = now_yr & & t2_md = = now_md ) ;
if ( off_mo )
ImGui : : PushStyleColor ( ImGuiCol_Text , col_dis ) ;
if ( t1_or_t2 ) {
ImGui : : PushStyleColor ( ImGuiCol_Button , col_dis ) ;
ImGui : : PushStyleColor ( ImGuiCol_Text , col_txt ) ;
}
ImGui : : PushID ( i * 7 + j ) ;
snprintf ( buff , 32 , " %d " , day ) ;
if ( now_yr = = min_yr - 1 | | now_yr = = max_yr + 1 ) {
ImGui : : Dummy ( cell_size ) ;
}
else if ( ImGui : : Button ( buff , cell_size ) & & ! clk ) {
* t = MakeTime ( now_yr , now_mo , now_md ) ;
clk = true ;
}
ImGui : : PopID ( ) ;
if ( t1_or_t2 )
ImGui : : PopStyleColor ( 2 ) ;
if ( off_mo )
ImGui : : PopStyleColor ( ) ;
if ( j ! = 6 )
ImGui : : SameLine ( ) ;
day + + ;
}
}
}
// month widget
else if ( * level = = 1 ) {
* t = FloorTime ( * t , ImPlotTimeUnit_Mo ) ;
GetTime ( * t , & Tm ) ;
int this_yr = Tm . tm_year + 1900 ;
snprintf ( buff , 32 , " %d " , this_yr ) ;
if ( ImGui : : Button ( buff ) )
* level = 2 ;
BeginDisabledControls ( this_yr < = min_yr ) ;
ImGui : : SameLine ( 5 * cell_size . x ) ;
if ( ImGui : : ArrowButtonEx ( " ##Up " , ImGuiDir_Up , cell_size ) )
* t = AddTime ( * t , ImPlotTimeUnit_Yr , - 1 ) ;
EndDisabledControls ( this_yr < = min_yr ) ;
ImGui : : SameLine ( ) ;
BeginDisabledControls ( this_yr > = max_yr ) ;
if ( ImGui : : ArrowButtonEx ( " ##Down " , ImGuiDir_Down , cell_size ) )
* t = AddTime ( * t , ImPlotTimeUnit_Yr , 1 ) ;
EndDisabledControls ( this_yr > = max_yr ) ;
// ImGui::Dummy(cell_size);
cell_size . x * = 7.0f / 4.0f ;
cell_size . y * = 7.0f / 3.0f ;
int mo = 0 ;
for ( int i = 0 ; i < 3 ; + + i ) {
for ( int j = 0 ; j < 4 ; + + j ) {
const bool t1_or_t2 = ( t1 ! = NULL & & t1_yr = = this_yr & & t1_mo = = mo ) | |
( t2 ! = NULL & & t2_yr = = this_yr & & t2_mo = = mo ) ;
if ( t1_or_t2 )
ImGui : : PushStyleColor ( ImGuiCol_Button , col_dis ) ;
2020-09-12 11:25:47 -04:00
if ( ImGui : : Button ( MONTH_ABRVS [ mo ] , cell_size ) & & ! clk ) {
2020-09-09 00:47:02 -04:00
* t = MakeTime ( this_yr , mo ) ;
* level = 0 ;
}
if ( t1_or_t2 )
ImGui : : PopStyleColor ( ) ;
if ( j ! = 3 )
ImGui : : SameLine ( ) ;
mo + + ;
}
}
}
else if ( * level = = 2 ) {
* t = FloorTime ( * t , ImPlotTimeUnit_Yr ) ;
int this_yr = GetYear ( * t ) ;
int yr = this_yr - this_yr % 20 ;
ImGui : : PushItemFlag ( ImGuiItemFlags_Disabled , true ) ;
snprintf ( buff , 32 , " %d-%d " , yr , yr + 19 ) ;
ImGui : : Button ( buff ) ;
ImGui : : PopItemFlag ( ) ;
ImGui : : SameLine ( 5 * cell_size . x ) ;
BeginDisabledControls ( yr < = min_yr ) ;
if ( ImGui : : ArrowButtonEx ( " ##Up " , ImGuiDir_Up , cell_size ) )
2020-09-09 20:17:19 -04:00
* t = MakeTime ( yr - 20 ) ;
2020-09-09 00:47:02 -04:00
EndDisabledControls ( yr < = min_yr ) ;
ImGui : : SameLine ( ) ;
BeginDisabledControls ( yr + 20 > = max_yr ) ;
if ( ImGui : : ArrowButtonEx ( " ##Down " , ImGuiDir_Down , cell_size ) )
2020-09-09 20:17:19 -04:00
* t = MakeTime ( yr + 20 ) ;
2020-09-09 00:47:02 -04:00
EndDisabledControls ( yr + 20 > = max_yr ) ;
// ImGui::Dummy(cell_size);
cell_size . x * = 7.0f / 4.0f ;
cell_size . y * = 7.0f / 5.0f ;
for ( int i = 0 ; i < 5 ; + + i ) {
for ( int j = 0 ; j < 4 ; + + j ) {
const bool t1_or_t2 = ( t1 ! = NULL & & t1_yr = = yr ) | | ( t2 ! = NULL & & t2_yr = = yr ) ;
if ( t1_or_t2 )
ImGui : : PushStyleColor ( ImGuiCol_Button , col_dis ) ;
snprintf ( buff , 32 , " %d " , yr ) ;
if ( yr < 1970 | | yr > 3000 ) {
ImGui : : Dummy ( cell_size ) ;
}
else if ( ImGui : : Button ( buff , cell_size ) ) {
2020-09-09 20:17:19 -04:00
* t = MakeTime ( yr ) ;
2020-09-09 00:47:02 -04:00
* level = 1 ;
}
if ( t1_or_t2 )
ImGui : : PopStyleColor ( ) ;
if ( j ! = 3 )
ImGui : : SameLine ( ) ;
yr + + ;
}
}
}
ImGui : : PopStyleVar ( ) ;
ImGui : : PopStyleColor ( ) ;
ImGui : : EndGroup ( ) ;
ImGui : : PopID ( ) ;
return clk ;
}
2020-09-08 01:56:00 -04:00
2020-10-14 23:07:27 -04:00
bool ShowTimePicker ( const char * id , ImPlotTime * t ) {
2020-09-09 00:47:02 -04:00
ImGui : : PushID ( id ) ;
tm & Tm = GImPlot - > Tm ;
2020-09-08 01:56:00 -04:00
GetTime ( * t , & Tm ) ;
2020-09-09 00:47:02 -04:00
static const char * nums [ ] = { " 00 " , " 01 " , " 02 " , " 03 " , " 04 " , " 05 " , " 06 " , " 07 " , " 08 " , " 09 " ,
" 10 " , " 11 " , " 12 " , " 13 " , " 14 " , " 15 " , " 16 " , " 17 " , " 18 " , " 19 " ,
" 20 " , " 21 " , " 22 " , " 23 " , " 24 " , " 25 " , " 26 " , " 27 " , " 28 " , " 29 " ,
" 30 " , " 31 " , " 32 " , " 33 " , " 34 " , " 35 " , " 36 " , " 37 " , " 38 " , " 39 " ,
" 40 " , " 41 " , " 42 " , " 43 " , " 44 " , " 45 " , " 46 " , " 47 " , " 48 " , " 49 " ,
" 50 " , " 51 " , " 52 " , " 53 " , " 54 " , " 55 " , " 56 " , " 57 " , " 58 " , " 59 " } ;
2020-09-08 01:56:00 -04:00
2020-09-09 00:47:02 -04:00
static const char * am_pm [ ] = { " am " , " pm " } ;
2020-09-08 01:56:00 -04:00
2020-10-14 23:07:27 -04:00
bool hour24 = GImPlot - > Style . Use24HourClock ;
2020-09-12 11:25:47 -04:00
int hr = hour24 ? Tm . tm_hour : ( ( Tm . tm_hour = = 0 | | Tm . tm_hour = = 12 ) ? 12 : Tm . tm_hour % 12 ) ;
2020-09-09 00:47:02 -04:00
int min = Tm . tm_min ;
int sec = Tm . tm_sec ;
int ap = Tm . tm_hour < 12 ? 0 : 1 ;
bool changed = false ;
2020-09-08 01:56:00 -04:00
2020-09-09 00:47:02 -04:00
ImVec2 spacing = ImGui : : GetStyle ( ) . ItemSpacing ;
spacing . x = 0 ;
float width = ImGui : : CalcTextSize ( " 888 " ) . x ;
float height = ImGui : : GetFrameHeight ( ) ;
2020-09-08 01:56:00 -04:00
2020-09-09 00:47:02 -04:00
ImGui : : PushStyleVar ( ImGuiStyleVar_ItemSpacing , spacing ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_ScrollbarSize , 2.0f ) ;
ImGui : : PushStyleColor ( ImGuiCol_FrameBg , ImVec4 ( 0 , 0 , 0 , 0 ) ) ;
2020-09-08 01:56:00 -04:00
ImGui : : PushStyleColor ( ImGuiCol_Button , ImVec4 ( 0 , 0 , 0 , 0 ) ) ;
2020-09-09 00:47:02 -04:00
ImGui : : PushStyleColor ( ImGuiCol_FrameBgHovered , ImGui : : GetStyleColorVec4 ( ImGuiCol_ButtonHovered ) ) ;
ImGui : : SetNextItemWidth ( width ) ;
if ( ImGui : : BeginCombo ( " ##hr " , nums [ hr ] , ImGuiComboFlags_NoArrowButton ) ) {
2020-09-12 11:25:47 -04:00
const int ia = hour24 ? 0 : 1 ;
const int ib = hour24 ? 24 : 13 ;
for ( int i = ia ; i < ib ; + + i ) {
2020-09-09 00:47:02 -04:00
if ( ImGui : : Selectable ( nums [ i ] , i = = hr ) ) {
hr = i ;
changed = true ;
}
}
ImGui : : EndCombo ( ) ;
2020-09-08 01:56:00 -04:00
}
ImGui : : SameLine ( ) ;
2020-09-09 00:47:02 -04:00
ImGui : : Text ( " : " ) ;
ImGui : : SameLine ( ) ;
ImGui : : SetNextItemWidth ( width ) ;
if ( ImGui : : BeginCombo ( " ##min " , nums [ min ] , ImGuiComboFlags_NoArrowButton ) ) {
for ( int i = 0 ; i < 60 ; + + i ) {
if ( ImGui : : Selectable ( nums [ i ] , i = = min ) ) {
min = i ;
changed = true ;
2020-09-08 01:56:00 -04:00
}
2020-09-09 00:47:02 -04:00
}
ImGui : : EndCombo ( ) ;
}
ImGui : : SameLine ( ) ;
ImGui : : Text ( " : " ) ;
ImGui : : SameLine ( ) ;
ImGui : : SetNextItemWidth ( width ) ;
if ( ImGui : : BeginCombo ( " ##sec " , nums [ sec ] , ImGuiComboFlags_NoArrowButton ) ) {
for ( int i = 0 ; i < 60 ; + + i ) {
if ( ImGui : : Selectable ( nums [ i ] , i = = sec ) ) {
sec = i ;
changed = true ;
2020-09-08 01:56:00 -04:00
}
}
2020-09-09 00:47:02 -04:00
ImGui : : EndCombo ( ) ;
}
2020-09-12 11:25:47 -04:00
if ( ! hour24 ) {
ImGui : : SameLine ( ) ;
if ( ImGui : : Button ( am_pm [ ap ] , ImVec2 ( height , height ) ) ) {
ap = 1 - ap ;
changed = true ;
}
2020-09-08 01:56:00 -04:00
}
2020-09-09 00:47:02 -04:00
ImGui : : PopStyleColor ( 3 ) ;
ImGui : : PopStyleVar ( 2 ) ;
ImGui : : PopID ( ) ;
2020-09-08 01:56:00 -04:00
2020-09-09 00:47:02 -04:00
if ( changed ) {
2020-09-12 11:25:47 -04:00
if ( ! hour24 )
hr = hr % 12 + ap * 12 ;
2020-09-09 00:47:02 -04:00
Tm . tm_hour = hr ;
Tm . tm_min = min ;
Tm . tm_sec = sec ;
* t = MkTime ( & Tm ) ;
}
return changed ;
2020-09-08 01:56:00 -04:00
}
2020-08-24 12:06:29 -04:00
void StyleColorsAuto ( ImPlotStyle * dst ) {
ImPlotStyle * style = dst ? dst : & ImPlot : : GetStyle ( ) ;
ImVec4 * colors = style - > Colors ;
2020-08-24 12:06:43 -04:00
style - > MinorAlpha = 0.25f ;
2020-08-24 12:06:29 -04:00
2020-08-30 18:12:36 -04:00
colors [ ImPlotCol_Line ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Fill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerOutline ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerFill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_ErrorBar ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_FrameBg ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_PlotBg ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_PlotBorder ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_LegendBg ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_LegendBorder ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_LegendText ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_TitleText ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_InlayText ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_PlotBorder ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_XAxis ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_XAxisGrid ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxis ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxisGrid ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxis2 ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxisGrid2 ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxis3 ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_YAxisGrid3 ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Selection ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Query ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Crosshairs ] = IMPLOT_AUTO_COL ;
2020-08-24 12:06:29 -04:00
}
void StyleColorsClassic ( ImPlotStyle * dst ) {
ImPlotStyle * style = dst ? dst : & ImPlot : : GetStyle ( ) ;
ImVec4 * colors = style - > Colors ;
2020-08-24 12:06:43 -04:00
style - > MinorAlpha = 0.5f ;
2020-08-24 12:06:29 -04:00
2020-08-30 18:12:36 -04:00
colors [ ImPlotCol_Line ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Fill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerOutline ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerFill ] = IMPLOT_AUTO_COL ;
2020-08-24 12:06:29 -04:00
colors [ ImPlotCol_ErrorBar ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_FrameBg ] = ImVec4 ( 0.43f , 0.43f , 0.43f , 0.39f ) ;
colors [ ImPlotCol_PlotBg ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.35f ) ;
colors [ ImPlotCol_PlotBorder ] = ImVec4 ( 0.50f , 0.50f , 0.50f , 0.50f ) ;
colors [ ImPlotCol_LegendBg ] = ImVec4 ( 0.11f , 0.11f , 0.14f , 0.92f ) ;
colors [ ImPlotCol_LegendBorder ] = ImVec4 ( 0.50f , 0.50f , 0.50f , 0.50f ) ;
colors [ ImPlotCol_LegendText ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_TitleText ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_InlayText ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_XAxis ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_XAxisGrid ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 0.25f ) ;
colors [ ImPlotCol_YAxis ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 0.25f ) ;
colors [ ImPlotCol_YAxis2 ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid2 ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 0.25f ) ;
colors [ ImPlotCol_YAxis3 ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid3 ] = ImVec4 ( 0.90f , 0.90f , 0.90f , 0.25f ) ;
colors [ ImPlotCol_Selection ] = ImVec4 ( 0.97f , 0.97f , 0.39f , 1.00f ) ;
colors [ ImPlotCol_Query ] = ImVec4 ( 0.00f , 1.00f , 0.59f , 1.00f ) ;
colors [ ImPlotCol_Crosshairs ] = ImVec4 ( 0.50f , 0.50f , 0.50f , 0.75f ) ;
}
void StyleColorsDark ( ImPlotStyle * dst ) {
ImPlotStyle * style = dst ? dst : & ImPlot : : GetStyle ( ) ;
ImVec4 * colors = style - > Colors ;
2020-08-24 12:06:43 -04:00
style - > MinorAlpha = 0.25f ;
2020-08-24 12:06:29 -04:00
2020-08-30 18:12:36 -04:00
colors [ ImPlotCol_Line ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Fill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerOutline ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerFill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_ErrorBar ] = IMPLOT_AUTO_COL ;
2020-08-24 12:06:29 -04:00
colors [ ImPlotCol_FrameBg ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.07f ) ;
colors [ ImPlotCol_PlotBg ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.50f ) ;
colors [ ImPlotCol_PlotBorder ] = ImVec4 ( 0.43f , 0.43f , 0.50f , 0.50f ) ;
colors [ ImPlotCol_LegendBg ] = ImVec4 ( 0.08f , 0.08f , 0.08f , 0.94f ) ;
colors [ ImPlotCol_LegendBorder ] = ImVec4 ( 0.43f , 0.43f , 0.50f , 0.50f ) ;
colors [ ImPlotCol_LegendText ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_TitleText ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_InlayText ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_XAxis ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_XAxisGrid ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.25f ) ;
colors [ ImPlotCol_YAxis ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.25f ) ;
colors [ ImPlotCol_YAxis2 ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid2 ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.25f ) ;
colors [ ImPlotCol_YAxis3 ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid3 ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.25f ) ;
colors [ ImPlotCol_Selection ] = ImVec4 ( 1.00f , 0.60f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_Query ] = ImVec4 ( 0.00f , 1.00f , 0.44f , 1.00f ) ;
colors [ ImPlotCol_Crosshairs ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.50f ) ;
}
void StyleColorsLight ( ImPlotStyle * dst ) {
ImPlotStyle * style = dst ? dst : & ImPlot : : GetStyle ( ) ;
ImVec4 * colors = style - > Colors ;
2020-08-24 12:06:43 -04:00
style - > MinorAlpha = 1.0f ;
2020-08-30 18:12:36 -04:00
colors [ ImPlotCol_Line ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_Fill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerOutline ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_MarkerFill ] = IMPLOT_AUTO_COL ;
colors [ ImPlotCol_ErrorBar ] = IMPLOT_AUTO_COL ;
2020-08-24 12:06:29 -04:00
colors [ ImPlotCol_FrameBg ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_PlotBg ] = ImVec4 ( 0.42f , 0.57f , 1.00f , 0.13f ) ;
colors [ ImPlotCol_PlotBorder ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.00f ) ;
colors [ ImPlotCol_LegendBg ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 0.98f ) ;
colors [ ImPlotCol_LegendBorder ] = ImVec4 ( 0.82f , 0.82f , 0.82f , 0.80f ) ;
colors [ ImPlotCol_LegendText ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_TitleText ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_InlayText ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_XAxis ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_XAxisGrid ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_YAxis ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
colors [ ImPlotCol_YAxisGrid ] = ImVec4 ( 1.00f , 1.00f , 1.00f , 1.00f ) ;
colors [ ImPlotCol_YAxis2 ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
2020-08-24 14:59:57 -04:00
colors [ ImPlotCol_YAxisGrid2 ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.50f ) ;
2020-08-24 12:06:29 -04:00
colors [ ImPlotCol_YAxis3 ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 1.00f ) ;
2020-08-24 14:59:57 -04:00
colors [ ImPlotCol_YAxisGrid3 ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.50f ) ;
2020-08-24 12:06:29 -04:00
colors [ ImPlotCol_Selection ] = ImVec4 ( 0.82f , 0.64f , 0.03f , 1.00f ) ;
colors [ ImPlotCol_Query ] = ImVec4 ( 0.00f , 0.84f , 0.37f , 1.00f ) ;
colors [ ImPlotCol_Crosshairs ] = ImVec4 ( 0.00f , 0.00f , 0.00f , 0.50f ) ;
}
2020-08-24 00:45:42 -04:00
2020-05-29 17:31:50 -04:00
} // namespace ImPlot