add terminate pseudostate + test
This commit is contained in:
@@ -107,6 +107,7 @@ CyberiadaNode* Element::to_node() const
|
|||||||
case elementInitial: node->type = cybNodeInitial; break;
|
case elementInitial: node->type = cybNodeInitial; break;
|
||||||
case elementFinal: node->type = cybNodeFinal; break;
|
case elementFinal: node->type = cybNodeFinal; break;
|
||||||
case elementChoice: node->type = cybNodeChoice; break;
|
case elementChoice: node->type = cybNodeChoice; break;
|
||||||
|
case elementTerminate: node->type = cybNodeTerminate; break;
|
||||||
default:
|
default:
|
||||||
std::cerr << id << " " << type << std::endl;
|
std::cerr << id << " " << type << std::endl;
|
||||||
CYB_ASSERT(false);
|
CYB_ASSERT(false);
|
||||||
@@ -131,6 +132,7 @@ std::ostream& Element::dump(std::ostream& os) const
|
|||||||
case elementInitial: type_str = "Initial"; break;
|
case elementInitial: type_str = "Initial"; break;
|
||||||
case elementFinal: type_str = "Final"; break;
|
case elementFinal: type_str = "Final"; break;
|
||||||
case elementChoice: type_str = "Choice"; break;
|
case elementChoice: type_str = "Choice"; break;
|
||||||
|
case elementTerminate: type_str = "Terminate"; break;
|
||||||
case elementTransition: type_str = "Transition"; break;
|
case elementTransition: type_str = "Transition"; break;
|
||||||
default:
|
default:
|
||||||
CYB_ASSERT(false);
|
CYB_ASSERT(false);
|
||||||
@@ -508,9 +510,11 @@ Vertex::Vertex(Element* _parent, ElementType _type, const ID& _id, const Name& _
|
|||||||
|
|
||||||
std::ostream& Vertex::dump(std::ostream& os) const
|
std::ostream& Vertex::dump(std::ostream& os) const
|
||||||
{
|
{
|
||||||
|
Element::dump(os);
|
||||||
if (has_geometry()) {
|
if (has_geometry()) {
|
||||||
os << ", geometry: " << geometry_point;
|
os << ", geometry: " << geometry_point;
|
||||||
}
|
}
|
||||||
|
os << "}";
|
||||||
return os;
|
return os;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -728,7 +732,8 @@ std::list<const Vertex*> ElementCollection::get_vertexes() const
|
|||||||
elementCompositeState,
|
elementCompositeState,
|
||||||
elementInitial,
|
elementInitial,
|
||||||
elementFinal,
|
elementFinal,
|
||||||
elementChoice };
|
elementChoice,
|
||||||
|
elementTerminate};
|
||||||
std::list<const Vertex*> result;
|
std::list<const Vertex*> result;
|
||||||
ConstElementList vertexes = find_elements_by_types(types);
|
ConstElementList vertexes = find_elements_by_types(types);
|
||||||
for (ConstElementList::const_iterator i = vertexes.begin(); i != vertexes.end(); i++) {
|
for (ConstElementList::const_iterator i = vertexes.begin(); i != vertexes.end(); i++) {
|
||||||
@@ -745,7 +750,8 @@ std::list<Vertex*> ElementCollection::get_vertexes()
|
|||||||
elementCompositeState,
|
elementCompositeState,
|
||||||
elementInitial,
|
elementInitial,
|
||||||
elementFinal,
|
elementFinal,
|
||||||
elementChoice };
|
elementChoice,
|
||||||
|
elementTerminate};
|
||||||
std::list<Vertex*> result;
|
std::list<Vertex*> result;
|
||||||
ElementList vertexes = find_elements_by_types(types);
|
ElementList vertexes = find_elements_by_types(types);
|
||||||
for (ElementList::const_iterator i = vertexes.begin(); i != vertexes.end(); i++) {
|
for (ElementList::const_iterator i = vertexes.begin(); i != vertexes.end(); i++) {
|
||||||
@@ -853,12 +859,18 @@ InitialPseudostate::InitialPseudostate(Element* _parent, const ID& _id, const Na
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
std::ostream& InitialPseudostate::dump(std::ostream& os) const
|
// -----------------------------------------------------------------------------
|
||||||
|
// Terminate pseudostate
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
TerminatePseudostate::TerminatePseudostate(Element* _parent, const ID& _id, const Point& p):
|
||||||
|
Pseudostate(_parent, elementTerminate, _id, p)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
TerminatePseudostate::TerminatePseudostate(Element* _parent, const ID& _id, const Name& _name, const Point& p):
|
||||||
|
Pseudostate(_parent, elementTerminate, _id, _name, p)
|
||||||
{
|
{
|
||||||
Element::dump(os);
|
|
||||||
Vertex::dump(os);
|
|
||||||
os << "}";
|
|
||||||
return os;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
@@ -914,14 +926,6 @@ FinalState::FinalState(Element* _parent, const ID& _id, const Name& _name, const
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
std::ostream& FinalState::dump(std::ostream& os) const
|
|
||||||
{
|
|
||||||
Element::dump(os);
|
|
||||||
Vertex::dump(os);
|
|
||||||
os << "}";
|
|
||||||
return os;
|
|
||||||
}
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
// State
|
// State
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
@@ -1333,6 +1337,36 @@ ChoicePseudostate* Document::new_choice(ElementCollection* _parent, const ID& _i
|
|||||||
return choice;
|
return choice;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TerminatePseudostate* Document::new_terminate(ElementCollection* _parent, const Point& p)
|
||||||
|
{
|
||||||
|
check_parent_element(_parent);
|
||||||
|
|
||||||
|
TerminatePseudostate* term = new TerminatePseudostate(_parent, generate_vertex_id(_parent), p);
|
||||||
|
_parent->add_element(term);
|
||||||
|
return term;
|
||||||
|
}
|
||||||
|
|
||||||
|
TerminatePseudostate* Document::new_terminate(ElementCollection* _parent, const Name& _name, const Point& p)
|
||||||
|
{
|
||||||
|
check_parent_element(_parent);
|
||||||
|
check_nonempty_string(_name);
|
||||||
|
|
||||||
|
TerminatePseudostate* term = new TerminatePseudostate(_parent, generate_vertex_id(_parent), _name, p);
|
||||||
|
_parent->add_element(term);
|
||||||
|
return term;
|
||||||
|
}
|
||||||
|
|
||||||
|
TerminatePseudostate* Document::new_terminate(ElementCollection* _parent, const ID& _id, const Name& _name, const Point& p)
|
||||||
|
{
|
||||||
|
check_parent_element(_parent);
|
||||||
|
check_nonempty_string(_name);
|
||||||
|
check_id_uniqueness(_id);
|
||||||
|
|
||||||
|
TerminatePseudostate* term = new TerminatePseudostate(_parent, _id, _name, p);
|
||||||
|
_parent->add_element(term);
|
||||||
|
return term;
|
||||||
|
}
|
||||||
|
|
||||||
Transition* Document::new_transition(StateMachine* sm, Element* source, Element* target,
|
Transition* Document::new_transition(StateMachine* sm, Element* source, Element* target,
|
||||||
const Action& action, const Polyline& pl,
|
const Action& action, const Polyline& pl,
|
||||||
const Point& sp, const Point& tp,
|
const Point& sp, const Point& tp,
|
||||||
@@ -1553,6 +1587,7 @@ void Document::check_transition_source(const Element* element) const
|
|||||||
element->get_type() == elementComment ||
|
element->get_type() == elementComment ||
|
||||||
element->get_type() == elementFormalComment ||
|
element->get_type() == elementFormalComment ||
|
||||||
element->get_type() == elementFinal ||
|
element->get_type() == elementFinal ||
|
||||||
|
element->get_type() == elementTerminate ||
|
||||||
element->get_type() == elementTransition) {
|
element->get_type() == elementTransition) {
|
||||||
throw ParametersException("Bad source for transition");
|
throw ParametersException("Bad source for transition");
|
||||||
}
|
}
|
||||||
@@ -1698,6 +1733,15 @@ void Document::import_nodes_recursively(ElementCollection* collection, Cyberiada
|
|||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case cybNodeTerminate:
|
||||||
|
if (n->title) {
|
||||||
|
element = new TerminatePseudostate(collection, n->id, n->title, point);
|
||||||
|
} else {
|
||||||
|
element = new TerminatePseudostate(collection, n->id, point);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
case cybNodeFinal:
|
case cybNodeFinal:
|
||||||
if (n->title) {
|
if (n->title) {
|
||||||
element = new FinalState(collection, n->id, n->title, point);
|
element = new FinalState(collection, n->id, n->title, point);
|
||||||
|
|||||||
@@ -49,6 +49,7 @@ namespace Cyberiada {
|
|||||||
elementInitial, // initial pseudostate
|
elementInitial, // initial pseudostate
|
||||||
elementFinal, // final state
|
elementFinal, // final state
|
||||||
elementChoice, // choice pseudostate
|
elementChoice, // choice pseudostate
|
||||||
|
elementTerminate, // terminate pseudostate
|
||||||
elementTransition // transition
|
elementTransition // transition
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -329,9 +330,6 @@ namespace Cyberiada {
|
|||||||
public:
|
public:
|
||||||
InitialPseudostate(Element* parent, const ID& id, const Point& p = Point());
|
InitialPseudostate(Element* parent, const ID& id, const Point& p = Point());
|
||||||
InitialPseudostate(Element* parent, const ID& id, const Name& name, const Point& p = Point());
|
InitialPseudostate(Element* parent, const ID& id, const Name& name, const Point& p = Point());
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual std::ostream& dump(std::ostream& os) const;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
@@ -359,6 +357,15 @@ namespace Cyberiada {
|
|||||||
Color color;
|
Color color;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
// Initial pseudostate
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
class TerminatePseudostate: public Pseudostate {
|
||||||
|
public:
|
||||||
|
TerminatePseudostate(Element* parent, const ID& id, const Point& p = Point());
|
||||||
|
TerminatePseudostate(Element* parent, const ID& id, const Name& name, const Point& p = Point());
|
||||||
|
};
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
// Final state
|
// Final state
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
@@ -366,10 +373,6 @@ namespace Cyberiada {
|
|||||||
public:
|
public:
|
||||||
FinalState(Element* parent, const ID& id, const Point& point = Point());
|
FinalState(Element* parent, const ID& id, const Point& point = Point());
|
||||||
FinalState(Element* parent, const ID& id, const Name& name, const Point& point = Point());
|
FinalState(Element* parent, const ID& id, const Name& name, const Point& point = Point());
|
||||||
|
|
||||||
protected:
|
|
||||||
|
|
||||||
virtual std::ostream& dump(std::ostream& os) const;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
@@ -547,8 +550,8 @@ namespace Cyberiada {
|
|||||||
State* new_state(ElementCollection* parent, const ID& id, const String& state_name,
|
State* new_state(ElementCollection* parent, const ID& id, const String& state_name,
|
||||||
const Rect& r = Rect(), const Color& color = Color());
|
const Rect& r = Rect(), const Color& color = Color());
|
||||||
InitialPseudostate* new_initial(ElementCollection* parent, const Point& p = Point());
|
InitialPseudostate* new_initial(ElementCollection* parent, const Point& p = Point());
|
||||||
InitialPseudostate* new_initial(ElementCollection* parent, const Name& initial_name, const Point& p = Point());
|
InitialPseudostate* new_initial(ElementCollection* parent, const Name& name, const Point& p = Point());
|
||||||
InitialPseudostate* new_initial(ElementCollection* parent, const ID& id, const Name& initial_name, const Point& p = Point());
|
InitialPseudostate* new_initial(ElementCollection* parent, const ID& id, const Name& name, const Point& p = Point());
|
||||||
FinalState* new_final(ElementCollection* parent, const Point& point = Point());
|
FinalState* new_final(ElementCollection* parent, const Point& point = Point());
|
||||||
FinalState* new_final(ElementCollection* parent, const Name& name, const Point& point = Point());
|
FinalState* new_final(ElementCollection* parent, const Name& name, const Point& point = Point());
|
||||||
FinalState* new_final(ElementCollection* parent, const ID& id, const Name& name, const Point& point = Point());
|
FinalState* new_final(ElementCollection* parent, const ID& id, const Name& name, const Point& point = Point());
|
||||||
@@ -558,6 +561,9 @@ namespace Cyberiada {
|
|||||||
const Rect& r = Rect(), const Color& color = Color());
|
const Rect& r = Rect(), const Color& color = Color());
|
||||||
ChoicePseudostate* new_choice(ElementCollection* parent, const ID& id, const Name& name,
|
ChoicePseudostate* new_choice(ElementCollection* parent, const ID& id, const Name& name,
|
||||||
const Rect& r = Rect(), const Color& color = Color());
|
const Rect& r = Rect(), const Color& color = Color());
|
||||||
|
TerminatePseudostate* new_terminate(ElementCollection* parent, const Point& p = Point());
|
||||||
|
TerminatePseudostate* new_terminate(ElementCollection* parent, const Name& name, const Point& p = Point());
|
||||||
|
TerminatePseudostate* new_terminate(ElementCollection* parent, const ID& id, const Name& name, const Point& p = Point());
|
||||||
Transition* new_transition(StateMachine* sm, Element* source, Element* target,
|
Transition* new_transition(StateMachine* sm, Element* source, Element* target,
|
||||||
const Action& action, const Polyline& pl = Polyline(),
|
const Action& action, const Polyline& pl = Polyline(),
|
||||||
const Point& sp = Point(), const Point& tp = Point(),
|
const Point& sp = Point(), const Point& tp = Point(),
|
||||||
|
|||||||
50
tests/15-output.graphml
Normal file
50
tests/15-output.graphml
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<graphml xmlns="http://graphml.graphdrawing.org/xmlns">
|
||||||
|
<data key="gFormat">Cyberiada-GraphML-1.0</data>
|
||||||
|
<key id="gFormat" for="graphml" attr.name="format" attr.type="string"/>
|
||||||
|
<key id="dName" for="graph" attr.name="name" attr.type="string"/>
|
||||||
|
<key id="dName" for="node" attr.name="name" attr.type="string"/>
|
||||||
|
<key id="dStateMachine" for="graph" attr.name="stateMachine" attr.type="string"/>
|
||||||
|
<key id="dSubmachineState" for="node" attr.name="submachineState" attr.type="string"/>
|
||||||
|
<key id="dGeometry" for="graph" attr.name="geometry"/>
|
||||||
|
<key id="dGeometry" for="node" attr.name="geometry"/>
|
||||||
|
<key id="dGeometry" for="edge" attr.name="geometry"/>
|
||||||
|
<key id="dSourcePoint" for="edge" attr.name="sourcePoint"/>
|
||||||
|
<key id="dTargetPoint" for="edge" attr.name="targetPoint"/>
|
||||||
|
<key id="dLabelGeometry" for="edge" attr.name="labelGeometry"/>
|
||||||
|
<key id="dNote" for="node" attr.name="note" attr.type="string"/>
|
||||||
|
<key id="dVertex" for="node" attr.name="vertex" attr.type="string"/>
|
||||||
|
<key id="dData" for="node" attr.name="data" attr.type="string"/>
|
||||||
|
<key id="dData" for="edge" attr.name="data" attr.type="string"/>
|
||||||
|
<key id="dMarkup" for="node" attr.name="markup" attr.type="string"/>
|
||||||
|
<key id="dColor" for="node" attr.name="color" attr.type="string"/>
|
||||||
|
<key id="dColor" for="edge" attr.name="color" attr.type="string"/>
|
||||||
|
<key id="dPivot" for="edge" attr.name="pivot" attr.type="string"/>
|
||||||
|
<key id="dChunk" for="edge" attr.name="chunk" attr.type="string"/>
|
||||||
|
<graph id="G0" edgedefault="directed">
|
||||||
|
<data key="dName">SM</data>
|
||||||
|
<node id="nMeta">
|
||||||
|
<data key="dNote">formal</data>
|
||||||
|
<data key="dName">CGML_META</data>
|
||||||
|
<data key="dData">standardVersion/ 1.0
|
||||||
|
|
||||||
|
transitionOrder/ transitionFirst
|
||||||
|
|
||||||
|
eventPropagation/ block
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</node>
|
||||||
|
<node id="n0">
|
||||||
|
<data key="dVertex">terminate</data>
|
||||||
|
</node>
|
||||||
|
<node id="n1">
|
||||||
|
<data key="dName">State</data>
|
||||||
|
<graph id="n1:" edgedefault="directed">
|
||||||
|
<node id="n1::n0">
|
||||||
|
<data key="dVertex">terminate</data>
|
||||||
|
<data key="dName">Local term</data>
|
||||||
|
</node>
|
||||||
|
</graph>
|
||||||
|
</node>
|
||||||
|
</graph>
|
||||||
|
</graphml>
|
||||||
1
tests/15-output.txt
Normal file
1
tests/15-output.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Document: {id: '', name: '', format: 'Cyberiada-GraphML-1.0', meta: {standard version: '1.0', transition order: transition first, event propagation: block events}, elements: {State Machine: {id: 'G0', name: 'SM', elements: {Terminate: {id: 'n0'}, Composite State: {id: 'n1', name: 'State', elements: {Terminate: {id: 'n1::n0', name: 'Local term'}}}}}}
|
||||||
57
tests/15-terminate.cpp
Normal file
57
tests/15-terminate.cpp
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
/* -----------------------------------------------------------------------------
|
||||||
|
* The Cyberiada GraphML C++ library implemention
|
||||||
|
*
|
||||||
|
* The test
|
||||||
|
*
|
||||||
|
* Copyright (C) 2024 Alexey Fedoseev <aleksey@fedoseev.net>
|
||||||
|
*
|
||||||
|
* 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 https://www.gnu.org/licenses/
|
||||||
|
* ----------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include "cyberiadamlpp.h"
|
||||||
|
#include "testutils.h"
|
||||||
|
|
||||||
|
using namespace Cyberiada;
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
int main(int argc, char** argv)
|
||||||
|
{
|
||||||
|
Document d;
|
||||||
|
|
||||||
|
StateMachine* sm = d.new_state_machine("SM");
|
||||||
|
|
||||||
|
d.new_terminate(sm);
|
||||||
|
try {
|
||||||
|
// check id uniqueness
|
||||||
|
d.new_terminate(sm, "n0", "name");
|
||||||
|
} catch (const Cyberiada::ParametersException&){
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// check non-empty name
|
||||||
|
d.new_terminate(sm, "");
|
||||||
|
} catch (const Cyberiada::ParametersException&){
|
||||||
|
}
|
||||||
|
|
||||||
|
State* parent = d.new_state(sm, "State");
|
||||||
|
d.new_terminate(parent, "Local term");
|
||||||
|
|
||||||
|
try {
|
||||||
|
cout << d << endl;
|
||||||
|
d.save(string(argv[0]) + ".graphml", formatCyberiada10);
|
||||||
|
} catch (const Cyberiada::Exception&) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user