tview/lib/ImNodeFlow.h

1319 lines
40 KiB
C
Raw Normal View History

2024-06-12 09:20:04 -04:00
#ifndef IM_NODE_FLOW
#define IM_NODE_FLOW
#pragma once
#include "context_wrapper.h"
#include "imgui.h"
#include "imgui_bezier_math.h"
#include "imgui_internal.h"
#include <algorithm>
#include <cmath>
#include <functional>
#include <iostream>
#include <memory>
#include <string>
#include <unordered_map>
#include <utility>
#include <vector>
namespace ImFlow {
// -----------------------------------------------------------------------------------------------------------------
// HELPERS
/**
* @brief <BR>Draw a sensible bezier between two points
* @param p1 Starting point
* @param p2 Ending point
* @param color Color of the curve
* @param thickness Thickness of the curve
*/
inline static void smart_bezier(const ImVec2 &p1, const ImVec2 &p2, ImU32 color,
float thickness);
/**
* @brief <BR>Collider checker for smart_bezier
* @details Projects the point "p" orthogonally onto the bezier curve and
* checks if the distance is less than the given radius.
* @param p Point to be tested
* @param p1 Starting point of smart_bezier
* @param p2 Ending point of smart_bezier
* @param radius Lateral width of the hit box
* @return [TRUE] if "p" is inside the collider
*
* Intended to be used in union with smart_bezier();
*/
inline static bool smart_bezier_collider(const ImVec2 &p, const ImVec2 &p1,
const ImVec2 &p2, float radius);
// -----------------------------------------------------------------------------------------------------------------
// CLASSES PRE-DEFINITIONS
template <typename T> class InPin;
template <typename T> class OutPin;
class Pin;
class BaseNode;
class ImNodeFlow;
// -----------------------------------------------------------------------------------------------------------------
// PIN'S PROPERTIES
/**
* @brief <BR>Basic filters
* @details List of, ready to use, basic filters. It's possible to create more
* filters with the help of "ConnectionFilter_MakeCustom".
*/
enum ConnectionFilter_ {
ConnectionFilter_None = 0,
ConnectionFilter_SameNode = 1 << 1,
ConnectionFilter_Bool = 1 << 2,
ConnectionFilter_Int = 1 << 3,
ConnectionFilter_Float = 1 << 4,
ConnectionFilter_Double = 1 << 5,
ConnectionFilter_String = 1 << 6,
ConnectionFilter_MakeCustom = 1 << 7,
ConnectionFilter_Numbers =
ConnectionFilter_Int | ConnectionFilter_Float | ConnectionFilter_Double
};
typedef long ConnectionFilter;
typedef unsigned long long int PinUID;
/**
* @brief Extra pin's style setting
*/
struct PinStyleExtras {
/// @brief Top and bottom spacing
ImVec2 padding = ImVec2(3.f, 1.f);
/// @brief Border and background corner rounding
float bg_radius = 8.f;
/// @brief Border thickness
float border_thickness = 1.f;
/// @brief Background color
ImU32 bg_color = IM_COL32(23, 16, 16, 0);
/// @brief Background color when hovered
ImU32 bg_hover_color = IM_COL32(100, 100, 255, 70);
/// @brief Border color
ImU32 border_color = IM_COL32(255, 255, 255, 0);
/// @brief Link thickness
float link_thickness = 2.6f;
/// @brief Link thickness when dragged
float link_dragged_thickness = 2.2f;
/// @brief Link thickness when hovered
float link_hovered_thickness = 3.5f;
/// @brief Thickness of the outline of a selected link
float link_selected_outline_thickness = 0.5f;
/// @brief Color of the outline of a selected link
ImU32 outline_color = IM_COL32(80, 20, 255, 200);
/// @brief Spacing between pin content and socket
float socket_padding = 6.6f;
};
/**
* @brief Defines the visual appearance of a pin
*/
class PinStyle {
public:
PinStyle(ImU32 color, int socket_shape, float socket_radius,
float socket_hovered_radius, float socket_connected_radius,
float socket_thickness)
: color(color), socket_shape(socket_shape), socket_radius(socket_radius),
socket_hovered_radius(socket_hovered_radius),
socket_connected_radius(socket_connected_radius),
socket_thickness(socket_thickness) {}
/// @brief Socket and link color
ImU32 color;
/// @brief Socket shape ID
int socket_shape;
/// @brief Socket radius
float socket_radius;
/// @brief Socket radius when hovered
float socket_hovered_radius;
/// @brief Socket radius when connected
float socket_connected_radius;
/// @brief Socket outline thickness when empty
float socket_thickness;
/// @brief List of less common properties
PinStyleExtras extra;
public:
/// @brief <BR>Default cyan style
static std::shared_ptr<PinStyle> cyan() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(87, 155, 185, 255), 0, 4.f, 4.67f, 3.7f, 1.f));
}
/// @brief <BR>Default green style
static std::shared_ptr<PinStyle> green() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(90, 191, 93, 255), 4, 4.f, 4.67f, 4.2f, 1.3f));
}
/// @brief <BR>Default blue style
static std::shared_ptr<PinStyle> blue() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(90, 117, 191, 255), 0, 4.f, 4.67f, 3.7f, 1.f));
}
/// @brief <BR>Default brown style
static std::shared_ptr<PinStyle> brown() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(191, 134, 90, 255), 0, 4.f, 4.67f, 3.7f, 1.f));
}
/// @brief <BR>Default red style
static std::shared_ptr<PinStyle> red() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(191, 90, 90, 255), 0, 4.f, 4.67f, 3.7f, 1.f));
}
/// @brief <BR>Default white style
static std::shared_ptr<PinStyle> white() {
return std::make_shared<PinStyle>(
PinStyle(IM_COL32(255, 255, 255, 255), 5, 4.f, 4.67f, 4.2f, 1.f));
}
};
// -----------------------------------------------------------------------------------------------------------------
// NODE'S PROPERTIES
typedef uintptr_t NodeUID;
/**
* @brief Defines the visual appearance of a node
*/
class NodeStyle {
public:
NodeStyle(ImU32 header_bg, ImColor header_title_color, float radius)
: header_bg(header_bg), header_title_color(header_title_color),
radius(radius) {}
/// @brief Body's background color
ImU32 bg = IM_COL32(55, 64, 75, 255);
/// @brief Header's background color
ImU32 header_bg;
/// @brief Header title color
ImColor header_title_color;
/// @brief Border color
ImU32 border_color = IM_COL32(30, 38, 41, 128);
/// @brief Border color when selected
ImU32 border_selected_color = IM_COL32(170, 190, 205, 255);
/// @brief Body's content padding (Left Top Right Bottom)
ImVec4 padding = ImVec4(13.7f, 6.f, 13.7f, 2.f);
/// @brief Edges rounding
float radius;
/// @brief Border thickness
float border_thickness = -1.35f;
/// @brief Border thickness when selected
float border_selected_thickness = 2.f;
public:
/// @brief <BR>Default cyan style
static std::shared_ptr<NodeStyle> cyan() {
return std::make_shared<NodeStyle>(IM_COL32(71, 142, 173, 255),
ImColor(233, 241, 244, 255), 6.5f);
}
/// @brief <BR>Default green style
static std::shared_ptr<NodeStyle> green() {
return std::make_shared<NodeStyle>(IM_COL32(90, 191, 93, 255),
ImColor(233, 241, 244, 255), 6.5f);
}
/// @brief <BR>Default red style
static std::shared_ptr<NodeStyle> red() {
return std::make_shared<NodeStyle>(IM_COL32(191, 90, 90, 255),
ImColor(233, 241, 244, 255), 6.f);
}
/// @brief <BR>Default brown style
static std::shared_ptr<NodeStyle> brown() {
return std::make_shared<NodeStyle>(IM_COL32(191, 134, 90, 255),
ImColor(233, 241, 244, 255), 6.5f);
}
};
// -----------------------------------------------------------------------------------------------------------------
// LINK
/**
* @brief Link between two Pins of two different Nodes
*/
class Link {
public:
/**
* @brief <BR>Construct a link
* @param left Pointer to the output Pin of the Link
* @param right Pointer to the input Pin of the Link
* @param inf Pointer to the Handler that contains the Link
*/
explicit Link(Pin *left, Pin *right, ImNodeFlow *inf)
: m_left(left), m_right(right), m_inf(inf) {}
/**
* @brief <BR>Destruction of a link
* @details Deletes references of this links form connected pins
*/
~Link();
/**
* @brief <BR>Looping function to update the Link
* @details Draws the Link and updates Hovering and Selected status.
*/
void update();
/**
* @brief <BR>Get Left pin of the link
* @return Pointer to the Pin
*/
[[nodiscard]] Pin *left() const { return m_left; }
/**
* @brief <BR>Get Right pin of the link
* @return Pointer to the Pin
*/
[[nodiscard]] Pin *right() const { return m_right; }
/**
* @brief <BR>Get hovering status
* @return [TRUE] If the link is hovered in the current frame
*/
[[nodiscard]] bool isHovered() const { return m_hovered; }
/**
* @brief <BR>Get selected status
* @return [TRUE] If the link is selected in the current frame
*/
[[nodiscard]] bool isSelected() const { return m_selected; }
private:
ImNodeFlow *m_inf;
Pin *m_left;
Pin *m_right;
bool m_hovered = false;
bool m_selected = false;
};
// -----------------------------------------------------------------------------------------------------------------
// HANDLER
/**
* @brief Grid's the color parameters
*/
struct InfColors {
/// @brief Background of the grid
ImU32 background = IM_COL32(33, 41, 45, 255);
/// @brief Main lines of the grid
ImU32 grid = IM_COL32(200, 200, 200, 40);
/// @brief Secondary lines
ImU32 subGrid = IM_COL32(200, 200, 200, 10);
};
/**
* @brief ALl the grid's appearance parameters. Sizes + Colors
*/
struct InfStyler {
/// @brief Size of main grid
float grid_size = 50.f;
/// @brief Sub-grid divisions for Node snapping
float grid_subdivisions = 5.f;
/// @brief ImNodeFlow colors
InfColors colors;
};
/**
* @brief Main node editor
* @details Handles the infinite grid, nodes and links. Also handles all the
* logic.
*/
class ImNodeFlow {
private:
static int m_instances;
public:
/**
* @brief <BR>Instantiate a new editor with default name.
* <BR> Editor name will be "FlowGrid + the number of editors"
*/
ImNodeFlow() : ImNodeFlow("FlowGrid" + std::to_string(m_instances)) {}
/**
* @brief <BR>Instantiate a new editor with given name
* @details Creates a new Node Editor with the given name.
* @param name Name of the editor
*/
explicit ImNodeFlow(std::string name) : m_name(std::move(name)) {
m_instances++;
m_context.config().extra_window_wrapper = true;
m_context.config().color = m_style.colors.background;
}
/**
* @brief <BR>Handler loop
* @details Main update function. Refreshes all the logic and draws
* everything. Must be called every frame.
*/
void update();
/**
* @brief <BR>Add a node to the grid
* @tparam T Derived class of <BaseNode> to be added
* @tparam Params types of optional args to forward to derived class ctor
* @param pos Position of the Node in grid coordinates
* @param args Optional arguments to be forwarded to derived class ctor
* @return Shared pointer of the pushed type to the newly added node
*
* Inheritance is checked at compile time, \<T> MUST be derived from BaseNode.
*/
template <typename T, typename... Params>
std::shared_ptr<T> addNode(const ImVec2 &pos, Params &&...args);
/**
* @brief <BR>Add a node to the grid
* @tparam T Derived class of <BaseNode> to be added
* @tparam Params types of optional args to forward to derived class ctor
* @param pos Position of the Node in screen coordinates
* @param args Optional arguments to be forwarded to derived class ctor
* @return Shared pointer of the pushed type to the newly added node
*
* Inheritance is checked at compile time, \<T> MUST be derived from BaseNode.
*/
template <typename T, typename... Params>
std::shared_ptr<T> placeNodeAt(const ImVec2 &pos, Params &&...args);
/**
* @brief <BR>Add a node to the grid using mouse position
* @tparam T Derived class of <BaseNode> to be added
* @tparam Params types of optional args to forward to derived class ctor
* @param args Optional arguments to be forwarded to derived class ctor
* @return Shared pointer of the pushed type to the newly added node
*
* Inheritance is checked at compile time, \<T> MUST be derived from BaseNode.
*/
template <typename T, typename... Params>
std::shared_ptr<T> placeNode(Params &&...args);
/**
* @brief <BR>Add link to the handler internal list
* @param link Reference to the link
*/
void addLink(std::shared_ptr<Link> &link);
/**
* @brief <BR>Pop-up when link is "dropped"
* @details Sets the content of a pop-up that can be displayed when dragging a
* link in the open instead of onto another pin.
* @details If "key = ImGuiKey_None" the pop-up will always open when a link
* is dropped.
* @param content Function or Lambda containing only the contents of the
* pop-up and the subsequent logic
* @param key Optional key required in order to open the pop-up
*/
void droppedLinkPopUpContent(std::function<void(Pin *dragged)> content,
ImGuiKey key = ImGuiKey_None) {
m_droppedLinkPopUp = std::move(content);
m_droppedLinkPupUpComboKey = key;
}
/**
* @brief <BR>Pop-up when right-clicking
* @details Sets the content of a pop-up that can be displayed when
* right-clicking on the grid.
* @param content Function or Lambda containing only the contents of the
* pop-up and the subsequent logic
*/
void rightClickPopUpContent(std::function<void(BaseNode *node)> content) {
m_rightClickPopUp = std::move(content);
}
/**
* @brief <BR>Get mouse clicking status
* @return [TRUE] if mouse is clicked and click hasn't been consumed
*/
[[nodiscard]] bool getSingleUseClick() const { return m_singleUseClick; }
/**
* @brief <BR>Consume the click for the given frame
*/
void consumeSingleUseClick() { m_singleUseClick = false; }
/**
* @brief <BR>Get editor's name
* @return Const reference to editor's name
*/
const std::string &getName() { return m_name; }
/**
* @brief <BR>Get editor's position
* @return Const reference to editor's position in screen coordinates
*/
const ImVec2 &getPos() { return m_context.origin(); }
/**
* @brief <BR>Get editor's grid scroll
* @details Scroll is the offset from the origin of the grid, changes while
* navigating the grid.
* @return Const reference to editor's grid scroll
*/
const ImVec2 &getScroll() { return m_context.scroll(); }
/**
* @brief <BR>Get editor's list of nodes
* @return Const reference to editor's internal nodes list
*/
std::unordered_map<NodeUID, std::shared_ptr<BaseNode>> &getNodes() {
return m_nodes;
}
/**
* @brief <BR>Get nodes count
* @return Number of nodes present in the editor
*/
uint32_t getNodesCount() { return (uint32_t)m_nodes.size(); }
/**
* @brief <BR>Get editor's list of links
* @return Const reference to editor's internal links list
*/
const std::vector<std::weak_ptr<Link>> &getLinks() { return m_links; }
/**
* @brief <BR>Get zooming viewport
* @return Const reference to editor's internal viewport for zoom support
*/
ContainedContext &getGrid() { return m_context; }
/**
* @brief <BR>Get dragging status
* @return [TRUE] if a Node is being dragged around the grid
*/
[[nodiscard]] bool isNodeDragged() const { return m_draggingNode; }
/**
* @brief <BR>Get current style
* @return Reference to style variables
*/
InfStyler &getStyle() { return m_style; }
/**
* @brief <BR>Set editor's size
* @param size Editor's size. Set to (0, 0) to auto-fit.
*/
void setSize(const ImVec2 &size) { m_context.config().size = size; }
/**
* @brief <BR>Set dragging status
* @param state New dragging state
*
* The new state will only be updated one at the start of each frame.
*/
void draggingNode(bool state) { m_draggingNodeNext = state; }
/**
* @brief <BR>Set what pin is being hovered
* @param hovering Pointer to the hovered pin
*/
void hovering(Pin *hovering) { m_hovering = hovering; }
/**
* @brief <BR>Set what node is being hovered
* @param hovering Pointer to the hovered node
*/
void hoveredNode(BaseNode *hovering) { m_hoveredNode = hovering; }
/**
* @brief <BR>Convert coordinates from screen to grid
* @param p Point in screen coordinates to be converted
* @return Point in grid's coordinates
*/
ImVec2 screen2grid(const ImVec2 &p);
/**
* @brief <BR>Convert coordinates from grid to screen
* @param p Point in grid's coordinates to be converted
* @return Point in screen coordinates
*/
ImVec2 grid2screen(const ImVec2 &p);
/**
* @brief <BR>Check if mouse is on selected node
* @return [TRUE] if the mouse is hovering a selected node
*/
bool on_selected_node();
/**
* @brief <BR>Check if mouse is on a free point on the grid
* @return [TRUE] if the mouse is not hovering a node or a link
*/
bool on_free_space();
private:
std::string m_name;
ContainedContext m_context;
bool m_singleUseClick = false;
std::unordered_map<NodeUID, std::shared_ptr<BaseNode>> m_nodes;
std::vector<std::weak_ptr<Link>> m_links;
std::function<void(Pin *dragged)> m_droppedLinkPopUp;
ImGuiKey m_droppedLinkPupUpComboKey = ImGuiKey_None;
Pin *m_droppedLinkLeft = nullptr;
std::function<void(BaseNode *node)> m_rightClickPopUp;
BaseNode *m_hoveredNodeAux = nullptr;
BaseNode *m_hoveredNode = nullptr;
bool m_draggingNode = false, m_draggingNodeNext = false;
Pin *m_hovering = nullptr;
Pin *m_dragOut = nullptr;
InfStyler m_style;
};
// -----------------------------------------------------------------------------------------------------------------
// BASE NODE
/**
* @brief Parent class for custom nodes
* @details Main class from which custom nodes can be created. All interactions
* with the main grid are handled internally.
*/
class BaseNode {
public:
BaseNode() = default;
/**
* @brief <BR>Main loop of the node
* @details Updates position, hovering and selected status, and renders the
* node. Must be called each frame.
*/
void update();
/**
* @brief <BR>Content of the node
* @details Function to be implemented by derived custom nodes.
* Must contain the body of the node. If left empty the node will
* only have input and output pins.
*/
virtual void draw() {}
/**
* @brief <BR>Add an Input to the node
* @details Will add an Input pin to the node with the given name and data
* type. <BR> <BR> In this case the name of the pin will also be its UID. <BR>
* <BR> The UID must be unique only in the context of the current node's
* inputs.
* @tparam T Type of the data the pin will handle
* @param name Name of the pin
* @param defReturn Default return value when the pin is not connected
* @param filter Connection filter
* @param style Style of the pin
* @return Shared pointer to the newly added pin
*/
template <typename T>
std::shared_ptr<InPin<T>>
addIN(const std::string &name, T defReturn,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Add an Input to the node
* @details Will add an Input pin to the node with the given name and data
* type. <BR> <BR> The UID must be unique only in the context of the current
* node's inputs.
* @tparam T Type of the data the pin will handle
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @param name Name of the pin
* @param defReturn Default return value when the pin is not connected
* @param filter Connection filter
* @param style Style of the pin
* @return Shared pointer to the newly added pin
*/
template <typename T, typename U>
std::shared_ptr<InPin<T>>
addIN_uid(const U &uid, const std::string &name, T defReturn,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Remove input pin
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
*/
template <typename U> void dropIN(const U &uid);
/**
* @brief <BR>Remove input pin
* @param uid Unique identifier of the pin
*/
void dropIN(const char *uid);
/**
* @brief <BR>Show a temporary input pin
* @details Will show an input pin with the given name.
* The pin is created the first time showIN is called and kept alive
* as long as showIN is called each frame. <BR> <BR> In this case the name of
* the pin will also be its UID. <BR> <BR> The UID must be unique only in the
* context of the current node's inputs.
* @tparam T Type of the data the pin will handle
* @param name Name of the pin
* @param defReturn Default return value when the pin is not connected
* @param filter Connection filter
* @param style Style of the pin
* @return Const reference to the value of the connected link for the current
* frame of defReturn
*/
template <typename T>
const T &showIN(const std::string &name, T defReturn,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Show a temporary input pin
* @details Will show an input pin with the given name and UID.
* The pin is created the first time showIN_uid is called and kept
* alive as long as showIN_uid is called each frame. <BR> <BR> The UID must be
* unique only in the context of the current node's inputs.
* @tparam T Type of the data the pin will handle
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @param name Name of the pin
* @param defReturn Default return value when the pin is not connected
* @param filter Connection filter
* @param style Style of the pin
* @return Const reference to the value of the connected link for the current
* frame of defReturn
*/
template <typename T, typename U>
const T &showIN_uid(const U &uid, const std::string &name, T defReturn,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Add an Output to the node
* @details Must be called in the node constructor. WIll add an Output pin to
* the node with the given name and data type. <BR> <BR> In this case the name
* of the pin will also be its UID. <BR> <BR> The UID must be unique only in
* the context of the current node's outputs.
* @tparam T Type of the data the pin will handle
* @param name Name of the pin
* @param filter Connection filter
* @param style Style of the pin
* @return Shared pointer to the newly added pin. Must be used to set the
* behaviour
*/
template <typename T>
[[nodiscard]] std::shared_ptr<OutPin<T>>
addOUT(const std::string &name,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Add an Output to the node
* @details Must be called in the node constructor. WIll add an Output pin to
* the node with the given name and data type. <BR> <BR> The UID must be
* unique only in the context of the current node's outputs.
* @tparam T Type of the data the pin will handle
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @param name Name of the pin
* @param filter Connection filter
* @param style Style of the pin
* @return Shared pointer to the newly added pin. Must be used to set the
* behaviour
*/
template <typename T, typename U>
[[nodiscard]] std::shared_ptr<OutPin<T>>
addOUT_uid(const U &uid, const std::string &name,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Remove output pin
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
*/
template <typename U> void dropOUT(const U &uid);
/**
* @brief <BR>Remove output pin
* @param uid Unique identifier of the pin
*/
void dropOUT(const char *uid);
/**
* @brief <BR>Show a temporary output pin
* @details Will show an output pin with the given name.
* The pin is created the first time showOUT is called and kept alive
* as long as showOUT is called each frame. <BR> <BR> In this case the name of
* the pin will also be its UID. <BR> <BR> The UID must be unique only in the
* context of the current node's outputs.
* @tparam T Type of the data the pin will handle
* @param name Name of the pin
* @param behaviour Function or lambda expression used to calculate output
* value
* @param filter Connection filter
* @param style Style of the pin
*/
template <typename T>
void showOUT(const std::string &name, std::function<T()> behaviour,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Show a temporary output pin
* @details Will show an output pin with the given name.
* The pin is created the first time showOUT_uid is called and kept
* alive as long as showOUT_uid is called each frame. <BR> <BR> The UID must
* be unique only in the context of the current node's outputs.
* @tparam T Type of the data the pin will handle
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @param name Name of the pin
* @param behaviour Function or lambda expression used to calculate output
* value
* @param filter Connection filter
* @param style Style of the pin
*/
template <typename T, typename U>
void showOUT_uid(const U &uid, const std::string &name,
std::function<T()> behaviour,
ConnectionFilter filter = ConnectionFilter_None,
std::shared_ptr<PinStyle> style = nullptr);
/**
* @brief <BR>Get Input value from an InPin
* @details Get a reference to the value of an input pin, the value is stored
* in the output pin at the other end of the link.
* @tparam T Data type
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @return Const reference to the value
*/
template <typename T, typename U> const T &getInVal(const U &uid);
/**
* @brief <BR>Get Input value from an InPin
* @details Get a reference to the value of an input pin, the value is stored
* in the output pin at the other end of the link.
* @tparam T Data type
* @param uid Unique identifier of the pin
* @return Const reference to the value
*/
template <typename T> const T &getInVal(const char *uid);
/**
* @brief <BR>Get generic reference to input pin
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @return Generic pointer to the pin
*/
template <typename U> Pin *inPin(const U &uid);
/**
* @brief <BR>Get generic reference to input pin
* @param uid Unique identifier of the pin
* @return Generic pointer to the pin
*/
Pin *inPin(const char *uid);
/**
* @brief <BR>Get generic reference to output pin
* @tparam U Type of the UID
* @param uid Unique identifier of the pin
* @return Generic pointer to the pin
*/
template <typename U> Pin *outPin(const U &uid);
/**
* @brief <BR>Get generic reference to output pin
* @param uid Unique identifier of the pin
* @return Generic pointer to the pin
*/
Pin *outPin(const char *uid);
/**
* @brief <BR>Get internal input pins list
* @return Const reference to node's internal list
*/
const std::vector<std::shared_ptr<Pin>> &getIns() { return m_ins; }
/**
* @brief <BR>Get internal output pins list
* @return Const reference to node's internal list
*/
const std::vector<std::shared_ptr<Pin>> &getOuts() { return m_outs; }
/**
* @brief <BR>Delete itself
*/
void destroy() { m_destroyed = true; }
/*
* @brief <BR>Get if node must be deleted
*/
[[nodiscard]] bool toDestroy() const { return m_destroyed; }
/**
* @brief <BR>Get hovered status
* @return [TRUE] if the mouse is hovering the node
*/
bool isHovered();
/**
* @brief <BR>Get node's UID
* @return Node's unique identifier
*/
[[nodiscard]] NodeUID getUID() const { return m_uid; }
/**
* @brief <BR>Get node name
* @return Const reference to the node's name
*/
const std::string &getName() { return m_title; }
/**
* @brief <BR>Get node size
* @return Const reference to the node's size
*/
const ImVec2 &getSize() { return m_size; }
/**
* @brief <BR>Get node position
* @return Const reference to the node's position
*/
const ImVec2 &getPos() { return m_pos; }
/**
* @brief <BR>Get grid handler bound to node
* @return Pointer to the handler
*/
ImNodeFlow *getHandler() { return m_inf; }
/**
* @brief <BR>Get node's style
* @return Shared pointer to the node's style
*/
const std::shared_ptr<NodeStyle> &getStyle() { return m_style; }
/**
* @brief <BR>Get selected status
* @return [TRUE] if the node is selected
*/
[[nodiscard]] bool isSelected() const { return m_selected; }
/**
* @brief <BR>Get dragged status
* @return [TRUE] if the node is being dragged
*/
[[nodiscard]] bool isDragged() const { return m_dragged; }
/**
* @brief <BR>Set node's uid
* @param uid Node's unique identifier
*/
BaseNode *setUID(NodeUID uid) {
m_uid = uid;
return this;
}
/**
* @brief <BR>Set node's name
* @param name New title
*/
BaseNode *setTitle(const std::string &title) {
m_title = title;
return this;
}
/**
* @brief <BR>Set node's position
* @param pos Position in grid coordinates
*/
BaseNode *setPos(const ImVec2 &pos) {
m_pos = pos;
m_posTarget = pos;
return this;
}
/**
* @brief <BR>Set ImNodeFlow handler
* @param inf Grid handler for the node
*/
BaseNode *setHandler(ImNodeFlow *inf) {
m_inf = inf;
return this;
}
/**
* @brief Set node's style
* @param style New style
*/
BaseNode *setStyle(std::shared_ptr<NodeStyle> style) {
m_style = std::move(style);
return this;
}
/**
* @brief <BR>Set selected status
* @param state New selected state
*
* Status only updates when updatePublicStatus() is called
*/
BaseNode *selected(bool state) {
m_selectedNext = state;
return this;
}
/**
* @brief <BR>Update the isSelected status of the node
*/
void updatePublicStatus() { m_selected = m_selectedNext; }
private:
NodeUID m_uid = 0;
std::string m_title;
ImVec2 m_pos, m_posTarget;
ImVec2 m_size;
ImNodeFlow *m_inf = nullptr;
std::shared_ptr<NodeStyle> m_style;
bool m_selected = false, m_selectedNext = false;
bool m_dragged = false;
bool m_destroyed = false;
std::vector<std::shared_ptr<Pin>> m_ins;
std::vector<std::pair<int, std::shared_ptr<Pin>>> m_dynamicIns;
std::vector<std::shared_ptr<Pin>> m_outs;
std::vector<std::pair<int, std::shared_ptr<Pin>>> m_dynamicOuts;
};
// -----------------------------------------------------------------------------------------------------------------
// PINS
/**
* @brief Pins type identifier
*/
enum PinType { PinType_Input, PinType_Output };
/**
* @brief Generic base class for pins
*/
class Pin {
public:
/**
* @brief <BR>Generic pin constructor
* @param name Name of the pin
* @param filter Connection filter
* @param kind Specifies Input or Output
* @param parent Pointer to the Node containing the pin
* @param inf Pointer to the Grid Handler the pin is in (same as parent)
* @param style Style of the pin
*/
explicit Pin(PinUID uid, std::string name, ConnectionFilter filter,
PinType kind, BaseNode *parent, ImNodeFlow **inf,
std::shared_ptr<PinStyle> style)
: m_uid(uid), m_name(std::move(name)), m_filter(filter), m_type(kind),
m_parent(parent), m_inf(inf), m_style(std::move(style)) {
if (!m_style)
m_style = PinStyle::cyan();
}
/**
* @brief <BR>Main loop of the pin
* @details Updates position, hovering and dragging status, and renders the
* pin. Must be called each frame.
*/
void update();
/**
* @brief <BR>Draw default pin's socket
*/
void drawSocket();
/**
* @brief <BR>Draw default pin's decoration (border, bg, and hover overlay)
*/
void drawDecoration();
/**
* @brief <BR>Used by output pins to calculate their values
*/
virtual void resolve() {}
/**
* @brief <BR>Custom render function to override Pin appearance
* @param r Function or lambda expression with new ImGui rendering
*/
Pin *renderer(std::function<void(Pin *p)> r) {
m_renderer = std::move(r);
return this;
}
/**
* @brief <BR>Create link between pins
* @param other Pointer to the other pin
*/
virtual void createLink(Pin *other) = 0;
/**
* @brief <BR>Set the reference to a link
* @param link Smart pointer to the link
*/
virtual void setLink(std::shared_ptr<Link> &link) {}
/**
* @brief <BR>Delete link reference
*/
virtual void deleteLink() = 0;
/**
* @brief <BR>Get connected status
* @return [TRUE] if the pin is connected
*/
virtual bool isConnected() = 0;
/**
* @brief <BR>Get pin's link
* @return Weak_ptr reference to pin's link
*/
virtual std::weak_ptr<Link> getLink() { return std::weak_ptr<Link>{}; }
/**
* @brief <BR>Get pin's UID
* @return Unique identifier of the pin
*/
[[nodiscard]] PinUID getUid() const { return m_uid; }
/**
* @brief <BR>Get pin's name
* @return Const reference to pin's name
*/
const std::string &getName() { return m_name; }
/**
* @brief <BR>Get pin's position
* @return Const reference to pin's position in grid coordinates
*/
[[nodiscard]] const ImVec2 &getPos() { return m_pos; }
/**
* @brief <BR>Get pin's hit-box size
* @return Const reference to pin's hit-box size
*/
[[nodiscard]] const ImVec2 &getSize() { return m_size; }
/**
* @brief <BR>Get pin's parent node
* @return Generic type pointer to pin's parent node. (Node that contains it)
*/
BaseNode *getParent() { return m_parent; }
/**
* @brief <BR>Get pin's type
* @return The pin type. Either Input or Output
*/
PinType getType() { return m_type; }
/**
* @brief <BR>Get pin's connection filter
* @return Pin's connection filter configuration
*/
[[nodiscard]] ConnectionFilter getFilter() const { return m_filter; }
/**
* @brief <BR>Get pin's style
* @return Smart pointer to pin's style
*/
std::shared_ptr<PinStyle> &getStyle() { return m_style; }
/**
* @brief <BR>Get pin's link attachment point (socket)
* @return Grid coordinates to the attachment point between the link and the
* pin's socket
*/
virtual ImVec2 pinPoint() = 0;
/**
* @brief <BR>Calculate pin's width pre-rendering
* @return The with of the pin once it will be rendered
*/
float calcWidth() { return ImGui::CalcTextSize(m_name.c_str()).x; }
/**
* @brief <BR>Set pin's position
* @param pos Position in screen coordinates
*/
void setPos(ImVec2 pos) { m_pos = pos; }
protected:
PinUID m_uid;
std::string m_name;
ImVec2 m_pos = ImVec2(0.f, 0.f);
ImVec2 m_size = ImVec2(0.f, 0.f);
PinType m_type;
ConnectionFilter m_filter;
std::shared_ptr<PinStyle> m_style;
BaseNode *m_parent = nullptr;
ImNodeFlow **m_inf;
std::function<void(Pin *p)> m_renderer;
};
/**
* @brief Input specific pin
* @details Derived from the generic class Pin. The input pin owns the link
* pointer.
* @tparam T Data type handled by the pin
*/
template <class T> class InPin : public Pin {
public:
/**
* @brief <BR>Input pin constructor
* @param name Name of the pin
* @param filter Connection filter
* @param parent Pointer to the Node containing the pin
* @param defReturn Default return value when the pin is not connected
* @param inf Pointer to the Grid Handler the pin is in (same as parent)
* @param style Style of the pin
*/
explicit InPin(PinUID uid, const std::string &name, ConnectionFilter filter,
BaseNode *parent, T defReturn, ImNodeFlow **inf,
std::shared_ptr<PinStyle> style)
: Pin(uid, name, filter, PinType_Input, parent, inf, style),
m_emptyVal(defReturn) {}
/**
* @brief <BR>Create link between pins
* @param other Pointer to the other pin
*/
void createLink(Pin *other) override;
/**
* @brief <BR>Delete the link connected to the pin
*/
void deleteLink() override { m_link.reset(); }
/**
* @brief <BR>Get connected status
* @return [TRUE] is pin is connected to a link
*/
bool isConnected() override { return m_link != nullptr; }
/**
* @brief <BR>Get pin's link
* @return Weak_ptr reference to the link connected to the pin
*/
std::weak_ptr<Link> getLink() override { return m_link; }
/**
* @brief <BR>Get pin's link attachment point (socket)
* @return Grid coordinates to the attachment point between the link and the
* pin's socket
*/
ImVec2 pinPoint() override {
return m_pos + ImVec2(-m_style->extra.socket_padding, m_size.y / 2);
}
/**
* @brief <BR>Get value carried by the connected link
* @return Reference to the value of the connected OutPin. Or the default
* value if not connected
*/
const T &val();
private:
std::shared_ptr<Link> m_link;
T m_emptyVal;
};
/**
* @brief Output specific pin
* @details Derived from the generic class Pin. The output pin handles the
* logic.
* @tparam T Data type handled by the pin
*/
template <class T> class OutPin : public Pin {
public:
/**
* @brief <BR>Output pin constructor
* @param name Name of the pin
* @param filter Connection filter
* @param parent Pointer to the Node containing the pin
* @param inf Pointer to the Grid Handler the pin is in (same as parent)
* @param style Style of the pin
*/
explicit OutPin(PinUID uid, const std::string &name, ConnectionFilter filter,
BaseNode *parent, ImNodeFlow **inf,
std::shared_ptr<PinStyle> style)
: Pin(uid, name, filter, PinType_Output, parent, inf, style) {}
/**
* @brief <BR>When parent gets deleted, remove the links
*/
~OutPin() {
for (auto &l : m_links)
if (!l.expired())
l.lock()->right()->deleteLink();
}
/**
* @brief <BR>Calculate output value based on set behaviour
*/
void resolve() override { m_val = m_behaviour(); }
/**
* @brief <BR>Create link between pins
* @param other Pointer to the other pin
*/
void createLink(Pin *other) override;
/**
* @brief <BR>Add a connected link to the internal list
* @param link Pointer to the link
*/
void setLink(std::shared_ptr<Link> &link) override;
/**
* @brief <BR>Delete any expired weak pointers to a (now deleted) link
*/
void deleteLink() override;
/**
* @brief <BR>Get connected status
* @return [TRUE] is pin is connected to one or more links
*/
bool isConnected() override { return !m_links.empty(); }
/**
* @brief <BR>Get pin's link attachment point (socket)
* @return Grid coordinates to the attachment point between the link and the
* pin's socket
*/
ImVec2 pinPoint() override {
return m_pos +
ImVec2(m_size.x + m_style->extra.socket_padding, m_size.y / 2);
}
/**
* @brief <BR>Get output value
* @return Const reference to the internal value of the pin
*/
const T &val();
/**
* @brief <BR>Set logic to calculate output value
* @details Used to define the pin behaviour. This is what gets the data from
* the parent's inputs, and applies the needed logic.
* @param func Function or lambda expression used to calculate output value
*/
OutPin<T> *behaviour(std::function<T()> func) {
m_behaviour = std::move(func);
return this;
}
private:
std::vector<std::weak_ptr<Link>> m_links;
std::function<T()> m_behaviour;
T m_val;
};
} // namespace ImFlow
#include "./ImNodeFlow.inl"
#endif