1
0
Fork 0
mirror of https://github.com/gwm17/implot.git synced 2024-11-13 22:48:50 -05:00
implot/implot.cpp

2710 lines
113 KiB
C++
Raw Normal View History

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-05-11 07:12:22 -04:00
// ImPlot v0.2 WIP
/*
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/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`
and `PlotScatter` (however, `PlotLine` can still be used to plot scatter points as `Plot` did before.)
- 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-13 10:11:25 -04:00
- Functions that were prefixed or decorated with the word "Plot" have been truncated. E.g., `ImGui::PlotBar` 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-04-27 11:27:59 -04:00
#ifndef IMGUI_DEFINE_MATH_OPERATORS
#define IMGUI_DEFINE_MATH_OPERATORS
#endif
#include "implot.h"
#include "imgui_internal.h"
2020-05-11 07:12:22 -04:00
#ifdef _MSC_VER
#define sprintf sprintf_s
#endif
2020-04-27 11:27:59 -04:00
#define IM_NORMALIZE2F_OVER_ZERO(VX, VY) \
{ \
float d2 = VX * VX + VY * VY; \
if (d2 > 0.0f) { \
float inv_len = 1.0f / ImSqrt(d2); \
VX *= inv_len; \
VY *= inv_len; \
} \
}
// Special Color used to specific that a plot item color should set determined automatically.
#define IM_COL_AUTO ImVec4(0,0,0,-1)
2020-05-11 07:12:22 -04:00
// The maximum number of support y-axes
#define MAX_Y_AXES 3
2020-04-27 11:27:59 -04:00
ImPlotStyle::ImPlotStyle() {
LineWeight = 1;
Marker = ImPlotMarker_None;
MarkerSize = 4;
2020-04-27 11:27:59 -04:00
MarkerWeight = 1;
ErrorBarSize = 5;
ErrorBarWeight = 1.5;
DigitalBitHeight = 8;
2020-04-27 11:27:59 -04:00
Colors[ImPlotCol_Line] = IM_COL_AUTO;
Colors[ImPlotCol_Fill] = IM_COL_AUTO;
Colors[ImPlotCol_MarkerOutline] = IM_COL_AUTO;
Colors[ImPlotCol_MarkerFill] = IM_COL_AUTO;
Colors[ImPlotCol_ErrorBar] = IM_COL_AUTO;
Colors[ImPlotCol_FrameBg] = IM_COL_AUTO;
Colors[ImPlotCol_PlotBg] = IM_COL_AUTO;
Colors[ImPlotCol_PlotBorder] = IM_COL_AUTO;
Colors[ImPlotCol_XAxis] = IM_COL_AUTO;
Colors[ImPlotCol_YAxis] = IM_COL_AUTO;
2020-05-11 07:12:22 -04:00
Colors[ImPlotCol_YAxis2] = IM_COL_AUTO;
Colors[ImPlotCol_YAxis3] = IM_COL_AUTO;
2020-04-27 11:27:59 -04:00
Colors[ImPlotCol_Selection] = ImVec4(1,1,0,1);
2020-05-04 03:09:33 -04:00
Colors[ImPlotCol_Query] = ImVec4(0,1,0,1);
}
2020-05-11 07:12:22 -04:00
ImPlotRange::ImPlotRange() : Min(NAN), Max(NAN) {}
bool ImPlotRange::Contains(float v) const {
return v >= Min && v <= Max;
}
2020-05-11 07:12:22 -04:00
float ImPlotRange::Size() const {
return Max - Min;
}
ImPlotLimits::ImPlotLimits() {}
bool ImPlotLimits::Contains(const ImVec2& p) const {
return X.Contains(p.x) && Y.Contains(p.y);
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
ImVec2 ImPlotLimits::Size() const {
return ImVec2(X.Size(),Y.Size());
}
namespace ImPlot {
2020-04-27 11:27:59 -04:00
namespace {
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// Private Utils
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
/// Returns true if a flag is set
template <typename TSet, typename TFlag>
inline bool HasFlag(TSet set, TFlag flag) {
return (set & flag) == flag;
}
/// Flips a flag in a flagset
2020-05-13 10:11:25 -04:00
template <typename TSet, typename TFlag>
2020-04-27 11:27:59 -04:00
inline void FlipFlag(TSet& set, TFlag flag) {
HasFlag(set, flag) ? set &= ~flag : set |= flag;
}
/// Linearly remaps float x from [x0 x1] to [y0 y1].
inline float Remap(float x, float x0, float x1, float y0, float y1) {
return y0 + (x - x0) * (y1 - y0) / (x1 - x0);
}
2020-05-03 01:24:10 -04:00
/// Turns NANs to 0s
2020-04-27 11:27:59 -04:00
inline float ConstrainNan(float val) {
2020-05-11 07:12:22 -04:00
return isnan(val) ? 0 : val;
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
/// Turns INFINITYs to FLT_MAXs
2020-04-27 11:27:59 -04:00
inline float ConstrainInf(float val) {
return val == INFINITY ? FLT_MAX : val == -INFINITY ? -FLT_MAX : val;
}
2020-05-03 01:24:10 -04:00
/// Turns numbers less than or equal to 0 to 0.001 (sort of arbitrary, is there a better way?)
2020-04-27 11:27:59 -04:00
inline float ConstrainLog(float val) {
2020-05-03 01:24:10 -04:00
return val <= 0 ? 0.001f : val;
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
/// Returns true if val is NAN or INFINITY
2020-04-27 11:27:59 -04:00
inline bool NanOrInf(float val) {
2020-05-11 07:12:22 -04:00
return val == INFINITY || val == -INFINITY || isnan(val);
2020-04-27 11:27:59 -04:00
}
/// Utility function to that rounds x to powers of 2,5 and 10 for generating axis labels
/// Taken from Graphics Gems 1 Chapter 11.2, "Nice Numbers for Graph Labels"
inline double NiceNum(double x, bool round) {
double f; /* fractional part of x */
double nf; /* nice, rounded fraction */
int expv = (int)floor(log10(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);
}
/// Draws vertical text. The position is the bottom left of the text rect.
inline void AddTextVertical(ImDrawList *DrawList, const char *text, ImVec2 pos, ImU32 text_color) {
pos.x = IM_ROUND(pos.x);
pos.y = IM_ROUND(pos.y);
ImFont * font = GImGui->Font;
const ImFontGlyph *glyph;
for (char c = *text++; c; c = *text++) {
2020-04-27 11:27:59 -04:00
glyph = font->FindGlyph(c);
if (!glyph)
continue;
DrawList->PrimReserve(6, 4);
DrawList->PrimQuadUV(
pos + ImVec2(glyph->Y0, -glyph->X0), pos + ImVec2(glyph->Y0, -glyph->X1),
pos + ImVec2(glyph->Y1, -glyph->X1), pos + ImVec2(glyph->Y1, -glyph->X0),
ImVec2(glyph->U0, glyph->V0), ImVec2(glyph->U1, glyph->V0),
ImVec2(glyph->U1, glyph->V1), ImVec2(glyph->U0, glyph->V1), text_color);
pos.y -= glyph->AdvanceX;
}
}
/// Calculates the size of vertical text
inline ImVec2 CalcTextSizeVertical(const char *text) {
2020-05-12 05:19:04 -04:00
ImVec2 sz = ImGui::CalcTextSize(text);
2020-04-27 11:27:59 -04:00
return ImVec2(sz.y, sz.x);
}
2020-05-03 01:24:10 -04:00
} // private namespace
//-----------------------------------------------------------------------------
// Forwards
//-----------------------------------------------------------------------------
ImVec4 NextColor();
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
// Structs
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
/// Tick mark info
struct ImTick {
2020-05-13 10:11:25 -04:00
ImTick(double value, bool major, bool render_label = true) {
2020-04-27 11:27:59 -04:00
PlotPos = value;
Major = major;
RenderLabel = render_label;
}
double PlotPos;
float PixelPos;
ImVec2 Size;
int TextOffset;
bool Major;
bool RenderLabel;
2020-04-27 11:27:59 -04:00
};
struct ImPlotItem {
2020-05-03 01:24:10 -04:00
ImPlotItem() {
2020-05-13 10:11:25 -04:00
Show = true;
2020-05-03 01:24:10 -04:00
Highlight = false;
2020-05-13 10:11:25 -04:00
Color = NextColor();
NameOffset = -1;
ID = 0;
2020-05-03 01:24:10 -04:00
}
2020-04-27 11:27:59 -04:00
~ImPlotItem() { ID = 0; }
bool Show;
2020-04-28 01:38:52 -04:00
bool Highlight;
2020-04-27 11:27:59 -04:00
ImVec4 Color;
int NameOffset;
ImGuiID ID;
};
/// Plot axis structure. You shouldn't need to construct this!
struct ImPlotAxis {
2020-05-13 10:11:25 -04:00
ImPlotAxis() {
2020-04-27 11:27:59 -04:00
Dragging = false;
2020-05-11 07:12:22 -04:00
Range.Min = 0;
Range.Max = 1;
2020-05-13 10:11:25 -04:00
Divisions = 3;
Subdivisions = 10;
Flags = PreviousFlags = ImPlotAxisFlags_Default;
2020-04-27 11:27:59 -04:00
}
bool Dragging;
2020-05-11 07:12:22 -04:00
ImPlotRange Range;
2020-04-27 11:27:59 -04:00
int Divisions;
int Subdivisions;
ImPlotAxisFlags Flags, PreviousFlags;
2020-04-27 11:27:59 -04:00
};
/// Holds Plot state information that must persist between frames
2020-05-12 05:19:04 -04:00
struct ImPlotState {
ImPlotState() {
2020-04-30 09:45:03 -04:00
Selecting = Querying = Queried = DraggingQuery = false;
2020-04-28 21:17:26 -04:00
SelectStart = QueryStart = ImVec2(0,0);
2020-05-11 07:12:22 -04:00
Flags = PreviousFlags = ImPlotFlags_Default;
2020-04-27 11:27:59 -04:00
ColorIdx = 0;
2020-05-11 07:12:22 -04:00
CurrentYAxis = 0;
2020-04-27 11:27:59 -04:00
}
ImPool<ImPlotItem> Items;
ImRect BB_Legend;
ImVec2 SelectStart;
bool Selecting;
2020-04-28 21:17:26 -04:00
bool Querying;
bool Queried;
bool DraggingQuery;
2020-04-28 21:17:26 -04:00
ImVec2 QueryStart;
2020-04-30 09:45:03 -04:00
ImRect QueryRect; // relative to BB_grid!!
2020-05-11 07:12:22 -04:00
2020-04-27 11:27:59 -04:00
ImPlotAxis XAxis;
2020-05-11 07:12:22 -04:00
ImPlotAxis YAxis[MAX_Y_AXES];
ImPlotFlags Flags, PreviousFlags;
2020-04-27 11:27:59 -04:00
int ColorIdx;
2020-05-11 07:12:22 -04:00
int CurrentYAxis;
2020-04-27 11:27:59 -04:00
};
struct ImNextPlotData {
ImGuiCond XRangeCond;
2020-05-11 07:12:22 -04:00
ImGuiCond YRangeCond[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
bool HasXRange;
2020-05-11 07:12:22 -04:00
bool HasYRange[MAX_Y_AXES];
ImPlotRange X;
ImPlotRange Y[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
};
/// Holds Plot state information that must persist only between calls to BeginPlot()/EndPlot()
struct ImPlotContext {
2020-05-11 07:12:22 -04:00
ImPlotContext() : RenderX(), RenderY() {
2020-04-27 11:27:59 -04:00
CurrentPlot = NULL;
2020-05-11 07:12:22 -04:00
FitThisFrame = FitX = false;
2020-05-12 05:19:04 -04:00
RestorePalette();
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
2020-05-13 10:11:25 -04:00
/// ALl Plots
2020-05-12 05:19:04 -04:00
ImPool<ImPlotState> Plots;
2020-04-27 11:27:59 -04:00
/// Current Plot
2020-05-12 05:19:04 -04:00
ImPlotState* CurrentPlot;
2020-04-27 11:27:59 -04:00
// Legend
2020-05-13 10:11:25 -04:00
ImVector<int> LegendIndices;
2020-05-03 01:24:10 -04:00
ImGuiTextBuffer LegendLabels;
2020-05-13 10:11:25 -04:00
// Bounding regions
2020-04-27 11:27:59 -04:00
ImRect BB_Frame;
ImRect BB_Canvas;
ImRect BB_Grid;
2020-05-03 01:24:10 -04:00
// Cached Colors
2020-05-13 10:11:25 -04:00
ImU32 Col_Frame, Col_Bg, Col_Border,
Col_Txt, Col_TxtDis,
2020-04-27 11:27:59 -04:00
Col_SlctBg, Col_SlctBd,
2020-05-11 09:57:36 -04:00
Col_QryBg, Col_QryBd;
2020-05-11 07:12:22 -04:00
struct AxisColor {
AxisColor() : Major(), Minor(), Txt() {}
ImU32 Major, Minor, Txt;
};
AxisColor Col_X;
AxisColor Col_Y[MAX_Y_AXES];
// Tick marks
ImVector<ImTick> XTicks, YTicks[MAX_Y_AXES];
ImGuiTextBuffer XTickLabels, YTickLabels[MAX_Y_AXES];
float AxisLabelReference[MAX_Y_AXES];
2020-05-03 01:24:10 -04:00
// Transformation cache
2020-05-11 07:12:22 -04:00
ImRect PixelRange[MAX_Y_AXES];
// linear scale (slope)
float Mx;
float My[MAX_Y_AXES];
// log scale denominator
float LogDenX;
float LogDenY[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
// Data extents
2020-05-11 07:12:22 -04:00
ImPlotRange ExtentsX;
ImPlotRange ExtentsY[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
int VisibleItemCount;
2020-05-11 07:12:22 -04:00
bool FitThisFrame; bool FitX;
bool FitY[MAX_Y_AXES] = {};
// Hover states
bool Hov_Frame;
2020-05-15 09:01:00 -04:00
bool Hov_Grid;
2020-04-27 11:27:59 -04:00
// Render flags
2020-05-11 07:12:22 -04:00
bool RenderX, RenderY[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
// Mouse pos
2020-05-11 07:12:22 -04:00
ImVec2 LastMousePos[MAX_Y_AXES];
2020-04-27 11:27:59 -04:00
// Style
ImVector<ImVec4> ColorMap;
ImPlotStyle Style;
ImVector<ImGuiColorMod> ColorModifiers; // Stack for PushStyleColor()/PopStyleColor()
ImVector<ImGuiStyleMod> StyleModifiers; // Stack for PushStyleVar()/PopStyleVar()
2020-05-13 10:11:25 -04:00
ImNextPlotData NextPlotData;
2020-04-29 10:32:35 -04:00
// Digital plot item count
int DigitalPlotItemCnt;
int DigitalPlotOffset;
2020-04-27 11:27:59 -04:00
};
/// Global plot context
static ImPlotContext gp;
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// Utils
//-----------------------------------------------------------------------------
/// Returns the next unused default plot color
ImVec4 NextColor() {
2020-05-11 07:12:22 -04:00
ImVec4 col = gp.ColorMap[gp.CurrentPlot->ColorIdx % gp.ColorMap.size()];
2020-05-03 01:24:10 -04:00
gp.CurrentPlot->ColorIdx++;
return col;
2020-04-28 01:38:52 -04:00
}
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
inline void FitPoint(const ImVec2& p) {
2020-05-11 07:12:22 -04:00
ImPlotRange* extents_x = &gp.ExtentsX;
ImPlotRange* extents_y = &gp.ExtentsY[gp.CurrentPlot->CurrentYAxis];
2020-05-03 01:24:10 -04:00
if (!NanOrInf(p.x)) {
2020-05-11 07:12:22 -04:00
extents_x->Min = p.x < extents_x->Min ? p.x : extents_x->Min;
extents_x->Max = p.x > extents_x->Max ? p.x : extents_x->Max;
2020-05-03 01:24:10 -04:00
}
if (!NanOrInf(p.y)) {
2020-05-11 07:12:22 -04:00
extents_y->Min = p.y < extents_y->Min ? p.y : extents_y->Min;
extents_y->Max = p.y > extents_y->Max ? p.y : extents_y->Max;
2020-05-03 01:24:10 -04:00
}
}
//-----------------------------------------------------------------------------
// Coordinate Transforms
//-----------------------------------------------------------------------------
inline void UpdateTransformCache() {
// get pixels for transforms
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.PixelRange[i] = ImRect(HasFlag(gp.CurrentPlot->XAxis.Flags, ImPlotAxisFlags_Invert) ? gp.BB_Grid.Max.x : gp.BB_Grid.Min.x,
HasFlag(gp.CurrentPlot->YAxis[i].Flags, ImPlotAxisFlags_Invert) ? gp.BB_Grid.Min.y : gp.BB_Grid.Max.y,
HasFlag(gp.CurrentPlot->XAxis.Flags, ImPlotAxisFlags_Invert) ? gp.BB_Grid.Min.x : gp.BB_Grid.Max.x,
HasFlag(gp.CurrentPlot->YAxis[i].Flags, ImPlotAxisFlags_Invert) ? gp.BB_Grid.Max.y : gp.BB_Grid.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();
}
gp.LogDenX = log10(gp.CurrentPlot->XAxis.Range.Max / gp.CurrentPlot->XAxis.Range.Min);
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.LogDenY[i] = log10(gp.CurrentPlot->YAxis[i].Range.Max / gp.CurrentPlot->YAxis[i].Range.Min);
}
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-05-11 07:12:22 -04:00
inline ImVec2 PixelsToPlot(float x, float y, int y_axis_in = -1) {
2020-05-03 01:24:10 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "PixelsToPlot() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-11 07:12:22 -04:00
const int y_axis = y_axis_in >= 0 ? y_axis_in : gp.CurrentPlot->CurrentYAxis;
2020-05-03 01:24:10 -04:00
ImVec2 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;
if (HasFlag(gp.CurrentPlot->XAxis.Flags, ImPlotAxisFlags_LogScale)) {
2020-05-11 07:12:22 -04:00
float t = (plt.x - gp.CurrentPlot->XAxis.Range.Min) / gp.CurrentPlot->XAxis.Range.Size();
plt.x = pow(10.0f, t * gp.LogDenX) * gp.CurrentPlot->XAxis.Range.Min;
2020-05-03 01:24:10 -04:00
}
if (HasFlag(gp.CurrentPlot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale)) {
2020-05-11 07:12:22 -04:00
float t = (plt.y - gp.CurrentPlot->YAxis[y_axis].Range.Min) / gp.CurrentPlot->YAxis[y_axis].Range.Size();
plt.y = pow(10.0f, t * gp.LogDenY[y_axis]) * gp.CurrentPlot->YAxis[y_axis].Range.Min;
2020-05-03 01:24:10 -04:00
}
return plt;
}
2020-05-15 09:05:02 -04:00
ImVec2 PixelsToPlot(const ImVec2& pix, int y_axis) {
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-05-11 07:12:22 -04:00
inline ImVec2 PlotToPixels(float x, float y, int y_axis_in = -1) {
2020-05-03 01:24:10 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "PlotToPixels() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-11 07:12:22 -04:00
const int y_axis = y_axis_in >= 0 ? y_axis_in : gp.CurrentPlot->CurrentYAxis;
2020-05-03 01:24:10 -04:00
ImVec2 pix;
if (HasFlag(gp.CurrentPlot->XAxis.Flags, ImPlotAxisFlags_LogScale)) {
2020-05-11 07:12:22 -04:00
float t = log10(x / gp.CurrentPlot->XAxis.Range.Min) / gp.LogDenX;
x = ImLerp(gp.CurrentPlot->XAxis.Range.Min, gp.CurrentPlot->XAxis.Range.Max, t);
2020-05-13 10:11:25 -04:00
}
if (HasFlag(gp.CurrentPlot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale)) {
2020-05-11 07:12:22 -04:00
float t = log10(y / gp.CurrentPlot->YAxis[y_axis].Range.Min) / gp.LogDenY[y_axis];
y = ImLerp(gp.CurrentPlot->YAxis[y_axis].Range.Min, gp.CurrentPlot->YAxis[y_axis].Range.Max, t);
2020-05-03 01:24:10 -04:00
}
2020-05-11 07:12:22 -04:00
pix.x = gp.PixelRange[y_axis].Min.x + gp.Mx * (x - gp.CurrentPlot->XAxis.Range.Min);
pix.y = 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-05-11 07:12:22 -04:00
ImVec2 PlotToPixels(const ImVec2& plt, int y_axis) {
return PlotToPixels(plt.x, plt.y, y_axis);
2020-05-03 01:24:10 -04:00
}
2020-05-15 09:05:02 -04:00
// Transformer structs
2020-05-03 01:24:10 -04:00
struct Plt2PixLinLin {
2020-05-11 07:12:22 -04:00
Plt2PixLinLin(int y_axis_in) : y_axis(y_axis_in) {}
2020-05-03 01:24:10 -04:00
2020-05-11 07:12:22 -04:00
ImVec2 operator()(const ImVec2& plt) { return (*this)(plt.x, plt.y); }
ImVec2 operator()(float x, float y) {
return ImVec2( gp.PixelRange[y_axis].Min.x + gp.Mx * (x - gp.CurrentPlot->XAxis.Range.Min),
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
}
2020-05-11 07:12:22 -04:00
int y_axis;
};
2020-05-03 01:24:10 -04:00
struct Plt2PixLogLin {
2020-05-11 07:12:22 -04:00
Plt2PixLogLin(int y_axis_in) : y_axis(y_axis_in) {}
ImVec2 operator()(const ImVec2& plt) { return (*this)(plt.x, plt.y); }
ImVec2 operator()(float x, float y) {
float t = log10(x / gp.CurrentPlot->XAxis.Range.Min) / gp.LogDenX;
x = ImLerp(gp.CurrentPlot->XAxis.Range.Min, gp.CurrentPlot->XAxis.Range.Max, t);
return ImVec2( gp.PixelRange[y_axis].Min.x + gp.Mx * (x - gp.CurrentPlot->XAxis.Range.Min),
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
}
2020-05-11 07:12:22 -04:00
int y_axis;
2020-05-03 01:24:10 -04:00
};
struct Plt2PixLinLog {
2020-05-11 07:12:22 -04:00
Plt2PixLinLog(int y_axis_in) : y_axis(y_axis_in) {}
ImVec2 operator()(const ImVec2& plt) { return (*this)(plt.x, plt.y); }
ImVec2 operator()(float x, float y) {
float t = log10(y / gp.CurrentPlot->YAxis[y_axis].Range.Min) / gp.LogDenY[y_axis];
y = ImLerp(gp.CurrentPlot->YAxis[y_axis].Range.Min, gp.CurrentPlot->YAxis[y_axis].Range.Max, t);
return ImVec2( gp.PixelRange[y_axis].Min.x + gp.Mx * (x - gp.CurrentPlot->XAxis.Range.Min),
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
}
2020-05-11 07:12:22 -04:00
int y_axis;
2020-05-03 01:24:10 -04:00
};
struct Plt2PixLogLog {
2020-05-11 07:12:22 -04:00
Plt2PixLogLog(int y_axis_in) : y_axis(y_axis_in) {}
ImVec2 operator()(const ImVec2& plt) { return (*this)(plt.x, plt.y); }
ImVec2 operator()(float x, float y) {
float t = log10(x / gp.CurrentPlot->XAxis.Range.Min) / gp.LogDenX;
x = ImLerp(gp.CurrentPlot->XAxis.Range.Min, gp.CurrentPlot->XAxis.Range.Max, t);
t = log10(y / gp.CurrentPlot->YAxis[y_axis].Range.Min) / gp.LogDenY[y_axis];
y = ImLerp(gp.CurrentPlot->YAxis[y_axis].Range.Min, gp.CurrentPlot->YAxis[y_axis].Range.Max, t);
return ImVec2( gp.PixelRange[y_axis].Min.x + gp.Mx * (x - gp.CurrentPlot->XAxis.Range.Min),
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
}
2020-05-11 07:12:22 -04:00
int y_axis;
2020-05-03 01:24:10 -04:00
};
//-----------------------------------------------------------------------------
// Legend Utils
//-----------------------------------------------------------------------------
ImPlotItem* RegisterItem(const char* label_id) {
ImGuiID id = ImGui::GetID(label_id);
ImPlotItem* item = gp.CurrentPlot->Items.GetOrAddByKey(id);
int idx = gp.CurrentPlot->Items.GetIndex(item);
item->ID = id;
gp.LegendIndices.push_back(idx);
item->NameOffset = gp.LegendLabels.size();
gp.LegendLabels.append(label_id, label_id + strlen(label_id) + 1);
if (item->Show)
gp.VisibleItemCount++;
return item;
}
int GetLegendCount() {
return gp.LegendIndices.size();
}
ImPlotItem* GetLegendItem(int i) {
return gp.CurrentPlot->Items.GetByIndex(gp.LegendIndices[i]);
}
const char* GetLegendLabel(int i) {
ImPlotItem* item = gp.CurrentPlot->Items.GetByIndex(gp.LegendIndices[i]);
IM_ASSERT(item->NameOffset != -1 && item->NameOffset < gp.LegendLabels.Buf.Size);
return gp.LegendLabels.Buf.Data + item->NameOffset;
}
//-----------------------------------------------------------------------------
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-05-11 07:12:22 -04:00
inline void GetTicks(const ImPlotRange& scale, int nMajor, int nMinor, bool logscale, ImVector<ImTick> &out) {
2020-04-27 11:27:59 -04:00
out.shrink(0);
if (logscale) {
2020-05-11 07:12:22 -04:00
if (scale.Min <= 0 || scale.Max <= 0)
2020-04-27 11:27:59 -04:00
return;
2020-05-11 07:12:22 -04:00
int exp_min = (int)(ImFloor(log10(scale.Min)));
int exp_max = (int)(ImCeil(log10(scale.Max)));
2020-04-27 11:27:59 -04:00
for (int e = exp_min - 1; e < exp_max + 1; ++e) {
double major1 = ImPow(10, (double)(e));
double major2 = ImPow(10, (double)(e + 1));
double interval = (major2 - major1) / 9;
2020-05-11 07:12:22 -04:00
if (major1 >= (scale.Min - FLT_EPSILON) && major1 <= (scale.Max + FLT_EPSILON))
2020-04-27 11:27:59 -04:00
out.push_back(ImTick(major1, true));
for (int i = 1; i < 9; ++i) {
double minor = major1 + i * interval;
2020-05-11 07:12:22 -04:00
if (minor >= (scale.Min - FLT_EPSILON) && minor <= (scale.Max + FLT_EPSILON))
2020-04-27 11:27:59 -04:00
out.push_back(ImTick(minor, false, false));
}
}
}
else {
2020-05-11 07:12:22 -04:00
const double range = NiceNum(scale.Max - scale.Min, 0);
2020-04-27 11:27:59 -04:00
const double interval = NiceNum(range / (nMajor - 1), 1);
2020-05-11 07:12:22 -04:00
const double graphmin = floor(scale.Min / interval) * interval;
const double graphmax = ceil(scale.Max / interval) * interval;
2020-04-27 11:27:59 -04:00
for (double major = graphmin; major < graphmax + 0.5 * interval; major += interval) {
2020-05-11 07:12:22 -04:00
if (major >= scale.Min && major <= scale.Max)
2020-04-27 11:27:59 -04:00
out.push_back(ImTick(major, true));
for (int i = 1; i < nMinor; ++i) {
double minor = major + i * interval / nMinor;
2020-05-11 07:12:22 -04:00
if (minor >= scale.Min && minor <= scale.Max)
2020-04-27 11:27:59 -04:00
out.push_back(ImTick(minor, false));
}
}
}
}
inline void LabelTicks(ImVector<ImTick> &ticks, bool scientific, ImGuiTextBuffer& buffer) {
buffer.Buf.resize(0);
char temp[32];
for (int t = 0; t < ticks.Size; t++) {
ImTick *tk = &ticks[t];
if (tk->RenderLabel) {
tk->TextOffset = buffer.size();
2020-04-27 11:27:59 -04:00
if (scientific)
sprintf(temp, "%.0e", tk->PlotPos);
2020-04-27 11:27:59 -04:00
else
sprintf(temp, "%g", tk->PlotPos);
2020-04-27 11:27:59 -04:00
buffer.append(temp, temp + strlen(temp) + 1);
tk->Size = ImGui::CalcTextSize(buffer.Buf.Data + tk->TextOffset);
2020-04-27 11:27:59 -04:00
}
}
}
2020-05-11 07:12:22 -04:00
namespace {
struct AxisState {
ImPlotAxis* axis;
bool has_range;
ImGuiCond range_cond;
bool present;
int present_so_far;
bool flip;
bool lock_min;
bool lock_max;
bool lock;
AxisState(ImPlotAxis& axis_in, bool has_range_in, ImGuiCond range_cond_in,
bool present_in, int previous_present)
: axis(&axis_in),
has_range(has_range_in),
range_cond(range_cond_in),
present(present_in),
present_so_far(previous_present + (present ? 1 : 0)),
flip(HasFlag(axis->Flags, ImPlotAxisFlags_Invert)),
lock_min(HasFlag(axis->Flags, ImPlotAxisFlags_LockMin)),
lock_max(HasFlag(axis->Flags, ImPlotAxisFlags_LockMax)),
2020-05-11 07:12:22 -04:00
lock(present && ((lock_min && lock_max) || (has_range && range_cond == ImGuiCond_Always))) {}
AxisState()
: axis(),
has_range(),
range_cond(),
present(),
present_so_far(),
flip(),
lock_min(),
lock_max(),
lock() {}
};
void UpdateAxisColor(int axis_flag, ImPlotContext::AxisColor* col) {
const ImVec4 col_Axis = gp.Style.Colors[axis_flag].w == -1 ? ImGui::GetStyle().Colors[ImGuiCol_Text] * ImVec4(1, 1, 1, 0.25f) : gp.Style.Colors[axis_flag];
2020-05-12 05:19:04 -04:00
col->Major = ImGui::GetColorU32(col_Axis);
col->Minor = ImGui::GetColorU32(col_Axis * ImVec4(1, 1, 1, 0.25f));
col->Txt = ImGui::GetColorU32(ImVec4(col_Axis.x, col_Axis.y, col_Axis.z, 1));
2020-05-11 07:12:22 -04:00
}
ImRect GetAxisScale(int y_axis, float tx, float ty, float zoom_rate) {
return ImRect(
PixelsToPlot(gp.BB_Grid.Min - gp.BB_Grid.GetSize() * ImVec2(tx * zoom_rate, ty * zoom_rate), y_axis),
PixelsToPlot(gp.BB_Grid.Max + gp.BB_Grid.GetSize() * ImVec2((1 - tx) * zoom_rate, (1 - ty) * zoom_rate), y_axis));
}
class YPadCalculator {
public:
YPadCalculator(const AxisState* axis_states, const float* max_label_widths, float txt_off)
: AxisStates(axis_states), MaxLabelWidths(max_label_widths), TxtOff(txt_off) {}
float operator()(int y_axis) {
2020-05-12 05:19:04 -04:00
ImPlotState& plot = *gp.CurrentPlot;
2020-05-11 07:12:22 -04:00
if (!AxisStates[y_axis].present) { return 0; }
// If we have more than 1 axis present before us, then we need
// extra space to account for our tick bar.
float pad_result = 0;
if (AxisStates[y_axis].present_so_far >= 3) {
pad_result += 6.0f;
}
if (!HasFlag(plot.YAxis[y_axis].Flags, ImPlotAxisFlags_TickLabels)) {
2020-05-11 07:12:22 -04:00
return pad_result;
}
pad_result += MaxLabelWidths[y_axis] + TxtOff;
return pad_result;
}
private:
const AxisState* const AxisStates;
const float* const MaxLabelWidths;
const float TxtOff;
};
} // namespace
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
bool BeginPlot(const char* title, const char* x_label, const char* y_label, const ImVec2& size, ImPlotFlags flags, ImPlotAxisFlags x_flags, ImPlotAxisFlags y_flags, ImPlotAxisFlags y2_flags, ImPlotAxisFlags y3_flags) {
2020-04-27 11:27:59 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot == NULL, "Mismatched BeginPlot()/EndPlot()!");
// FRONT MATTER -----------------------------------------------------------
ImGuiContext &G = *GImGui;
ImGuiWindow * Window = G.CurrentWindow;
if (Window->SkipItems) {
gp.NextPlotData = ImNextPlotData();
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-05-13 10:11:25 -04:00
bool just_created = gp.Plots.GetByKey(ID) == NULL;
2020-04-27 11:27:59 -04:00
gp.CurrentPlot = gp.Plots.GetOrAddByKey(ID);
2020-05-12 05:19:04 -04:00
ImPlotState &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;
plot.YAxis[0].Flags = y_flags;
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-05-11 07:12:22 -04:00
if (y_flags != plot.YAxis[0].PreviousFlags)
plot.YAxis[0].PreviousFlags = y_flags;
if (y2_flags != plot.YAxis[1].PreviousFlags)
plot.YAxis[1].PreviousFlags = y2_flags;
if (y3_flags != plot.YAxis[2].PreviousFlags)
plot.YAxis[2].PreviousFlags = y3_flags;
}
plot.PreviousFlags = flags;
plot.XAxis.PreviousFlags = x_flags;
plot.YAxis[0].PreviousFlags = y_flags;
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-05-16 10:11:29 -04:00
const float default_w = 400;
const float default_h = 300;
2020-05-03 01:24:10 -04:00
if (!HasFlag(plot.Flags, ImPlotFlags_NoChild)) {
2020-05-16 10:11:29 -04:00
ImGui::BeginChild(title, ImVec2(size.x == 0 ? default_w : size.x, size.y == 0 ? default_h : size.y));
2020-05-03 01:24:10 -04:00
Window = ImGui::GetCurrentWindow();
Window->ScrollMax.y = 1.0f;
}
ImDrawList &DrawList = *Window->DrawList;
2020-04-27 11:27:59 -04:00
// NextPlotData -----------------------------------------------------------
if (gp.NextPlotData.HasXRange) {
if (just_created || gp.NextPlotData.XRangeCond == ImGuiCond_Always)
{
2020-05-11 07:12:22 -04:00
plot.XAxis.Range = gp.NextPlotData.X;
2020-04-27 11:27:59 -04:00
}
}
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (gp.NextPlotData.HasYRange[i]) {
if (just_created || gp.NextPlotData.YRangeCond[i] == ImGuiCond_Always)
{
plot.YAxis[i].Range = gp.NextPlotData.Y[i];
}
2020-04-27 11:27:59 -04:00
}
}
// AXIS STATES ------------------------------------------------------------
2020-05-11 07:12:22 -04:00
AxisState x(plot.XAxis, gp.NextPlotData.HasXRange, gp.NextPlotData.XRangeCond, true, 0);
AxisState y[MAX_Y_AXES];
y[0] = AxisState(plot.YAxis[0], gp.NextPlotData.HasYRange[0], gp.NextPlotData.YRangeCond[0], true, 0);
y[1] = AxisState(plot.YAxis[1], gp.NextPlotData.HasYRange[1], gp.NextPlotData.YRangeCond[1],
HasFlag(plot.Flags, ImPlotFlags_YAxis2), y[0].present_so_far);
y[2] = AxisState(plot.YAxis[2], gp.NextPlotData.HasYRange[2], gp.NextPlotData.YRangeCond[2],
HasFlag(plot.Flags, ImPlotFlags_YAxis3), y[1].present_so_far);
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
const bool lock_plot = x.lock && y[0].lock && y[1].lock && y[2].lock;
2020-04-27 11:27:59 -04:00
// CONSTRAINTS ------------------------------------------------------------
2020-05-11 07:12:22 -04:00
plot.XAxis.Range.Min = ConstrainNan(ConstrainInf(plot.XAxis.Range.Min));
plot.XAxis.Range.Max = ConstrainNan(ConstrainInf(plot.XAxis.Range.Max));
for (int i = 0; i < MAX_Y_AXES; i++) {
plot.YAxis[i].Range.Min = ConstrainNan(ConstrainInf(plot.YAxis[i].Range.Min));
plot.YAxis[i].Range.Max = ConstrainNan(ConstrainInf(plot.YAxis[i].Range.Max));
}
2020-04-27 11:27:59 -04:00
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
plot.XAxis.Range.Min = ConstrainLog(plot.XAxis.Range.Min);
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
plot.XAxis.Range.Max = ConstrainLog(plot.XAxis.Range.Max);
for (int i = 0; i < MAX_Y_AXES; i++) {
if (HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
plot.YAxis[i].Range.Min = ConstrainLog(plot.YAxis[i].Range.Min);
if (HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
plot.YAxis[i].Range.Max = ConstrainLog(plot.YAxis[i].Range.Max);
}
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
if (plot.XAxis.Range.Max <= plot.XAxis.Range.Min)
plot.XAxis.Range.Max = plot.XAxis.Range.Min + FLT_EPSILON;
for (int i = 0; i < MAX_Y_AXES; i++) {
if (plot.YAxis[i].Range.Max <= plot.YAxis[i].Range.Min)
plot.YAxis[i].Range.Max = plot.YAxis[i].Range.Min + FLT_EPSILON;
}
2020-04-27 11:27:59 -04:00
// adaptive divisions
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_Adaptive)) {
2020-04-27 11:27:59 -04:00
plot.XAxis.Divisions = (int)IM_ROUND(0.003 * gp.BB_Canvas.GetWidth());
if (plot.XAxis.Divisions < 2)
2020-05-13 10:11:25 -04:00
plot.XAxis.Divisions = 2;
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_Adaptive)) {
2020-05-11 07:12:22 -04:00
plot.YAxis[i].Divisions = (int)IM_ROUND(0.003 * gp.BB_Canvas.GetHeight());
if (plot.YAxis[i].Divisions < 2)
plot.YAxis[i].Divisions = 2;
}
2020-04-27 11:27:59 -04:00
}
// COLORS -----------------------------------------------------------------
2020-05-12 05:19:04 -04:00
gp.Col_Frame = gp.Style.Colors[ImPlotCol_FrameBg].w == -1 ? ImGui::GetColorU32(ImGuiCol_FrameBg) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_FrameBg]);
gp.Col_Bg = gp.Style.Colors[ImPlotCol_PlotBg].w == -1 ? ImGui::GetColorU32(ImGuiCol_WindowBg) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_PlotBg]);
gp.Col_Border = gp.Style.Colors[ImPlotCol_PlotBorder].w == -1 ? ImGui::GetColorU32(ImGuiCol_Text, 0.5f) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_PlotBorder]);
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
UpdateAxisColor(ImPlotCol_XAxis, &gp.Col_X);
UpdateAxisColor(ImPlotCol_YAxis, &gp.Col_Y[0]);
UpdateAxisColor(ImPlotCol_YAxis2, &gp.Col_Y[1]);
UpdateAxisColor(ImPlotCol_YAxis3, &gp.Col_Y[2]);
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
gp.Col_Txt = ImGui::GetColorU32(ImGuiCol_Text);
gp.Col_TxtDis = ImGui::GetColorU32(ImGuiCol_TextDisabled);
gp.Col_SlctBg = ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Selection] * ImVec4(1,1,1,0.25f));
gp.Col_SlctBd = ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Selection]);
gp.Col_QryBg = ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Query] * ImVec4(1,1,1,0.25f));
gp.Col_QryBd = ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Query]);
2020-04-27 11:27:59 -04:00
// BB AND HOVER -----------------------------------------------------------
// frame
2020-05-16 10:11:29 -04:00
const ImVec2 frame_size = ImGui::CalcItemSize(size, default_w, default_h);
2020-04-27 11:27:59 -04:00
gp.BB_Frame = ImRect(Window->DC.CursorPos, Window->DC.CursorPos + frame_size);
2020-05-12 05:19:04 -04:00
ImGui::ItemSize(gp.BB_Frame);
if (!ImGui::ItemAdd(gp.BB_Frame, 0, &gp.BB_Frame)) {
2020-04-27 11:27:59 -04:00
gp.NextPlotData = ImNextPlotData();
gp.CurrentPlot = NULL;
2020-05-03 01:24:10 -04:00
if (!HasFlag(plot.Flags, ImPlotFlags_NoChild))
ImGui::EndChild();
2020-04-27 11:27:59 -04:00
return false;
}
2020-05-12 05:19:04 -04:00
gp.Hov_Frame = ImGui::ItemHoverable(gp.BB_Frame, ID);
ImGui::RenderFrame(gp.BB_Frame.Min, gp.BB_Frame.Max, gp.Col_Frame, true, Style.FrameRounding);
2020-04-27 11:27:59 -04:00
// canvas bb
2020-05-13 10:11:25 -04:00
gp.BB_Canvas = ImRect(gp.BB_Frame.Min + Style.WindowPadding, gp.BB_Frame.Max - Style.WindowPadding);
2020-04-27 11:27:59 -04:00
gp.RenderX = (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_GridLines) ||
HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickMarks) ||
HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickLabels)) && plot.XAxis.Divisions > 1;
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.RenderY[i] =
y[i].present &&
2020-05-13 10:11:25 -04:00
(HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_GridLines) ||
HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_TickMarks) ||
HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_TickLabels)) && plot.YAxis[i].Divisions > 1;
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
// get ticks
if (gp.RenderX)
GetTicks(plot.XAxis.Range, plot.XAxis.Divisions, plot.XAxis.Subdivisions, HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_LogScale), gp.XTicks);
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (gp.RenderY[i]) {
GetTicks(plot.YAxis[i].Range, plot.YAxis[i].Divisions, plot.YAxis[i].Subdivisions, HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_LogScale), gp.YTicks[i]);
2020-05-11 07:12:22 -04:00
}
}
2020-04-27 11:27:59 -04:00
// label ticks
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickLabels))
LabelTicks(gp.XTicks, HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_Scientific), gp.XTickLabels);
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
float max_label_width[MAX_Y_AXES] = {};
for (int i = 0; i < MAX_Y_AXES; i++) {
if (y[i].present && HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_TickLabels)) {
LabelTicks(gp.YTicks[i], HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_Scientific), gp.YTickLabels[i]);
for (int t = 0; t < gp.YTicks[i].Size; t++) {
ImTick *yt = &gp.YTicks[i][t];
max_label_width[i] = yt->Size.x > max_label_width[i] ? yt->Size.x : max_label_width[i];
2020-05-11 07:12:22 -04:00
}
}
2020-04-27 11:27:59 -04:00
}
// grid bb
2020-05-12 05:19:04 -04:00
const ImVec2 title_size = ImGui::CalcTextSize(title, NULL, true);
2020-04-27 11:27:59 -04:00
const float txt_off = 5;
2020-05-12 05:19:04 -04:00
const float txt_height = ImGui::GetTextLineHeight();
2020-04-27 11:27:59 -04:00
const float pad_top = title_size.x > 0.0f ? txt_height + txt_off : 0;
const float pad_bot = (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickLabels) ? txt_height + txt_off : 0) + (x_label ? txt_height + txt_off : 0);
2020-05-11 07:12:22 -04:00
YPadCalculator y_axis_pad(y, max_label_width, txt_off);
const float pad_left = y_axis_pad(0) + (y_label ? txt_height + txt_off : 0);
const float pad_right = y_axis_pad(1) + y_axis_pad(2);
2020-05-11 09:57:36 -04:00
gp.BB_Grid = ImRect(gp.BB_Canvas.Min + ImVec2(pad_left, pad_top), gp.BB_Canvas.Max - ImVec2(pad_right, pad_bot));
gp.Hov_Grid = gp.BB_Grid.Contains(IO.MousePos);
2020-04-27 11:27:59 -04:00
// axis region bbs
2020-05-11 07:12:22 -04:00
const ImRect xAxisRegion_bb(gp.BB_Grid.Min + ImVec2(10, 0), ImVec2(gp.BB_Grid.Max.x, gp.BB_Frame.Max.y) - ImVec2(10, 0));
2020-04-27 11:27:59 -04:00
const bool hov_x_axis_region = xAxisRegion_bb.Contains(IO.MousePos);
2020-05-11 07:12:22 -04:00
// The left labels are referenced to the left of the bounding box.
gp.AxisLabelReference[0] = gp.BB_Grid.Min.x;
// If Y axis 1 is present, its labels will be referenced to the
// right of the bounding box.
gp.AxisLabelReference[1] = gp.BB_Grid.Max.x;
// The third axis may be either referenced to the right of the
// bounding box, or 6 pixels further past the end of the 2nd axis.
gp.AxisLabelReference[2] =
!y[1].present ?
gp.BB_Grid.Max.x :
(gp.AxisLabelReference[1] + y_axis_pad(1) + 6);
ImRect yAxisRegion_bb[MAX_Y_AXES];
yAxisRegion_bb[0] = ImRect(ImVec2(gp.BB_Frame.Min.x, gp.BB_Grid.Min.y), ImVec2(gp.BB_Grid.Min.x + 6, gp.BB_Grid.Max.y - 10));
2020-05-11 07:12:22 -04:00
// The auxiliary y axes are off to the right of the BB grid.
yAxisRegion_bb[1] = ImRect(ImVec2(gp.BB_Grid.Max.x - 6, gp.BB_Grid.Min.y),
2020-05-11 07:12:22 -04:00
gp.BB_Grid.Max + ImVec2(y_axis_pad(1), 0));
yAxisRegion_bb[2] = ImRect(ImVec2(gp.AxisLabelReference[2] - 6, gp.BB_Grid.Min.y),
2020-05-11 07:12:22 -04:00
yAxisRegion_bb[1].Max + ImVec2(y_axis_pad(2), 0));
ImRect centralRegion(ImVec2(gp.BB_Grid.Min.x + 6, gp.BB_Grid.Min.y),
ImVec2(gp.BB_Grid.Max.x - 6, gp.BB_Grid.Max.y));
2020-05-13 10:11:25 -04:00
2020-05-11 07:12:22 -04:00
const bool hov_y_axis_region[MAX_Y_AXES] = {
y[0].present && (yAxisRegion_bb[0].Contains(IO.MousePos) || centralRegion.Contains(IO.MousePos)),
y[1].present && (yAxisRegion_bb[1].Contains(IO.MousePos) || centralRegion.Contains(IO.MousePos)),
y[2].present && (yAxisRegion_bb[2].Contains(IO.MousePos) || centralRegion.Contains(IO.MousePos)),
};
const bool any_hov_y_axis_region = hov_y_axis_region[0] || hov_y_axis_region[1] || hov_y_axis_region[2];
2020-04-27 11:27:59 -04:00
// legend hovered from last frame
2020-05-13 10:11:25 -04:00
const bool hov_legend = HasFlag(plot.Flags, ImPlotFlags_Legend) ? gp.Hov_Frame && plot.BB_Legend.Contains(IO.MousePos) : false;
2020-04-30 09:45:03 -04:00
bool hov_query = false;
2020-05-11 07:12:22 -04:00
if (gp.Hov_Frame && gp.Hov_Grid && plot.Queried && !plot.Querying) {
ImRect bb_query = plot.QueryRect;
bb_query.Min += gp.BB_Grid.Min;
bb_query.Max += gp.BB_Grid.Min;
2020-04-30 09:45:03 -04:00
hov_query = bb_query.Contains(IO.MousePos);
}
// QUERY DRAG -------------------------------------------------------------
if (plot.DraggingQuery && (IO.MouseReleased[0] || !IO.MouseDown[0])) {
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-05-11 07:12:22 -04:00
if (gp.Hov_Frame && gp.Hov_Grid && hov_query && !plot.DraggingQuery && !plot.Selecting && !hov_legend) {
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;
if (IO.MouseDown[0] && !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-05-11 07:12:22 -04:00
2020-04-27 11:27:59 -04:00
// end drags
if (plot.XAxis.Dragging && (IO.MouseReleased[0] || !IO.MouseDown[0])) {
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-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (plot.YAxis[i].Dragging && (IO.MouseReleased[0] || !IO.MouseDown[0])) {
plot.YAxis[i].Dragging = false;
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-05-11 07:12:22 -04:00
if (!x.lock && plot.XAxis.Dragging) {
ImVec2 plot_tl = PixelsToPlot(gp.BB_Grid.Min - IO.MouseDelta, 0);
ImVec2 plot_br = PixelsToPlot(gp.BB_Grid.Max - IO.MouseDelta, 0);
if (!x.lock_min)
plot.XAxis.Range.Min = x.flip ? plot_br.x : plot_tl.x;
if (!x.lock_max)
plot.XAxis.Range.Max = x.flip ? plot_tl.x : plot_br.x;
}
for (int i = 0; i < MAX_Y_AXES; i++) {
if (!y[i].lock && plot.YAxis[i].Dragging) {
ImVec2 plot_tl = PixelsToPlot(gp.BB_Grid.Min - IO.MouseDelta, i);
ImVec2 plot_br = PixelsToPlot(gp.BB_Grid.Max - IO.MouseDelta, i);
if (!y[i].lock_min)
plot.YAxis[i].Range.Min = y[i].flip ? plot_tl.y : plot_br.y;
if (!y[i].lock_max)
plot.YAxis[i].Range.Max = y[i].flip ? plot_br.y : plot_tl.y;
}
}
// Set the mouse cursor based on which axes are moving.
int direction = 0;
2020-05-11 09:57:36 -04:00
if (!x.lock && plot.XAxis.Dragging) {
2020-05-11 07:12:22 -04:00
direction |= (1 << 1);
}
for (int i = 0; i < MAX_Y_AXES; i++) {
if (!y[i].present) { continue; }
if (!y[i].lock && plot.YAxis[i].Dragging) {
direction |= (1 << 2);
break;
}
}
if (direction == 0) {
2020-04-27 11:27:59 -04:00
ImGui::SetMouseCursor(ImGuiMouseCursor_NotAllowed);
2020-05-11 07:12:22 -04:00
} else if (direction == (1 << 1)) {
2020-04-27 11:27:59 -04:00
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW);
2020-05-11 07:12:22 -04:00
} else if (direction == (1 << 2)) {
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeNS);
} else {
2020-04-27 11:27:59 -04:00
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeAll);
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
// start drag
2020-05-11 07:12:22 -04:00
if (!drag_in_progress && gp.Hov_Frame && IO.MouseDragMaxDistanceSqr[0] > 5 && !plot.Selecting && !hov_legend && !hov_query && !plot.DraggingQuery) {
if (hov_x_axis_region) {
plot.XAxis.Dragging = true;
}
for (int i = 0; i < MAX_Y_AXES; i++) {
if (hov_y_axis_region[i]) {
plot.YAxis[i].Dragging = true;
}
}
}
2020-04-27 11:27:59 -04:00
// SCROLL INPUT -----------------------------------------------------------
2020-05-11 07:12:22 -04:00
if (gp.Hov_Frame && (hov_x_axis_region || any_hov_y_axis_region) && IO.MouseWheel != 0) {
2020-05-03 01:24:10 -04:00
UpdateTransformCache();
2020-04-27 11:27:59 -04:00
float zoom_rate = 0.1f;
2020-05-13 10:11:25 -04:00
if (IO.MouseWheel > 0)
zoom_rate = (-zoom_rate) / (1.0f + (2.0f * zoom_rate));
2020-04-27 11:27:59 -04:00
float tx = Remap(IO.MousePos.x, gp.BB_Grid.Min.x, gp.BB_Grid.Max.x, 0, 1);
float ty = Remap(IO.MousePos.y, gp.BB_Grid.Min.y, gp.BB_Grid.Max.y, 0, 1);
2020-05-11 07:12:22 -04:00
if (hov_x_axis_region && !x.lock) {
ImRect axis_scale = GetAxisScale(0, tx, ty, zoom_rate);
const ImVec2& plot_tl = axis_scale.Min;
const ImVec2& plot_br = axis_scale.Max;
if (!x.lock_min)
plot.XAxis.Range.Min = x.flip ? plot_br.x : plot_tl.x;
if (!x.lock_max)
plot.XAxis.Range.Max = x.flip ? plot_tl.x : plot_br.x;
}
for (int i = 0; i < MAX_Y_AXES; i++) {
if (hov_y_axis_region[i] && !y[i].lock) {
ImRect axis_scale = GetAxisScale(i, tx, ty, zoom_rate);
const ImVec2& plot_tl = axis_scale.Min;
const ImVec2& plot_br = axis_scale.Max;
if (!y[i].lock_min)
plot.YAxis[i].Range.Min = y[i].flip ? plot_tl.y : plot_br.y;
if (!y[i].lock_max)
plot.YAxis[i].Range.Max = y[i].flip ? plot_br.y : plot_tl.y;
}
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-04-28 21:17:26 -04:00
if (plot.Selecting && (IO.MouseReleased[1] || !IO.MouseDown[1])) {
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-05-12 05:19:04 -04:00
if (HasFlag(plot.Flags, ImPlotFlags_BoxSelect) && ImFabs(select_size.x) > 2 && ImFabs(select_size.y) > 2) {
2020-05-03 01:24:10 -04:00
ImVec2 p1 = PixelsToPlot(plot.SelectStart);
ImVec2 p2 = PixelsToPlot(IO.MousePos);
2020-05-11 07:12:22 -04:00
if (!x.lock_min && !IO.KeyAlt)
plot.XAxis.Range.Min = ImMin(p1.x, p2.x);
if (!x.lock_max && !IO.KeyAlt)
plot.XAxis.Range.Max = ImMax(p1.x, p2.x);
for (int i = 0; i < MAX_Y_AXES; i++) {
p1 = PixelsToPlot(plot.SelectStart, i);
p2 = PixelsToPlot(IO.MousePos, i);
if (!y[i].lock_min && !IO.KeyShift)
plot.YAxis[i].Range.Min = ImMin(p1.y, p2.y);
if (!y[i].lock_max && !IO.KeyShift)
plot.YAxis[i].Range.Max = ImMax(p1.y, p2.y);
}
2020-05-13 10:11:25 -04:00
}
2020-04-27 11:27:59 -04:00
plot.Selecting = false;
}
// bad selection
2020-05-13 10:11:25 -04:00
if (plot.Selecting && (!HasFlag(plot.Flags, ImPlotFlags_BoxSelect) || lock_plot) && ImLengthSqr(plot.SelectStart - IO.MousePos) > 4) {
2020-04-27 11:27:59 -04:00
ImGui::SetMouseCursor(ImGuiMouseCursor_NotAllowed);
}
// cancel selection
2020-04-28 21:17:26 -04:00
if (plot.Selecting && (IO.MouseClicked[0] || IO.MouseDown[0])) {
2020-04-27 11:27:59 -04:00
plot.Selecting = false;
}
// begin selection or query
2020-04-27 11:27:59 -04:00
if (gp.Hov_Frame && gp.Hov_Grid && IO.MouseClicked[1]) {
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-04-30 09:45:03 -04:00
plot.QueryRect.Min.x = IO.KeyAlt ? gp.BB_Grid.Min.x : ImMin(plot.QueryStart.x, IO.MousePos.x);
plot.QueryRect.Max.x = IO.KeyAlt ? gp.BB_Grid.Max.x : ImMax(plot.QueryStart.x, IO.MousePos.x);
2020-04-28 21:17:26 -04:00
plot.QueryRect.Min.y = IO.KeyShift ? gp.BB_Grid.Min.y : ImMin(plot.QueryStart.y, IO.MousePos.y);
plot.QueryRect.Max.y = IO.KeyShift ? gp.BB_Grid.Max.y : ImMax(plot.QueryStart.y, IO.MousePos.y);
2020-05-11 07:12:22 -04:00
2020-04-30 09:45:03 -04:00
plot.QueryRect.Min -= gp.BB_Grid.Min;
plot.QueryRect.Max -= gp.BB_Grid.Min;
2020-04-28 21:17:26 -04:00
}
// end query
if (plot.Querying && (IO.MouseReleased[2] || IO.MouseReleased[1])) {
plot.Querying = false;
if (plot.QueryRect.GetWidth() > 2 && plot.QueryRect.GetHeight() > 2) {
plot.Queried = true;
}
else {
plot.Queried = false;
}
}
// begin query
2020-05-11 07:12:22 -04:00
if (HasFlag(plot.Flags, ImPlotFlags_Query) && (gp.Hov_Frame && gp.Hov_Grid && IO.MouseClicked[2])) {
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-05-11 07:12:22 -04:00
if (HasFlag(plot.Flags, ImPlotFlags_Query) && plot.Selecting && IO.KeyCtrl) {
2020-04-28 21:17:26 -04:00
plot.Selecting = false;
plot.QueryRect = ImRect(0,0,0,0);
plot.Querying = true;
2020-04-28 21:17:26 -04:00
plot.Queried = true;
plot.QueryStart = plot.SelectStart;
}
2020-05-12 05:19:04 -04:00
if (HasFlag(plot.Flags, ImPlotFlags_BoxSelect) && plot.Querying && !IO.KeyCtrl && !IO.MouseDown[2]) {
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-05-13 10:11:25 -04:00
2020-04-27 11:27:59 -04:00
// DOUBLE CLICK -----------------------------------------------------------
2020-05-11 07:12:22 -04:00
if ( IO.MouseDoubleClicked[0] && gp.Hov_Frame && (hov_x_axis_region || any_hov_y_axis_region) && !hov_legend && !hov_query) {
2020-04-27 11:27:59 -04:00
gp.FitThisFrame = true;
gp.FitX = hov_x_axis_region;
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.FitY[i] = hov_y_axis_region[i];
}
}
else {
2020-05-13 10:11:25 -04:00
gp.FitThisFrame = false;
gp.FitX = false;
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.FitY[i] = false;
}
}
2020-04-27 11:27:59 -04:00
// FOCUS ------------------------------------------------------------------
// focus window
if ((IO.MouseClicked[0] || IO.MouseClicked[1]) && gp.Hov_Frame)
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-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.LastMousePos[i] = PixelsToPlot(IO.MousePos, i);
}
2020-04-27 11:27:59 -04:00
// RENDER -----------------------------------------------------------------
// grid bg
DrawList.AddRectFilled(gp.BB_Grid.Min, gp.BB_Grid.Max, gp.Col_Bg);
// render axes
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
2020-04-27 11:27:59 -04:00
// transform ticks
if (gp.RenderX) {
for (int t = 0; t < gp.XTicks.Size; t++) {
ImTick *xt = &gp.XTicks[t];
xt->PixelPos = PlotToPixels((float)xt->PlotPos, 0, 0).x;
}
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (gp.RenderY[i]) {
for (int t = 0; t < gp.YTicks[i].Size; t++) {
ImTick *yt = &gp.YTicks[i][t];
yt->PixelPos = PlotToPixels(0, (float)yt->PlotPos, i).y;
}
2020-05-11 07:12:22 -04:00
}
2020-04-27 11:27:59 -04:00
}
// render grid
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_GridLines)) {
for (int t = 0; t < gp.XTicks.Size; t++) {
ImTick *xt = &gp.XTicks[t];
DrawList.AddLine(ImVec2(xt->PixelPos, gp.BB_Grid.Min.y), ImVec2(xt->PixelPos, gp.BB_Grid.Max.y), xt->Major ? gp.Col_X.Major : gp.Col_X.Minor, 1);
}
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (y[i].present && HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_GridLines)) {
for (int t = 0; t < gp.YTicks[i].Size; t++) {
ImTick *yt = &gp.YTicks[i][t];
DrawList.AddLine(ImVec2(gp.BB_Grid.Min.x, yt->PixelPos), ImVec2(gp.BB_Grid.Max.x, yt->PixelPos), yt->Major ? gp.Col_Y[i].Major : gp.Col_Y[i].Minor, 1);
}
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
if (title_size.x > 0.0f) {
2020-05-12 05:19:04 -04:00
ImGui::RenderText(ImVec2(gp.BB_Canvas.GetCenter().x - title_size.x * 0.5f, gp.BB_Canvas.Min.y), title, NULL, true);
2020-04-27 11:27:59 -04:00
}
// render labels
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickLabels)) {
2020-05-12 05:19:04 -04:00
ImGui::PushClipRect(gp.BB_Frame.Min, gp.BB_Frame.Max, true);
for (int t = 0; t < gp.XTicks.Size; t++) {
ImTick *xt = &gp.XTicks[t];
if (xt->RenderLabel && xt->PixelPos >= gp.BB_Grid.Min.x - 1 && xt->PixelPos <= gp.BB_Grid.Max.x + 1)
DrawList.AddText(ImVec2(xt->PixelPos - xt->Size.x * 0.5f, gp.BB_Grid.Max.y + txt_off), gp.Col_X.Txt, gp.XTickLabels.Buf.Data + xt->TextOffset);
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
ImGui::PopClipRect();
2020-04-27 11:27:59 -04:00
}
if (x_label) {
2020-05-12 05:19:04 -04:00
const ImVec2 xLabel_size = ImGui::CalcTextSize(x_label);
2020-04-27 11:27:59 -04:00
const ImVec2 xLabel_pos(gp.BB_Grid.GetCenter().x - xLabel_size.x * 0.5f,
gp.BB_Canvas.Max.y - txt_height);
2020-05-11 07:12:22 -04:00
DrawList.AddText(xLabel_pos, gp.Col_X.Txt, x_label);
}
2020-05-12 05:19:04 -04:00
ImGui::PushClipRect(gp.BB_Frame.Min, gp.BB_Frame.Max, true);
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (y[i].present && HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_TickLabels)) {
2020-05-11 07:12:22 -04:00
const float x_start =
gp.AxisLabelReference[i] +
((i == 0) ?
(-txt_off - max_label_width[0]) :
txt_off);
for (int t = 0; t < gp.YTicks[i].Size; t++) {
ImTick *yt = &gp.YTicks[i][t];
if (yt->RenderLabel && yt->PixelPos >= gp.BB_Grid.Min.y - 1 && yt->PixelPos <= gp.BB_Grid.Max.y + 1) {
ImVec2 start(x_start, yt->PixelPos - 0.5f * yt->Size.y);
DrawList.AddText(start, gp.Col_Y[i].Txt, gp.YTickLabels[i].Buf.Data + yt->TextOffset);
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-04-27 11:27:59 -04:00
if (y_label) {
const ImVec2 yLabel_size = CalcTextSizeVertical(y_label);
const ImVec2 yLabel_pos(gp.BB_Canvas.Min.x, gp.BB_Grid.GetCenter().y + yLabel_size.y * 0.5f);
2020-05-11 07:12:22 -04:00
AddTextVertical(&DrawList, y_label, yLabel_pos, gp.Col_Y[0].Txt);
2020-04-27 11:27:59 -04:00
}
// PREP -------------------------------------------------------------------
// push plot ID into stack
2020-05-12 05:19:04 -04:00
ImGui::PushID(ID);
2020-04-27 11:27:59 -04:00
// reset items count
gp.VisibleItemCount = 0;
// reset extents
2020-05-11 07:12:22 -04:00
gp.ExtentsX.Min = INFINITY;
gp.ExtentsX.Max = -INFINITY;
for (int i = 0; i < MAX_Y_AXES; i++) {
gp.ExtentsY[i].Min = INFINITY;
gp.ExtentsY[i].Max = -INFINITY;
}
2020-04-27 11:27:59 -04:00
// clear item names
2020-05-03 01:24:10 -04:00
gp.LegendLabels.Buf.resize(0);
2020-04-29 10:32:35 -04:00
// reset digital plot items count
gp.DigitalPlotItemCnt = 0;
gp.DigitalPlotOffset = 0;
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
inline void AxisMenu(ImPlotAxis& Axis) {
ImGui::PushItemWidth(75);
bool lock_min = HasFlag(Axis.Flags, ImPlotAxisFlags_LockMin);
bool lock_max = HasFlag(Axis.Flags, ImPlotAxisFlags_LockMax);
bool invert = HasFlag(Axis.Flags, ImPlotAxisFlags_Invert);
bool logscale = HasFlag(Axis.Flags, ImPlotAxisFlags_LogScale);
bool grid = HasFlag(Axis.Flags, ImPlotAxisFlags_GridLines);
bool ticks = HasFlag(Axis.Flags, ImPlotAxisFlags_TickMarks);
bool labels = HasFlag(Axis.Flags, ImPlotAxisFlags_TickLabels);
2020-05-13 10:11:25 -04:00
if (ImGui::Checkbox("##LockMin", &lock_min))
FlipFlag(Axis.Flags, ImPlotAxisFlags_LockMin);
2020-04-27 11:27:59 -04:00
ImGui::SameLine();
if (lock_min) {
ImGui::PushItemFlag(ImGuiItemFlags_Disabled, true);
ImGui::PushStyleVar(ImGuiStyleVar_Alpha, ImGui::GetStyle().Alpha * 0.25f);
}
2020-05-11 07:12:22 -04:00
ImGui::DragFloat("Min", &Axis.Range.Min, 0.01f + 0.01f * (Axis.Range.Size()), -INFINITY, Axis.Range.Max - FLT_EPSILON);
2020-04-27 11:27:59 -04:00
if (lock_min) {
ImGui::PopItemFlag();
ImGui::PopStyleVar(); }
if (ImGui::Checkbox("##LockMax", &lock_max))
FlipFlag(Axis.Flags, ImPlotAxisFlags_LockMax);
2020-04-27 11:27:59 -04:00
ImGui::SameLine();
if (lock_max) {
ImGui::PushItemFlag(ImGuiItemFlags_Disabled, true);
ImGui::PushStyleVar(ImGuiStyleVar_Alpha, ImGui::GetStyle().Alpha * 0.25f); }
2020-05-11 07:12:22 -04:00
ImGui::DragFloat("Max", &Axis.Range.Max, 0.01f + 0.01f * (Axis.Range.Size()), Axis.Range.Min + FLT_EPSILON, INFINITY);
2020-04-27 11:27:59 -04:00
if (lock_max) {
ImGui::PopItemFlag();
ImGui::PopStyleVar();
}
ImGui::Separator();
if (ImGui::Checkbox("Invert", &invert))
FlipFlag(Axis.Flags, ImPlotAxisFlags_Invert);
2020-04-27 11:27:59 -04:00
if (ImGui::Checkbox("Log Scale", &logscale))
FlipFlag(Axis.Flags, ImPlotAxisFlags_LogScale);
2020-04-27 11:27:59 -04:00
ImGui::Separator();
if (ImGui::Checkbox("Grid Lines", &grid))
FlipFlag(Axis.Flags, ImPlotAxisFlags_GridLines);
2020-04-27 11:27:59 -04:00
if (ImGui::Checkbox("Tick Marks", &ticks))
FlipFlag(Axis.Flags, ImPlotAxisFlags_TickMarks);
2020-04-27 11:27:59 -04:00
if (ImGui::Checkbox("Labels", &labels))
FlipFlag(Axis.Flags, ImPlotAxisFlags_TickLabels);
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
void PlotContextMenu(ImPlotState& plot) {
2020-05-13 10:11:25 -04:00
if (ImGui::BeginMenu("X-Axis")) {
ImGui::PushID("X");
2020-04-27 11:27:59 -04:00
AxisMenu(plot.XAxis);
2020-05-12 05:19:04 -04:00
ImGui::PopID();
2020-04-27 11:27:59 -04:00
ImGui::EndMenu();
}
2020-05-11 07:12:22 -04:00
for (int i = 0; i < MAX_Y_AXES; i++) {
if (i == 1 && !HasFlag(plot.Flags, ImPlotFlags_YAxis2)) {
continue;
}
if (i == 2 && !HasFlag(plot.Flags, ImPlotFlags_YAxis3)) {
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-05-11 07:12:22 -04:00
AxisMenu(plot.YAxis[i]);
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-05-12 05:19:04 -04:00
if (ImGui::MenuItem("Box Select",NULL,HasFlag(plot.Flags, ImPlotFlags_BoxSelect))) {
FlipFlag(plot.Flags, ImPlotFlags_BoxSelect);
}
if (ImGui::MenuItem("Query",NULL,HasFlag(plot.Flags, ImPlotFlags_Query))) {
FlipFlag(plot.Flags, ImPlotFlags_Query);
2020-04-27 11:27:59 -04:00
}
2020-04-28 21:17:26 -04:00
if (ImGui::MenuItem("Crosshairs",NULL,HasFlag(plot.Flags, ImPlotFlags_Crosshairs))) {
FlipFlag(plot.Flags, ImPlotFlags_Crosshairs);
2020-04-27 11:27:59 -04:00
}
if (ImGui::MenuItem("Mouse Position",NULL,HasFlag(plot.Flags, ImPlotFlags_MousePos))) {
FlipFlag(plot.Flags, ImPlotFlags_MousePos);
}
2020-04-28 21:17:26 -04:00
if (ImGui::MenuItem("Cull Data",NULL,HasFlag(plot.Flags, ImPlotFlags_CullData))) {
FlipFlag(plot.Flags, ImPlotFlags_CullData);
}
if (ImGui::MenuItem("Anti-Aliased Lines",NULL,HasFlag(plot.Flags, ImPlotFlags_AntiAliased))) {
FlipFlag(plot.Flags, ImPlotFlags_AntiAliased);
2020-04-27 11:27:59 -04:00
}
ImGui::EndMenu();
}
if (ImGui::MenuItem("Legend",NULL,HasFlag(plot.Flags, ImPlotFlags_Legend))) {
FlipFlag(plot.Flags, ImPlotFlags_Legend);
}
#if 0
if (ImGui::BeginMenu("Metrics")) {
ImGui::PushItemWidth(75);
ImGui::LabelText("Plots", "%d", gp.Plots.GetSize());
ImGui::LabelText("Color Modifiers", "%d", gp.ColorModifiers.size());
ImGui::LabelText("Style Modifiers", "%d", gp.StyleModifiers.size());
ImGui::PopItemWidth();
ImGui::EndMenu();
}
#endif
}
2020-05-11 07:12:22 -04:00
namespace {
class BufferWriter {
public:
BufferWriter(char* buffer, size_t size) : Buffer(buffer), Pos(0), Size(size) {}
void Write(const char* fmt, ...) IM_FMTARGS(2) {
va_list argp;
va_start(argp, fmt);
VWrite(fmt, argp);
va_end(argp);
}
private:
void VWrite(const char* fmt, va_list argp) {
const int written = ::vsnprintf(&Buffer[Pos], Size - Pos - 1, fmt, argp);
if (written > 0)
Pos += ImMin(size_t(written), Size-Pos-1);
2020-05-11 07:12:22 -04:00
}
char* const Buffer;
size_t Pos;
const size_t Size;
};
}
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-05-13 10:11:25 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "Mismatched BeginPlot()/EndPlot()!");
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
ImPlotState &plot = *gp.CurrentPlot;
2020-04-27 11:27:59 -04:00
ImGuiContext &G = *GImGui;
ImGuiWindow * Window = G.CurrentWindow;
ImDrawList & DrawList = *Window->DrawList;
2020-05-12 05:19:04 -04:00
const ImGuiIO & IO = ImGui::GetIO();
2020-04-27 11:27:59 -04:00
// AXIS STATES ------------------------------------------------------------
2020-05-11 07:12:22 -04:00
AxisState x(plot.XAxis, gp.NextPlotData.HasXRange, gp.NextPlotData.XRangeCond, true, 0);
AxisState y[MAX_Y_AXES];
y[0] = AxisState(plot.YAxis[0], gp.NextPlotData.HasYRange[0], gp.NextPlotData.YRangeCond[0], true, 0);
y[1] = AxisState(plot.YAxis[1], gp.NextPlotData.HasYRange[1], gp.NextPlotData.YRangeCond[1],
HasFlag(plot.Flags, ImPlotFlags_YAxis2), y[0].present_so_far);
y[2] = AxisState(plot.YAxis[2], gp.NextPlotData.HasYRange[2], gp.NextPlotData.YRangeCond[2],
HasFlag(plot.Flags, ImPlotFlags_YAxis3), y[1].present_so_far);
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
const bool lock_plot = x.lock && y[0].lock && y[1].lock && y[2].lock;
const bool any_y_locked = y[0].lock || y[1].lock || y[2].lock;
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
// FINAL RENDER -----------------------------------------------------------
2020-04-28 21:17:26 -04:00
// render ticks
2020-05-11 07:12:22 -04:00
PushPlotClipRect();
if (HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_TickMarks)) {
for (int t = 0; t < gp.XTicks.Size; t++) {
ImTick *xt = &gp.XTicks[t];
DrawList.AddLine(ImVec2(xt->PixelPos, gp.BB_Grid.Max.y),ImVec2(xt->PixelPos, gp.BB_Grid.Max.y - (xt->Major ? 10.0f : 5.0f)), gp.Col_Border, 1);
}
2020-04-28 21:17:26 -04:00
}
2020-05-11 07:12:22 -04:00
PopPlotClipRect();
ImGui::PushClipRect(gp.BB_Grid.Min, ImVec2(gp.BB_Frame.Max.x, gp.BB_Grid.Max.y), true);
2020-05-11 07:12:22 -04:00
int axis_count = 0;
for (int i = 0; i < MAX_Y_AXES; i++) {
if (!y[i].present) { continue; }
axis_count++;
if (!HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_TickMarks)) { continue; }
2020-05-11 07:12:22 -04:00
float x_start = gp.AxisLabelReference[i];
float direction = (i == 0) ? 1.0f : -1.0f;
bool no_major = axis_count >= 3;
for (int t = 0; t < gp.YTicks[i].Size; t++) {
ImTick *yt = &gp.YTicks[i][t];
ImVec2 start = ImVec2(x_start, yt->PixelPos);
2020-05-11 07:12:22 -04:00
DrawList.AddLine(
start,
start + ImVec2(direction * ((!no_major && yt->Major) ? 10.0f : 5.0f), 0),
2020-05-11 07:12:22 -04:00
gp.Col_Border, 1);
}
if (axis_count >= 3) {
// Draw a bar next to the ticks to act as a visual separator.
DrawList.AddLine(
ImVec2(x_start, gp.BB_Grid.Min.y),
ImVec2(x_start, gp.BB_Grid.Max.y),
gp.Col_Border, 1);
}
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
PushPlotClipRect();
2020-04-28 21:17:26 -04:00
// render selection/query
if (plot.Selecting) {
ImRect select_bb(ImMin(IO.MousePos, plot.SelectStart), ImMax(IO.MousePos, plot.SelectStart));
2020-05-12 05:19:04 -04:00
if (plot.Selecting && !lock_plot && HasFlag(plot.Flags, ImPlotFlags_BoxSelect)) {
2020-04-28 21:17:26 -04:00
if (IO.KeyAlt && IO.KeyShift && select_bb.GetWidth() > 2 && select_bb.GetHeight() > 2) {
DrawList.AddRectFilled(gp.BB_Grid.Min, gp.BB_Grid.Max, gp.Col_SlctBg);
DrawList.AddRect( gp.BB_Grid.Min, gp.BB_Grid.Max, gp.Col_SlctBd);
}
2020-05-11 07:12:22 -04:00
else if ((x.lock || IO.KeyAlt) && select_bb.GetHeight() > 2) {
2020-04-28 21:17:26 -04:00
DrawList.AddRectFilled(ImVec2(gp.BB_Grid.Min.x, select_bb.Min.y), ImVec2(gp.BB_Grid.Max.x, select_bb.Max.y), gp.Col_SlctBg);
DrawList.AddRect( ImVec2(gp.BB_Grid.Min.x, select_bb.Min.y), ImVec2(gp.BB_Grid.Max.x, select_bb.Max.y), gp.Col_SlctBd);
}
2020-05-11 07:12:22 -04:00
else if ((any_y_locked || IO.KeyShift) && select_bb.GetWidth() > 2) {
2020-04-28 21:17:26 -04:00
DrawList.AddRectFilled(ImVec2(select_bb.Min.x, gp.BB_Grid.Min.y), ImVec2(select_bb.Max.x, gp.BB_Grid.Max.y), gp.Col_SlctBg);
DrawList.AddRect( ImVec2(select_bb.Min.x, gp.BB_Grid.Min.y), ImVec2(select_bb.Max.x, gp.BB_Grid.Max.y), gp.Col_SlctBd);
}
else if (select_bb.GetWidth() > 2 && select_bb.GetHeight() > 2) {
DrawList.AddRectFilled(select_bb.Min, select_bb.Max, gp.Col_SlctBg);
DrawList.AddRect( select_bb.Min, select_bb.Max, gp.Col_SlctBd);
}
}
}
2020-05-11 07:12:22 -04:00
if (plot.Querying || plot.Queried) {
2020-04-28 21:17:26 -04:00
if (plot.QueryRect.GetWidth() > 2 && plot.QueryRect.GetHeight() > 2) {
2020-04-30 09:45:03 -04:00
DrawList.AddRectFilled(plot.QueryRect.Min + gp.BB_Grid.Min, plot.QueryRect.Max + gp.BB_Grid.Min, gp.Col_QryBg);
DrawList.AddRect( plot.QueryRect.Min + gp.BB_Grid.Min, plot.QueryRect.Max + gp.BB_Grid.Min, gp.Col_QryBd);
2020-05-13 10:11:25 -04:00
}
2020-04-28 21:17:26 -04:00
}
else if (plot.Queried) {
2020-05-11 07:12:22 -04:00
ImRect bb_query = plot.QueryRect;
bb_query.Min += gp.BB_Grid.Min;
bb_query.Max += gp.BB_Grid.Min;
DrawList.AddRectFilled(bb_query.Min, bb_query.Max, gp.Col_QryBg);
DrawList.AddRect( bb_query.Min, bb_query.Max, gp.Col_QryBd);
}
2020-04-27 11:27:59 -04:00
// render legend
2020-05-12 05:19:04 -04:00
const float txt_ht = ImGui::GetTextLineHeight();
2020-04-27 11:27:59 -04:00
const ImVec2 legend_offset(10, 10);
const ImVec2 legend_padding(5, 5);
const float legend_icon_size = txt_ht;
ImRect legend_content_bb;
2020-05-03 01:24:10 -04:00
int nItems = GetLegendCount();
2020-04-27 11:27:59 -04:00
bool hov_legend = false;
if (HasFlag(plot.Flags, ImPlotFlags_Legend) && nItems > 0) {
// get max width
float max_label_width = 0;
for (int i = 0; i < nItems; ++i) {
2020-05-03 01:24:10 -04:00
const char* label = GetLegendLabel(i);
2020-05-12 05:19:04 -04:00
ImVec2 labelWidth = ImGui::CalcTextSize(label, NULL, true);
2020-04-27 11:27:59 -04:00
max_label_width = labelWidth.x > max_label_width ? labelWidth.x : max_label_width;
}
legend_content_bb = ImRect(gp.BB_Grid.Min + legend_offset, gp.BB_Grid.Min + legend_offset + ImVec2(max_label_width, nItems * txt_ht));
plot.BB_Legend = ImRect(legend_content_bb.Min, legend_content_bb.Max + legend_padding * 2 + ImVec2(legend_icon_size, 0));
hov_legend = HasFlag(plot.Flags, ImPlotFlags_Legend) ? gp.Hov_Frame && plot.BB_Legend.Contains(IO.MousePos) : false;
// render legend box
2020-05-12 05:19:04 -04:00
DrawList.AddRectFilled(plot.BB_Legend.Min, plot.BB_Legend.Max, ImGui::GetColorU32(ImGuiCol_PopupBg));
2020-04-27 11:27:59 -04:00
DrawList.AddRect(plot.BB_Legend.Min, plot.BB_Legend.Max, gp.Col_Border);
// render each legend item
for (int i = 0; i < nItems; ++i) {
2020-05-03 01:24:10 -04:00
ImPlotItem* item = GetLegendItem(i);
2020-04-27 11:27:59 -04:00
ImRect icon_bb;
icon_bb.Min = legend_content_bb.Min + legend_padding + ImVec2(0, i * txt_ht) + ImVec2(2, 2);
icon_bb.Max = legend_content_bb.Min + legend_padding + ImVec2(0, i * txt_ht) + ImVec2(legend_icon_size - 2, legend_icon_size - 2);
2020-04-28 01:38:52 -04:00
ImRect label_bb;
label_bb.Min = legend_content_bb.Min + legend_padding + ImVec2(0, i * txt_ht) + ImVec2(2, 2);
label_bb.Max = legend_content_bb.Min + legend_padding + ImVec2(0, i * txt_ht) + ImVec2(legend_content_bb.Max.x, legend_icon_size - 2);
ImU32 col_hl_txt;
if (HasFlag(plot.Flags, ImPlotFlags_Highlight) && hov_legend && (icon_bb.Contains(IO.MousePos) || label_bb.Contains(IO.MousePos))) {
item->Highlight = true;
2020-05-12 05:19:04 -04:00
col_hl_txt = ImGui::GetColorU32(ImLerp(G.Style.Colors[ImGuiCol_Text], item->Color, 0.25f));
2020-04-28 01:38:52 -04:00
}
else
{
item->Highlight = false;
col_hl_txt = gp.Col_Txt;
}
2020-04-27 11:27:59 -04:00
ImU32 iconColor;
if (hov_legend && icon_bb.Contains(IO.MousePos)) {
2020-05-11 07:12:22 -04:00
ImVec4 colAlpha = item->Color;
2020-04-27 11:27:59 -04:00
colAlpha.w = 0.5f;
2020-05-12 05:19:04 -04:00
iconColor = item->Show ? ImGui::GetColorU32(colAlpha)
: ImGui::GetColorU32(ImGuiCol_TextDisabled, 0.5f);
2020-04-27 11:27:59 -04:00
if (IO.MouseClicked[0])
item->Show = !item->Show;
} else {
2020-05-12 05:19:04 -04:00
iconColor = item->Show ? ImGui::GetColorU32(item->Color) : gp.Col_TxtDis;
2020-04-27 11:27:59 -04:00
}
DrawList.AddRectFilled(icon_bb.Min, icon_bb.Max, iconColor, 1);
2020-05-03 01:24:10 -04:00
const char* label = GetLegendLabel(i);
2020-05-12 05:19:04 -04:00
const char* text_display_end = ImGui::FindRenderedTextEnd(label, NULL);
2020-04-27 11:27:59 -04:00
if (label != text_display_end)
DrawList.AddText(legend_content_bb.Min + legend_padding + ImVec2(legend_icon_size, i * txt_ht), item->Show ? col_hl_txt : gp.Col_TxtDis, label, text_display_end);
2020-04-27 11:27:59 -04:00
}
}
// render crosshairs
if (HasFlag(plot.Flags, ImPlotFlags_Crosshairs) && gp.Hov_Grid && gp.Hov_Frame &&
2020-05-11 07:12:22 -04:00
!(plot.XAxis.Dragging || any_y_dragging) && !plot.Selecting && !plot.Querying && !hov_legend) {
2020-04-27 11:27:59 -04:00
ImGui::SetMouseCursor(ImGuiMouseCursor_None);
ImVec2 xy = IO.MousePos;
ImVec2 h1(gp.BB_Grid.Min.x, xy.y);
ImVec2 h2(xy.x - 5, xy.y);
ImVec2 h3(xy.x + 5, xy.y);
ImVec2 h4(gp.BB_Grid.Max.x, xy.y);
ImVec2 v1(xy.x, gp.BB_Grid.Min.y);
ImVec2 v2(xy.x, xy.y - 5);
ImVec2 v3(xy.x, xy.y + 5);
ImVec2 v4(xy.x, gp.BB_Grid.Max.y);
DrawList.AddLine(h1, h2, gp.Col_Border);
DrawList.AddLine(h3, h4, gp.Col_Border);
DrawList.AddLine(v1, v2, gp.Col_Border);
DrawList.AddLine(v3, v4, gp.Col_Border);
}
// render mouse pos
if (HasFlag(plot.Flags, ImPlotFlags_MousePos) && gp.Hov_Grid) {
char buffer[128] = {};
2020-05-11 07:12:22 -04:00
BufferWriter writer(buffer, sizeof(buffer));
writer.Write("%.2f,%.2f", gp.LastMousePos[0].x, gp.LastMousePos[0].y);
if (HasFlag(plot.Flags, ImPlotFlags_YAxis2)) {
writer.Write(",(%.2f)", gp.LastMousePos[1].y);
}
if (HasFlag(plot.Flags, ImPlotFlags_YAxis3)) {
writer.Write(",(%.2f)", gp.LastMousePos[2].y);
}
2020-05-12 05:19:04 -04:00
ImVec2 size = ImGui::CalcTextSize(buffer);
2020-04-27 11:27:59 -04:00
ImVec2 pos = gp.BB_Grid.Max - size - ImVec2(5, 5);
DrawList.AddText(pos, gp.Col_Txt, buffer);
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
2020-04-27 11:27:59 -04:00
// render border
DrawList.AddRect(gp.BB_Grid.Min, gp.BB_Grid.Max, gp.Col_Border);
// FIT DATA --------------------------------------------------------------
2020-04-28 21:17:26 -04:00
if (gp.FitThisFrame && (gp.VisibleItemCount > 0 || plot.Queried)) {
if (gp.FitX && !HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_LockMin) && !NanOrInf(gp.ExtentsX.Min)) {
2020-05-11 07:12:22 -04:00
plot.XAxis.Range.Min = gp.ExtentsX.Min;
}
if (gp.FitX && !HasFlag(plot.XAxis.Flags, ImPlotAxisFlags_LockMax) && !NanOrInf(gp.ExtentsX.Max)) {
2020-05-11 07:12:22 -04:00
plot.XAxis.Range.Max = gp.ExtentsX.Max;
}
for (int i = 0; i < MAX_Y_AXES; i++) {
if (gp.FitY[i] && !HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_LockMin) && !NanOrInf(gp.ExtentsY[i].Min)) {
2020-05-11 07:12:22 -04:00
plot.YAxis[i].Range.Min = gp.ExtentsY[i].Min;
}
if (gp.FitY[i] && !HasFlag(plot.YAxis[i].Flags, ImPlotAxisFlags_LockMax) && !NanOrInf(gp.ExtentsY[i].Max)) {
2020-05-11 07:12:22 -04:00
plot.YAxis[i].Range.Max = gp.ExtentsY[i].Max;
}
}
2020-04-27 11:27:59 -04:00
}
// CONTEXT MENU -----------------------------------------------------------
if (HasFlag(plot.Flags, ImPlotFlags_ContextMenu) && gp.Hov_Frame && gp.Hov_Grid && IO.MouseDoubleClicked[1] && !hov_legend)
ImGui::OpenPopup("##Context");
if (ImGui::BeginPopup("##Context")) {
PlotContextMenu(plot);
ImGui::EndPopup();
}
// CLEANUP ----------------------------------------------------------------
// Reset legend items
2020-05-03 01:24:10 -04:00
gp.LegendIndices.shrink(0);
2020-04-27 11:27:59 -04:00
// Null current plot/data
gp.CurrentPlot = NULL;
// Reset next plot data
gp.NextPlotData = ImNextPlotData();
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-04-27 11:27:59 -04:00
// End child window
2020-05-03 01:24:10 -04:00
if (!HasFlag(plot.Flags, ImPlotFlags_NoChild))
ImGui::EndChild();
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-05-11 07:12:22 -04:00
void SetNextPlotLimits(float x_min, float x_max, float y_min, float y_max, ImGuiCond cond) {
SetNextPlotLimitsX(x_min, x_max, cond);
SetNextPlotLimitsY(y_min, y_max, cond);
2020-04-27 11:27:59 -04:00
}
2020-05-11 07:12:22 -04:00
void SetNextPlotLimitsX(float x_min, float x_max, ImGuiCond cond) {
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-05-11 07:12:22 -04:00
void SetNextPlotLimitsY(float y_min, float y_max, ImGuiCond cond, int y_axis) {
IM_ASSERT_USER_ERROR(y_axis >= 0 && y_axis < MAX_Y_AXES, "y_axis Needs to be between 0 and MAX_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;
}
void SetPlotYAxis(int y_axis) {
IM_ASSERT_USER_ERROR(y_axis >= 0 && y_axis < MAX_Y_AXES, "y_axis Needs to be between 0 and MAX_Y_AXES");
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "SetPlotYAxis() Needs to be called between BeginPlot() and EndPlot()!");
gp.CurrentPlot->CurrentYAxis = y_axis;
2020-04-27 11:27:59 -04:00
}
2020-04-30 09:45:03 -04:00
ImVec2 GetPlotPos() {
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "GetPlotPos() Needs to be called between BeginPlot() and EndPlot()!");
return gp.BB_Grid.Min;
}
ImVec2 GetPlotSize() {
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "GetPlotSize() Needs to be called between BeginPlot() and EndPlot()!");
return gp.BB_Grid.GetSize();
}
void PushPlotClipRect() {
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "PushPlotClipRect() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-12 05:19:04 -04:00
ImGui::PushClipRect(gp.BB_Grid.Min, gp.BB_Grid.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-04-28 21:17:26 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "IsPlotHovered() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-13 10:11:25 -04:00
return gp.Hov_Grid;
2020-04-28 21:17:26 -04:00
}
2020-05-11 07:12:22 -04:00
ImVec2 GetPlotMousePos(int y_axis_in) {
IM_ASSERT_USER_ERROR(y_axis_in >= -1 && y_axis_in < MAX_Y_AXES, "y_axis needs to between -1 and MAX_Y_AXES");
2020-04-28 21:17:26 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "GetPlotMousePos() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-11 07:12:22 -04:00
const int y_axis = y_axis_in >= 0 ? y_axis_in : gp.CurrentPlot->CurrentYAxis;
2020-05-13 10:11:25 -04:00
return gp.LastMousePos[y_axis];
2020-04-28 21:17:26 -04:00
}
2020-04-27 11:27:59 -04:00
2020-05-11 07:12:22 -04:00
ImPlotLimits GetPlotLimits(int y_axis_in) {
IM_ASSERT_USER_ERROR(y_axis_in >= -1 && y_axis_in < MAX_Y_AXES, "y_axis needs to between -1 and MAX_Y_AXES");
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "GetPlotLimits() Needs to be called between BeginPlot() and EndPlot()!");
const int y_axis = y_axis_in >= 0 ? y_axis_in : gp.CurrentPlot->CurrentYAxis;
2020-05-12 05:19:04 -04:00
ImPlotState& 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;
}
bool IsPlotQueried() {
2020-04-28 21:17:26 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "IsPlotQueried() Needs to be called between BeginPlot() and EndPlot()!");
return gp.CurrentPlot->Queried;
}
2020-05-11 07:12:22 -04:00
ImPlotLimits GetPlotQuery(int y_axis_in) {
IM_ASSERT_USER_ERROR(y_axis_in >= -1 && y_axis_in < MAX_Y_AXES, "y_axis needs to between -1 and MAX_Y_AXES");
2020-04-28 21:17:26 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "GetPlotQuery() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-12 05:19:04 -04:00
ImPlotState& plot = *gp.CurrentPlot;
2020-05-11 07:12:22 -04:00
const int y_axis = y_axis_in >= 0 ? y_axis_in : gp.CurrentPlot->CurrentYAxis;
UpdateTransformCache();
ImVec2 p1 = PixelsToPlot(plot.QueryRect.Min + gp.BB_Grid.Min, y_axis);
ImVec2 p2 = PixelsToPlot(plot.QueryRect.Max + gp.BB_Grid.Min, y_axis);
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-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
struct ImPlotStyleVarInfo {
ImGuiDataType Type;
ImU32 Count;
ImU32 Offset;
void* GetVarPtr(ImPlotStyle* style) const { return (void*)((unsigned char*)style + Offset); }
};
2020-05-13 10:11:25 -04:00
static const ImPlotStyleVarInfo GPlotStyleVarInfo[] =
2020-04-27 11:27:59 -04:00
{
2020-04-29 10:32:35 -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, 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
2020-04-27 11:27:59 -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-05-12 05:19:04 -04:00
ImPlotStyle& GetStyle() {
2020-04-27 11:27:59 -04:00
return gp.Style;
}
2020-05-12 05:19:04 -04:00
void SetPalette(const ImVec4* colors, int num_colors) {
2020-04-27 11:27:59 -04:00
gp.ColorMap.shrink(0);
gp.ColorMap.reserve(num_colors);
for (int i = 0; i < num_colors; ++i) {
gp.ColorMap.push_back(colors[i]);
}
}
/// Returns the next unused default plot color
2020-05-12 05:19:04 -04:00
void RestorePalette() {
2020-04-27 11:27:59 -04:00
static ImVec4 default_colors[10] = {
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-04-27 11:27:59 -04:00
};
2020-05-12 05:19:04 -04:00
SetPalette(default_colors, 10);
2020-04-27 11:27:59 -04:00
}
2020-05-12 05:19:04 -04:00
void PushStyleColor(ImPlotCol idx, ImU32 col) {
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-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-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-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-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-05-12 05:19:04 -04:00
void PopStyleVar(int count) {
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-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// RENDERING FUNCTIONS
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
#define SQRT_1_2 0.70710678118f
#define SQRT_3_2 0.86602540378f
inline void TransformMarker(ImVec2* points, int n, const ImVec2& c, float s) {
for (int i = 0; i < n; ++i) {
points[i].x = c.x + points[i].x * s;
points[i].y = c.y + points[i].y * s;
}
}
inline void MarkerGeneral(ImDrawList& DrawList, ImVec2* points, int n, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
TransformMarker(points, n, c, s);
if (fill)
DrawList.AddConvexPolyFilled(points, n, col_fill);
if (outline && !(fill && col_outline == col_fill)) {
for (int i = 0; i < n; ++i)
DrawList.AddLine(points[i], points[(i+1)%n], col_outline, weight);
}
}
2020-05-11 07:12:22 -04:00
inline void MarkerCircle(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[10] = {ImVec2(1.0f, 0.0f),
ImVec2(0.809017f, 0.58778524f),
ImVec2(0.30901697f, 0.95105654f),
ImVec2(-0.30901703f, 0.9510565f),
ImVec2(-0.80901706f, 0.5877852f),
ImVec2(-1.0f, 0.0f),
ImVec2(-0.80901694f, -0.58778536f),
ImVec2(-0.3090171f, -0.9510565f),
ImVec2(0.30901712f, -0.9510565f),
ImVec2(0.80901694f, -0.5877853f)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 10, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerDiamond(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[4] = {ImVec2(1, 0), ImVec2(0, -1), ImVec2(-1, 0), ImVec2(0, 1)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 4, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerSquare(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[4] = {ImVec2(SQRT_1_2,SQRT_1_2),ImVec2(SQRT_1_2,-SQRT_1_2),ImVec2(-SQRT_1_2,-SQRT_1_2),ImVec2(-SQRT_1_2,SQRT_1_2)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 4, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerUp(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[3] = {ImVec2(SQRT_3_2,0.5f),ImVec2(0,-1),ImVec2(-SQRT_3_2,0.5f)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 3, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerDown(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[3] = {ImVec2(SQRT_3_2,-0.5f),ImVec2(0,1),ImVec2(-SQRT_3_2,-0.5f)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 3, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerLeft(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[3] = {ImVec2(-1,0), ImVec2(0.5, SQRT_3_2), ImVec2(0.5, -SQRT_3_2)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 3, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerRight(ImDrawList& DrawList, const ImVec2& c, float s, bool outline, ImU32 col_outline, bool fill, ImU32 col_fill, float weight) {
ImVec2 marker[3] = {ImVec2(1,0), ImVec2(-0.5, SQRT_3_2), ImVec2(-0.5, -SQRT_3_2)};
2020-04-27 11:27:59 -04:00
MarkerGeneral(DrawList, marker, 3, c, s, outline, col_outline, fill, col_fill, weight);
}
inline void MarkerAsterisk(ImDrawList& DrawList, const ImVec2& c, float s, bool /*outline*/, ImU32 col_outline, bool /*fill*/, ImU32 /*col_fill*/, float weight) {
2020-04-27 11:27:59 -04:00
ImVec2 marker[6] = {{SQRT_3_2, 0.5f}, {0, -1}, {-SQRT_3_2, 0.5f}, {SQRT_3_2, -0.5f}, {0, 1}, {-SQRT_3_2, -0.5f}};
TransformMarker(marker, 6, c, s);
DrawList.AddLine(marker[0], marker[5], col_outline, weight);
DrawList.AddLine(marker[1], marker[4], col_outline, weight);
DrawList.AddLine(marker[2], marker[3], col_outline, weight);
}
inline void MarkerPlus(ImDrawList& DrawList, const ImVec2& c, float s, bool /*outline*/, ImU32 col_outline, bool /*fill*/, ImU32 /*col_fill*/, float weight) {
2020-04-27 11:27:59 -04:00
ImVec2 marker[4] = {{1, 0}, {0, -1}, {-1, 0}, {0, 1}};
TransformMarker(marker, 4, c, s);
DrawList.AddLine(marker[0], marker[2], col_outline, weight);
DrawList.AddLine(marker[1], marker[3], col_outline, weight);
}
inline void MarkerCross(ImDrawList& DrawList, const ImVec2& c, float s, bool /*outline*/, ImU32 col_outline, bool /*fill*/, ImU32 /*col_fill*/, float weight) {
2020-04-27 11:27:59 -04:00
ImVec2 marker[4] = {{SQRT_1_2,SQRT_1_2},{SQRT_1_2,-SQRT_1_2},{-SQRT_1_2,-SQRT_1_2},{-SQRT_1_2,SQRT_1_2}};
TransformMarker(marker, 4, c, s);
DrawList.AddLine(marker[0], marker[2], col_outline, weight);
DrawList.AddLine(marker[1], marker[3], col_outline, weight);
}
2020-05-03 01:24:10 -04:00
template <typename Transformer, typename Getter>
2020-05-11 07:12:22 -04:00
inline void RenderMarkers(Transformer transformer, ImDrawList& DrawList, Getter getter, int count, int offset, bool rend_mk_line, ImU32 col_mk_line, bool rend_mk_fill, ImU32 col_mk_fill, bool cull) {
2020-05-03 01:24:10 -04:00
int idx = offset;
2020-05-13 10:11:25 -04:00
for (int i = 0; i < count; ++i) {
2020-05-03 01:24:10 -04:00
ImVec2 c;
2020-05-11 07:12:22 -04:00
c = transformer(getter(idx));
2020-05-03 01:24:10 -04:00
idx = (idx + 1) % count;
if (!cull || gp.BB_Grid.Contains(c)) {
// TODO: Optimize the loop and if statements, this is atrocious
2020-05-13 10:11:25 -04:00
if (HasFlag(gp.Style.Marker, ImPlotMarker_Circle))
2020-05-11 07:12:22 -04:00
MarkerCircle(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Square))
2020-05-13 10:11:25 -04:00
MarkerSquare(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Diamond))
2020-05-03 01:24:10 -04:00
MarkerDiamond(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Up))
2020-05-13 10:11:25 -04:00
MarkerUp(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Down))
MarkerDown(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Left))
2020-05-13 10:11:25 -04:00
MarkerLeft(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Right))
MarkerRight(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Cross))
2020-05-13 10:11:25 -04:00
MarkerCross(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Plus))
2020-05-13 10:11:25 -04:00
MarkerPlus(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
if (HasFlag(gp.Style.Marker, ImPlotMarker_Asterisk))
2020-05-13 10:11:25 -04:00
MarkerAsterisk(DrawList, c, gp.Style.MarkerSize, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, gp.Style.MarkerWeight);
2020-05-03 01:24:10 -04:00
}
2020-05-13 10:11:25 -04:00
}
2020-05-03 01:24:10 -04:00
}
inline void RenderLine(ImDrawList& DrawList, const ImVec2& p1, const ImVec2& p2, float line_weight, ImU32 col_line, ImVec2 uv) {
// http://assemblyrequired.crashworks.org/timing-square-root/
float dx = p2.x - p1.x;
float dy = p2.y - p1.y;
IM_NORMALIZE2F_OVER_ZERO(dx, dy);
dx *= (line_weight * 0.5f);
2020-05-13 10:11:25 -04:00
dy *= (line_weight * 0.5f);
2020-05-03 01:24:10 -04:00
DrawList._VtxWritePtr[0].pos.x = p1.x + dy;
DrawList._VtxWritePtr[0].pos.y = p1.y - dx;
DrawList._VtxWritePtr[0].uv = uv;
DrawList._VtxWritePtr[0].col = col_line;
DrawList._VtxWritePtr[1].pos.x = p2.x + dy;
DrawList._VtxWritePtr[1].pos.y = p2.y - dx;
DrawList._VtxWritePtr[1].uv = uv;
DrawList._VtxWritePtr[1].col = col_line;
DrawList._VtxWritePtr[2].pos.x = p2.x - dy;
DrawList._VtxWritePtr[2].pos.y = p2.y + dx;
DrawList._VtxWritePtr[2].uv = uv;
DrawList._VtxWritePtr[2].col = col_line;
DrawList._VtxWritePtr[3].pos.x = p1.x - dy;
DrawList._VtxWritePtr[3].pos.y = p1.y + dx;
DrawList._VtxWritePtr[3].uv = uv;
DrawList._VtxWritePtr[3].col = col_line;
DrawList._VtxWritePtr += 4;
DrawList._IdxWritePtr[0] = (ImDrawIdx)(DrawList._VtxCurrentIdx);
DrawList._IdxWritePtr[1] = (ImDrawIdx)(DrawList._VtxCurrentIdx + 1);
DrawList._IdxWritePtr[2] = (ImDrawIdx)(DrawList._VtxCurrentIdx + 2);
DrawList._IdxWritePtr[3] = (ImDrawIdx)(DrawList._VtxCurrentIdx);
DrawList._IdxWritePtr[4] = (ImDrawIdx)(DrawList._VtxCurrentIdx + 2);
DrawList._IdxWritePtr[5] = (ImDrawIdx)(DrawList._VtxCurrentIdx + 3);
DrawList._IdxWritePtr += 6;
DrawList._VtxCurrentIdx += 4;
}
inline void RenderLineAA(ImDrawList& DrawList, const ImVec2& p1, const ImVec2& p2, float line_weight, ImU32 col_line) {
DrawList.AddLine(p1, p2, col_line, line_weight);
}
template <typename Transformer, typename Getter>
inline void RenderLineStrip(Transformer transformer, ImDrawList& DrawList, Getter getter, int count, int offset, float line_weight, ImU32 col_line, bool cull) {
2020-05-03 01:24:10 -04:00
// render line segments
2020-05-13 10:11:25 -04:00
offset %= count;
if (offset < 0) offset += count; // shift negative offset to positive range
int i_start = offset + 1;
if (i_start >= count ) i_start -= count;
int i_end = offset + count;
if (i_end >= count) i_end -= count;
const int segments = count - 1;
ImVec2 p1 = transformer(getter(offset));
if (HasFlag(gp.CurrentPlot->Flags, ImPlotFlags_AntiAliased)) {
for (int i1 = i_start; i1 != i_end; i1 = i1 + 1 < count ? i1 + 1 : i1 + 1 - count) {
ImVec2 p2 = transformer(getter(i1));
if (!cull || gp.BB_Grid.Overlaps(ImRect(ImMin(p1,p2), ImMax(p1,p2))))
RenderLineAA(DrawList, p1, p2, line_weight, col_line);
p1 = p2;
}
}
else {
const ImVec2 uv = DrawList._Data->TexUvWhitePixel;
DrawList.PrimReserve(segments * 6, segments * 4);
int segments_culled = 0;
for (int i1 = i_start; i1 != i_end; i1 = i1 + 1 < count ? i1 + 1 : i1 + 1 - count) {
ImVec2 p2 = transformer(getter(i1));
if (!cull || gp.BB_Grid.Overlaps(ImRect(ImMin(p1, p2), ImMax(p1, p2))))
RenderLine(DrawList, p1, p2, line_weight, col_line, uv);
else
segments_culled++;
p1 = p2;
}
if (segments_culled > 0)
DrawList.PrimUnreserve(segments_culled * 6, segments_culled * 4);
}
2020-05-03 01:24:10 -04:00
}
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// DATA GETTERS
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
inline float StrideIndex(const float* data, int idx, int stride) {
return *(const float*)(const void*)((const unsigned char*)data + (size_t)idx * stride);
}
struct GetterYs {
GetterYs(const float* ys, int stride) { Ys = ys; Stride = stride; }
2020-04-27 11:27:59 -04:00
const float* Ys;
int Stride;
2020-05-03 01:24:10 -04:00
inline ImVec2 operator()(int idx) {
return ImVec2((float)idx, StrideIndex(Ys, idx, Stride));
2020-04-27 11:27:59 -04:00
}
};
2020-05-03 01:24:10 -04:00
struct Getter2D {
Getter2D(const float* xs, const float* ys, int stride) { Xs = xs; Ys = ys; Stride = stride; }
const float* Xs;
const float* Ys;
int Stride;
inline ImVec2 operator()(int idx) {
return ImVec2(StrideIndex(Xs, idx, Stride), StrideIndex(Ys, idx, Stride));
}
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
struct GetterImVec2 {
GetterImVec2(const ImVec2* data) { Data = data; }
inline ImVec2 operator()(int idx) { return Data[idx]; }
const ImVec2* Data;
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
struct GetterFuncPtrImVec2 {
GetterFuncPtrImVec2(ImVec2 (*g)(void* data, int idx), void* d) { getter = g; data = d;}
ImVec2 operator()(int idx) { return getter(data, idx); }
ImVec2 (*getter)(void* data, int idx);
void* data;
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
struct GetterFuncPtrImVec4 {
GetterFuncPtrImVec4(ImVec4 (*g)(void* data, int idx), void* d) { getter = g; data = d;}
ImVec4 operator()(int idx) { return getter(data, idx); }
ImVec4 (*getter)(void* data, int idx);
void* data;
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// PLOT
//-----------------------------------------------------------------------------
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
template <typename Getter>
inline void PlotEx(const char* label_id, Getter getter, int count, int offset)
2020-04-27 11:27:59 -04:00
{
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "Plot() Needs to be called between BeginPlot() and EndPlot()!");
2020-05-12 05:19:04 -04:00
ImPlotState* plot = gp.CurrentPlot;
2020-05-11 07:12:22 -04:00
const int y_axis = plot->CurrentYAxis;
2020-05-03 01:24:10 -04:00
ImPlotItem* item = RegisterItem(label_id);
2020-04-27 11:27:59 -04:00
if (!item->Show)
return;
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
const bool rend_line = gp.Style.Colors[ImPlotCol_Line].w != 0 && gp.Style.LineWeight > 0;
const bool rend_mk_line = gp.Style.Colors[ImPlotCol_MarkerOutline].w != 0 && gp.Style.MarkerWeight > 0;
const bool rend_mk_fill = gp.Style.Colors[ImPlotCol_MarkerFill].w != 0;
2020-05-12 05:19:04 -04:00
ImU32 col_line = gp.Style.Colors[ImPlotCol_Line].w == -1 ? ImGui::GetColorU32(item->Color) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Line]);
ImU32 col_mk_line = gp.Style.Colors[ImPlotCol_MarkerOutline].w == -1 ? col_line : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_MarkerOutline]);
ImU32 col_mk_fill = gp.Style.Colors[ImPlotCol_MarkerFill].w == -1 ? col_line : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_MarkerFill]);
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
const float line_weight = item->Highlight ? gp.Style.LineWeight * 2 : gp.Style.LineWeight;
2020-04-27 11:27:59 -04:00
if (gp.Style.Colors[ImPlotCol_Line].w != -1)
item->Color = gp.Style.Colors[ImPlotCol_Line];
2020-05-11 07:12:22 -04:00
bool cull = HasFlag(plot->Flags, ImPlotFlags_CullData);
2020-05-03 01:24:10 -04:00
2020-04-27 11:27:59 -04:00
// find data extents
if (gp.FitThisFrame) {
for (int i = 0; i < count; ++i) {
2020-05-03 01:24:10 -04:00
ImVec2 p = getter(i);
FitPoint(p);
2020-04-27 11:27:59 -04:00
}
}
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
2020-04-27 11:27:59 -04:00
if (count > 1 && rend_line) {
if (HasFlag(plot->XAxis.Flags, ImPlotAxisFlags_LogScale) && HasFlag(plot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale))
RenderLineStrip(Plt2PixLogLog(y_axis), DrawList, getter, count, offset, line_weight, col_line, cull);
else if (HasFlag(plot->XAxis.Flags, ImPlotAxisFlags_LogScale))
RenderLineStrip(Plt2PixLogLin(y_axis), DrawList, getter, count, offset, line_weight, col_line, cull);
else if (HasFlag(plot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale))
RenderLineStrip(Plt2PixLinLog(y_axis), DrawList, getter, count, offset, line_weight, col_line, cull);
2020-05-03 01:24:10 -04:00
else
RenderLineStrip(Plt2PixLinLin(y_axis), DrawList, getter, count, offset, line_weight, col_line, cull);
2020-05-03 01:24:10 -04:00
}
2020-05-04 03:09:33 -04:00
// render markers
if (gp.Style.Marker != ImPlotMarker_None) {
if (HasFlag(plot->XAxis.Flags, ImPlotAxisFlags_LogScale) && HasFlag(plot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
RenderMarkers(Plt2PixLogLog(y_axis), DrawList, getter, count, offset, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, cull);
else if (HasFlag(plot->XAxis.Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
RenderMarkers(Plt2PixLogLin(y_axis), DrawList, getter, count, offset, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, cull);
else if (HasFlag(plot->YAxis[y_axis].Flags, ImPlotAxisFlags_LogScale))
2020-05-11 07:12:22 -04:00
RenderMarkers(Plt2PixLinLog(y_axis), DrawList, getter, count, offset, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, cull);
2020-05-03 01:24:10 -04:00
else
2020-05-11 07:12:22 -04:00
RenderMarkers(Plt2PixLinLin(y_axis), DrawList, getter, count, offset, rend_mk_line, col_mk_line, rend_mk_fill, col_mk_fill, cull);
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
}
void PlotLine(const char* label_id, const float* values, int count, int offset, int stride) {
2020-05-03 01:24:10 -04:00
GetterYs getter(values,stride);
PlotEx(label_id, getter, count, offset);
2020-04-29 10:32:35 -04:00
}
void PlotLine(const char* label_id, const float* xs, const float* ys, int count, int offset, int stride) {
2020-05-03 01:24:10 -04:00
Getter2D getter(xs,ys,stride);
return PlotEx(label_id, getter, count, offset);
2020-04-27 11:27:59 -04:00
}
2020-04-30 09:45:03 -04:00
void PlotLine(const char* label_id, const ImVec2* data, int count, int offset) {
2020-05-03 01:24:10 -04:00
GetterImVec2 getter(data);
return PlotEx(label_id, getter, count, offset);
2020-04-27 11:27:59 -04:00
}
2020-04-30 09:45:03 -04:00
void PlotLine(const char* label_id, ImVec2 (*getter_func)(void* data, int idx), void* data, int count, int offset) {
2020-05-03 01:24:10 -04:00
GetterFuncPtrImVec2 getter(getter_func,data);
return PlotEx(label_id, getter, count, offset);
2020-04-29 10:32:35 -04:00
}
//-----------------------------------------------------------------------------
// PLOT SCATTER
//-----------------------------------------------------------------------------
void PlotScatter(const char* label_id, const float* values, int count, int offset, int stride) {
int pops = 1;
PushStyleVar(ImPlotStyleVar_LineWeight, 0);
if (GetStyle().Marker == ImPlotMarker_None) {
PushStyleVar(ImPlotStyleVar_Marker, ImPlotMarker_Circle);
pops++;
}
PlotLine(label_id, values, count, offset, stride);
PopStyleVar(pops);
}
void PlotScatter(const char* label_id, const float* xs, const float* ys, int count, int offset, int stride) {
int pops = 1;
PushStyleVar(ImPlotStyleVar_LineWeight, 0);
if (GetStyle().Marker == ImPlotMarker_None) {
PushStyleVar(ImPlotStyleVar_Marker, ImPlotMarker_Circle);
pops++;
}
PlotLine(label_id, xs, ys, count, offset, stride);
PopStyleVar(pops);
}
void PlotScatter(const char* label_id, const ImVec2* data, int count, int offset) {
int pops = 1;
PushStyleVar(ImPlotStyleVar_LineWeight, 0);
if (GetStyle().Marker == ImPlotMarker_None) {
PushStyleVar(ImPlotStyleVar_Marker, ImPlotMarker_Circle);
pops++;
}
PlotLine(label_id, data, count, offset);
PopStyleVar(pops);
}
void PlotScatter(const char* label_id, ImVec2 (*getter)(void* data, int idx), void* data, int count, int offset) {
int pops = 1;
PushStyleVar(ImPlotStyleVar_LineWeight, 0);
if (GetStyle().Marker == ImPlotMarker_None) {
PushStyleVar(ImPlotStyleVar_Marker, ImPlotMarker_Circle);
pops++;
}
PlotLine(label_id, getter, data, count, offset);
PopStyleVar(pops);
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// PLOT BAR
//-----------------------------------------------------------------------------
2020-04-29 10:32:35 -04:00
2020-05-03 01:24:10 -04:00
struct GetterBarV {
const float* Ys; float XShift; int Stride;
GetterBarV(const float* ys, float xshift, int stride) { Ys = ys; XShift = xshift; Stride = stride; }
inline ImVec2 operator()(int idx) { return ImVec2((float)idx + XShift, StrideIndex(Ys, idx, Stride)); }
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
struct GetterBarH {
const float* Xs; float YShift; int Stride;
GetterBarH(const float* xs, float yshift, int stride) { Xs = xs; YShift = yshift; Stride = stride; }
inline ImVec2 operator()(int idx) { return ImVec2(StrideIndex(Xs, idx, Stride), (float)idx + YShift); }
};
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
template <typename Getter>
void PlotBarEx(const char* label_id, Getter getter, int count, float width, int offset) {
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "Bar() Needs to be called between BeginPlot() and EndPlot()!");
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
ImPlotItem* item = RegisterItem(label_id);
2020-04-27 11:27:59 -04:00
if (!item->Show)
return;
2020-05-12 05:19:04 -04:00
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
2020-04-27 11:27:59 -04:00
bool rend_line = gp.Style.Colors[ImPlotCol_Line].w != 0 && gp.Style.LineWeight > 0;
bool rend_fill = gp.Style.Colors[ImPlotCol_Fill].w != 0;
2020-05-12 05:19:04 -04:00
ImU32 col_line = gp.Style.Colors[ImPlotCol_Line].w == -1 ? ImGui::GetColorU32(item->Color) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Line]);
ImU32 col_fill = gp.Style.Colors[ImPlotCol_Fill].w == -1 ? col_line : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Fill]);
2020-04-27 11:27:59 -04:00
if (rend_fill && col_line == col_fill)
rend_line = false;
if (gp.Style.Colors[ImPlotCol_Line].w != -1)
item->Color = gp.Style.Colors[ImPlotCol_Line];
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
2020-04-27 11:27:59 -04:00
float half_width = width * 0.5f;
// find data extents
if (gp.FitThisFrame) {
for (int i = 0; i < count; ++i) {
2020-05-03 01:24:10 -04:00
ImVec2 p = getter(i);
FitPoint(ImVec2(p.x - half_width, p.y));
FitPoint(ImVec2(p.x + half_width, 0));
2020-04-27 11:27:59 -04:00
}
}
int idx = offset;
2020-05-13 10:11:25 -04:00
for (int i = 0; i < count; ++i) {
2020-04-27 11:27:59 -04:00
ImVec2 p;
2020-05-03 01:24:10 -04:00
p = getter(idx);
2020-04-27 11:27:59 -04:00
idx = (idx + 1) % count;
if (p.y == 0)
continue;
2020-05-03 01:24:10 -04:00
ImVec2 a = PlotToPixels(p.x - half_width, p.y);
ImVec2 b = PlotToPixels(p.x + half_width, 0);
2020-04-27 11:27:59 -04:00
if (rend_fill)
DrawList.AddRectFilled(a, b, col_fill);
if (rend_line)
DrawList.AddRect(a, b, col_line);
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
2020-04-27 11:27:59 -04:00
}
void PlotBar(const char* label_id, const float* values, int count, float width, float shift, int offset, int stride) {
2020-05-03 01:24:10 -04:00
GetterBarV getter(values,shift,stride);
PlotBarEx(label_id, getter, count, width, offset);
2020-04-27 11:27:59 -04:00
}
void PlotBar(const char* label_id, const float* xs, const float* ys, int count, float width, int offset, int stride) {
2020-05-03 01:24:10 -04:00
Getter2D getter(xs,ys,stride);
PlotBarEx(label_id, getter, count, width, offset);
2020-04-27 11:27:59 -04:00
}
void PlotBar(const char* label_id, ImVec2 (*getter_func)(void* data, int idx), void* data, int count, float width, int offset) {
2020-05-03 01:24:10 -04:00
GetterFuncPtrImVec2 getter(getter_func, data);
PlotBarEx(label_id, getter, count, width, offset);
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
template <typename Getter>
void PlotBarHEx(const char* label_id, Getter getter, int count, float height, int offset) {
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "BarH() Needs to be called between BeginPlot() and EndPlot()!");
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
ImPlotItem* item = RegisterItem(label_id);
2020-04-27 11:27:59 -04:00
if (!item->Show)
return;
2020-05-12 05:19:04 -04:00
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
2020-04-27 11:27:59 -04:00
bool rend_line = gp.Style.Colors[ImPlotCol_Line].w != 0 && gp.Style.LineWeight > 0;
bool rend_fill = gp.Style.Colors[ImPlotCol_Fill].w != 0;
2020-05-12 05:19:04 -04:00
ImU32 col_line = gp.Style.Colors[ImPlotCol_Line].w == -1 ? ImGui::GetColorU32(item->Color) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Line]);
ImU32 col_fill = gp.Style.Colors[ImPlotCol_Fill].w == -1 ? col_line : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_Fill]);
2020-04-27 11:27:59 -04:00
if (rend_fill && col_line == col_fill)
rend_line = false;
if (gp.Style.Colors[ImPlotCol_Line].w != -1)
item->Color = gp.Style.Colors[ImPlotCol_Line];
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
2020-04-27 11:27:59 -04:00
float half_height = height * 0.5f;
// find data extents
if (gp.FitThisFrame) {
for (int i = 0; i < count; ++i) {
2020-05-03 01:24:10 -04:00
ImVec2 p = getter(i);
FitPoint(ImVec2(0, p.y - half_height));
FitPoint(ImVec2(p.x, p.y + half_height));
2020-04-27 11:27:59 -04:00
}
}
int idx = offset;
2020-05-13 10:11:25 -04:00
for (int i = 0; i < count; ++i) {
2020-04-27 11:27:59 -04:00
ImVec2 p;
2020-05-03 01:24:10 -04:00
p = getter(idx);
2020-04-27 11:27:59 -04:00
idx = (idx + 1) % count;
if (p.x == 0)
continue;
2020-05-03 01:24:10 -04:00
ImVec2 a = PlotToPixels(0, p.y - half_height);
ImVec2 b = PlotToPixels(p.x, p.y + half_height);
2020-04-27 11:27:59 -04:00
if (rend_fill)
DrawList.AddRectFilled(a, b, col_fill);
if (rend_line)
DrawList.AddRect(a, b, col_line);
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
2020-04-27 11:27:59 -04:00
}
void PlotBarH(const char* label_id, const float* values, int count, float height, float shift, int offset, int stride) {
2020-05-03 01:24:10 -04:00
GetterBarH getter(values,shift,stride);
PlotBarHEx(label_id, getter, count, height, offset);
2020-04-27 11:27:59 -04:00
}
void PlotBarH(const char* label_id, const float* xs, const float* ys, int count, float height, int offset, int stride) {
2020-05-03 01:24:10 -04:00
Getter2D getter(xs,ys,stride);
PlotBarHEx(label_id, getter, count, height, offset);
2020-04-27 11:27:59 -04:00
}
void PlotBarH(const char* label_id, ImVec2 (*getter_func)(void* data, int idx), void* data, int count, float height, int offset) {
2020-05-03 01:24:10 -04:00
GetterFuncPtrImVec2 getter(getter_func, data);
PlotBarHEx(label_id, getter, count, height, offset);
2020-04-27 11:27:59 -04:00
}
2020-05-03 01:24:10 -04:00
//-----------------------------------------------------------------------------
// PLOT ERROR BARS
//-----------------------------------------------------------------------------
struct GetterError {
const float* Xs; const float* Ys; const float* Neg; const float* Pos; int Stride;
GetterError(const float* xs, const float* ys, const float* neg, const float* pos, int stride) {
Xs = xs; Ys = ys; Neg = neg; Pos = pos; Stride = stride;
}
ImVec4 operator()(int idx) {
2020-05-13 10:11:25 -04:00
return ImVec4(StrideIndex(Xs, idx, Stride),
2020-05-03 01:24:10 -04:00
StrideIndex(Ys, idx, Stride),
StrideIndex(Neg, idx, Stride),
StrideIndex(Pos, idx, Stride));
}
};
template <typename Getter>
void PlotErrorBarsEx(const char* label_id, Getter getter, int count, int offset) {
2020-05-12 05:19:04 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "ErrorBars() Needs to be called between BeginPlot() and EndPlot()!");
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
ImGuiID id = ImGui::GetID(label_id);
2020-04-27 11:27:59 -04:00
ImPlotItem* item = gp.CurrentPlot->Items.GetByKey(id);
if (item != NULL && item->Show == false)
return;
2020-05-12 05:19:04 -04:00
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
2020-04-27 11:27:59 -04:00
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
2020-04-27 11:27:59 -04:00
2020-05-12 05:19:04 -04:00
const ImU32 col = gp.Style.Colors[ImPlotCol_ErrorBar].w == -1 ? ImGui::GetColorU32(ImGuiCol_Text) : ImGui::GetColorU32(gp.Style.Colors[ImPlotCol_ErrorBar]);
2020-04-27 11:27:59 -04:00
const bool rend_whisker = gp.Style.ErrorBarSize > 0;
const float half_whisker = gp.Style.ErrorBarSize * 0.5f;
// find data extents
if (gp.FitThisFrame) {
for (int i = 0; i < count; ++i) {
2020-05-03 01:24:10 -04:00
ImVec4 e = getter(i);
FitPoint(ImVec2(e.x , e.y - e.z));
FitPoint(ImVec2(e.x , e.y + e.w ));
2020-04-27 11:27:59 -04:00
}
}
int idx = offset;
for (int i = 0; i < count; ++i) {
ImVec4 e;
2020-05-03 01:24:10 -04:00
e = getter(idx);
2020-04-27 11:27:59 -04:00
idx = (idx + 1) % count;
2020-05-03 01:24:10 -04:00
ImVec2 p1 = PlotToPixels(e.x, e.y - e.z);
ImVec2 p2 = PlotToPixels(e.x, e.y + e.w);
2020-04-27 11:27:59 -04:00
DrawList.AddLine(p1,p2,col, gp.Style.ErrorBarWeight);
if (rend_whisker) {
DrawList.AddLine(p1 - ImVec2(half_whisker, 0), p1 + ImVec2(half_whisker, 0), col, gp.Style.ErrorBarWeight);
DrawList.AddLine(p2 - ImVec2(half_whisker, 0), p2 + ImVec2(half_whisker, 0), col, gp.Style.ErrorBarWeight);
}
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
}
2020-04-27 11:27:59 -04:00
void PlotErrorBars(const char* label_id, const float* xs, const float* ys, const float* err, int count, int offset, int stride) {
2020-05-03 01:24:10 -04:00
GetterError getter(xs, ys, err, err, stride);
PlotErrorBarsEx(label_id, getter, count, offset);
}
void PlotErrorBars(const char* label_id, const float* xs, const float* ys, const float* neg, const float* pos, int count, int offset, int stride) {
2020-05-03 01:24:10 -04:00
GetterError getter(xs, ys, neg, pos, stride);
PlotErrorBarsEx(label_id, getter, count, offset);
}
void PlotErrorBars(const char* label_id, ImVec4 (*getter_func)(void* data, int idx), void* data, int count, int offset) {
2020-05-03 01:24:10 -04:00
GetterFuncPtrImVec4 getter(getter_func, data);
PlotErrorBarsEx(label_id, getter, count, offset);
}
//-----------------------------------------------------------------------------
// PLOT MISC
//-----------------------------------------------------------------------------
inline void DrawPieSlice(ImDrawList& DrawList, const ImVec2& center, float radius, float a0, float a1, ImU32 col) {
static const float resolution = 50 / (2 * IM_PI);
static ImVec2 buffer[50];
buffer[0] = PlotToPixels(center);
int n = ImMax(3, (int)((a1 - a0) * resolution));
float da = (a1 - a0) / (n - 1);
for (int i = 0; i < n; ++i) {
float a = a0 + i * da;
buffer[i + 1] = PlotToPixels(center.x + radius * cos(a), center.y + radius * sin(a));
}
DrawList.AddConvexPolyFilled(buffer, n + 1, col);
}
void PlotPieChart(const char** label_ids, float* values, int count, const ImVec2& center, float radius, bool show_percents, float angle0) {
2020-05-12 05:19:04 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "PieChart() Needs to be called between BeginPlot() and EndPlot()!");
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
2020-05-03 01:24:10 -04:00
float sum = 0;
for (int i = 0; i < count; ++i)
sum += values[i];
2020-05-13 10:11:25 -04:00
2020-05-03 01:24:10 -04:00
const bool normalize = sum > 1.0f;
PushPlotClipRect();
float a0 = angle0 * 2 * IM_PI / 360.0f;
float a1 = angle0 * 2 * IM_PI / 360.0f;
for (int i = 0; i < count; ++i) {
ImPlotItem* item = RegisterItem(label_ids[i]);
2020-05-12 05:19:04 -04:00
ImU32 col = ImGui::GetColorU32(item->Color);
2020-05-03 01:24:10 -04:00
float percent = normalize ? values[i] / sum : values[i];
a1 = a0 + 2 * IM_PI * percent;
if (item->Show) {
if (percent < 0.5) {
DrawPieSlice(DrawList, center, radius, a0, a1, col);
}
else {
DrawPieSlice(DrawList, center, radius, a0, a0 + (a1 - a0) * 0.5f, col);
DrawPieSlice(DrawList, center, radius, a0 + (a1 - a0) * 0.5f, a1, col);
}
if (show_percents) {
char buffer[8];
2020-05-03 01:24:10 -04:00
sprintf(buffer, "%.0f%%", percent * 100);
2020-05-12 05:19:04 -04:00
ImVec2 size = ImGui::CalcTextSize(buffer);
2020-05-03 01:24:10 -04:00
float angle = a0 + (a1 - a0) * 0.5f;
ImVec2 pos = PlotToPixels(center.x + 0.5f * radius * cos(angle), center.y + 0.5f * radius * sin(angle));
DrawList.AddText(pos - size * 0.5f + ImVec2(1,1), IM_COL32(0,0,0,255), buffer);
DrawList.AddText(pos - size * 0.5f, IM_COL32(255,255,255,255), buffer);
2020-05-03 01:24:10 -04:00
}
}
a0 = a1;
2020-05-13 10:11:25 -04:00
}
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
2020-04-27 11:27:59 -04:00
}
void PlotText(const char* text, float x, float y, bool vertical, const ImVec2& pixel_offset) {
2020-05-12 05:19:04 -04:00
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "Text() Needs to be called between BeginPlot() and EndPlot()!");
2020-04-27 11:27:59 -04:00
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
2020-05-03 01:24:10 -04:00
PushPlotClipRect();
ImVec2 pos = PlotToPixels(ImVec2(x,y)) + pixel_offset;
if (vertical)
AddTextVertical(&DrawList, text, pos, gp.Col_Txt);
else
DrawList.AddText(pos, gp.Col_Txt, text);
2020-05-03 01:24:10 -04:00
PopPlotClipRect();
2020-04-27 11:27:59 -04:00
}
2020-05-04 05:14:44 -04:00
template <typename Getter>
inline void PlotDigitalEx(const char* label_id, Getter getter, int count, int offset)
{
IM_ASSERT_USER_ERROR(gp.CurrentPlot != NULL, "Plot() Needs to be called between BeginPlot() and EndPlot()!");
ImPlotItem* item = RegisterItem(label_id);
if (!item->Show)
return;
ImDrawList & DrawList = *ImGui::GetWindowDrawList();
const bool rend_line = gp.Style.Colors[ImPlotCol_Line].w != 0 && gp.Style.LineWeight > 0;
if (gp.Style.Colors[ImPlotCol_Line].w != -1)
item->Color = gp.Style.Colors[ImPlotCol_Line];
2020-05-13 05:10:14 -04:00
// find data extents
2020-05-04 05:14:44 -04:00
if (gp.FitThisFrame) {
for (int i = 0; i < count; ++i) {
ImVec2 p = getter(i);
2020-05-13 05:10:14 -04:00
FitPoint(ImVec2(p.x, 0));
2020-05-04 05:14:44 -04:00
}
}
ImGui::PushClipRect(gp.BB_Grid.Min, gp.BB_Grid.Max, true);
bool cull = HasFlag(gp.CurrentPlot->Flags, ImPlotFlags_CullData);
const float line_weight = item->Highlight ? gp.Style.LineWeight * 2 : gp.Style.LineWeight;
2020-05-11 07:12:22 -04:00
const int ax = gp.CurrentPlot->CurrentYAxis;
2020-05-04 05:14:44 -04:00
// render digital signals as "pixel bases" rectangles
if (count > 1 && rend_line) {
2020-05-04 08:17:39 -04:00
//
2020-05-11 07:12:22 -04:00
const float mx = (gp.PixelRange[ax].Max.x - gp.PixelRange[ax].Min.x) / gp.CurrentPlot->XAxis.Range.Size();
2020-05-04 05:14:44 -04:00
const int segments = count - 1;
int i1 = offset;
int pixYMax = 0;
2020-05-04 05:14:44 -04:00
for (int s = 0; s < segments; ++s) {
const int i2 = (i1 + 1) % count;
ImVec2 itemData1 = getter(i1);
ImVec2 itemData2 = getter(i2);
i1 = i2;
int pixY_0 = (int)(line_weight);
float pixY_1_float = gp.Style.DigitalBitHeight * ImMax(0.0f, itemData1.y);
int pixY_1 = (int)(pixY_1_float); //allow only positive values
int pixY_chPosOffset = (int)(ImMax(gp.Style.DigitalBitHeight, pixY_1_float) + gp.Style.DigitalBitGap);
pixYMax = ImMax(pixYMax, pixY_chPosOffset);
ImVec2 pMin, pMax;
2020-05-11 07:12:22 -04:00
pMin.x = gp.PixelRange[ax].Min.x + mx * (itemData1.x - gp.CurrentPlot->XAxis.Range.Min);
pMax.x = gp.PixelRange[ax].Min.x + mx * (itemData2.x - gp.CurrentPlot->XAxis.Range.Min);
int pixY_Offset = 20;//20 pixel from bottom due to mouse cursor label
2020-05-11 07:12:22 -04:00
pMin.y = (gp.PixelRange[ax].Min.y) + ((-gp.DigitalPlotOffset) - pixY_Offset);
pMax.y = (gp.PixelRange[ax].Min.y) + ((-gp.DigitalPlotOffset) - pixY_0 - pixY_1 - pixY_Offset);
//plot only one rectangle for same digital state
while (((s+2) < segments) && (itemData1.y == itemData2.y)) {
const int i3 = (i1 + 1) % count;
itemData2 = getter(i3);
2020-05-11 07:12:22 -04:00
pMax.x = gp.PixelRange[ax].Min.x + mx * (itemData2.x - gp.CurrentPlot->XAxis.Range.Min);
i1 = i3;
2020-05-04 08:17:39 -04:00
s++;
2020-05-13 10:11:25 -04:00
}
//do not extend plot outside plot range
2020-05-11 07:12:22 -04:00
if (pMin.x < gp.PixelRange[ax].Min.x) pMin.x = gp.PixelRange[ax].Min.x;
if (pMax.x < gp.PixelRange[ax].Min.x) pMax.x = gp.PixelRange[ax].Min.x;
if (pMin.x > gp.PixelRange[ax].Max.x) pMin.x = gp.PixelRange[ax].Max.x;
if (pMax.x > gp.PixelRange[ax].Max.x) pMax.x = gp.PixelRange[ax].Max.x;
//plot a rectangle that extends up to x2 with y1 height
if ((pMax.x > pMin.x) && (!cull || gp.BB_Grid.Contains(pMin) || gp.BB_Grid.Contains(pMax))) {
2020-05-11 07:12:22 -04:00
ImVec4 colAlpha = item->Color;
colAlpha.w = item->Highlight ? 1.0f : 0.9f;
2020-05-12 05:19:04 -04:00
DrawList.AddRectFilled(pMin, pMax, ImGui::GetColorU32(colAlpha));
2020-05-04 05:14:44 -04:00
}
}
gp.DigitalPlotItemCnt++;
gp.DigitalPlotOffset += pixYMax;
2020-05-13 10:11:25 -04:00
}
2020-05-04 05:14:44 -04:00
ImGui::PopClipRect();
}
void PlotDigital(const char* label_id, const float* xs, const float* ys, int count, int offset, int stride) {
2020-05-04 05:14:44 -04:00
Getter2D getter(xs,ys,stride);
return PlotDigitalEx(label_id, getter, count, offset);
}
void PlotDigital(const char* label_id, ImVec2 (*getter_func)(void* data, int idx), void* data, int count, int offset) {
2020-05-04 05:14:44 -04:00
GetterFuncPtrImVec2 getter(getter_func,data);
return PlotDigitalEx(label_id, getter, count, offset);
2020-04-27 11:27:59 -04:00
}
2020-05-13 11:25:26 -04:00
} // namespace ImPlot