⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dfdiagram.c

📁 这个工具集提供以下结构化分析和UML分析中所用的图形化绘图工具:ER-diagrams, data and event flow diagrams and state-transition diagr
💻 C
字号:
//////////////////////////////////////////////////////////////////////////////////// This file is part of Toolkit for Conceptual Modeling (TCM).// (c) copyright 1995, Vrije Universiteit Amsterdam.// Author: Frank Dehne (frank@cs.vu.nl).//// TCM 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 2 of the License, or// (at your option) any later version.//// TCM 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 TCM; if not, write to the Free Software// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA// 02111-1307, USA.////////////////////////////////////////////////////////////////////////////////#include "dfgraph.h"#include "dfviewer.h"#include "dfwindow.h"#include "dfstubs.h"#include "dfchecks.h"#include "line.h"#include "messagedialog.h"#include "replacedialog.h"#include "texteditdialog.h"#include "externalentity.h"#include "dataprocess.h"#include "datastore.h"#include "splitmergenode.h"#include "bidirectionaldataflow.h"#include "dfdiagram.h"#include "textbox.h"#include "square.h"#include "circle.h"#include "horizontalbar.h"#include "blackdot.h"#include "comment.h"#include <ctype.h>#include <stdlib.h>DFDiagram::DFDiagram(Config *c, DFWindow *d, DFViewer *v, DFGraph *g): 		Diagram(c,d,v,g) {	UpdateNodeType(1);	UpdateEdgeType(1);	GetReplaceDialog()->ManageNameOnlyToggle(True);	editProcess = 0;	dfChecks = new DFChecks(this,g);	minispecDialog = new TextEditDialog(GetMainWindow()->GetWidget());	minispecDialog->Initialize();	minispecDialog->SetTitle("Minispec Editor");	minispecDialog->SetCancelCallback(0, 0);	minispecDialog->SetOKCallback(DFStubs::MinispecOKCB, this);	minispecDialog->SetTextSize(12, 60);}DFDiagram::~DFDiagram() {	delete minispecDialog;	delete dfChecks;}Thing *DFDiagram::CreateThing(int classNr) {	Grafport *g = GetDiagramViewer()->GetGrafport();	ShapeView *v = GetDiagramViewer()->GetCurView();	DFGraph *dg = (DFGraph *)GetGraph();	Thing *thing = 0;	// view	if (classNr == Code::VIEW)		thing = new ShapeView(GetDiagramViewer());	// node shapes	else if (classNr == Code::SQUARE)		thing = new Square(v, g, 0, 0);	else if (classNr == Code::HORIZONTAL_BAR)		thing = new HorizontalBar(v, g, 0, 0);	else if (classNr == Code::CIRCLE) {		Circle *c = new Circle(v, g, 0, 0);		c->SetFixedIndexLabel(False);		thing = c;	}	else if (classNr == Code::BLACK_DOT)		thing = new BlackDot(v, g, 0, 0);	else if (classNr == Code::TEXT_BOX)		thing = new TextBox(v, g, 0, 0);	// lines	else if (classNr == Code::LINE)		thing = new Line(v, g, 0, 0, 0);	else if (classNr == Code::ARROW) {		Line *line = new Line(v, g, 0, 0, 0);		line->SetEnd1(LineEnd::EMPTY);		line->SetEnd2(LineEnd::FILLED_ARROW);		thing = line;	}	else if (classNr == Code::DOUBLE_ARROW) {		Line *line = new Line(v, g, 0, 0, 0);		line->SetEnd1(LineEnd::FILLED_ARROW);		line->SetEnd2(LineEnd::FILLED_ARROW);		thing = line;	}	// nodes	else if (classNr == Code::EXTERNAL_ENTITY)		thing = new ExternalEntity(dg);	else if (classNr == Code::DATA_STORE)		thing = new DataStore(dg);	else if (classNr == Code::DATA_PROCESS)		thing = new DataProcess(dg);	else if (classNr == Code::SPLIT_MERGE_NODE)		thing = new SplitMergeNode(dg);	else if (classNr == Code::COMMENT)		thing = new Comment(dg);	// edges	else if (classNr == Code::DATA_FLOW)		thing = new DataFlow(dg, 0, 0);	else if (classNr == Code::BIDIRECTIONAL_DATA_FLOW)		thing = new BidirectionalDataFlow(dg, 0, 0);	else		error("%s, line %d: impl error: "			"wrong class number %d\n", __FILE__, __LINE__, classNr);	return thing;}Node *DFDiagram::CreateNode(){	Node *node = 0;	DFGraph *dg = (DFGraph *)GetGraph();	if (GetNodeType() == Code::EXTERNAL_ENTITY)		node = new ExternalEntity(dg);	else if (GetNodeType() == Code::DATA_PROCESS)		node = new DataProcess(dg);	else if (GetNodeType() == Code::DATA_STORE)		node = new DataStore(dg);	else if (GetNodeType() == Code::SPLIT_MERGE_NODE)		node = new SplitMergeNode(dg);	else if (GetNodeType() == Code::COMMENT)		node = new Comment(dg);	else {		error("%s, line %d: unknown node type\n", 				__FILE__, __LINE__);	}	return node;}Edge *DFDiagram::CreateEdge(Subject *subj1, Subject *subj2){	if (!CheckEdgeConstraints(subj1, subj2))		return 0;	Edge *edge = 0;	DFGraph *dg = (DFGraph *)GetGraph();	if (GetEdgeType() == Code::DATA_FLOW)		edge = new DataFlow(dg, subj1, subj2);	else if (GetEdgeType() == Code::BIDIRECTIONAL_DATA_FLOW)		edge = new BidirectionalDataFlow(dg, subj1, subj2);	else		error("%s, line %d: unknown edge type\n", __FILE__, __LINE__);	return edge;}NodeShape *DFDiagram::CreateNodeShape(Node *node, int x, int y) {	NodeShape *shape = 0;	Grafport *g = GetDiagramViewer()->GetGrafport();	ShapeView *v = GetDiagramViewer()->GetCurView();	if (GetNodeShapeType() == Code::HORIZONTAL_BAR)		shape = new HorizontalBar(v, g, x, y);	else if (GetNodeShapeType() == Code::CIRCLE) {		Circle *c = new Circle(v, g, x, y);		c->SetFixedIndexLabel(False);		shape = c;	}	else if (GetNodeShapeType() == Code::SQUARE)		shape = new Square(v, g, x, y);	else if (GetNodeShapeType() == Code::BLACK_DOT)		shape = new BlackDot(v, g, x, y);	else if (GetNodeShapeType() == Code::TEXT_BOX)		shape = new TextBox(v, g, x, y);	else 		error("%s, line %d: impl error: "			"node shape type does not exist\n", __FILE__, __LINE__);	if (check(shape)) {		shape->SetSubject(node);		shape->SetTextShape();	}	return shape;}Line *DFDiagram::CreateLine(		Edge *edge, GShape *from, GShape *to, List<Point *> *l) {	Grafport *g = GetDiagramViewer()->GetGrafport();	ShapeView *v = GetDiagramViewer()->GetCurView();	Line *line = 0;	if (GetLineType() == Code::LINE)		line = new Line(v, g, from, to, l, IsCurve());	else		error("%s, line %d: impl error: "			"line type does not exist\n", __FILE__, __LINE__);	if (check(line)) {		line->SetSubject(edge);		line->SetTextShape();		line->SetEnd1(GetLineEnd1());		line->SetEnd2(GetLineEnd2());	}	return line;}void DFDiagram::UpdateNodeType(int num) {	((DiagramWindow *)GetMainWindow())->SetNodeName(num);	switch (num) {	case 1: SetNodeType(Code::DATA_PROCESS);		SetNodeShapeType(Code::CIRCLE);		break;	case 2: SetNodeType(Code::DATA_STORE);		SetNodeShapeType(Code::HORIZONTAL_BAR);		break;	case 3: SetNodeType(Code::EXTERNAL_ENTITY);		SetNodeShapeType(Code::SQUARE);		break;	case 4: SetNodeType(Code::SPLIT_MERGE_NODE);		SetNodeShapeType(Code::BLACK_DOT);		break;	case 5: SetNodeType(Code::COMMENT);		SetNodeShapeType(Code::TEXT_BOX);		break;	default:		error("%s, line %d: impl error: "			"unknown node type selected\n", __FILE__,__LINE__);	}}void DFDiagram::UpdateEdgeType(int num) {	((DiagramWindow *)GetMainWindow())->SetEdgeName(num);	SetLineType(Code::LINE);	switch(num) {	case 1: SetEdgeType(Code::DATA_FLOW);		SetLineEnd1(LineEnd::EMPTY);                SetLineEnd2(LineEnd::FILLED_ARROW);		break;	case 2: SetEdgeType(Code::BIDIRECTIONAL_DATA_FLOW);		SetLineEnd1(LineEnd::FILLED_ARROW);                SetLineEnd2(LineEnd::FILLED_ARROW);		break;	default:		error("%s, line %d: impl error: "			"unknown edge type selected\n", __FILE__,__LINE__);	}}bool DFDiagram::CheckEdgeConstraints(Subject *subj1, Subject *subj2) {	// Check possible connections (subj-subj-edge matrix).	if (!CheckConnection(subj1, subj2))		return False;	if (subj1 == subj2) {		ShowDialog(MessageDialog::ERROR, "Error",			"Cannot add a flow from and to the same node.");		return False;	}	int subj1Type = subj1->GetClassType();	int subj2Type = subj2->GetClassType();	// Cannot connect a data process with a splitmerge node	// when they are already connected (even in the opposite direction).	if ((GetEdgeType()==Code::DATA_FLOW||	     GetEdgeType()==Code::CONTINUOUS_DATA_FLOW) &&		(subj1Type==Code::DATA_PROCESS &&		 subj2Type==Code::SPLIT_MERGE_NODE ||		 subj1Type==Code::SPLIT_MERGE_NODE &&		 subj2Type==Code::DATA_PROCESS) &&		(GetGraph()->IsConnected(subj2, subj1) ||		 GetGraph()->IsConnected(subj1, subj2))) {		ShowDialog(MessageDialog::ERROR, "Error",			"Cannot add another data flow here. ");		return False;	}	return True;}bool DFDiagram::HasIndexNode(int code) {	return (code == Code::DATA_PROCESS || code == Code::CONTROL_PROCESS);}bool DFDiagram::HasIndexShape(int code) {	return (code == Code::CIRCLE);}void DFDiagram::EditMinispec(Subject *s) {	if (s->GetClassType() != Code::DATA_PROCESS) {		ShowDialog(MessageDialog::ERROR, "Not a data process",			"Only data processes can have a minispec");		return;	}	if (minispecDialog->IsManaged()) {		ShowDialog(MessageDialog::WARNING, "Warning",			"You can only open one minispec editor at a time.");		return;	}	editProcess = (DataProcess *)s;	if (editProcess->IsProcessGroup()) {		ShowDialog(MessageDialog::ERROR, "A process group",			"This data process is a process group,\n"			"so it cannot have a minispec");		editProcess = 0;		return;	}	const string *n = editProcess->GetIndex();	const string *spec = editProcess->GetMinispec();	string x = "Minispec of process ";	string title = x + *n;	minispecDialog->SetTitle(&title);	minispecDialog->SetTextString(spec);	minispecDialog->Popup();}void DFDiagram::SetMinispec() {	SetStatus("action: Update minispec");	if (!check(editProcess))		return;	string t;	minispecDialog->GetTextString(&t);	editProcess->SetMinispec(&t);}void DFDiagram::SetDiagram(const string *dfd) {	DFGraph *dg = (DFGraph *)GetGraph();	string sdfd = *dfd;	if (((*dfd == "") || DFProcess::CorrectIndex(&sdfd)) && 	     (*dfd != "0")) {		dg->SetIndexPrefix(&sdfd);		((DFWindow *)GetMainWindow())->SetDiagram(dfd);		if (sdfd == "")			sdfd = "0";		string txt = "diagram " + sdfd + " is the current diagram";		SetStatus(&txt);	}	else {		const string s = *dg->GetIndexPrefix();		((DFWindow *)GetMainWindow())->SetDiagram(&s);		string txt = "'";		txt += *dfd;		txt += "' wrong syntax for \na data flow diagram index";		ShowDialog(MessageDialog::ERROR, "Error", &txt);	}}void DFDiagram::CheckDocument() {	chkbuf = "";	unsigned total = 0;	total += dfChecks->CheckNamelessNodes(Code::DATA_PROCESS, chkbuf);	total += dfChecks->CheckNamelessNodes(Code::DATA_STORE, chkbuf);	total += dfChecks->CheckNamelessNodes(Code::EXTERNAL_ENTITY, chkbuf);	total += dfChecks->CheckNamelessFlows(Code::DATA_FLOW, 		Code::EXTERNAL_ENTITY, Code::DATA_PROCESS, chkbuf);	total += dfChecks->CheckNamelessFlows(Code::DATA_FLOW, 		Code::DATA_PROCESS, Code::EXTERNAL_ENTITY, chkbuf);	total += dfChecks->CheckNamelessFlows(Code::DATA_FLOW, 		Code::DATA_PROCESS, Code::DATA_PROCESS, chkbuf);	total += dfChecks->CheckDoubleNodes(Code::DATA_PROCESS, chkbuf);	total += dfChecks->CheckDoubleNodes(Code::DATA_STORE, chkbuf);	total += dfChecks->CheckDoubleNodes(Code::EXTERNAL_ENTITY, chkbuf);	total += dfChecks->CheckDoubleIndexes(chkbuf);	total += dfChecks->CheckDataProcessCoherence(chkbuf);	total += dfChecks->CheckConnected(Code::DATA_STORE, False, chkbuf);	total += dfChecks->CheckConnected(Code::EXTERNAL_ENTITY, False, chkbuf);	total += dfChecks->CheckSplitMergeNodeCoherence(chkbuf);	total += dfChecks->CheckNamelessSplitMergeEdges(chkbuf);	total += dfChecks->CheckMinispecs(chkbuf);	ReportCheck(total, &chkbuf);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -