Skip to content
Snippets Groups Projects
GraphViewer.cpp 11.13 KiB
/*
 ------------------------------------------------------------------
 
 This file is part of the Open Ephys GUI
 Copyright (C) 2014 Open Ephys
 
 ------------------------------------------------------------------
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
 the Free Software Foundation, either version 3 of the License, or
 (at your option) any later version.
 
 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 GNU General Public License for more details.
 
 You should have received a copy of the GNU General Public License
 along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
 */

#include "GraphViewer.h"


static const Font FONT_LABEL    ("Paragraph",  50, Font::plain);
static const Font FONT_VERSION  ("Small Text", 14, Font::plain);


GraphViewer::GraphViewer()
{
    JUCEApplication* app = JUCEApplication::getInstance();
    currentVersionText = "GUI version " + app->getApplicationVersion();
    
    rootNum = 0;
}


GraphViewer::~GraphViewer()
{
}


void GraphViewer::addNode (GenericEditor* editor)
{
    GraphNode* gn = new GraphNode (editor, this);
    addAndMakeVisible (gn);
    availableNodes.add (gn);
    
    gn->setBounds (20, 20, 150, 50);
    updateNodeLocations();
}


void GraphViewer::removeNode (GenericEditor* editor)
{
    availableNodes.remove (getIndexOfEditor (editor));
    
    updateNodeLocations();
}


void GraphViewer::removeAllNodes()
{
    availableNodes.clear();
    
    updateNodeLocations();
}

// void GenericEditor::updateNodeName(GenericEditor* editor)
// {
//     GraphNode* n = getNodeForEditor(editor);
//     n->repaint();
// }


void GraphViewer::updateNodeLocations()
{
    const int numAvailableNodes = availableNodes.size();
    
    // If this was made just for future purpose it would be better to execute this only in debug mode
#if JUCE_DEBUG
    // set the initial locations
    for (int i = 0; i < numAvailableNodes; ++i)
    {
    }
#endif
    
    rootNum = 0;
    
    // do initial layout
    for (int i = 0; i < numAvailableNodes; ++i)
    {
        checkLayout (availableNodes[i]);
    }
    
    // check for overlap
    for (int i = 0; i < numAvailableNodes; ++i)
    {
        for (int j = 0; j < numAvailableNodes; ++j)
        {
            if (j != i)
            {
                if (availableNodes[j]->getLevel() == availableNodes[i]->getLevel()
                    && availableNodes[j]->getHorzShift() == availableNodes[i]->getHorzShift())
                {
                    availableNodes[j]->setHorzShift (availableNodes[j]->getHorzShift() + 1);
                }
            }
        }
    }
    
    repaint();
}


void GraphViewer::checkLayout (GraphNode* gn)
{
    if (gn != nullptr)
    {
        GraphNode* sourceNode = nullptr;
        
        if (gn->isMerger())
        {
            Array<GenericEditor*> editors = gn->getConnectedEditors();
            
            int level1 = 0;
            int level2 = 0;
            
            if (editors[0] != nullptr)
            {
                level1 = getNodeForEditor (editors[0])->getLevel();
            }
            
            if (editors[1] != nullptr)
            {
                level2 = getNodeForEditor (editors[1])->getLevel();
            }
            
            // std::cout << "LEVEL1 = " << level1 << " LEVEL2 = " << level2 << std::endl;
            
            sourceNode = level1 > level2 ? getNodeForEditor (editors[0])
            : getNodeForEditor (editors[1]); // choose the higher source
        }
        else
        {
            sourceNode = getNodeForEditor (gn->getSource());
        }
        
        if (! sourceNode)
        {
            gn->setLevel (0);
            gn->setHorzShift (rootNum);
            
            ++rootNum;
        }
        else if (sourceNode->isSplitter())
        {
            Array<GenericEditor*> editors = sourceNode->getConnectedEditors();
            
            const int newHorizontalShift = gn->hasEditor (editors[1])
            ? sourceNode->getHorzShift() + 1
            : sourceNode->getHorzShift();
            
            // increase level
            gn->setLevel (sourceNode->getLevel() + 1);
            // increase horizontal shift
            gn->setHorzShift (newHorizontalShift);
        }
        else
        {
            // increase level
            gn->setLevel (sourceNode->getLevel() + 1);
            // use the same horizontal shift
            gn->setHorzShift (sourceNode->getHorzShift());
        }
        
        checkLayout (getNodeForEditor (gn->getDest()));
    }
}


int GraphViewer::getIndexOfEditor (GenericEditor* editor) const
{
    int index = -1;
    
    const int numAvailableNodes = availableNodes.size();
    for (int i = 0; i < numAvailableNodes; ++i)
    {
        if (availableNodes[i]->hasEditor (editor))
        {
            return i;
        }
    }
    
    return index;
}


GraphNode* GraphViewer::getNodeForEditor (GenericEditor* editor) const
{
    int indexOfEditor = getIndexOfEditor (editor);
    
    if (indexOfEditor > -1)
        return availableNodes[indexOfEditor];
    else
        return nullptr;
}


int GraphViewer::nodesAtLevel (int level) const
{
    int numNodes = 0;
    
    for (auto& node : availableNodes)
    {
        // Level value is not used.
        // It seems we should check if current level of each node is equal to the passed as param level
        // TODO : research this question and either delete this comment or fix the bug
        if (node->getLevel() == numNodes)
            ++numNodes;
    }
    
    return numNodes;
}


