/trunk/source/Habanero.Base/Criteria/HabaneroStringBuilder.cs

# · C# · 304 lines · 185 code · 19 blank · 100 comment · 32 complexity · 01f6f50bf283578f32870752a0dc39b0 MD5 · raw file

  1. //---------------------------------------------------------------------------------
  2. // Copyright (C) 2008 Chillisoft Solutions
  3. //
  4. // This file is part of the Habanero framework.
  5. //
  6. // Habanero is a free framework: you can redistribute it and/or modify
  7. // it under the terms of the GNU Lesser General Public License as published by
  8. // the Free Software Foundation, either version 3 of the License, or
  9. // (at your option) any later version.
  10. //
  11. // The Habanero framework is distributed in the hope that it will be useful,
  12. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. // GNU Lesser General Public License for more details.
  15. //
  16. // You should have received a copy of the GNU Lesser General Public License
  17. // along with the Habanero framework. If not, see <http://www.gnu.org/licenses/>.
  18. //---------------------------------------------------------------------------------
  19. using System;
  20. using System.Collections.Generic;
  21. using System.Text;
  22. namespace Habanero.Base
  23. {
  24. /// <summary>
  25. /// Wraps StringBuilder, adding some additional string manipulation
  26. /// functions to parse expressions and remove quoted sections from a string.
  27. /// Some example scenarios that would need to be resolved would be: <br/>
  28. /// companyname = 'henry''s station' and ....<br/>
  29. /// companyname = 'brad and partners'<br/>
  30. /// companyname = 'peter (pty) ltd'<br/>
  31. /// NB: Syntax like companyname = 'henry's station' will result in
  32. /// incorrectly parsing of quoted sections, so it is essential that the
  33. /// client of this class parses this to 'henry''s station' prior to
  34. /// using this class.
  35. /// </summary>
  36. public class HabaneroStringBuilder
  37. {
  38. private StringBuilder _string;
  39. private List<QuotedSection> _quotedSections = new List<QuotedSection>(); //this stores an
  40. // array list of all quoted sections that have been removed from
  41. // the original string, along with the position the quoted section
  42. // was removed from (so that it can be replaced when required)
  43. private String[] _quotes = new String[] {"'", "\""};
  44. /// <summary>
  45. /// Constructor to initialise a new builder
  46. /// </summary>
  47. public HabaneroStringBuilder() : base()
  48. {
  49. _string = new StringBuilder();
  50. }
  51. /// <summary>
  52. /// Constructor to initialise a new builder with an initial string
  53. /// </summary>
  54. /// <param name="s">The initial string</param>
  55. public HabaneroStringBuilder(String s) : base()
  56. {
  57. _string = new StringBuilder(s);
  58. }
  59. /// <summary>
  60. /// Sets the list of quote types to recognise (' and " by default)
  61. /// </summary>
  62. /// <param name="quotes">The quotes</param>
  63. public void SetQuotes(string[] quotes)
  64. {
  65. _quotes = quotes;
  66. }
  67. /// <summary>
  68. /// Removes all quoted sections from the string and stores them
  69. /// so that they can be replaced later using PutBackQuotedSections()
  70. /// </summary>
  71. /// <returns>Returns a HabaneroStringBuilder string without quotes</returns>
  72. public HabaneroStringBuilder RemoveQuotedSections()
  73. {
  74. _quotedSections.Clear();
  75. String newString = _string.ToString();
  76. List<QuotedSection> doubleQuotedSections = new List<QuotedSection>();
  77. foreach (String quote in _quotes)
  78. {
  79. int posDouble = 0;
  80. do
  81. {
  82. posDouble = newString.IndexOf(quote + quote, posDouble);
  83. if (posDouble != -1)
  84. {
  85. doubleQuotedSections.Add(new QuotedSection(posDouble, quote));
  86. newString = newString.Remove(posDouble, (quote + quote).Length);
  87. }
  88. } while (posDouble != -1);
  89. }
  90. int offset = 0;
  91. foreach (String quote in _quotes)
  92. {
  93. int pos = newString.IndexOf(quote);
  94. while (pos != -1)
  95. {
  96. int pos2 = newString.Substring(pos + 1, newString.Length - pos - 1).IndexOf(quote);
  97. if (pos2 != -1)
  98. {
  99. int quoteLength = pos2 + 2;
  100. string quoteString = newString.Substring(pos, quoteLength);
  101. int quoteOffset = 0;
  102. for (int i = 0; i < doubleQuotedSections.Count; )
  103. {
  104. QuotedSection doubleQuotedSection = doubleQuotedSections[i];
  105. int quotePos = doubleQuotedSection.pos - offset - pos;
  106. if (quotePos < 0 || quotePos > quoteLength)
  107. {
  108. //Outside the quoted string
  109. i++;
  110. } else
  111. {
  112. //Within the quoted string
  113. quoteString = quoteString.Insert(quotePos + quoteOffset, doubleQuotedSection.quotedSection);
  114. quoteOffset += doubleQuotedSection.quotedSection.Length;
  115. doubleQuotedSections.Remove(doubleQuotedSection);
  116. }
  117. }
  118. offset += quoteLength;
  119. _quotedSections.Add(new QuotedSection(pos, quoteString));
  120. newString = newString.Remove(pos, quoteLength);
  121. pos = newString.IndexOf(quote);
  122. }
  123. else
  124. {
  125. pos = -1;
  126. }
  127. }
  128. }
  129. _string = new StringBuilder(newString);
  130. if (doubleQuotedSections.Count > 0)
  131. {
  132. for (int i = doubleQuotedSections.Count - 1; i >= 0; i--)
  133. {
  134. QuotedSection doubleQuotedSection = doubleQuotedSections[i];
  135. _string.Insert(doubleQuotedSection.pos, doubleQuotedSection.quotedSection);
  136. //Add the offset from this insert to each quoted section that follows.
  137. for( int j = 0; j < _quotedSections.Count; j++)
  138. {
  139. if (_quotedSections[j].pos > doubleQuotedSection.pos)
  140. {
  141. _quotedSections.Insert(j, new QuotedSection(
  142. _quotedSections[j].pos + doubleQuotedSection.quotedSection.Length,
  143. _quotedSections[j].quotedSection));
  144. _quotedSections.RemoveAt(j + 1);
  145. }
  146. }
  147. }
  148. }
  149. return this;
  150. }
  151. /// <summary>
  152. /// Replaces all stored quoted sections into the string in their
  153. /// original positions (these would have been removed using
  154. /// RemoveQuotedSections())
  155. /// </summary>
  156. /// <returns>Returns a HabaneroStringBuilder object with previously
  157. /// removed quoted sections put back</returns>
  158. public HabaneroStringBuilder PutBackQuotedSections()
  159. {
  160. if (_quotedSections.Count > 0)
  161. {
  162. for (int i = _quotedSections.Count - 1; i >= 0; i--)
  163. {
  164. _string.Insert((_quotedSections[i]).pos,
  165. (_quotedSections[i]).quotedSection);
  166. }
  167. _quotedSections.Clear();
  168. }
  169. return this;
  170. }
  171. /// <summary>
  172. /// Returns the string being held
  173. /// </summary>
  174. /// <returns>Returns the string</returns>
  175. public override String ToString()
  176. {
  177. return _string.ToString();
  178. }
  179. /// <summary>
  180. /// Manages a quoted section that has been temporarily removed
  181. /// </summary>
  182. internal struct QuotedSection
  183. {
  184. public int pos;
  185. public String quotedSection;
  186. /// <summary>
  187. /// Constructor to initialise the quoted section
  188. /// </summary>
  189. /// <param name="pos">The position of the quote</param>
  190. /// <param name="quotedSection">The quoted section as a string</param>
  191. public QuotedSection(int pos, String quotedSection)
  192. {
  193. this.pos = pos;
  194. this.quotedSection = quotedSection;
  195. }
  196. }
  197. /// <summary>
  198. /// Returns a sub-string of the string
  199. /// </summary>
  200. /// <param name="startIndex">The starting index</param>
  201. /// <param name="length">The length to return</param>
  202. /// <returns>Returns a HabaneroStringBuilder object</returns>
  203. public HabaneroStringBuilder Substring(int startIndex, int length)
  204. {
  205. HabaneroStringBuilder newHabaneroStringBuilder = new HabaneroStringBuilder(_string.ToString().Substring(startIndex, length));
  206. if (_quotedSections != null)
  207. {
  208. foreach (QuotedSection quote in _quotedSections)
  209. {
  210. if ((quote.pos >= startIndex) && (quote.pos <= startIndex + length))
  211. {
  212. newHabaneroStringBuilder._quotedSections.Add(new QuotedSection(quote.pos - startIndex, quote.quotedSection));
  213. }
  214. }
  215. }
  216. return newHabaneroStringBuilder;
  217. }
  218. /// <summary>
  219. /// Returns a sub-string of the string, beginning at the start index
  220. /// </summary>
  221. /// <param name="startIndex">The start index to begin from</param>
  222. /// <returns>Returns a HabaneroStringBuilder object</returns>
  223. public HabaneroStringBuilder Substring(int startIndex)
  224. {
  225. HabaneroStringBuilder newHabaneroStringBuilder = new HabaneroStringBuilder(_string.ToString().Substring(startIndex));
  226. if (_quotedSections != null)
  227. {
  228. foreach (QuotedSection quote in _quotedSections)
  229. {
  230. if (quote.pos >= startIndex)
  231. {
  232. newHabaneroStringBuilder._quotedSections.Add(new QuotedSection(quote.pos - startIndex, quote.quotedSection));
  233. }
  234. }
  235. }
  236. return newHabaneroStringBuilder;
  237. }
  238. /// <summary>
  239. /// Returns the index position of the given string segment
  240. /// </summary>
  241. /// <param name="value">The string segment to search for</param>
  242. /// <returns>Returns the index position if found, or -1</returns>
  243. public int IndexOf(String value)
  244. {
  245. return _string.ToString().IndexOf(value);
  246. }
  247. /// <summary>
  248. /// Returns the index position of the given string segment, beginning
  249. /// the search at the specified start index point
  250. /// </summary>
  251. /// <param name="value">The string segment to search for</param>
  252. /// <param name="startIndex">The start index point to begin searching
  253. /// from </param>
  254. /// <returns>Returns the index position if found, or -1</returns>
  255. public int IndexOf(String value, int startIndex)
  256. {
  257. return _string.ToString().IndexOf(value, startIndex);
  258. }
  259. /// <summary>
  260. /// Drops the outer quotes from the string
  261. /// </summary>
  262. /// <returns>Returns the string after the outer quotes have been
  263. /// removed</returns>
  264. public HabaneroStringBuilder DropOuterQuotes()
  265. {
  266. if (_string.Length > 0)
  267. {
  268. if ((_string[0] == '\'') && (_string[_string.Length - 1] == '\''))
  269. {
  270. _string.Remove(0, 1);
  271. if (_string.Length != 0)
  272. {
  273. _string.Remove(_string.Length - 1, 1);
  274. }
  275. }
  276. }
  277. return this;
  278. }
  279. internal List<QuotedSection> QuotedSections
  280. {
  281. get { return _quotedSections; }
  282. set { _quotedSections = value; }
  283. }
  284. }
  285. }