-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathHypergraph.m
117 lines (81 loc) · 3.66 KB
/
Hypergraph.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
Package["SetReplace`"]
PackageImport["GeneralUtilities`"]
PackageExport["Hypergraph"]
PackageExport["HypergraphQ"]
PackageExport["HypergraphSymmetry"]
SetRelatedSymbolGroup[Hypergraph, HypergraphQ, HypergraphSymmetry, EdgeList, VertexList];
(* HypergraphQ *)
SetUsage @ "HypergraphQ[hg$] yields True if hg$ is a valid Hypergraph object and False otherwise.";
SyntaxInformation[HypergraphQ] = {"ArgumentsPattern" -> {expr_}};
HypergraphQ[expr_Hypergraph] := System`Private`HoldNoEntryQ[expr];
HypergraphQ[_] = False;
(* HypergraphSymmetry *)
$hypergraphSymmetries = {"Ordered", "Unordered", "Cyclic"(*, "Directed"*)};
SetUsage @ "HypergraphSymmetry[hg$] returns the symmetry of the hypergraph hg$.";
HypergraphSymmetry[HoldPattern[Hypergraph[hyperedges_, symmetry_] ? HypergraphQ]] := symmetry;
(* Hypergraph *)
SetUsage[Hypergraph, "
Hypergraph[{he$1, he$2, $$}] yields an ordered hypergraph with hyperedges he$j.
Hypergraph[$$, sym$] returns a hypergraph with symmetry sym$.
* Valid hypergraph symmetries include: " <> listToSentence[$hypergraphSymmetries] <> ".
"];
SyntaxInformation[Hypergraph] = {"ArgumentsPattern" -> {hyperedges_, symmetry_.}};
Hypergraph /: Information`GetInformation[obj_Hypergraph ? HypergraphQ] :=
<|
"ObjectType" -> Hypergraph,
"Symmetry" -> HypergraphSymmetry[obj],
"VertexCount" -> VertexCount[obj],
"EdgeCount" -> EdgeCount[obj]
|>;
((expr : Hypergraph[args___]) ? System`Private`HoldEntryQ) /; CheckArguments[expr, {1, 2}] :=
With[{
result = Catch[hypergraph[args],
_ ? FailureQ,
message[Hypergraph, #, <|"expr" -> HoldForm[expr]|>] &]
},
result /; !FailureQ[result]
];
hypergraph[hyperedges : {___List}, symmetry : Alternatives @@ $hypergraphSymmetries] :=
System`Private`ConstructNoEntry[Hypergraph, hyperedges, symmetry];
hypergraph[hyperedges_] := hypergraph[hyperedges, "Ordered"];
declareMessage[Hypergraph::invalidHyperedges,
"The argument at position 1 in `expr` should be a list of of lists."];
hypergraph[hyperedges_, symmetry : Alternatives @@ $hypergraphSymmetries] :=
throw[Failure["invalidHyperedges"]];
declareMessage[Hypergraph::invalidSymmetry,
"The argument at position 2 in `expr` should be a supported symmetry: `symmetries`."];
hypergraph[hyperedges_, symmetry_] :=
throw[Failure["invalidSymmetry", <|"symmetries" -> $hypergraphSymmetries|>]];
(* Accessors *)
Hypergraph /: EdgeList[HoldPattern[Hypergraph[hyperedges_, _] ? HypergraphQ]] := hyperedges;
Hypergraph /: EdgeCount[hg_Hypergraph ? HypergraphQ] := Length[EdgeList[hg]];
Hypergraph /: VertexList[hg_Hypergraph ? HypergraphQ] := DeleteDuplicates[Catenate[EdgeList[hg]]];
Hypergraph /: VertexCount[hg_Hypergraph ? HypergraphQ] := Length[VertexList[hg]];
(* Normal *)
Hypergraph /: Normal[hg_Hypergraph ? HypergraphQ] := EdgeList[hg];
(* SameQ *)
Hypergraph /: SameQ[hg1_Hypergraph, hg2_Hypergraph] := Normal[hg1] === Normal[hg2];
(* Boxes *)
disablePlotQ = TrueQ[EdgeCount[#] > 100] &;
getIcon[hg_] /; (!disablePlotQ[hg] && MemberQ[$edgeTypes, HypergraphSymmetry[hg]]) :=
HypergraphPlot[EdgeList[hg], HypergraphSymmetry[hg], ImageSize -> {29, 29}];
getIcon[_] = style[$lightTheme][$evolutionObjectIcon];
Hypergraph /: MakeBoxes[hg_Hypergraph ? HypergraphQ, fmt_] :=
Module[{collapsed, expanded},
collapsed = BoxForm`SummaryItem /@ {
{"VertexCount: ", VertexCount[hg]},
{"EdgeCount: ", EdgeCount[hg]}
};
expanded = BoxForm`SummaryItem /@ {
{"Symmetry: ", HypergraphSymmetry[hg]}
};
BoxForm`ArrangeSummaryBox[
Hypergraph,
hg,
getIcon[hg],
collapsed,
expanded,
fmt,
"Interpretable" -> True
]
];