-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathTopicMultiMap.cs
160 lines (138 loc) · 8.62 KB
/
TopicMultiMap.cs
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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
/*==============================================================================================================================
| Author Ignia, LLC
| Client Ignia, LLC
| Project Topics Library
\=============================================================================================================================*/
using System.Collections.ObjectModel;
namespace OnTopic.Collections.Specialized {
/*============================================================================================================================
| CLASS: TOPIC MULTIMAP
\---------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// The <see cref="TopicMultiMap"/> offers support for a keyed collection where each key is mapped to a collection of <see
/// cref="Topic"/> instances, thus supporting a 1:n relationship with zero or more topics, organized by key.
/// </summary>
public class TopicMultiMap: KeyedCollection<string, KeyValuesPair<string, TopicCollection>> {
/*==========================================================================================================================
| CONSTRUCTOR
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Constructs a new instance of a <see cref="TopicMultiMap"/> class.
/// </summary>
public TopicMultiMap(): base(StringComparer.OrdinalIgnoreCase) {
}
/*==========================================================================================================================
| METHOD: CONTAINS
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Determines if the <paramref name="topic"/> exists in a collection with the supplied <paramref name="key"/>.
/// </summary>
/// <returns>
/// Returns <c>true</c> if the <see cref="Topic"/> exists in the specified collection. Returns <c>false</c> if the
/// collection doesn't exist.
/// </returns>
public bool Contains(string key, Topic topic) => Contains(key) && this[key].Values.Contains(topic);
/*==========================================================================================================================
| METHOD: GET VALUES
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Retrieves a list of <see cref="Topic"/> objects grouped by a specific <paramref name="key"/>.
/// </summary>
/// <remarks>
/// Returns a reference to the underlying <see cref="Collection{Topic}"/> collection.
/// </remarks>
/// <param name="key">The key of the collection to be returned.</param>
public TopicCollection GetValues(string key) {
Contract.Requires<ArgumentNullException>(!String.IsNullOrWhiteSpace(key), nameof(key));
if (Contains(key)) {
return this[key].Values;
}
return new();
}
/// <inheritdoc cref="GetValues(String)"/>
[ExcludeFromCodeCoverage]
[Obsolete($"The {nameof(GetTopics)} method has been renamed to {nameof(GetValues)}.", true)]
public TopicCollection GetTopics(string key) => GetValues(key);
/*==========================================================================================================================
| METHOD: ADD
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Adds a <paramref name="topic"/> to a collection with the supplied <paramref name="key"/>. If the collection with
/// <paramref name="key"/> doesn't exist, it will be established.
/// </summary>
public void Add(string key, Topic topic) {
/*------------------------------------------------------------------------------------------------------------------------
| Validate parameters
\-----------------------------------------------------------------------------------------------------------------------*/
Contract.Requires(key, nameof(key));
Contract.Requires(topic, nameof(topic));
/*------------------------------------------------------------------------------------------------------------------------
| Ensure collection is established
\-----------------------------------------------------------------------------------------------------------------------*/
if (!Contains(key)) {
Add(new(key, new()));
}
/*------------------------------------------------------------------------------------------------------------------------
| Add topic, if it hasn't already been added
\-----------------------------------------------------------------------------------------------------------------------*/
if (!Contains(key, topic)) {
this[key].Values.Add(topic);
}
}
/*==========================================================================================================================
| METHOD: REMOVE
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Removes a <paramref name="topic"/> from a collection with the supplied <paramref name="key"/>.
/// </summary>
/// <param name="key">The key of the collection.</param>
/// <param name="topic">The <see cref="Topic"/> to be removed.</param>
/// <returns>
/// Returns <c>true</c> if the <see cref="Topic"/> is removed; returns <c>false</c> if either the <paramref name="key"/>
/// or the <paramref name="topic"/> cannot be found.
/// </returns>
public bool Remove(string key, Topic topic) {
/*------------------------------------------------------------------------------------------------------------------------
| Validate contracts
\-----------------------------------------------------------------------------------------------------------------------*/
Contract.Requires<ArgumentNullException>(!String.IsNullOrWhiteSpace(key), nameof(key));
Contract.Requires(topic, nameof(topic));
/*------------------------------------------------------------------------------------------------------------------------
| Validate key
\-----------------------------------------------------------------------------------------------------------------------*/
var topics = GetValues(key);
if (topics is null || !topics.Contains(topic)) {
return false;
}
/*------------------------------------------------------------------------------------------------------------------------
| Remove topic
\-----------------------------------------------------------------------------------------------------------------------*/
topics.Remove(topic);
/*------------------------------------------------------------------------------------------------------------------------
| Remove true
\-----------------------------------------------------------------------------------------------------------------------*/
return true;
}
/*==========================================================================================================================
| METHOD: CLEAR
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Removes all <see cref="Topic"/> objects grouped by a specific <paramref name="key"/>.
/// </summary>
/// <param name="key">The key of the collection to be cleared.</param>
public void Clear(string key) => GetValues(key).Clear();
/*==========================================================================================================================
| OVERRIDE: GET KEY FOR ITEM
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Method must be overridden for the <see cref="TopicMultiMap"/> to extract the keys from the items.
/// </summary>
/// <param name="item">The <see cref="KeyValuesPair{TKey, TValue}"/> object from which to extract the key.</param>
/// <returns>The key for the specified collection item.</returns>
[ExcludeFromCodeCoverage]
protected override sealed string GetKeyForItem(KeyValuesPair<string, TopicCollection> item) {
Contract.Requires(item, "The item must be available in order to derive its key.");
return item.Key;
}
} //Class
} //Namespace