int GraphViewer::getHorizontalShift (GraphNode* gn) const
{
    int shift = 0;
    
    for (auto& node : availableNodes)
    {
        if (node == gn)
            break;
        else
            if (node->getLevel() == gn->getLevel())
                ++shift;
    }
    
    return shift;
}


void GraphViewer::paint (Graphics& g)
{
    g.fillAll (Colours::darkgrey);
    
    g.setFont (FONT_LABEL);
    
    g.setColour (Colours::grey);
    
    
    g.drawFittedText ("open ephys", 40, 40, getWidth()-50, getHeight()-60, Justification::bottomRight, 100);
    
    g.setFont (FONT_VERSION);
    g.drawFittedText (currentVersionText, 40, 40, getWidth()-50, getHeight()-45, Justification::bottomRight, 100);
    
    // Draw connections
    const int numAvailableNodes = availableNodes.size();
    for (int i = 0; i < numAvailableNodes; ++i)
    {
        if (! availableNodes[i]->isSplitter())
        {
            if (availableNodes[i]->getDest() != nullptr)
            {
                int indexOfDest = getIndexOfEditor (availableNodes[i]->getDest());
                
                if (indexOfDest > -1)
                    connectNodes (i, indexOfDest, g);
            }
        }
        else
        {
            Array<GenericEditor*> editors = availableNodes[i]->getConnectedEditors();
            
            for (int path = 0; path < 2; ++path)
            {
                int indexOfDest = getIndexOfEditor (editors[path]);
                
                if (indexOfDest > -1)
                    connectNodes (i, indexOfDest, g);
            }
        }
    }
}


void GraphViewer::connectNodes (int node1, int node2, Graphics& g)
{
    
    Point<float> start  = availableNodes[node1]->getCenterPoint();
    Point<float> end    = availableNodes[node2]->getCenterPoint();
    
    Path linePath;
    float x1 = start.getX();
    float y1 = start.getY();
    float x2 = end.getX();
    float y2 = end.getY();
    
    linePath.startNewSubPath (x1, y1);
    linePath.cubicTo (x1, y1 + (y2 - y1) * 0.9f,
                      x2, y1 + (y2 - y1) * 0.1f,
                      x2, y2);
    
    PathStrokeType stroke (2.0f);
    g.strokePath (linePath, stroke);
}

/// ------------------------------------------------------

GraphNode::GraphNode (GenericEditor* ed, GraphViewer* g)
: editor        (ed)
, gv            (g)
, isMouseOver   (false)
{
}


GraphNode::~GraphNode()
{
}


int GraphNode::getLevel() const
{
    // int level = -1;
    
    // GenericEditor* ed = editor;
    
    // while (ed != nullptr)
    // {
    //     level += 1;
    //     ed = ed->getSourceEditor();
    // }
    
    return vertShift;
}


void GraphNode::setLevel (int level)
{
    setBounds (getX(), 20 + level * 40, getWidth(), getHeight());
    
    vertShift = level;
}

int GraphNode::getHorzShift() const
{
    return horzShift; //gv->getHorizontalShift(this);
}


void GraphNode::setHorzShift (int shift)
{
    setBounds (20 + shift * 140, getY(), getWidth(), getHeight());
    
    horzShift = shift;
}


void GraphNode::mouseEnter (const MouseEvent& m)
{
    isMouseOver = true;
    
    repaint();
}


void GraphNode::mouseExit (const MouseEvent& m)
{
    isMouseOver = false;
    
    repaint();
}


void GraphNode::mouseDown (const MouseEvent& m)
{
    editor->makeVisible();
}


bool GraphNode::hasEditor (GenericEditor* ed) const
{
    if (ed == editor)
        return true;
    else
        return false;
}


bool GraphNode::isSplitter() const
{
    return editor->isSplitter();
}


bool GraphNode::isMerger() const
{
    return editor->isMerger();
}


GenericEditor* GraphNode::getDest() const
{
    return editor->getDestEditor();
}


GenericEditor* GraphNode::getSource() const
{
    return editor->getSourceEditor();
}


Array<GenericEditor*> GraphNode::getConnectedEditors() const
{
    return editor->getConnectedEditors();
}


const String GraphNode::getName() const
{
    return editor->getDisplayName();
}


Point<float> GraphNode::getCenterPoint() const
{
    Point<float> center = Point<float> (getX() + 10, getY() + 10);
    
    return center;
}


void GraphNode::switchIO (int path)
{
}


void GraphNode::updateBoundaries()
{
    int horzShift = gv->getHorizontalShift (this);
    
    setBounds (20 + horzShift * 140, 20 + getLevel() * 40, 150, 50);
}


void GraphNode::paint (Graphics& g)
{
    Array<bool> recordStatuses = editor->getRecordStatusArray();
    
    Path recordPath;
    
    const int numRecordStatuses = recordStatuses.size();
    for (int i = 0; i < numRecordStatuses; ++i)
    {
        float stepSize      = 2 * float_Pi / numRecordStatuses;
        float startRadians  = stepSize * i;
        float endRadians    = startRadians + stepSize;
        
        if (recordStatuses[i])
            recordPath.addPieSegment (0, 0, 20, 20, startRadians, endRadians, 0.5);
    }
    
    g.setColour (Colours::red);
    g.fillPath (recordPath);
    
    g.setColour (isMouseOver ? Colours::yellow : editor->getBackgroundColor());
    g.fillEllipse (2, 2, 16, 16);
    
    g.drawText (getName(), 25, 0, getWidth() - 25, 20, Justification::left, true);
}