#pragma once #include "FractoriumPch.h" /// /// CurvesGraphicsView and EllipseItem classes. /// class EllipseItem; /// /// Enumeration used for setting values on a specific curve. /// enum class CurveIndex : et { ALL, RED, GREEN, BLUE }; /// /// Derivation to display points on bezier cuves for the user to drag. /// Selection logic and updating is handled by the base class. /// Changes here will affect the current ember and vice versa. /// The points, axis lines and pens are kept as members and the curves /// themselves are drawn on the fly during each paint event. /// Pointers to these are kept in arrays to make manipulating them in loops easier. /// Note that this must work off a fixed rect size, hence why the control is not resizeable. /// class CurvesGraphicsView : public QGraphicsView { Q_OBJECT public: CurvesGraphicsView(QWidget* parent = nullptr); void PointChanged(int curveIndex, int pointIndex, const QPointF& point); QPointF Get(int curveIndex, int pointIndex); void Set(int curveIndex, int pointIndex, const QPointF& point); void Set(Curves& curves); void SetTop(CurveIndex curveIndex); size_t SelectedCurveIndex() const noexcept { return m_Index; } Q_SIGNALS: void PointChangedSignal(int curveIndex, int pointIndex, const QPointF& point); void PointAddedSignal(size_t curveIndex, const QPointF& point); void PointRemovedSignal(size_t curveIndex, int pointIndex); protected: void paintEvent(QPaintEvent* e) override; void mousePressEvent(QMouseEvent* e) override; size_t m_Index = 0; QPen m_APen; QPen m_RPen; QPen m_GPen; QPen m_BPen; QPen m_AxisPen; std::vector m_AllP; std::vector m_RedP; std::vector m_GrnP; std::vector m_BluP; QGraphicsLineItem* m_XLine; QGraphicsLineItem* m_YLine; std::array m_Pens; QGraphicsScene m_Scene; QRectF m_OriginalRect; std::array, 4> m_Points; }; /// /// Derivation for draggable points needed to trigger an event whenever the item is changed. /// Custom drawing is also done to omit drawing a selection rectangle. /// class EllipseItem : public QGraphicsEllipseItem { public: /// /// Construct the point and specify the curve index it's part of, as well as the /// point index within the curve. /// /// Pass to the parent /// The curve's index this point is a part of, 0-3. /// The point index within the curve /// The graphics view this point is displayed on /// The parent widget of this item EllipseItem(const QRectF& rect, int curveIndex, int pointIndex, CurvesGraphicsView* viewParent, QGraphicsItem* parent = nullptr) : QGraphicsEllipseItem(rect, parent) { m_CurveIndex = curveIndex; m_PointIndex = pointIndex; m_ViewParent = viewParent; setFlag(QGraphicsItem::ItemSendsScenePositionChanges); setFlag(QGraphicsItem::ItemIsSelectable); setFlag(QGraphicsItem::ItemIsMovable); setPen(Qt::NoPen); } /// /// Set whether this item is selectable, which means this curve is the current one. /// /// True if selected, else false. void SetCurrent(bool b) { setFlag(QGraphicsItem::ItemIsMovable, b); setZValue(b ? 2 : 1); } /// /// Index properties, getters only. /// int CurveIndex() const noexcept { return m_CurveIndex; } int PointIndex() const noexcept { return m_PointIndex; } protected: /// /// Overridden paint event to disable the selection rectangle. /// /// Unused and just passed to QGraphicsEllipseItem::paint() /// Drawing options used which will have the QStyle::State_Selected flag unset /// Unused and just passed to QGraphicsEllipseItem::paint() void paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget) override { if (option != nullptr && widget != nullptr) { QStyleOptionGraphicsItem myOption(*option); myOption.state &= ~QStyle::State_Selected; QGraphicsEllipseItem::paint(painter, &myOption, widget); } } /// /// Overridden itemChange event to notify the parent control that it has moved. /// Movement is also restricted to the scene rect. /// /// Action is only taken if this value equals ItemPositionChange /// The new position. This will be clamped to the scene rect. /// The new position QVariant itemChange(GraphicsItemChange change, const QVariant& value) override { if ((change == ItemPositionChange) && scene()) { //Value is the new position. auto newPos = value.toPointF(); const auto rect = scene()->sceneRect(); if (!rect.contains(newPos)) { //Keep the item inside the scene rect. newPos.setX(qMin(rect.right(), qMax(newPos.x(), rect.left()))); newPos.setY(qMin(rect.bottom(), qMax(newPos.y(), rect.top()))); } m_ViewParent->PointChanged(m_CurveIndex, m_PointIndex, newPos); return newPos; } return QGraphicsEllipseItem::itemChange(change, value); } int m_CurveIndex; int m_PointIndex; CurvesGraphicsView* m_ViewParent; };