@@ -17,6 +17,7 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
17
17
PropertyTypes. FixedSize => :fixedsize ,
18
18
PropertyTypes. Graph => :graph ,
19
19
PropertyTypes. Hankel => :hankel ,
20
+ PropertyTypes. Hermitian => :hermitian ,
20
21
PropertyTypes. Hessenberg => :hessenberg ,
21
22
PropertyTypes. IllConditioned => :illcond ,
22
23
PropertyTypes. Indefinite => :indefinite ,
@@ -30,6 +31,7 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
30
31
PropertyTypes. Orthogonal => :orthogonal ,
31
32
PropertyTypes. Positive => :positive ,
32
33
PropertyTypes. PositiveDefinite => :posdef ,
34
+ PropertyTypes. PositiveSemidefinite => :possemidef ,
33
35
PropertyTypes. Random => :random ,
34
36
PropertyTypes. RankDeficient => :rankdef ,
35
37
PropertyTypes. Rectangular => :rectangular ,
@@ -42,10 +44,73 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
42
44
PropertyTypes. TotallyPositive => :totpos ,
43
45
PropertyTypes. Triangular => :triangular ,
44
46
PropertyTypes. Tridiagonal => :tridiagonal ,
45
- PropertyTypes. Unimodular => :unimodular
47
+ PropertyTypes. Unimodular => :unimodular ,
48
+ PropertyTypes. Unitary => :unitary
46
49
)
47
50
)
48
51
52
+ # Handle property implications
53
+ struct PropertyImplication
54
+ ifPresent:: Set{Property}
55
+ ifAbsent:: Set{Property}
56
+ thenAdd:: Set{Property}
57
+ end
58
+
59
+ const IMPLICATIONS = [
60
+ PropertyImplication (Set (Property (present)), Set (Property (absent)), Set (Property (toAdd)))
61
+ for (present:: Vector{Symbol} , absent:: Vector{Symbol} , toAdd:: Vector{Symbol} ) in [
62
+ ([:bidiagonal ], [], [:hessenberg , :sparse , :triangular , :tridiagonal ]),
63
+ ([:circulant ], [], [:toeplitz , :eigen ]),
64
+ ([:complex ], [], [:normal ]),
65
+ ([:correlation ], [], [:symmetric , :possemidef ]),
66
+ ([:diagdom ], [], [:posdef ]),
67
+ ([:hankel ], [], [:symmetric ]),
68
+ ([:hermitian ], [], [:normal ]),
69
+ ([:infdiv ], [], [:possemidef ]),
70
+ ([:involutory ], [], [:inverse ]),
71
+ ([:involutory , :symmetric ], [], [:orthofonal ]),
72
+ ([:involutory , :normal ], [:complex ], [:symmetric , :orthogonal ]),
73
+ ([:involutory , :normal , :complex ], [], [:hermitian , :unitary ]),
74
+ ([:orthogonal ], [:complex ], [:unitary ]),
75
+ ([:positive ], [], [:nonneg ]),
76
+ ([:posdef ], [], [:possemidef ]),
77
+ ([:posdef ], [:complex ], [:symmetric ]),
78
+ ([:posdef ], [:complex ], [:hermitian ]),
79
+ ([:symmetric ], [], [:normal ]),
80
+ ([:symmetric ], [:complex ], [:hermitian ]),
81
+ ([:TotallyNonnegative ], [], [:nonneg ]),
82
+ ([:TotallyNonnegative , :hermitian ], [], [:possemidef ]),
83
+ ([:totallyPositive ], [], [:positive ]),
84
+ ([:totallyPositive , :hermitian ], [], [:posdef ]),
85
+ ([:tridiagonal ], [], [:hessenberg ]),
86
+ ([:unimodular ], [], [:integer ]),
87
+ ([:unitary ], [:complex ], [:symmetric ])
88
+ ]
89
+ ]
90
+
91
+ """
92
+ add_implied_properties(properties::Set{Symbol})
93
+
94
+ Add implied properties to a set of properties.
95
+
96
+ # Examples
97
+ ```julia-repl
98
+ julia> add_implied_properties(Set([:posdef]))
99
+ """
100
+ function add_implied_properties! (properties:: Vector{Property} )
101
+ changed = true
102
+ while changed
103
+ changed = false
104
+ for imp in IMPLICATIONS
105
+ if imp. ifPresent ⊆ properties && isempty (imp. ifAbsent ∩ properties) && ! (imp. thenAdd ⊆ properties)
106
+ append! (properties, setdiff (imp. thenAdd, properties))
107
+ changed = true
108
+ end
109
+ end
110
+ end
111
+ return properties
112
+ end
113
+
49
114
"""
50
115
list_properties()
51
116
@@ -154,6 +219,7 @@ julia> register_properties(Matrix, [PropertyTypes.Symmetric(), PropertyTypes.Inv
154
219
function register_properties (T:: Type , props:: Vector{Property} )
155
220
# check props
156
221
check_properties_exist (props... )
222
+ add_implied_properties! (props)
157
223
158
224
# register properties
159
225
@eval properties (:: Type{<:$T} ) = $ props
0 commit comments