/Nebo/Model/User.cs

# · C# · 497 lines · 375 code · 47 blank · 75 comment · 68 complexity · 42b3f561f58a725c664b4f1f5020a7c2 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.Specialized;
  4. using System.ComponentModel;
  5. using System.Text;
  6. using System.Text.RegularExpressions;
  7. namespace MetaBuilders.Irc
  8. {
  9. /// <summary>
  10. /// Represents a User on an irc server.
  11. /// </summary>
  12. [Serializable]
  13. public sealed class User : INotifyPropertyChanged
  14. {
  15. #region CTor
  16. /// <summary>
  17. /// Initializes a new instance of the <see cref="User"/> class.
  18. /// </summary>
  19. public User()
  20. {
  21. this.modes.CollectionChanged += delegate( object sender, NotifyCollectionChangedEventArgs e ) {
  22. this.PropChanged( "Modes" );
  23. };
  24. }
  25. /// <summary>
  26. /// Initializes a new instance of the <see cref="User"/> class with the given mask string
  27. /// </summary>
  28. /// <param name="mask">The mask string to parse.</param>
  29. public User( String mask ) : this()
  30. {
  31. Parse( mask );
  32. }
  33. #endregion
  34. #region Properties
  35. /// <summary>
  36. /// Gets or sets the nickname of the User
  37. /// </summary>
  38. public String Nick
  39. {
  40. get
  41. {
  42. return nick;
  43. }
  44. set
  45. {
  46. if ( nick != value )
  47. {
  48. nick = value;
  49. PropChanged( "Nick" );
  50. }
  51. }
  52. }
  53. private String nick = "";
  54. /// <summary>
  55. /// Gets or sets the supposed real name of the User
  56. /// </summary>
  57. public String RealName
  58. {
  59. get
  60. {
  61. return realName;
  62. }
  63. set
  64. {
  65. if ( realName != value )
  66. {
  67. realName = value;
  68. PropChanged( "RealName" );
  69. }
  70. }
  71. }
  72. private String realName = "";
  73. /// <summary>
  74. /// Gets or sets the Password the User will use on the server
  75. /// </summary>
  76. public String Password
  77. {
  78. get
  79. {
  80. return password;
  81. }
  82. set
  83. {
  84. if ( password != value )
  85. {
  86. password = value;
  87. PropChanged( "Password" );
  88. }
  89. }
  90. }
  91. private String password = "";
  92. /// <summary>
  93. /// Gets or sets the username of the User on her local server.
  94. /// </summary>
  95. public String UserName
  96. {
  97. get
  98. {
  99. return username;
  100. }
  101. set
  102. {
  103. if ( username != value )
  104. {
  105. username = value;
  106. PropChanged( "UserName" );
  107. }
  108. }
  109. }
  110. private String username = "";
  111. /// <summary>
  112. /// Gets or sets the hostname of the local machine of this User
  113. /// </summary>
  114. public String HostName
  115. {
  116. get
  117. {
  118. return hostname;
  119. }
  120. set
  121. {
  122. if ( hostname != value )
  123. {
  124. hostname = value;
  125. PropChanged( "HostName" );
  126. }
  127. }
  128. }
  129. private String hostname = "";
  130. /// <summary>
  131. /// Gets or sets the online status of this User
  132. /// </summary>
  133. public UserOnlineStatus OnlineStatus
  134. {
  135. get
  136. {
  137. return onlineStatus;
  138. }
  139. set
  140. {
  141. if ( onlineStatus != value )
  142. {
  143. onlineStatus = value;
  144. PropChanged( "OnlineStatus" );
  145. }
  146. }
  147. }
  148. private UserOnlineStatus onlineStatus;
  149. /// <summary>
  150. /// Gets or sets the away message of this User
  151. /// </summary>
  152. public String AwayMessage
  153. {
  154. get
  155. {
  156. return awayMessage;
  157. }
  158. set
  159. {
  160. if ( awayMessage != value )
  161. {
  162. awayMessage = value;
  163. PropChanged( "AwayMessage" );
  164. }
  165. }
  166. }
  167. private String awayMessage;
  168. /// <summary>
  169. /// Gets or sets the name of the server which the User is connected to.
  170. /// </summary>
  171. public String ServerName
  172. {
  173. get
  174. {
  175. return serverName;
  176. }
  177. set
  178. {
  179. if ( serverName != value )
  180. {
  181. serverName = value;
  182. PropChanged( "ServerName" );
  183. }
  184. }
  185. }
  186. private String serverName;
  187. /// <summary>
  188. /// Gets or sets if the User is an IRC Operator
  189. /// </summary>
  190. public Boolean IrcOperator
  191. {
  192. get
  193. {
  194. return ircOperator;
  195. }
  196. set
  197. {
  198. if ( ircOperator != value )
  199. {
  200. ircOperator = value;
  201. PropChanged( "IrcOperator" );
  202. }
  203. }
  204. }
  205. private Boolean ircOperator;
  206. /// <summary>
  207. /// Gets the modes which apply to the user.
  208. /// </summary>
  209. public MetaBuilders.Irc.Messages.Modes.UserModeCollection Modes
  210. {
  211. get
  212. {
  213. return modes;
  214. }
  215. }
  216. private MetaBuilders.Irc.Messages.Modes.UserModeCollection modes = new MetaBuilders.Irc.Messages.Modes.UserModeCollection();
  217. #endregion
  218. #region Methods
  219. /// <summary>
  220. /// Represents this User's information as an irc mask
  221. /// </summary>
  222. /// <returns></returns>
  223. public override String ToString()
  224. {
  225. StringBuilder result = new StringBuilder();
  226. result.Append( Nick );
  227. if ( !String.IsNullOrEmpty( this.UserName ) )
  228. {
  229. result.Append( "!" );
  230. result.Append( this.UserName );
  231. }
  232. if ( !String.IsNullOrEmpty( this.HostName ) )
  233. {
  234. result.Append( "@" );
  235. result.Append( this.HostName );
  236. }
  237. return result.ToString();
  238. }
  239. /// <summary>
  240. /// Represents this User's information with a guarenteed nick!user@host format.
  241. /// </summary>
  242. public String ToNickUserHostString()
  243. {
  244. String finalNick = ( String.IsNullOrEmpty( this.Nick ) ) ? "*" : this.Nick;
  245. String user = ( String.IsNullOrEmpty( this.UserName ) ) ? "*" : this.UserName;
  246. String host = ( String.IsNullOrEmpty( this.HostName ) ) ? "*" : this.HostName;
  247. return finalNick + "!" + user + "@" + host;
  248. }
  249. /// <summary>
  250. /// Determines wether the current user mask matches the given user mask.
  251. /// </summary>
  252. /// <param name="wildcardMask">The wild-card filled mask to compare with the current.</param>
  253. /// <returns>True if this mask is described by the given wildcard Mask. False if not.</returns>
  254. public Boolean IsMatch( User wildcardMask )
  255. {
  256. if ( wildcardMask == null )
  257. {
  258. return false;
  259. }
  260. //Fist we'll return quickly if they are exact matches
  261. if ( this.Nick == wildcardMask.Nick && this.UserName == wildcardMask.UserName && this.HostName == wildcardMask.HostName )
  262. {
  263. return true;
  264. }
  265. return ( true
  266. && Regex.IsMatch( this.Nick, makeRegexPattern( wildcardMask.Nick ), RegexOptions.IgnoreCase )
  267. && Regex.IsMatch( this.UserName, makeRegexPattern( wildcardMask.UserName ), RegexOptions.IgnoreCase )
  268. && Regex.IsMatch( this.HostName, makeRegexPattern( wildcardMask.HostName ), RegexOptions.IgnoreCase )
  269. );
  270. }
  271. /// <summary>
  272. /// Decides if the given user address matches the given address mask.
  273. /// </summary>
  274. /// <param name="actualMask">The user address mask to compare match.</param>
  275. /// <param name="wildcardMask">The address mask containing wildcards to match with.</param>
  276. /// <returns>True if <parmref>actualMask</parmref> is contained within ( or described with ) the <paramref>wildcardMask</paramref>. False if not.</returns>
  277. public static Boolean IsMatch( String actualMask, String wildcardMask )
  278. {
  279. return new User( actualMask ).IsMatch( new User( wildcardMask ) );
  280. }
  281. /// <summary>
  282. /// Parses the given string as a mask to populate this user object.
  283. /// </summary>
  284. /// <param name="rawMask">The mask to parse.</param>
  285. public void Parse( String rawMask )
  286. {
  287. this.Reset();
  288. String mask = rawMask;
  289. Int32 indexOfBang = mask.IndexOf( "!", StringComparison.Ordinal );
  290. Int32 indexOfAt = mask.LastIndexOf( "@", StringComparison.Ordinal );
  291. if ( indexOfAt > 1 )
  292. {
  293. this.HostName = mask.Substring( indexOfAt + 1 );
  294. mask = mask.Substring( 0, indexOfAt );
  295. }
  296. if ( indexOfBang != -1 )
  297. {
  298. this.UserName = mask.Substring( indexOfBang + 1 );
  299. mask = mask.Substring( 0, indexOfBang );
  300. }
  301. if ( !string.IsNullOrEmpty( mask ) )
  302. {
  303. String newNick = mask;
  304. String firstLetter = newNick.Substring( 0, 1 );
  305. if ( ChannelStatus.Exists( firstLetter ) )
  306. {
  307. newNick = newNick.Substring( 1 );
  308. }
  309. this.Nick = newNick;
  310. }
  311. }
  312. /// <summary>
  313. /// Resets the User properties to the default values
  314. /// </summary>
  315. public void Reset()
  316. {
  317. this.Nick = "";
  318. this.UserName = "";
  319. this.HostName = "";
  320. this.OnlineStatus = UserOnlineStatus.Online;
  321. this.AwayMessage = "";
  322. this.IrcOperator = false;
  323. this.Modes.Clear();
  324. this.Password = "";
  325. this.RealName = "";
  326. this.ServerName = "";
  327. this.UserName = "";
  328. this.dirtyProperties.Clear();
  329. }
  330. /// <summary>
  331. /// Merges the properties of the given User onto this User.
  332. /// </summary>
  333. public void MergeWith( User user )
  334. {
  335. if ( user == null )
  336. {
  337. return;
  338. }
  339. if ( user.IsDirty( "OnlineStatus" ) && !this.IsDirty( "OnlineStatus" ) )
  340. {
  341. this.OnlineStatus = user.OnlineStatus;
  342. }
  343. if ( user.IsDirty( "AwayMessage" ) && !this.IsDirty( "AwayMessage" ) )
  344. {
  345. this.AwayMessage = user.AwayMessage;
  346. }
  347. if ( user.IsDirty( "HostName" ) && !this.IsDirty( "HostName" ) )
  348. {
  349. this.HostName = user.HostName;
  350. }
  351. if ( user.IsDirty( "Nick" ) && !this.IsDirty( "Nick" ) )
  352. {
  353. this.Nick = user.Nick;
  354. }
  355. if ( user.IsDirty( "Password" ) && !this.IsDirty( "Password" ) )
  356. {
  357. this.Password = user.Password;
  358. }
  359. if ( user.IsDirty( "RealName" ) && !this.IsDirty( "RealName" ) )
  360. {
  361. this.RealName = user.RealName;
  362. }
  363. if ( user.IsDirty( "UserName" ) && !this.IsDirty( "UserName" ) )
  364. {
  365. this.UserName = user.UserName;
  366. }
  367. if ( user.IsDirty( "ServerName" ) && !this.IsDirty( "ServerName" ) )
  368. {
  369. this.ServerName = user.ServerName;
  370. }
  371. if ( user.IsDirty( "IrcOperator" ) && !this.IsDirty( "IrcOperator" ) )
  372. {
  373. this.IrcOperator = user.IrcOperator;
  374. }
  375. }
  376. /// <summary>
  377. /// Copies the properties of the given User onto this User.
  378. /// </summary>
  379. public void CopyFrom( User user )
  380. {
  381. if ( user.IsDirty( "OnlineStatus" ) )
  382. {
  383. this.OnlineStatus = user.OnlineStatus;
  384. }
  385. if ( user.IsDirty( "AwayMessage" ) )
  386. {
  387. this.AwayMessage = user.AwayMessage;
  388. }
  389. if ( user.IsDirty( "HostName" ) )
  390. {
  391. this.HostName = user.HostName;
  392. }
  393. if ( user.IsDirty( "Nick" ) )
  394. {
  395. this.Nick = user.Nick;
  396. }
  397. if ( user.IsDirty( "Password" ) )
  398. {
  399. this.Password = user.Password;
  400. }
  401. if ( user.IsDirty( "RealName" ) )
  402. {
  403. this.RealName = user.RealName;
  404. }
  405. if ( user.IsDirty( "UserName" ) )
  406. {
  407. this.UserName = user.UserName;
  408. }
  409. if ( user.IsDirty( "ServerName" ) )
  410. {
  411. this.ServerName = user.ServerName;
  412. }
  413. if ( user.IsDirty( "IrcOperator" ) )
  414. {
  415. this.IrcOperator = user.IrcOperator;
  416. }
  417. }
  418. private static String makeRegexPattern( String wildcardString )
  419. {
  420. return Regex.Escape( wildcardString ).Replace( @"\*", @".*" ).Replace( @"\?", @"." );
  421. }
  422. #endregion
  423. #region INotifyPropertyChanged Members
  424. /// <summary>
  425. /// Raised when a property on the instance has changed.
  426. /// </summary>
  427. public event PropertyChangedEventHandler PropertyChanged;
  428. private void OnPropertyChanged( PropertyChangedEventArgs e )
  429. {
  430. if ( this.PropertyChanged != null )
  431. {
  432. this.PropertyChanged( this, e );
  433. }
  434. }
  435. #endregion
  436. private void PropChanged( String propertyName )
  437. {
  438. if ( !dirtyProperties.Contains( propertyName ) )
  439. {
  440. dirtyProperties.Add( propertyName );
  441. }
  442. this.OnPropertyChanged( new PropertyChangedEventArgs( propertyName ) );
  443. }
  444. private Boolean IsDirty( String propertyName )
  445. {
  446. return dirtyProperties.Contains( propertyName );
  447. }
  448. private List<String> dirtyProperties = new List<string>();
  449. }
  450. }