DefaultContext.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832
  1. //
  2. // System.Xml.XPath.DefaultContext & support classes
  3. //
  4. // Author:
  5. // Piers Haken ([email protected])
  6. //
  7. // (C) 2002 Piers Haken
  8. //
  9. using System;
  10. using System.Collections;
  11. using System.Xml;
  12. using System.Xml.XPath;
  13. using System.Xml.Xsl;
  14. using System.Text;
  15. namespace System.Xml.XPath
  16. {
  17. internal class XPathFunctions
  18. {
  19. public static bool ToBoolean (object arg)
  20. {
  21. if (arg == null)
  22. throw new ArgumentNullException ();
  23. if (arg is bool)
  24. return (bool) arg;
  25. if (arg is double)
  26. {
  27. double dArg = (double) arg;
  28. return (dArg != 0.0 && !double.IsNaN (dArg));
  29. }
  30. if (arg is string)
  31. return ((string) arg).Length != 0;
  32. if (arg is BaseIterator)
  33. {
  34. BaseIterator iter = (BaseIterator) arg;
  35. return iter.MoveNext ();
  36. }
  37. throw new ArgumentException ();
  38. }
  39. [MonoTODO]
  40. public static string ToString (object arg)
  41. {
  42. if (arg == null)
  43. throw new ArgumentNullException ();
  44. if (arg is string)
  45. return (string) arg;
  46. if (arg is bool)
  47. return ((bool) arg) ? "true" : "false";
  48. if (arg is double)
  49. return ((double) arg).ToString ("R", System.Globalization.NumberFormatInfo.InvariantInfo);
  50. if (arg is BaseIterator)
  51. {
  52. BaseIterator iter = (BaseIterator) arg;
  53. if (!iter.MoveNext ())
  54. return "";
  55. return iter.Current.Value;
  56. }
  57. throw new ArgumentException ();
  58. }
  59. [MonoTODO]
  60. public static double ToNumber (object arg)
  61. {
  62. if (arg == null)
  63. throw new ArgumentNullException ();
  64. if (arg is string)
  65. {
  66. try
  67. {
  68. return XmlConvert.ToDouble ((string) arg); // TODO: spec? convert string to number
  69. }
  70. catch (System.FormatException)
  71. {
  72. return double.NaN;
  73. }
  74. }
  75. if (arg is BaseIterator)
  76. arg = ToString (arg); // follow on
  77. if (arg is double)
  78. return (double) arg;
  79. if (arg is bool)
  80. return Convert.ToDouble ((bool) arg);
  81. throw new ArgumentException ();
  82. }
  83. public static double ToNumber (string arg)
  84. {
  85. try {
  86. return XmlConvert.ToDouble ((string) arg); // TODO: spec? convert string to number
  87. } catch (System.OverflowException) {
  88. return double.NaN;
  89. } catch (System.FormatException) {
  90. return double.NaN;
  91. }
  92. }
  93. }
  94. internal abstract class XPathFunction : Expression
  95. {
  96. public XPathFunction (FunctionArguments args) {}
  97. }
  98. internal class XPathFunctionLast : XPathFunction
  99. {
  100. public XPathFunctionLast (FunctionArguments args) : base (args)
  101. {
  102. if (args != null)
  103. throw new XPathException ("last takes 0 args");
  104. }
  105. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  106. public override object Evaluate (BaseIterator iter)
  107. {
  108. return (double) iter.Count;
  109. }
  110. }
  111. internal class XPathFunctionPosition : XPathFunction
  112. {
  113. public XPathFunctionPosition (FunctionArguments args) : base (args)
  114. {
  115. if (args != null)
  116. throw new XPathException ("position takes 0 args");
  117. }
  118. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  119. public override object Evaluate (BaseIterator iter)
  120. {
  121. return (double) iter.CurrentPosition;
  122. }
  123. }
  124. internal class XPathFunctionCount : XPathFunction
  125. {
  126. Expression arg0;
  127. public XPathFunctionCount (FunctionArguments args) : base (args)
  128. {
  129. if (args == null || args.Tail != null)
  130. throw new XPathException ("count takes 1 arg");
  131. arg0 = args.Arg;
  132. }
  133. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  134. public override object Evaluate (BaseIterator iter)
  135. {
  136. return (double) arg0.EvaluateNodeSet (iter).Count;
  137. }
  138. public override bool EvaluateBoolean (BaseIterator iter)
  139. {
  140. if (arg0.GetReturnType (iter) == XPathResultType.NodeSet)
  141. return arg0.EvaluateBoolean (iter);
  142. return arg0.EvaluateNodeSet (iter).MoveNext ();
  143. }
  144. }
  145. internal class XPathFunctionId : XPathFunction
  146. {
  147. Expression arg0;
  148. public XPathFunctionId (FunctionArguments args) : base (args)
  149. {
  150. if (args == null || args.Tail != null)
  151. throw new XPathException ("id takes 1 arg");
  152. arg0 = args.Arg;
  153. }
  154. public Expression Id { get { return arg0; } }
  155. private static char [] rgchWhitespace = {' ', '\t', '\r', '\n'};
  156. public override XPathResultType ReturnType { get { return XPathResultType.NodeSet; }}
  157. [MonoTODO]
  158. public override object Evaluate (BaseIterator iter)
  159. {
  160. String strArgs;
  161. object val = arg0.Evaluate (iter);
  162. BaseIterator valItr = val as BaseIterator;
  163. if (valItr != null)
  164. {
  165. strArgs = "";
  166. while (!valItr.MoveNext ())
  167. strArgs += valItr.Current.Value + " ";
  168. }
  169. else
  170. strArgs = XPathFunctions.ToString (val);
  171. XPathNavigator n = iter.Current.Clone ();
  172. ArrayList rgNodes = new ArrayList ();
  173. // FIXME: sort
  174. // SortedList rgNodes = new SortedList (XPathComparer.Instance);
  175. foreach (string strArg in strArgs.Split (rgchWhitespace))
  176. {
  177. if (n.MoveToId (strArg)) {
  178. rgNodes.Add (n.Clone ());
  179. // XPathNavigator clone = n.Clone ();
  180. // rgNodes.Add (clone, clone);
  181. }
  182. }
  183. return new EnumeratorIterator (iter, rgNodes.GetEnumerator ());
  184. // return new EnumeratorIterator (iter, rgNodes.Values.GetEnumerator ());
  185. }
  186. }
  187. internal class XPathFunctionLocalName : XPathFunction
  188. {
  189. Expression arg0;
  190. public XPathFunctionLocalName (FunctionArguments args) : base (args)
  191. {
  192. if (args != null) {
  193. arg0 = args.Arg;
  194. if (args.Tail != null)
  195. throw new XPathException ("local-name takes 1 or zero args");
  196. }
  197. }
  198. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  199. public override object Evaluate (BaseIterator iter)
  200. {
  201. if (arg0 == null)
  202. return iter.Current.LocalName;
  203. BaseIterator argNs = arg0.EvaluateNodeSet (iter);
  204. if (argNs == null || !argNs.MoveNext ())
  205. return "";
  206. return argNs.Current.LocalName;
  207. }
  208. }
  209. internal class XPathFunctionNamespaceUri : XPathFunction
  210. {
  211. Expression arg0;
  212. public XPathFunctionNamespaceUri (FunctionArguments args) : base (args)
  213. {
  214. if (args != null) {
  215. arg0 = args.Arg;
  216. if (args.Tail != null)
  217. throw new XPathException ("namespace-uri takes 1 or zero args");
  218. }
  219. }
  220. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  221. public override object Evaluate (BaseIterator iter)
  222. {
  223. if (arg0 == null)
  224. return iter.Current.NamespaceURI;
  225. BaseIterator argNs = arg0.EvaluateNodeSet (iter);
  226. if (argNs == null || !argNs.MoveNext ())
  227. return "";
  228. return argNs.Current.NamespaceURI;
  229. }
  230. }
  231. internal class XPathFunctionName : XPathFunction
  232. {
  233. Expression arg0;
  234. public XPathFunctionName (FunctionArguments args) : base (args)
  235. {
  236. if (args != null) {
  237. arg0 = args.Arg;
  238. if (args.Tail != null)
  239. throw new XPathException ("name takes 1 or zero args");
  240. }
  241. }
  242. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  243. public override object Evaluate (BaseIterator iter)
  244. {
  245. if (arg0 == null)
  246. return iter.Current.Name;
  247. BaseIterator argNs = arg0.EvaluateNodeSet (iter);
  248. if (argNs == null || !argNs.MoveNext ())
  249. return "";
  250. return argNs.Current.Name;
  251. }
  252. }
  253. internal class XPathFunctionString : XPathFunction
  254. {
  255. Expression arg0;
  256. public XPathFunctionString (FunctionArguments args) : base (args)
  257. {
  258. if (args != null) {
  259. arg0 = args.Arg;
  260. if (args.Tail != null)
  261. throw new XPathException ("boolean takes 1 or zero args");
  262. }
  263. }
  264. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  265. public override object Evaluate (BaseIterator iter)
  266. {
  267. if (arg0 == null)
  268. return iter.Current.Value;
  269. return arg0.EvaluateString (iter);
  270. }
  271. }
  272. internal class XPathFunctionConcat : XPathFunction
  273. {
  274. ArrayList rgs;
  275. public XPathFunctionConcat (FunctionArguments args) : base (args)
  276. {
  277. if (args == null || args.Tail == null)
  278. throw new XPathException ("concat takes 2 or more args");
  279. args.ToArrayList (rgs = new ArrayList ());
  280. }
  281. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  282. public override object Evaluate (BaseIterator iter)
  283. {
  284. StringBuilder sb = new StringBuilder ();
  285. int len = rgs.Count;
  286. for (int i = 0; i < len; i++)
  287. sb.Append (((Expression)rgs[i]).EvaluateString (iter));
  288. return sb.ToString ();
  289. }
  290. }
  291. internal class XPathFunctionStartsWith : XPathFunction
  292. {
  293. Expression arg0, arg1;
  294. public XPathFunctionStartsWith (FunctionArguments args) : base (args)
  295. {
  296. if (args == null || args.Tail == null || args.Tail.Tail != null)
  297. throw new XPathException ("starts-with takes 2 args");
  298. arg0 = args.Arg;
  299. arg1 = args.Tail.Arg;
  300. }
  301. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  302. public override object Evaluate (BaseIterator iter)
  303. {
  304. return arg0.EvaluateString (iter).StartsWith (arg1.EvaluateString (iter));
  305. }
  306. }
  307. internal class XPathFunctionContains : XPathFunction
  308. {
  309. Expression arg0, arg1;
  310. public XPathFunctionContains (FunctionArguments args) : base (args)
  311. {
  312. if (args == null || args.Tail == null || args.Tail.Tail != null)
  313. throw new XPathException ("contains takes 2 args");
  314. arg0 = args.Arg;
  315. arg1 = args.Tail.Arg;
  316. }
  317. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  318. public override object Evaluate (BaseIterator iter)
  319. {
  320. return arg0.EvaluateString (iter).IndexOf (arg1.EvaluateString (iter)) != -1;
  321. }
  322. }
  323. internal class XPathFunctionSubstringBefore : XPathFunction
  324. {
  325. Expression arg0, arg1;
  326. public XPathFunctionSubstringBefore (FunctionArguments args) : base (args)
  327. {
  328. if (args == null || args.Tail == null || args.Tail.Tail != null)
  329. throw new XPathException ("substring-before takes 2 args");
  330. arg0 = args.Arg;
  331. arg1 = args.Tail.Arg;
  332. }
  333. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  334. public override object Evaluate (BaseIterator iter)
  335. {
  336. string str1 = arg0.EvaluateString (iter);
  337. string str2 = arg1.EvaluateString (iter);
  338. int ich = str1.IndexOf (str2);
  339. if (ich <= 0)
  340. return "";
  341. return str1.Substring (0, ich);
  342. }
  343. }
  344. internal class XPathFunctionSubstringAfter : XPathFunction
  345. {
  346. Expression arg0, arg1;
  347. public XPathFunctionSubstringAfter (FunctionArguments args) : base (args)
  348. {
  349. if (args == null || args.Tail == null || args.Tail.Tail != null)
  350. throw new XPathException ("substring-after takes 2 args");
  351. arg0 = args.Arg;
  352. arg1 = args.Tail.Arg;
  353. }
  354. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  355. public override object Evaluate (BaseIterator iter)
  356. {
  357. string str1 = arg0.EvaluateString (iter);
  358. string str2 = arg1.EvaluateString (iter);
  359. int ich = str1.IndexOf (str2);
  360. if (ich < 0)
  361. return "";
  362. return str1.Substring (ich + str2.Length);
  363. }
  364. }
  365. internal class XPathFunctionSubstring : XPathFunction
  366. {
  367. Expression arg0, arg1, arg2;
  368. public XPathFunctionSubstring (FunctionArguments args) : base (args)
  369. {
  370. if (args == null || args.Tail == null || (args.Tail.Tail != null && args.Tail.Tail.Tail != null))
  371. throw new XPathException ("substring takes 2 or 3 args");
  372. arg0 = args.Arg;
  373. arg1 = args.Tail.Arg;
  374. if (args.Tail.Tail != null)
  375. arg2= args.Tail.Tail.Arg;
  376. }
  377. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  378. [MonoTODO]
  379. public override object Evaluate (BaseIterator iter)
  380. {
  381. // TODO: check this, what the hell were they smoking?
  382. string str = arg0.EvaluateString (iter);
  383. double ich = Math.Round (arg1.EvaluateNumber (iter)) - 1;
  384. if (Double.IsNaN (ich) || ich >= (double) str.Length)
  385. return "";
  386. if (arg2 == null)
  387. {
  388. if (ich < 0)
  389. ich = 0.0;
  390. return str.Substring ((int) ich);
  391. }
  392. else
  393. {
  394. double cch = Math.Round (arg2.EvaluateNumber (iter));
  395. if (Double.IsNaN (cch))
  396. return "";
  397. if (ich < 0.0 || cch < 0.0)
  398. {
  399. cch = ich + cch;
  400. if (cch <= 0.0)
  401. return "";
  402. ich = 0.0;
  403. }
  404. double cchMax = (double) str.Length - ich;
  405. if (cch > cchMax)
  406. cch = cchMax;
  407. return str.Substring ((int) ich, (int) cch);
  408. }
  409. }
  410. }
  411. internal class XPathFunctionStringLength : XPathFunction
  412. {
  413. Expression arg0;
  414. public XPathFunctionStringLength (FunctionArguments args) : base (args)
  415. {
  416. if (args != null) {
  417. arg0 = args.Arg;
  418. if (args.Tail != null)
  419. throw new XPathException ("string-length takes 1 or zero args");
  420. }
  421. }
  422. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  423. public override object Evaluate (BaseIterator iter)
  424. {
  425. string str;
  426. if (arg0 != null)
  427. str = arg0.EvaluateString (iter);
  428. else
  429. str = iter.Current.Value;
  430. return (double) str.Length;
  431. }
  432. }
  433. internal class XPathFunctionNormalizeSpace : XPathFunction
  434. {
  435. Expression arg0;
  436. public XPathFunctionNormalizeSpace (FunctionArguments args) : base (args)
  437. {
  438. if (args != null) {
  439. arg0 = args.Arg;
  440. if (args.Tail != null)
  441. throw new XPathException ("string-length takes 1 or zero args");
  442. }
  443. }
  444. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  445. [MonoTODO]
  446. public override object Evaluate (BaseIterator iter)
  447. {
  448. string str;
  449. if (arg0 == null)
  450. str = arg0.EvaluateString (iter);
  451. else
  452. str = iter.Current.Value;
  453. System.Text.StringBuilder sb = new System.Text.StringBuilder ();
  454. bool fSpace = false;
  455. foreach (char ch in str)
  456. {
  457. if (ch == ' ' || ch == '\t' || ch == '\r' || ch == '\n')
  458. {
  459. fSpace = true;
  460. }
  461. else
  462. {
  463. if (fSpace)
  464. {
  465. fSpace = false;
  466. if (sb.Length > 0)
  467. sb.Append (' ');
  468. }
  469. sb.Append (ch);
  470. }
  471. }
  472. return sb.ToString ();
  473. }
  474. }
  475. internal class XPathFunctionTranslate : XPathFunction
  476. {
  477. Expression arg0, arg1, arg2;
  478. public XPathFunctionTranslate (FunctionArguments args) : base (args)
  479. {
  480. if (args == null || args.Tail == null || args.Tail.Tail == null || args.Tail.Tail.Tail != null)
  481. throw new XPathException ("translate takes 3 args");
  482. arg0 = args.Arg;
  483. arg1 = args.Tail.Arg;
  484. arg2= args.Tail.Tail.Arg;
  485. }
  486. public override XPathResultType ReturnType { get { return XPathResultType.String; }}
  487. public override object Evaluate (BaseIterator iter)
  488. {
  489. string s0 = arg0.EvaluateString (iter);
  490. string s1 = arg1.EvaluateString (iter);
  491. string s2 = arg2.EvaluateString (iter);
  492. StringBuilder ret = new StringBuilder (s0.Length);
  493. int pos = 0, len = s0.Length, s2len = s2.Length;
  494. while (pos < len) {
  495. int idx = s1.IndexOf (s0 [pos]);
  496. if (idx != -1) {
  497. if (idx < s2len)
  498. ret.Append (s2 [idx]);
  499. }
  500. else
  501. ret.Append (s0 [pos]);
  502. pos++;
  503. }
  504. return ret.ToString ();
  505. }
  506. }
  507. internal class XPathFunctionBoolean : XPathFunction
  508. {
  509. Expression arg0;
  510. public XPathFunctionBoolean (FunctionArguments args) : base (args)
  511. {
  512. if (args != null) {
  513. arg0 = args.Arg;
  514. if (args.Tail != null)
  515. throw new XPathException ("boolean takes 1 or zero args");
  516. }
  517. }
  518. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  519. public override object Evaluate (BaseIterator iter)
  520. {
  521. if (arg0 == null)
  522. return XPathFunctions.ToBoolean (iter.Current.Value);
  523. return arg0.EvaluateBoolean (iter);
  524. }
  525. }
  526. internal class XPathFunctionNot : XPathFunction
  527. {
  528. Expression arg0;
  529. public XPathFunctionNot (FunctionArguments args) : base (args)
  530. {
  531. if (args == null || args.Tail != null)
  532. throw new XPathException ("not takes one arg");
  533. arg0 = args.Arg;
  534. }
  535. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  536. public override object Evaluate (BaseIterator iter)
  537. {
  538. return !arg0.EvaluateBoolean (iter);
  539. }
  540. }
  541. internal class XPathFunctionTrue : XPathFunction
  542. {
  543. public XPathFunctionTrue (FunctionArguments args) : base (args)
  544. {
  545. if (args != null)
  546. throw new XPathException ("true takes 0 args");
  547. }
  548. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  549. public override object Evaluate (BaseIterator iter)
  550. {
  551. return true;
  552. }
  553. }
  554. internal class XPathFunctionFalse : XPathFunction
  555. {
  556. public XPathFunctionFalse (FunctionArguments args) : base (args)
  557. {
  558. if (args != null)
  559. throw new XPathException ("false takes 0 args");
  560. }
  561. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  562. public override object Evaluate (BaseIterator iter)
  563. {
  564. return false;
  565. }
  566. }
  567. internal class XPathFunctionLang : XPathFunction
  568. {
  569. Expression arg0;
  570. public XPathFunctionLang (FunctionArguments args) : base (args)
  571. {
  572. if (args == null || args.Tail != null)
  573. throw new XPathException ("lang takes one arg");
  574. arg0 = args.Arg;
  575. }
  576. public override XPathResultType ReturnType { get { return XPathResultType.Boolean; }}
  577. public override object Evaluate (BaseIterator iter)
  578. {
  579. string lang = arg0.EvaluateString (iter).ToLower ();
  580. string actualLang = iter.Current.XmlLang.ToLower ();
  581. return lang == actualLang || lang == (actualLang.Split ('-')[0]);
  582. }
  583. }
  584. internal class XPathFunctionNumber : XPathFunction
  585. {
  586. Expression arg0;
  587. public XPathFunctionNumber (FunctionArguments args) : base (args)
  588. {
  589. if (args != null) {
  590. arg0 = args.Arg;
  591. if (args.Tail != null)
  592. throw new XPathException ("number takes 1 or zero args");
  593. }
  594. }
  595. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  596. public override object Evaluate (BaseIterator iter)
  597. {
  598. if (arg0 == null)
  599. return XPathFunctions.ToNumber (iter.Current.Value);
  600. return arg0.EvaluateNumber (iter);
  601. }
  602. }
  603. internal class XPathFunctionSum : XPathFunction
  604. {
  605. Expression arg0;
  606. public XPathFunctionSum (FunctionArguments args) : base (args)
  607. {
  608. if (args == null || args.Tail != null)
  609. throw new XPathException ("sum takes one arg");
  610. arg0 = args.Arg;
  611. }
  612. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  613. public override object Evaluate (BaseIterator iter)
  614. {
  615. XPathNodeIterator itr = arg0.EvaluateNodeSet (iter);
  616. double sum = 0;
  617. while (itr.MoveNext ())
  618. sum += XPathFunctions.ToNumber (itr.Current.Value);
  619. return sum;
  620. }
  621. }
  622. internal class XPathFunctionFloor : XPathFunction
  623. {
  624. Expression arg0;
  625. public XPathFunctionFloor (FunctionArguments args) : base (args)
  626. {
  627. if (args == null || args.Tail != null)
  628. throw new XPathException ("floor takes one arg");
  629. arg0 = args.Arg;
  630. }
  631. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  632. public override object Evaluate (BaseIterator iter)
  633. {
  634. return Math.Floor (arg0.EvaluateNumber (iter));
  635. }
  636. }
  637. internal class XPathFunctionCeil : XPathFunction
  638. {
  639. Expression arg0;
  640. public XPathFunctionCeil (FunctionArguments args) : base (args)
  641. {
  642. if (args == null || args.Tail != null)
  643. throw new XPathException ("ceil takes one arg");
  644. arg0 = args.Arg;
  645. }
  646. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  647. public override object Evaluate (BaseIterator iter)
  648. {
  649. return Math.Ceiling (arg0.EvaluateNumber (iter));
  650. }
  651. }
  652. internal class XPathFunctionRound : XPathFunction
  653. {
  654. Expression arg0;
  655. public XPathFunctionRound (FunctionArguments args) : base (args)
  656. {
  657. if (args == null || args.Tail != null)
  658. throw new XPathException ("round takes one arg");
  659. arg0 = args.Arg;
  660. }
  661. public override XPathResultType ReturnType { get { return XPathResultType.Number; }}
  662. public override object Evaluate (BaseIterator iter)
  663. {
  664. double arg = arg0.EvaluateNumber (iter);
  665. if (arg < -0.5 || arg > 0)
  666. return Math.Floor (arg + 0.5);
  667. return Math.Round (arg);
  668. }
  669. }
  670. }