yellow-brick-road 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (233) hide show
  1. data/README.rst +148 -0
  2. data/Rakefile +37 -0
  3. data/lib/tasks/yellow-brick-road_tasks.rake +4 -0
  4. data/lib/yellow-brick-road.rb +6 -0
  5. data/lib/yellow-brick-road/config.rb +15 -0
  6. data/lib/yellow-brick-road/directive_processor.rb +68 -0
  7. data/lib/yellow-brick-road/engine.rb +22 -0
  8. data/lib/yellow-brick-road/soy_processor.rb +56 -0
  9. data/lib/yellow-brick-road/utils.rb +38 -0
  10. data/lib/yellow-brick-road/version.rb +3 -0
  11. data/test/dummy/Rakefile +7 -0
  12. data/test/dummy/app/assets/javascripts/application.js +12 -0
  13. data/test/dummy/app/assets/javascripts/closure-deps.js +4 -0
  14. data/test/dummy/app/assets/javascripts/my-closure/simple.js.soy +14 -0
  15. data/test/dummy/app/assets/javascripts/my-closure/start.js +25 -0
  16. data/test/dummy/app/assets/stylesheets/application.css +7 -0
  17. data/test/dummy/app/controllers/application_controller.rb +7 -0
  18. data/test/dummy/app/helpers/application_helper.rb +2 -0
  19. data/test/dummy/app/views/application/index.html.erb +0 -0
  20. data/test/dummy/app/views/layouts/application.html.erb +18 -0
  21. data/test/dummy/config.ru +4 -0
  22. data/test/dummy/config/application.rb +51 -0
  23. data/test/dummy/config/boot.rb +10 -0
  24. data/test/dummy/config/environment.rb +5 -0
  25. data/test/dummy/config/environments/development.rb +30 -0
  26. data/test/dummy/config/environments/production.rb +60 -0
  27. data/test/dummy/config/environments/test.rb +39 -0
  28. data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
  29. data/test/dummy/config/initializers/inflections.rb +10 -0
  30. data/test/dummy/config/initializers/mime_types.rb +5 -0
  31. data/test/dummy/config/initializers/secret_token.rb +7 -0
  32. data/test/dummy/config/initializers/session_store.rb +8 -0
  33. data/test/dummy/config/initializers/wrap_parameters.rb +10 -0
  34. data/test/dummy/config/initializers/yellow_brick_road.rb +2 -0
  35. data/test/dummy/config/locales/en.yml +5 -0
  36. data/test/dummy/config/routes.rb +58 -0
  37. data/test/dummy/log/development.log +13924 -0
  38. data/test/dummy/log/test.log +0 -0
  39. data/test/dummy/public/404.html +26 -0
  40. data/test/dummy/public/422.html +26 -0
  41. data/test/dummy/public/500.html +26 -0
  42. data/test/dummy/public/favicon.ico +0 -0
  43. data/test/dummy/script/rails +6 -0
  44. data/test/dummy/tmp/cache/assets/BE0/120/sprockets%2F751842b8c6750008c7310d182600d173 +142 -0
  45. data/test/dummy/tmp/cache/assets/C19/950/sprockets%2F142437f1d8d9424007b4a882a5429333 +1372 -0
  46. data/test/dummy/tmp/cache/assets/C4B/D00/sprockets%2F400f22c081529179ce3d079457de3009 +806 -0
  47. data/test/dummy/tmp/cache/assets/C5A/EE0/sprockets%2Fb711429ed948c503b718d077037780f8 +0 -0
  48. data/test/dummy/tmp/cache/assets/C5B/A80/sprockets%2F508569b474262724c61a461f7777dab7 +1118 -0
  49. data/test/dummy/tmp/cache/assets/C6C/660/sprockets%2F2c298b3b02232a21527680685a3efc30 +9289 -0
  50. data/test/dummy/tmp/cache/assets/C72/8A0/sprockets%2F167b265129e30d87d253c406db305c60 +293 -0
  51. data/test/dummy/tmp/cache/assets/C7E/9F0/sprockets%2F89862076204c62c4593ac20de32da909 +9 -0
  52. data/test/dummy/tmp/cache/assets/C8B/5F0/sprockets%2Fc6a4470b5c21e285e829a99365839b24 +0 -0
  53. data/test/dummy/tmp/cache/assets/C92/D20/sprockets%2F2e618f7805f445889aec94885a500f03 +457 -0
  54. data/test/dummy/tmp/cache/assets/C98/FD0/sprockets%2Fb11442af041f96e87a43a1dc11231745 +283 -0
  55. data/test/dummy/tmp/cache/assets/CA3/520/sprockets%2F5379d7143c6c52b11b88dc0ab5436133 +277 -0
  56. data/test/dummy/tmp/cache/assets/CA5/450/sprockets%2F6bb727c9312a749134ad67323a317f0d +73 -0
  57. data/test/dummy/tmp/cache/assets/CA5/4F0/sprockets%2Feeb7de7771527700af194c0441d29101 +709 -0
  58. data/test/dummy/tmp/cache/assets/CA6/E90/sprockets%2F611f68180f43c4181f06ae5c5f8201e2 +1546 -0
  59. data/test/dummy/tmp/cache/assets/CA7/310/sprockets%2F45664cf816315200b574e029fde6f10a +0 -0
  60. data/test/dummy/tmp/cache/assets/CA9/9D0/sprockets%2F2672e32464cf7267c4ba3d028f54b153 +224 -0
  61. data/test/dummy/tmp/cache/assets/CAB/5A0/sprockets%2F7f50e0289f150c8636ac9253129bc13c +2556 -0
  62. data/test/dummy/tmp/cache/assets/CB5/7E0/sprockets%2F42ff6672683b2029233a800e7539eeee +474 -0
  63. data/test/dummy/tmp/cache/assets/CB6/DC0/sprockets%2F2f9882155bb2d4d3ab5d708951857c60 +494 -0
  64. data/test/dummy/tmp/cache/assets/CBB/680/sprockets%2F1dc336d96fb52df34b458185559922b5 +1018 -0
  65. data/test/dummy/tmp/cache/assets/CBC/640/sprockets%2F67d2e0d9e5129d237e575d2780c64b47 +1260 -0
  66. data/test/dummy/tmp/cache/assets/CBE/550/sprockets%2Fd680cac830e0b3408ba910f0b0421147 +25 -0
  67. data/test/dummy/tmp/cache/assets/CC7/790/sprockets%2F69941f32a12e4f99d4a57f65386d870d +608 -0
  68. data/test/dummy/tmp/cache/assets/CCB/F80/sprockets%2Fa865701ef2ec41155e524772c31a1a2b +1088 -0
  69. data/test/dummy/tmp/cache/assets/CCE/580/sprockets%2F879411ed27ed1c557d57853d8f579b56 +0 -0
  70. data/test/dummy/tmp/cache/assets/CD0/070/sprockets%2F6748fe8481965f260d9c56b7f9f508a2 +1530 -0
  71. data/test/dummy/tmp/cache/assets/CD4/750/sprockets%2Faf3505141ecb3169ce41ce519d136924 +0 -0
  72. data/test/dummy/tmp/cache/assets/CD6/A90/sprockets%2F3f55ac75b9fb8426312116bcb940a580 +2539 -0
  73. data/test/dummy/tmp/cache/assets/CD9/6C0/sprockets%2F5e2458bc52da90ba349a66035e3b6752 +0 -0
  74. data/test/dummy/tmp/cache/assets/CD9/F50/sprockets%2F9008bf696500cfae1d61f045f209181e +256 -0
  75. data/test/dummy/tmp/cache/assets/CDB/B40/sprockets%2F53529a22c994570a0df4742c0bfe61f4 +0 -0
  76. data/test/dummy/tmp/cache/assets/CDC/D10/sprockets%2F1e775b4ff06b4401c07503ce95a839b5 +75 -0
  77. data/test/dummy/tmp/cache/assets/CDE/CD0/sprockets%2F141066798b4acf07053f7e3a6cb4e555 +1613 -0
  78. data/test/dummy/tmp/cache/assets/CE1/760/sprockets%2F245fe11803630fe30d0cf8a869886ab5 +357 -0
  79. data/test/dummy/tmp/cache/assets/CE1/FA0/sprockets%2F863d1650ef066e4a2168bc57c7c0e096 +0 -0
  80. data/test/dummy/tmp/cache/assets/CE2/310/sprockets%2Fd38075d5592ecaf82e43526c03b467c4 +1545 -0
  81. data/test/dummy/tmp/cache/assets/CE2/420/sprockets%2F30f2e8f30477e80cf6416dfe27307c07 +204 -0
  82. data/test/dummy/tmp/cache/assets/CE2/D70/sprockets%2F467c44b63644e0f44dd06a585f36f1a6 +522 -0
  83. data/test/dummy/tmp/cache/assets/CE3/670/sprockets%2Fddd11860b444cd0f9996be0c46762318 +796 -0
  84. data/test/dummy/tmp/cache/assets/CE4/0B0/sprockets%2Ff11535d785c0d34349c64d673bd4b28f +1105 -0
  85. data/test/dummy/tmp/cache/assets/CE4/590/sprockets%2Fd68edcc87a4ae302794093081b45b819 +10 -0
  86. data/test/dummy/tmp/cache/assets/CE4/D60/sprockets%2F61772e4a60b616f74da91b838a2f4f82 +0 -0
  87. data/test/dummy/tmp/cache/assets/CE6/DD0/sprockets%2F2fe073024a2bf26bd98458388b57af37 +1355 -0
  88. data/test/dummy/tmp/cache/assets/CE7/160/sprockets%2Fba1f9939f031b4356ec1869d40fc2747 +1261 -0
  89. data/test/dummy/tmp/cache/assets/CE8/C70/sprockets%2Fc6c9ba3c677b5e2af8520395192c9445 +173 -0
  90. data/test/dummy/tmp/cache/assets/CEA/8B0/sprockets%2Ff77e549cb6d37604105f35d4e67d8c21 +511 -0
  91. data/test/dummy/tmp/cache/assets/CEA/AA0/sprockets%2F5f816982c86d2e6b72b2f5f65c51d070 +1529 -0
  92. data/test/dummy/tmp/cache/assets/CEA/C10/sprockets%2F9dcd541e67c299ab076a44a2183872f8 +0 -0
  93. data/test/dummy/tmp/cache/assets/CF1/5A0/sprockets%2F1a697695edf2bb7b49a2896904218bc7 +115 -0
  94. data/test/dummy/tmp/cache/assets/CF4/480/sprockets%2F5e99c77e93f4a522c84357e62b25e0f7 +43 -0
  95. data/test/dummy/tmp/cache/assets/CF7/460/sprockets%2Fd12ea9733fe3c92456f57f9145569b9c +0 -0
  96. data/test/dummy/tmp/cache/assets/CF7/470/sprockets%2F2897897a166ca3369fecb88f83f211b5 +435 -0
  97. data/test/dummy/tmp/cache/assets/CFA/760/sprockets%2Fa9024adba14091e0635c6874d1db4e22 +0 -0
  98. data/test/dummy/tmp/cache/assets/D01/F30/sprockets%2Fe639a15b6e0cca37d12443b408e1166f +25 -0
  99. data/test/dummy/tmp/cache/assets/D02/9F0/sprockets%2Fb99eae308897fe88cb9414b96824098b +0 -0
  100. data/test/dummy/tmp/cache/assets/D03/330/sprockets%2F1e003cdb3e7dcc9307e84090ad457127 +454 -0
  101. data/test/dummy/tmp/cache/assets/D03/8B0/sprockets%2Ff74632bddf2c0b2018ca7b736309380e +365 -0
  102. data/test/dummy/tmp/cache/assets/D05/0A0/sprockets%2Fe57658220260db13eb5577aef42cb61b +257 -0
  103. data/test/dummy/tmp/cache/assets/D05/920/sprockets%2F909507434dcc270db4853e4c147f0aac +31 -0
  104. data/test/dummy/tmp/cache/assets/D08/510/sprockets%2Fa567be6cb6d7310096f1739b25a5a3f0 +50 -0
  105. data/test/dummy/tmp/cache/assets/D08/9F0/sprockets%2F1d61d2c89ca50957066bacc5b69011f5 +1424 -0
  106. data/test/dummy/tmp/cache/assets/D0A/790/sprockets%2Feb16913e6504c9b0d3be431de39e4751 +300 -0
  107. data/test/dummy/tmp/cache/assets/D0C/C50/sprockets%2Fd6613bee8b40d50459af6b52a7084f34 +796 -0
  108. data/test/dummy/tmp/cache/assets/D0D/030/sprockets%2Fba3f13b4a444679e8bc2549226ec743b +21 -0
  109. data/test/dummy/tmp/cache/assets/D0D/350/sprockets%2F2670bce036d485e15d059c0f1e98f24a +207 -0
  110. data/test/dummy/tmp/cache/assets/D13/270/sprockets%2F497cb163e6317e3fc1565d832f406cfb +348 -0
  111. data/test/dummy/tmp/cache/assets/D13/380/sprockets%2F786d003c9c7fb759dd26c1030c087cf6 +813 -0
  112. data/test/dummy/tmp/cache/assets/D13/7C0/sprockets%2F528d22310a9ab8e6fba08d82844ce795 +0 -0
  113. data/test/dummy/tmp/cache/assets/D15/F60/sprockets%2Fa28394e3f80365b5bc86794dd46daa22 +0 -0
  114. data/test/dummy/tmp/cache/assets/D18/500/sprockets%2Fd22c2d97d1db2154f2f7592906e957ea +1001 -0
  115. data/test/dummy/tmp/cache/assets/D1B/C70/sprockets%2F1c04848b1e1d6e8e33b8581f8c8128ff +223 -0
  116. data/test/dummy/tmp/cache/assets/D1C/600/sprockets%2Fa0601f99147f59ddd6266e6aff077e14 +0 -0
  117. data/test/dummy/tmp/cache/assets/D1E/470/sprockets%2Fb3933e694547b78bf6fb15f44a4623fe +53 -0
  118. data/test/dummy/tmp/cache/assets/D1F/480/sprockets%2F545164168becf42b289efc8708f1db68 +1278 -0
  119. data/test/dummy/tmp/cache/assets/D21/630/sprockets%2F0a14114729c14637f2e8f122acabd1ab +823 -0
  120. data/test/dummy/tmp/cache/assets/D24/A90/sprockets%2F21c8a0d48c51b8585ba03bc112ecd153 +692 -0
  121. data/test/dummy/tmp/cache/assets/D25/D30/sprockets%2F3b937f4c16d2aa0a28fc02a0e922712b +1441 -0
  122. data/test/dummy/tmp/cache/assets/D25/F30/sprockets%2Fe251527c0c018f4b4ad44b2ad7064fb0 +2255 -0
  123. data/test/dummy/tmp/cache/assets/D26/DE0/sprockets%2F761ac1bdb0739cc56fa0c1224b137b09 +0 -0
  124. data/test/dummy/tmp/cache/assets/D2A/C30/sprockets%2F978fe7268754f451c59b9ccb323ffa56 +0 -0
  125. data/test/dummy/tmp/cache/assets/D2B/0F0/sprockets%2Fbf223af03335cfc08a71e549e10c0e27 +2539 -0
  126. data/test/dummy/tmp/cache/assets/D2E/6D0/sprockets%2F9e7fda3affb389313cc50223f01c0324 +0 -0
  127. data/test/dummy/tmp/cache/assets/D2F/110/sprockets%2F3730ffe2ed34c1219d580bd1615ea7b0 +2239 -0
  128. data/test/dummy/tmp/cache/assets/D2F/F10/sprockets%2F9cac242f0bf5ec00079ea3a463e26552 +153 -0
  129. data/test/dummy/tmp/cache/assets/D30/1D0/sprockets%2Fd690a26a0b0ae138c1c48d257a7674bc +589 -0
  130. data/test/dummy/tmp/cache/assets/D32/A10/sprockets%2F13fe41fee1fe35b49d145bcc06610705 +0 -0
  131. data/test/dummy/tmp/cache/assets/D36/940/sprockets%2Fb4ba462ac9f4aac561c36e60367c1378 +572 -0
  132. data/test/dummy/tmp/cache/assets/D37/090/sprockets%2F2f2e71ebbc240014ebf648f0917e854a +798 -0
  133. data/test/dummy/tmp/cache/assets/D3A/440/sprockets%2F6d332243647c841dea36ed822cdfc23e +165 -0
  134. data/test/dummy/tmp/cache/assets/D3B/5A0/sprockets%2F82e05cf8c88ba8cb1c87e50d51372a03 +0 -0
  135. data/test/dummy/tmp/cache/assets/D41/B30/sprockets%2Fba1b93913dd01d83ac9a96df334456f8 +0 -0
  136. data/test/dummy/tmp/cache/assets/D43/D90/sprockets%2F04f3a5926d5a7672456dc7afa73d0c7f +38 -0
  137. data/test/dummy/tmp/cache/assets/D45/1F0/sprockets%2F1bfa855d9c10ff2431a9a548958cc5e4 +83 -0
  138. data/test/dummy/tmp/cache/assets/D45/B10/sprockets%2F643f3d9b216e48df952b7f150a46a7ec +60 -0
  139. data/test/dummy/tmp/cache/assets/D46/040/sprockets%2F199546844e5939721a5afcbcce4ea43a +143 -0
  140. data/test/dummy/tmp/cache/assets/D49/320/sprockets%2Fd8123eb229e34c9f507f7b1876f1c4ee +506 -0
  141. data/test/dummy/tmp/cache/assets/D49/750/sprockets%2Ff41ef1c0832b7294fb15588ff5c783cb +0 -0
  142. data/test/dummy/tmp/cache/assets/D49/930/sprockets%2F1071926698fa55abc71e1b6fd966dfe1 +72 -0
  143. data/test/dummy/tmp/cache/assets/D49/D10/sprockets%2Fb67b8d7cc7579352a91694f9ae10cdb0 +127 -0
  144. data/test/dummy/tmp/cache/assets/D4A/160/sprockets%2Fcbd8988b5e18d153eff6230d72e9046a +516 -0
  145. data/test/dummy/tmp/cache/assets/D4A/2B0/sprockets%2Ffcdf002d38fd938bf7226b46b76706c8 +1101 -0
  146. data/test/dummy/tmp/cache/assets/D4E/2F0/sprockets%2F77cb87ae57f0d8c25cdaa74181a64997 +1441 -0
  147. data/test/dummy/tmp/cache/assets/D4F/060/sprockets%2Fa9e66b39ada7394a29bf44f3a682f665 +511 -0
  148. data/test/dummy/tmp/cache/assets/D50/BD0/sprockets%2F88650dd57ef64075462e6dae757dbe2c +0 -0
  149. data/test/dummy/tmp/cache/assets/D53/CA0/sprockets%2Fa85032e82709a043fbb2ec00e04f2bbd +0 -0
  150. data/test/dummy/tmp/cache/assets/D54/BF0/sprockets%2F91970514ff528e8d2bfd81f1ec83c9c8 +240 -0
  151. data/test/dummy/tmp/cache/assets/D54/ED0/sprockets%2F71c9fa01091d432b131da3bb73faf3d4 +10 -0
  152. data/test/dummy/tmp/cache/assets/D56/500/sprockets%2F75ebadd035f1324b194034af92eac3a0 +0 -0
  153. data/test/dummy/tmp/cache/assets/D5A/900/sprockets%2F5729f77d97fdee53b1942cf17f6f05e4 +474 -0
  154. data/test/dummy/tmp/cache/assets/D5A/D50/sprockets%2F81bace3db2c2f1241175ff3c5009d08c +0 -0
  155. data/test/dummy/tmp/cache/assets/D5B/C70/sprockets%2Fe64119a9db7017b7ab3b0da1b6076f0d +44 -0
  156. data/test/dummy/tmp/cache/assets/D5E/4D0/sprockets%2F756f25e4ad861a050cdc41bb8414e4ab +126 -0
  157. data/test/dummy/tmp/cache/assets/D5E/730/sprockets%2Fd30caf70b387c4604326bdda32aeb549 +0 -0
  158. data/test/dummy/tmp/cache/assets/D5E/AA0/sprockets%2F790dd07caaaacb30ceb0174664e90817 +136 -0
  159. data/test/dummy/tmp/cache/assets/D64/D30/sprockets%2Fe4bd8f60a3a446274c3cb2f5d16563ac +590 -0
  160. data/test/dummy/tmp/cache/assets/D66/400/sprockets%2Fecb7401d20daf25605a7afad3a793778 +0 -0
  161. data/test/dummy/tmp/cache/assets/D67/250/sprockets%2F8422eb26855ca9c953a3bf1aeade8004 +0 -0
  162. data/test/dummy/tmp/cache/assets/D69/900/sprockets%2F28ae64f52cb18f77b1a1bd9c51293da4 +0 -0
  163. data/test/dummy/tmp/cache/assets/D6A/870/sprockets%2F1ece6a51b42f9280f4cee9020c94b72e +197 -0
  164. data/test/dummy/tmp/cache/assets/D6B/DD0/sprockets%2F9a0abc735e27fc61dab019788518eba4 +164 -0
  165. data/test/dummy/tmp/cache/assets/D6E/CC0/sprockets%2F8b4fbcec2475e7203b8859baf8e310a3 +0 -0
  166. data/test/dummy/tmp/cache/assets/D72/470/sprockets%2Fbc0c6d850e7c769e83d58eb3e6061b2a +2556 -0
  167. data/test/dummy/tmp/cache/assets/D72/9D0/sprockets%2F9402daf6da6f7eed331a9d78216cb761 +0 -0
  168. data/test/dummy/tmp/cache/assets/D74/070/sprockets%2F91ae68f698a39bd7d0d11150fc46e4cd +418 -0
  169. data/test/dummy/tmp/cache/assets/D74/7F0/sprockets%2F22dad1069ea6a8f29ae757c44e18ec83 +0 -0
  170. data/test/dummy/tmp/cache/assets/D75/3C0/sprockets%2F44e2ae51b97853d8ecda7b264a267c7d +1630 -0
  171. data/test/dummy/tmp/cache/assets/D77/780/sprockets%2Ffa2abdfb0b62867c346f79187b8e42d5 +823 -0
  172. data/test/dummy/tmp/cache/assets/D79/850/sprockets%2F4d540586ed379fd9a70fa0b7ce4f5b27 +306 -0
  173. data/test/dummy/tmp/cache/assets/D7A/B60/sprockets%2F5925c92f48caff053d08853dc69e5bba +0 -0
  174. data/test/dummy/tmp/cache/assets/D81/C90/sprockets%2F19aca5c835efce74878c23e581e2a89e +2239 -0
  175. data/test/dummy/tmp/cache/assets/D84/210/sprockets%2Fabd0103ccec2b428ac62c94e4c40b384 +11228 -0
  176. data/test/dummy/tmp/cache/assets/D8B/B10/sprockets%2Fbb7a2305a8f2832631863bbababb67be +301 -0
  177. data/test/dummy/tmp/cache/assets/D92/3B0/sprockets%2F93dc71f411faf72ec8ef0b5d217b51a0 +505 -0
  178. data/test/dummy/tmp/cache/assets/D92/EE0/sprockets%2Ffd051ea0abf6be6970754aa732e40c8b +289 -0
  179. data/test/dummy/tmp/cache/assets/D93/810/sprockets%2F7dc1ac5e1b83ba5144e34cf156d48a8a +471 -0
  180. data/test/dummy/tmp/cache/assets/D94/020/sprockets%2F35cb4ce22488fb5bdd260508ca47a3fb +239 -0
  181. data/test/dummy/tmp/cache/assets/D94/EC0/sprockets%2F8b900fba18f8e046f5f12e6d1f0ac15b +74 -0
  182. data/test/dummy/tmp/cache/assets/D95/470/sprockets%2F5be5d361a52575c433831b5e8cccbace +9273 -0
  183. data/test/dummy/tmp/cache/assets/D97/9D0/sprockets%2F524a2a94a88c1d392dabcba654e365fc +103 -0
  184. data/test/dummy/tmp/cache/assets/D9D/800/sprockets%2Ff8387220eb4dda9b29a37b5deb9f2a40 +54 -0
  185. data/test/dummy/tmp/cache/assets/D9E/DD0/sprockets%2Fb6b4ffad6485eda046b327a196e8c44a +0 -0
  186. data/test/dummy/tmp/cache/assets/DA1/0A0/sprockets%2F359c94aee6ba2e15fc9666aa4de311c8 +166 -0
  187. data/test/dummy/tmp/cache/assets/DA1/0D0/sprockets%2F798de3c28452e164a7fc1a3a6e8cdb4b +652 -0
  188. data/test/dummy/tmp/cache/assets/DA3/CB0/sprockets%2Fdfad6bb1857a8af91f98ba137c18504e +815 -0
  189. data/test/dummy/tmp/cache/assets/DA4/E10/sprockets%2Fbda7705734ace95f12a7d79df9c14af2 +1354 -0
  190. data/test/dummy/tmp/cache/assets/DA5/E30/sprockets%2F2e6707a119e94a784efbadca18d7e8d1 +129 -0
  191. data/test/dummy/tmp/cache/assets/DA6/4B0/sprockets%2F4f7a708b6aa2d42eb577493dea7cef55 +115 -0
  192. data/test/dummy/tmp/cache/assets/DA8/A10/sprockets%2F0ae5bbb4ca200e74db3b64d72c2be120 +0 -0
  193. data/test/dummy/tmp/cache/assets/DA9/3F0/sprockets%2Fdd0af9cfd65fb7d02fe778c8f6015361 +499 -0
  194. data/test/dummy/tmp/cache/assets/DA9/780/sprockets%2Fe3559a635e8f92dec7727a3db2ae7b4c +813 -0
  195. data/test/dummy/tmp/cache/assets/DAC/0E0/sprockets%2F04cef8243a9ef7321846debc977ea8db +607 -0
  196. data/test/dummy/tmp/cache/assets/DB1/260/sprockets%2Fb879ff7530bf8cc175e0c7e70dc4e79b +0 -0
  197. data/test/dummy/tmp/cache/assets/DB4/4C0/sprockets%2F30a76cbd98dedb56742cd8ecaf858757 +0 -0
  198. data/test/dummy/tmp/cache/assets/DBD/7E0/sprockets%2F236fb1ac5ed9f2e3fe32203acc3a842d +0 -0
  199. data/test/dummy/tmp/cache/assets/DC2/DF0/sprockets%2Fe233bcaba8beff15626c35cb5e0e0936 +845 -0
  200. data/test/dummy/tmp/cache/assets/DC4/600/sprockets%2F59c96aa6c1cbebc61bff6c050a5351d7 +225 -0
  201. data/test/dummy/tmp/cache/assets/DC5/030/sprockets%2Fcb2fe864a0177eec19017c43ce9aa4ad +187 -0
  202. data/test/dummy/tmp/cache/assets/DC8/A60/sprockets%2F9fcb9f5c3f679ce749ee4c3f93869ba6 +74 -0
  203. data/test/dummy/tmp/cache/assets/DCB/D40/sprockets%2Ffd5542bfb7660a8d8cac3a2e46fb01f8 +828 -0
  204. data/test/dummy/tmp/cache/assets/DCC/020/sprockets%2F0c129c5f2784c96fa1dc3da6c19c1efc +128 -0
  205. data/test/dummy/tmp/cache/assets/DCF/520/sprockets%2F975162fb97a3ec7033db5f2fdba4fcd3 +70 -0
  206. data/test/dummy/tmp/cache/assets/DD0/000/sprockets%2F0d5d631cce6f0b078bb29cf98b3d78bb +276 -0
  207. data/test/dummy/tmp/cache/assets/DD0/C90/sprockets%2F84eeca4ef8f5eb29b13f7e1ea3c71324 +591 -0
  208. data/test/dummy/tmp/cache/assets/DD6/710/sprockets%2F9e83f2ebbe3e349efca2de390e15d635 +204 -0
  209. data/test/dummy/tmp/cache/assets/DE1/830/sprockets%2Fd23d3cafef2e2a9055bc103dffc1a023 +208 -0
  210. data/test/dummy/tmp/cache/assets/DEB/470/sprockets%2Fae2de0fcbc2214b71d3f2a875d2bc8c3 +2255 -0
  211. data/test/dummy/tmp/cache/assets/DEE/690/sprockets%2F5d824bcadef29060cc13eed71af4b4d0 +550 -0
  212. data/test/dummy/tmp/cache/assets/DF0/5D0/sprockets%2F25a875eb0a5d4c49bcc3fa39ea0adb26 +567 -0
  213. data/test/dummy/tmp/cache/assets/DF3/E20/sprockets%2Fada770c49a5d6b38d1cc43ed4d51c1fc +0 -0
  214. data/test/dummy/tmp/cache/assets/DF5/C80/sprockets%2F03ad7d7b3a8feba9ed9be82ae2083b20 +340 -0
  215. data/test/dummy/tmp/cache/assets/DF6/560/sprockets%2Fafceb2b55eb108efe4446dfb02b519b8 +0 -0
  216. data/test/dummy/tmp/cache/assets/DFC/030/sprockets%2Fcbdd3679eaba219b4e1ff20677af1c6f +1371 -0
  217. data/test/dummy/tmp/cache/assets/E00/970/sprockets%2F8bc6dde8d05c382dd07cbcec70f54e09 +378 -0
  218. data/test/dummy/tmp/cache/assets/E04/890/sprockets%2F2f5173deea6c795b8fdde723bb4b63af +0 -0
  219. data/test/dummy/tmp/cache/assets/E12/2F0/sprockets%2F5c45062ecfbe512dcba176ba6ddacc00 +523 -0
  220. data/test/dummy/tmp/cache/assets/E21/5D0/sprockets%2Fd4fe6d611bfb0e63c04de5dbfb128a8a +0 -0
  221. data/test/dummy/tmp/cache/assets/E22/770/sprockets%2F2ccebcabfbb9b59159e4b73b3b30a74d +208 -0
  222. data/test/dummy/tmp/cache/assets/E31/A90/sprockets%2F26b98bbbc15eee9b370c7dcde88d16aa +394 -0
  223. data/test/dummy/tmp/cache/assets/E31/E30/sprockets%2Ffbdafcc4e2f44bb9fc653bcf9625b750 +284 -0
  224. data/test/dummy/tmp/cache/assets/E37/1C0/sprockets%2Fa5eb45fc86dafe2bee4e4fa3f53c4785 +0 -0
  225. data/test/dummy/tmp/cache/assets/E4C/960/sprockets%2F13fd6e4bfdf031acb3dedca8d0f407b4 +1277 -0
  226. data/test/dummy/tmp/cache/assets/E88/330/sprockets%2Fe02d0adb5529ffafcea1b9e5ccf69a5b +669 -0
  227. data/test/dummy/tmp/pids/server.pid +1 -0
  228. data/test/dummy/tmp/simple.js.js +20 -0
  229. data/test/dummy/tmp/soy-1325274421.js +20 -0
  230. data/test/dummy/tmp/soy-1325286267.js +20 -0
  231. data/test/test_helper.rb +10 -0
  232. data/test/yellow-brick-road_test.rb +7 -0
  233. metadata +538 -0
@@ -0,0 +1,2539 @@
1
+ o: ActiveSupport::Cache::Entry :@compressedF:@expires_in0:@created_atf1325458073.030408: @value{ I" length:EFi�CI" digest;
2
+ F"%727ecd3542acdd7f6cf9417a0cd1b810I" source;
3
+ FI"�C// Copyright 2006 The Closure Library Authors. All Rights Reserved.
4
+ //
5
+ // Licensed under the Apache License, Version 2.0 (the "License");
6
+ // you may not use this file except in compliance with the License.
7
+ // You may obtain a copy of the License at
8
+ //
9
+ // http://www.apache.org/licenses/LICENSE-2.0
10
+ //
11
+ // Unless required by applicable law or agreed to in writing, software
12
+ // distributed under the License is distributed on an "AS-IS" BASIS,
13
+ // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ // See the License for the specific language governing permissions and
15
+ // limitations under the License.
16
+
17
+ /**
18
+ * @fileoverview Utilities for manipulating the browser's Document Object Model
19
+ * Inspiration taken *heavily* from mochikit (http://mochikit.com/).
20
+ *
21
+ * You can use {@link goog.dom.DomHelper} to create new dom helpers that refer
22
+ * to a different document object. This is useful if you are working with
23
+ * frames or multiple windows.
24
+ *
25
+ */
26
+
27
+
28
+ // TODO(arv): Rename/refactor getTextContent and getRawTextContent. The problem
29
+ // is that getTextContent should mimic the DOM3 textContent. We should add a
30
+ // getInnerText (or getText) which tries to return the visible text, innerText.
31
+
32
+
33
+ goog.provide('goog.dom');
34
+ goog.provide('goog.dom.DomHelper');
35
+ goog.provide('goog.dom.NodeType');
36
+
37
+ goog.require('goog.array');
38
+ goog.require('goog.dom.BrowserFeature');
39
+ goog.require('goog.dom.TagName');
40
+ goog.require('goog.dom.classes');
41
+ goog.require('goog.math.Coordinate');
42
+ goog.require('goog.math.Size');
43
+ goog.require('goog.object');
44
+ goog.require('goog.string');
45
+ goog.require('goog.userAgent');
46
+
47
+
48
+ /**
49
+ * @define {boolean} Whether we know at compile time that the browser is in
50
+ * quirks mode.
51
+ */
52
+ goog.dom.ASSUME_QUIRKS_MODE = false;
53
+
54
+
55
+ /**
56
+ * @define {boolean} Whether we know at compile time that the browser is in
57
+ * standards compliance mode.
58
+ */
59
+ goog.dom.ASSUME_STANDARDS_MODE = false;
60
+
61
+
62
+ /**
63
+ * Whether we know the compatibility mode at compile time.
64
+ * @type {boolean}
65
+ * @private
66
+ */
67
+ goog.dom.COMPAT_MODE_KNOWN_ =
68
+ goog.dom.ASSUME_QUIRKS_MODE || goog.dom.ASSUME_STANDARDS_MODE;
69
+
70
+
71
+ /**
72
+ * Enumeration for DOM node types (for reference)
73
+ * @enum {number}
74
+ */
75
+ goog.dom.NodeType = {
76
+ ELEMENT: 1,
77
+ ATTRIBUTE: 2,
78
+ TEXT: 3,
79
+ CDATA_SECTION: 4,
80
+ ENTITY_REFERENCE: 5,
81
+ ENTITY: 6,
82
+ PROCESSING_INSTRUCTION: 7,
83
+ COMMENT: 8,
84
+ DOCUMENT: 9,
85
+ DOCUMENT_TYPE: 10,
86
+ DOCUMENT_FRAGMENT: 11,
87
+ NOTATION: 12
88
+ };
89
+
90
+
91
+ /**
92
+ * Gets the DomHelper object for the document where the element resides.
93
+ * @param {Node|Window=} opt_element If present, gets the DomHelper for this
94
+ * element.
95
+ * @return {!goog.dom.DomHelper} The DomHelper.
96
+ */
97
+ goog.dom.getDomHelper = function(opt_element) {
98
+ return opt_element ?
99
+ new goog.dom.DomHelper(goog.dom.getOwnerDocument(opt_element)) :
100
+ (goog.dom.defaultDomHelper_ ||
101
+ (goog.dom.defaultDomHelper_ = new goog.dom.DomHelper()));
102
+ };
103
+
104
+
105
+ /**
106
+ * Cached default DOM helper.
107
+ * @type {goog.dom.DomHelper}
108
+ * @private
109
+ */
110
+ goog.dom.defaultDomHelper_;
111
+
112
+
113
+ /**
114
+ * Gets the document object being used by the dom library.
115
+ * @return {!Document} Document object.
116
+ */
117
+ goog.dom.getDocument = function() {
118
+ return document;
119
+ };
120
+
121
+
122
+ /**
123
+ * Alias for getElementById. If a DOM node is passed in then we just return
124
+ * that.
125
+ * @param {string|Element} element Element ID or a DOM node.
126
+ * @return {Element} The element with the given ID, or the node passed in.
127
+ */
128
+ goog.dom.getElement = function(element) {
129
+ return goog.isString(element) ?
130
+ document.getElementById(element) : element;
131
+ };
132
+
133
+
134
+ /**
135
+ * Alias for getElement.
136
+ * @param {string|Element} element Element ID or a DOM node.
137
+ * @return {Element} The element with the given ID, or the node passed in.
138
+ * @deprecated Use {@link goog.dom.getElement} instead.
139
+ */
140
+ goog.dom.$ = goog.dom.getElement;
141
+
142
+
143
+ /**
144
+ * Looks up elements by both tag and class name, using browser native functions
145
+ * ({@code querySelectorAll}, {@code getElementsByTagName} or
146
+ * {@code getElementsByClassName}) where possible. This function
147
+ * is a useful, if limited, way of collecting a list of DOM elements
148
+ * with certain characteristics. {@code goog.dom.query} offers a
149
+ * more powerful and general solution which allows matching on CSS3
150
+ * selector expressions, but at increased cost in code size. If all you
151
+ * need is particular tags belonging to a single class, this function
152
+ * is fast and sleek.
153
+ *
154
+ * @see {goog.dom.query}
155
+ *
156
+ * @param {?string=} opt_tag Element tag name.
157
+ * @param {?string=} opt_class Optional class name.
158
+ * @param {Document|Element=} opt_el Optional element to look in.
159
+ * @return { {length: number} } Array-like list of elements (only a length
160
+ * property and numerical indices are guaranteed to exist).
161
+ */
162
+ goog.dom.getElementsByTagNameAndClass = function(opt_tag, opt_class, opt_el) {
163
+ return goog.dom.getElementsByTagNameAndClass_(document, opt_tag, opt_class,
164
+ opt_el);
165
+ };
166
+
167
+
168
+ /**
169
+ * Returns an array of all the elements with the provided className.
170
+ * @see {goog.dom.query}
171
+ * @param {string} className the name of the class to look for.
172
+ * @param {Document|Element=} opt_el Optional element to look in.
173
+ * @return { {length: number} } The items found with the class name provided.
174
+ */
175
+ goog.dom.getElementsByClass = function(className, opt_el) {
176
+ var parent = opt_el || document;
177
+ if (goog.dom.canUseQuerySelector_(parent)) {
178
+ return parent.querySelectorAll('.' + className);
179
+ } else if (parent.getElementsByClassName) {
180
+ return parent.getElementsByClassName(className);
181
+ }
182
+ return goog.dom.getElementsByTagNameAndClass_(
183
+ document, '*', className, opt_el);
184
+ };
185
+
186
+
187
+ /**
188
+ * Returns the first element with the provided className.
189
+ * @see {goog.dom.query}
190
+ * @param {string} className the name of the class to look for.
191
+ * @param {Element|Document=} opt_el Optional element to look in.
192
+ * @return {Element} The first item with the class name provided.
193
+ */
194
+ goog.dom.getElementByClass = function(className, opt_el) {
195
+ var parent = opt_el || document;
196
+ var retVal = null;
197
+ if (goog.dom.canUseQuerySelector_(parent)) {
198
+ retVal = parent.querySelector('.' + className);
199
+ } else {
200
+ retVal = goog.dom.getElementsByClass(className, opt_el)[0];
201
+ }
202
+ return retVal || null;
203
+ };
204
+
205
+
206
+ /**
207
+ * Prefer the standardized (http://www.w3.org/TR/selectors-api/), native and
208
+ * fast W3C Selectors API. However, the version of WebKit that shipped with
209
+ * Safari 3.1 and Chrome has a bug where it will not correctly match mixed-
210
+ * case class name selectors in quirks mode.
211
+ * @param {!Element|Document} parent The parent document object.
212
+ * @return {boolean} whether or not we can use parent.querySelector* APIs.
213
+ * @private
214
+ */
215
+ goog.dom.canUseQuerySelector_ = function(parent) {
216
+ return parent.querySelectorAll &&
217
+ parent.querySelector &&
218
+ (!goog.userAgent.WEBKIT || goog.dom.isCss1CompatMode_(document) ||
219
+ goog.userAgent.isVersion('528'));
220
+ };
221
+
222
+
223
+ /**
224
+ * Helper for {@code getElementsByTagNameAndClass}.
225
+ * @param {!Document} doc The document to get the elements in.
226
+ * @param {?string=} opt_tag Element tag name.
227
+ * @param {?string=} opt_class Optional class name.
228
+ * @param {Document|Element=} opt_el Optional element to look in.
229
+ * @return { {length: number} } Array-like list of elements (only a length
230
+ * property and numerical indices are guaranteed to exist).
231
+ * @private
232
+ */
233
+ goog.dom.getElementsByTagNameAndClass_ = function(doc, opt_tag, opt_class,
234
+ opt_el) {
235
+ var parent = opt_el || doc;
236
+ var tagName = (opt_tag && opt_tag != '*') ? opt_tag.toUpperCase() : '';
237
+
238
+ if (goog.dom.canUseQuerySelector_(parent) &&
239
+ (tagName || opt_class)) {
240
+ var query = tagName + (opt_class ? '.' + opt_class : '');
241
+ return parent.querySelectorAll(query);
242
+ }
243
+
244
+ // Use the native getElementsByClassName if available, under the assumption
245
+ // that even when the tag name is specified, there will be fewer elements to
246
+ // filter through when going by class than by tag name
247
+ if (opt_class && parent.getElementsByClassName) {
248
+ var els = parent.getElementsByClassName(opt_class);
249
+
250
+ if (tagName) {
251
+ var arrayLike = {};
252
+ var len = 0;
253
+
254
+ // Filter for specific tags if requested.
255
+ for (var i = 0, el; el = els[i]; i++) {
256
+ if (tagName == el.nodeName) {
257
+ arrayLike[len++] = el;
258
+ }
259
+ }
260
+ arrayLike.length = len;
261
+
262
+ return arrayLike;
263
+ } else {
264
+ return els;
265
+ }
266
+ }
267
+
268
+ var els = parent.getElementsByTagName(tagName || '*');
269
+
270
+ if (opt_class) {
271
+ var arrayLike = {};
272
+ var len = 0;
273
+ for (var i = 0, el; el = els[i]; i++) {
274
+ var className = el.className;
275
+ // Check if className has a split function since SVG className does not.
276
+ if (typeof className.split == 'function' &&
277
+ goog.array.contains(className.split(/\s+/), opt_class)) {
278
+ arrayLike[len++] = el;
279
+ }
280
+ }
281
+ arrayLike.length = len;
282
+ return arrayLike;
283
+ } else {
284
+ return els;
285
+ }
286
+ };
287
+
288
+
289
+ /**
290
+ * Alias for {@code getElementsByTagNameAndClass}.
291
+ * @param {?string=} opt_tag Element tag name.
292
+ * @param {?string=} opt_class Optional class name.
293
+ * @param {Element=} opt_el Optional element to look in.
294
+ * @return { {length: number} } Array-like list of elements (only a length
295
+ * property and numerical indices are guaranteed to exist).
296
+ * @deprecated Use {@link goog.dom.getElementsByTagNameAndClass} instead.
297
+ */
298
+ goog.dom.$$ = goog.dom.getElementsByTagNameAndClass;
299
+
300
+
301
+ /**
302
+ * Sets multiple properties on a node.
303
+ * @param {Element} element DOM node to set properties on.
304
+ * @param {Object} properties Hash of property:value pairs.
305
+ */
306
+ goog.dom.setProperties = function(element, properties) {
307
+ goog.object.forEach(properties, function(val, key) {
308
+ if (key == 'style') {
309
+ element.style.cssText = val;
310
+ } else if (key == 'class') {
311
+ element.className = val;
312
+ } else if (key == 'for') {
313
+ element.htmlFor = val;
314
+ } else if (key in goog.dom.DIRECT_ATTRIBUTE_MAP_) {
315
+ element.setAttribute(goog.dom.DIRECT_ATTRIBUTE_MAP_[key], val);
316
+ } else if (goog.string.startsWith(key, 'aria-')) {
317
+ element.setAttribute(key, val);
318
+ } else {
319
+ element[key] = val;
320
+ }
321
+ });
322
+ };
323
+
324
+
325
+ /**
326
+ * Map of attributes that should be set using
327
+ * element.setAttribute(key, val) instead of element[key] = val. Used
328
+ * by goog.dom.setProperties.
329
+ *
330
+ * @type {Object}
331
+ * @private
332
+ */
333
+ goog.dom.DIRECT_ATTRIBUTE_MAP_ = {
334
+ 'cellpadding': 'cellPadding',
335
+ 'cellspacing': 'cellSpacing',
336
+ 'colspan': 'colSpan',
337
+ 'rowspan': 'rowSpan',
338
+ 'valign': 'vAlign',
339
+ 'height': 'height',
340
+ 'width': 'width',
341
+ 'usemap': 'useMap',
342
+ 'frameborder': 'frameBorder',
343
+ 'maxlength': 'maxLength',
344
+ 'type': 'type'
345
+ };
346
+
347
+
348
+ /**
349
+ * Gets the dimensions of the viewport.
350
+ *
351
+ * Gecko Standards mode:
352
+ * docEl.clientWidth Width of viewport excluding scrollbar.
353
+ * win.innerWidth Width of viewport including scrollbar.
354
+ * body.clientWidth Width of body element.
355
+ *
356
+ * docEl.clientHeight Height of viewport excluding scrollbar.
357
+ * win.innerHeight Height of viewport including scrollbar.
358
+ * body.clientHeight Height of document.
359
+ *
360
+ * Gecko Backwards compatible mode:
361
+ * docEl.clientWidth Width of viewport excluding scrollbar.
362
+ * win.innerWidth Width of viewport including scrollbar.
363
+ * body.clientWidth Width of viewport excluding scrollbar.
364
+ *
365
+ * docEl.clientHeight Height of document.
366
+ * win.innerHeight Height of viewport including scrollbar.
367
+ * body.clientHeight Height of viewport excluding scrollbar.
368
+ *
369
+ * IE6/7 Standards mode:
370
+ * docEl.clientWidth Width of viewport excluding scrollbar.
371
+ * win.innerWidth Undefined.
372
+ * body.clientWidth Width of body element.
373
+ *
374
+ * docEl.clientHeight Height of viewport excluding scrollbar.
375
+ * win.innerHeight Undefined.
376
+ * body.clientHeight Height of document element.
377
+ *
378
+ * IE5 + IE6/7 Backwards compatible mode:
379
+ * docEl.clientWidth 0.
380
+ * win.innerWidth Undefined.
381
+ * body.clientWidth Width of viewport excluding scrollbar.
382
+ *
383
+ * docEl.clientHeight 0.
384
+ * win.innerHeight Undefined.
385
+ * body.clientHeight Height of viewport excluding scrollbar.
386
+ *
387
+ * Opera 9 Standards and backwards compatible mode:
388
+ * docEl.clientWidth Width of viewport excluding scrollbar.
389
+ * win.innerWidth Width of viewport including scrollbar.
390
+ * body.clientWidth Width of viewport excluding scrollbar.
391
+ *
392
+ * docEl.clientHeight Height of document.
393
+ * win.innerHeight Height of viewport including scrollbar.
394
+ * body.clientHeight Height of viewport excluding scrollbar.
395
+ *
396
+ * WebKit:
397
+ * Safari 2
398
+ * docEl.clientHeight Same as scrollHeight.
399
+ * docEl.clientWidth Same as innerWidth.
400
+ * win.innerWidth Width of viewport excluding scrollbar.
401
+ * win.innerHeight Height of the viewport including scrollbar.
402
+ * frame.innerHeight Height of the viewport exluding scrollbar.
403
+ *
404
+ * Safari 3 (tested in 522)
405
+ *
406
+ * docEl.clientWidth Width of viewport excluding scrollbar.
407
+ * docEl.clientHeight Height of viewport excluding scrollbar in strict mode.
408
+ * body.clientHeight Height of viewport excluding scrollbar in quirks mode.
409
+ *
410
+ * @param {Window=} opt_window Optional window element to test.
411
+ * @return {!goog.math.Size} Object with values 'width' and 'height'.
412
+ */
413
+ goog.dom.getViewportSize = function(opt_window) {
414
+ // TODO(arv): This should not take an argument
415
+ return goog.dom.getViewportSize_(opt_window || window);
416
+ };
417
+
418
+
419
+ /**
420
+ * Helper for {@code getViewportSize}.
421
+ * @param {Window} win The window to get the view port size for.
422
+ * @return {!goog.math.Size} Object with values 'width' and 'height'.
423
+ * @private
424
+ */
425
+ goog.dom.getViewportSize_ = function(win) {
426
+ var doc = win.document;
427
+
428
+ if (goog.userAgent.WEBKIT && !goog.userAgent.isVersion('500') &&
429
+ !goog.userAgent.MOBILE) {
430
+ // TODO(user): Sometimes we get something that isn't a valid window
431
+ // object. In this case we just revert to the current window. We need to
432
+ // figure out when this happens and find a real fix for it.
433
+ // See the comments on goog.dom.getWindow.
434
+ if (typeof win.innerHeight == 'undefined') {
435
+ win = window;
436
+ }
437
+ var innerHeight = win.innerHeight;
438
+ var scrollHeight = win.document.documentElement.scrollHeight;
439
+
440
+ if (win == win.top) {
441
+ if (scrollHeight < innerHeight) {
442
+ innerHeight -= 15; // Scrollbars are 15px wide on Mac
443
+ }
444
+ }
445
+ return new goog.math.Size(win.innerWidth, innerHeight);
446
+ }
447
+
448
+ var el = goog.dom.isCss1CompatMode_(doc) ? doc.documentElement : doc.body;
449
+
450
+ return new goog.math.Size(el.clientWidth, el.clientHeight);
451
+ };
452
+
453
+
454
+ /**
455
+ * Calculates the height of the document.
456
+ *
457
+ * @return {number} The height of the current document.
458
+ */
459
+ goog.dom.getDocumentHeight = function() {
460
+ return goog.dom.getDocumentHeight_(window);
461
+ };
462
+
463
+
464
+ /**
465
+ * Calculates the height of the document of the given window.
466
+ *
467
+ * Function code copied from the opensocial gadget api:
468
+ * gadgets.window.adjustHeight(opt_height)
469
+ *
470
+ * @private
471
+ * @param {Window} win The window whose document height to retrieve.
472
+ * @return {number} The height of the document of the given window.
473
+ */
474
+ goog.dom.getDocumentHeight_ = function(win) {
475
+ // NOTE(eae): This method will return the window size rather than the document
476
+ // size in webkit quirks mode.
477
+ var doc = win.document;
478
+ var height = 0;
479
+
480
+ if (doc) {
481
+ // Calculating inner content height is hard and different between
482
+ // browsers rendering in Strict vs. Quirks mode. We use a combination of
483
+ // three properties within document.body and document.documentElement:
484
+ // - scrollHeight
485
+ // - offsetHeight
486
+ // - clientHeight
487
+ // These values differ significantly between browsers and rendering modes.
488
+ // But there are patterns. It just takes a lot of time and persistence
489
+ // to figure out.
490
+
491
+ // Get the height of the viewport
492
+ var vh = goog.dom.getViewportSize_(win).height;
493
+ var body = doc.body;
494
+ var docEl = doc.documentElement;
495
+ if (goog.dom.isCss1CompatMode_(doc) && docEl.scrollHeight) {
496
+ // In Strict mode:
497
+ // The inner content height is contained in either:
498
+ // document.documentElement.scrollHeight
499
+ // document.documentElement.offsetHeight
500
+ // Based on studying the values output by different browsers,
501
+ // use the value that's NOT equal to the viewport height found above.
502
+ height = docEl.scrollHeight != vh ?
503
+ docEl.scrollHeight : docEl.offsetHeight;
504
+ } else {
505
+ // In Quirks mode:
506
+ // documentElement.clientHeight is equal to documentElement.offsetHeight
507
+ // except in IE. In most browsers, document.documentElement can be used
508
+ // to calculate the inner content height.
509
+ // However, in other browsers (e.g. IE), document.body must be used
510
+ // instead. How do we know which one to use?
511
+ // If document.documentElement.clientHeight does NOT equal
512
+ // document.documentElement.offsetHeight, then use document.body.
513
+ var sh = docEl.scrollHeight;
514
+ var oh = docEl.offsetHeight;
515
+ if (docEl.clientHeight != oh) {
516
+ sh = body.scrollHeight;
517
+ oh = body.offsetHeight;
518
+ }
519
+
520
+ // Detect whether the inner content height is bigger or smaller
521
+ // than the bounding box (viewport). If bigger, take the larger
522
+ // value. If smaller, take the smaller value.
523
+ if (sh > vh) {
524
+ // Content is larger
525
+ height = sh > oh ? sh : oh;
526
+ } else {
527
+ // Content is smaller
528
+ height = sh < oh ? sh : oh;
529
+ }
530
+ }
531
+ }
532
+
533
+ return height;
534
+ };
535
+
536
+
537
+ /**
538
+ * Gets the page scroll distance as a coordinate object.
539
+ *
540
+ * @param {Window=} opt_window Optional window element to test.
541
+ * @return {!goog.math.Coordinate} Object with values 'x' and 'y'.
542
+ * @deprecated Use {@link goog.dom.getDocumentScroll} instead.
543
+ */
544
+ goog.dom.getPageScroll = function(opt_window) {
545
+ var win = opt_window || goog.global || window;
546
+ return goog.dom.getDomHelper(win.document).getDocumentScroll();
547
+ };
548
+
549
+
550
+ /**
551
+ * Gets the document scroll distance as a coordinate object.
552
+ *
553
+ * @return {!goog.math.Coordinate} Object with values 'x' and 'y'.
554
+ */
555
+ goog.dom.getDocumentScroll = function() {
556
+ return goog.dom.getDocumentScroll_(document);
557
+ };
558
+
559
+
560
+ /**
561
+ * Helper for {@code getDocumentScroll}.
562
+ *
563
+ * @param {!Document} doc The document to get the scroll for.
564
+ * @return {!goog.math.Coordinate} Object with values 'x' and 'y'.
565
+ * @private
566
+ */
567
+ goog.dom.getDocumentScroll_ = function(doc) {
568
+ var el = goog.dom.getDocumentScrollElement_(doc);
569
+ var win = goog.dom.getWindow_(doc);
570
+ return new goog.math.Coordinate(win.pageXOffset || el.scrollLeft,
571
+ win.pageYOffset || el.scrollTop);
572
+ };
573
+
574
+
575
+ /**
576
+ * Gets the document scroll element.
577
+ * @return {Element} Scrolling element.
578
+ */
579
+ goog.dom.getDocumentScrollElement = function() {
580
+ return goog.dom.getDocumentScrollElement_(document);
581
+ };
582
+
583
+
584
+ /**
585
+ * Helper for {@code getDocumentScrollElement}.
586
+ * @param {!Document} doc The document to get the scroll element for.
587
+ * @return {Element} Scrolling element.
588
+ * @private
589
+ */
590
+ goog.dom.getDocumentScrollElement_ = function(doc) {
591
+ // Safari (2 and 3) needs body.scrollLeft in both quirks mode and strict mode.
592
+ return !goog.userAgent.WEBKIT && goog.dom.isCss1CompatMode_(doc) ?
593
+ doc.documentElement : doc.body;
594
+ };
595
+
596
+
597
+ /**
598
+ * Gets the window object associated with the given document.
599
+ *
600
+ * @param {Document=} opt_doc Document object to get window for.
601
+ * @return {!Window} The window associated with the given document.
602
+ */
603
+ goog.dom.getWindow = function(opt_doc) {
604
+ // TODO(arv): This should not take an argument.
605
+ return opt_doc ? goog.dom.getWindow_(opt_doc) : window;
606
+ };
607
+
608
+
609
+ /**
610
+ * Helper for {@code getWindow}.
611
+ *
612
+ * @param {!Document} doc Document object to get window for.
613
+ * @return {!Window} The window associated with the given document.
614
+ * @private
615
+ */
616
+ goog.dom.getWindow_ = function(doc) {
617
+ return doc.parentWindow || doc.defaultView;
618
+ };
619
+
620
+
621
+ /**
622
+ * Returns a dom node with a set of attributes. This function accepts varargs
623
+ * for subsequent nodes to be added. Subsequent nodes will be added to the
624
+ * first node as childNodes.
625
+ *
626
+ * So:
627
+ * <code>createDom('div', null, createDom('p'), createDom('p'));</code>
628
+ * would return a div with two child paragraphs
629
+ *
630
+ * @param {string} tagName Tag to create.
631
+ * @param {Object|Array.<string>|string=} opt_attributes If object, then a map
632
+ * of name-value pairs for attributes. If a string, then this is the
633
+ * className of the new element. If an array, the elements will be joined
634
+ * together as the className of the new element.
635
+ * @param {...Object|string|Array|NodeList} var_args Further DOM nodes or
636
+ * strings for text nodes. If one of the var_args is an array or NodeList,i
637
+ * its elements will be added as childNodes instead.
638
+ * @return {!Element} Reference to a DOM node.
639
+ */
640
+ goog.dom.createDom = function(tagName, opt_attributes, var_args) {
641
+ return goog.dom.createDom_(document, arguments);
642
+ };
643
+
644
+
645
+ /**
646
+ * Helper for {@code createDom}.
647
+ * @param {!Document} doc The document to create the DOM in.
648
+ * @param {!Arguments} args Argument object passed from the callers. See
649
+ * {@code goog.dom.createDom} for details.
650
+ * @return {!Element} Reference to a DOM node.
651
+ * @private
652
+ */
653
+ goog.dom.createDom_ = function(doc, args) {
654
+ var tagName = args[0];
655
+ var attributes = args[1];
656
+
657
+ // Internet Explorer is dumb: http://msdn.microsoft.com/workshop/author/
658
+ // dhtml/reference/properties/name_2.asp
659
+ // Also does not allow setting of 'type' attribute on 'input' or 'button'.
660
+ if (!goog.dom.BrowserFeature.CAN_ADD_NAME_OR_TYPE_ATTRIBUTES && attributes &&
661
+ (attributes.name || attributes.type)) {
662
+ var tagNameArr = ['<', tagName];
663
+ if (attributes.name) {
664
+ tagNameArr.push(' name="', goog.string.htmlEscape(attributes.name),
665
+ '"');
666
+ }
667
+ if (attributes.type) {
668
+ tagNameArr.push(' type="', goog.string.htmlEscape(attributes.type),
669
+ '"');
670
+
671
+ // Clone attributes map to remove 'type' without mutating the input.
672
+ var clone = {};
673
+ goog.object.extend(clone, attributes);
674
+ attributes = clone;
675
+ delete attributes.type;
676
+ }
677
+ tagNameArr.push('>');
678
+ tagName = tagNameArr.join('');
679
+ }
680
+
681
+ var element = doc.createElement(tagName);
682
+
683
+ if (attributes) {
684
+ if (goog.isString(attributes)) {
685
+ element.className = attributes;
686
+ } else if (goog.isArray(attributes)) {
687
+ goog.dom.classes.add.apply(null, [element].concat(attributes));
688
+ } else {
689
+ goog.dom.setProperties(element, attributes);
690
+ }
691
+ }
692
+
693
+ if (args.length > 2) {
694
+ goog.dom.append_(doc, element, args, 2);
695
+ }
696
+
697
+ return element;
698
+ };
699
+
700
+
701
+ /**
702
+ * Appends a node with text or other nodes.
703
+ * @param {!Document} doc The document to create new nodes in.
704
+ * @param {!Node} parent The node to append nodes to.
705
+ * @param {!Arguments} args The values to add. See {@code goog.dom.append}.
706
+ * @param {number} startIndex The index of the array to start from.
707
+ * @private
708
+ */
709
+ goog.dom.append_ = function(doc, parent, args, startIndex) {
710
+ function childHandler(child) {
711
+ // TODO(user): More coercion, ala MochiKit?
712
+ if (child) {
713
+ parent.appendChild(goog.isString(child) ?
714
+ doc.createTextNode(child) : child);
715
+ }
716
+ }
717
+
718
+ for (var i = startIndex; i < args.length; i++) {
719
+ var arg = args[i];
720
+ // TODO(attila): Fix isArrayLike to return false for a text node.
721
+ if (goog.isArrayLike(arg) && !goog.dom.isNodeLike(arg)) {
722
+ // If the argument is a node list, not a real array, use a clone,
723
+ // because forEach can't be used to mutate a NodeList.
724
+ goog.array.forEach(goog.dom.isNodeList(arg) ?
725
+ goog.array.clone(arg) : arg,
726
+ childHandler);
727
+ } else {
728
+ childHandler(arg);
729
+ }
730
+ }
731
+ };
732
+
733
+
734
+ /**
735
+ * Alias for {@code createDom}.
736
+ * @param {string} tagName Tag to create.
737
+ * @param {string|Object=} opt_attributes If object, then a map of name-value
738
+ * pairs for attributes. If a string, then this is the className of the new
739
+ * element.
740
+ * @param {...Object|string|Array|NodeList} var_args Further DOM nodes or
741
+ * strings for text nodes. If one of the var_args is an array, its
742
+ * children will be added as childNodes instead.
743
+ * @return {!Element} Reference to a DOM node.
744
+ * @deprecated Use {@link goog.dom.createDom} instead.
745
+ */
746
+ goog.dom.$dom = goog.dom.createDom;
747
+
748
+
749
+ /**
750
+ * Creates a new element.
751
+ * @param {string} name Tag name.
752
+ * @return {!Element} The new element.
753
+ */
754
+ goog.dom.createElement = function(name) {
755
+ return document.createElement(name);
756
+ };
757
+
758
+
759
+ /**
760
+ * Creates a new text node.
761
+ * @param {string} content Content.
762
+ * @return {!Text} The new text node.
763
+ */
764
+ goog.dom.createTextNode = function(content) {
765
+ return document.createTextNode(content);
766
+ };
767
+
768
+
769
+ /**
770
+ * Create a table.
771
+ * @param {number} rows The number of rows in the table. Must be >= 1.
772
+ * @param {number} columns The number of columns in the table. Must be >= 1.
773
+ * @param {boolean=} opt_fillWithNbsp If true, fills table entries with nsbps.
774
+ * @return {!Element} The created table.
775
+ */
776
+ goog.dom.createTable = function(rows, columns, opt_fillWithNbsp) {
777
+ return goog.dom.createTable_(document, rows, columns, !!opt_fillWithNbsp);
778
+ };
779
+
780
+
781
+ /**
782
+ * Create a table.
783
+ * @param {!Document} doc Document object to use to create the table.
784
+ * @param {number} rows The number of rows in the table. Must be >= 1.
785
+ * @param {number} columns The number of columns in the table. Must be >= 1.
786
+ * @param {boolean} fillWithNbsp If true, fills table entries with nsbps.
787
+ * @return {!Element} The created table.
788
+ * @private
789
+ */
790
+ goog.dom.createTable_ = function(doc, rows, columns, fillWithNbsp) {
791
+ var rowHtml = ['<tr>'];
792
+ for (var i = 0; i < columns; i++) {
793
+ rowHtml.push(fillWithNbsp ? '<td>&nbsp;</td>' : '<td></td>');
794
+ }
795
+ rowHtml.push('</tr>');
796
+ rowHtml = rowHtml.join('');
797
+ var totalHtml = ['<table>'];
798
+ for (i = 0; i < rows; i++) {
799
+ totalHtml.push(rowHtml);
800
+ }
801
+ totalHtml.push('</table>');
802
+
803
+ var elem = doc.createElement(goog.dom.TagName.DIV);
804
+ elem.innerHTML = totalHtml.join('');
805
+ return /** @type {!Element} */ (elem.removeChild(elem.firstChild));
806
+ };
807
+
808
+
809
+ /**
810
+ * Converts an HTML string into a document fragment.
811
+ *
812
+ * @param {string} htmlString The HTML string to convert.
813
+ * @return {!Node} The resulting document fragment.
814
+ */
815
+ goog.dom.htmlToDocumentFragment = function(htmlString) {
816
+ return goog.dom.htmlToDocumentFragment_(document, htmlString);
817
+ };
818
+
819
+
820
+ /**
821
+ * Helper for {@code htmlToDocumentFragment}.
822
+ *
823
+ * @param {!Document} doc The document.
824
+ * @param {string} htmlString The HTML string to convert.
825
+ * @return {!Node} The resulting document fragment.
826
+ * @private
827
+ */
828
+ goog.dom.htmlToDocumentFragment_ = function(doc, htmlString) {
829
+ var tempDiv = doc.createElement('div');
830
+ if (goog.dom.BrowserFeature.INNER_HTML_NEEDS_SCOPED_ELEMENT) {
831
+ tempDiv.innerHTML = '<br>' + htmlString;
832
+ tempDiv.removeChild(tempDiv.firstChild);
833
+ } else {
834
+ tempDiv.innerHTML = htmlString;
835
+ }
836
+ if (tempDiv.childNodes.length == 1) {
837
+ return /** @type {!Node} */ (tempDiv.removeChild(tempDiv.firstChild));
838
+ } else {
839
+ var fragment = doc.createDocumentFragment();
840
+ while (tempDiv.firstChild) {
841
+ fragment.appendChild(tempDiv.firstChild);
842
+ }
843
+ return fragment;
844
+ }
845
+ };
846
+
847
+
848
+ /**
849
+ * Returns the compatMode of the document.
850
+ * @return {string} The result is either CSS1Compat or BackCompat.
851
+ * @deprecated use goog.dom.isCss1CompatMode instead.
852
+ */
853
+ goog.dom.getCompatMode = function() {
854
+ return goog.dom.isCss1CompatMode() ? 'CSS1Compat' : 'BackCompat';
855
+ };
856
+
857
+
858
+ /**
859
+ * Returns true if the browser is in "CSS1-compatible" (standards-compliant)
860
+ * mode, false otherwise.
861
+ * @return {boolean} True if in CSS1-compatible mode.
862
+ */
863
+ goog.dom.isCss1CompatMode = function() {
864
+ return goog.dom.isCss1CompatMode_(document);
865
+ };
866
+
867
+
868
+ /**
869
+ * Returns true if the browser is in "CSS1-compatible" (standards-compliant)
870
+ * mode, false otherwise.
871
+ * @param {Document} doc The document to check.
872
+ * @return {boolean} True if in CSS1-compatible mode.
873
+ * @private
874
+ */
875
+ goog.dom.isCss1CompatMode_ = function(doc) {
876
+ if (goog.dom.COMPAT_MODE_KNOWN_) {
877
+ return goog.dom.ASSUME_STANDARDS_MODE;
878
+ }
879
+
880
+ return doc.compatMode == 'CSS1Compat';
881
+ };
882
+
883
+
884
+ /**
885
+ * Determines if the given node can contain children, intended to be used for
886
+ * HTML generation.
887
+ *
888
+ * IE natively supports node.canHaveChildren but has inconsistent behavior.
889
+ * Prior to IE8 the base tag allows children and in IE9 all nodes return true
890
+ * for canHaveChildren.
891
+ *
892
+ * In practice all non-IE browsers allow you to add children to any node, but
893
+ * the behavior is inconsistent:
894
+ *
895
+ * <pre>
896
+ * var a = document.createElement('br');
897
+ * a.appendChild(document.createTextNode('foo'));
898
+ * a.appendChild(document.createTextNode('bar'));
899
+ * console.log(a.childNodes.length); // 2
900
+ * console.log(a.innerHTML); // Chrome: "", IE9: "foobar", FF3.5: "foobar"
901
+ * </pre>
902
+ *
903
+ * TODO(user): Rename shouldAllowChildren() ?
904
+ *
905
+ * @param {Node} node The node to check.
906
+ * @return {boolean} Whether the node can contain children.
907
+ */
908
+ goog.dom.canHaveChildren = function(node) {
909
+ if (node.nodeType != goog.dom.NodeType.ELEMENT) {
910
+ return false;
911
+ }
912
+ switch (node.tagName) {
913
+ case goog.dom.TagName.APPLET:
914
+ case goog.dom.TagName.AREA:
915
+ case goog.dom.TagName.BASE:
916
+ case goog.dom.TagName.BR:
917
+ case goog.dom.TagName.COL:
918
+ case goog.dom.TagName.FRAME:
919
+ case goog.dom.TagName.HR:
920
+ case goog.dom.TagName.IMG:
921
+ case goog.dom.TagName.INPUT:
922
+ case goog.dom.TagName.IFRAME:
923
+ case goog.dom.TagName.ISINDEX:
924
+ case goog.dom.TagName.LINK:
925
+ case goog.dom.TagName.NOFRAMES:
926
+ case goog.dom.TagName.NOSCRIPT:
927
+ case goog.dom.TagName.META:
928
+ case goog.dom.TagName.OBJECT:
929
+ case goog.dom.TagName.PARAM:
930
+ case goog.dom.TagName.SCRIPT:
931
+ case goog.dom.TagName.STYLE:
932
+ return false;
933
+ }
934
+ return true;
935
+ };
936
+
937
+
938
+ /**
939
+ * Appends a child to a node.
940
+ * @param {Node} parent Parent.
941
+ * @param {Node} child Child.
942
+ */
943
+ goog.dom.appendChild = function(parent, child) {
944
+ parent.appendChild(child);
945
+ };
946
+
947
+
948
+ /**
949
+ * Appends a node with text or other nodes.
950
+ * @param {!Node} parent The node to append nodes to.
951
+ * @param {...goog.dom.Appendable} var_args The things to append to the node.
952
+ * If this is a Node it is appended as is.
953
+ * If this is a string then a text node is appended.
954
+ * If this is an array like object then fields 0 to length - 1 are appended.
955
+ */
956
+ goog.dom.append = function(parent, var_args) {
957
+ goog.dom.append_(goog.dom.getOwnerDocument(parent), parent, arguments, 1);
958
+ };
959
+
960
+
961
+ /**
962
+ * Removes all the child nodes on a DOM node.
963
+ * @param {Node} node Node to remove children from.
964
+ */
965
+ goog.dom.removeChildren = function(node) {
966
+ // Note: Iterations over live collections can be slow, this is the fastest
967
+ // we could find. The double parenthesis are used to prevent JsCompiler and
968
+ // strict warnings.
969
+ var child;
970
+ while ((child = node.firstChild)) {
971
+ node.removeChild(child);
972
+ }
973
+ };
974
+
975
+
976
+ /**
977
+ * Inserts a new node before an existing reference node (i.e. as the previous
978
+ * sibling). If the reference node has no parent, then does nothing.
979
+ * @param {Node} newNode Node to insert.
980
+ * @param {Node} refNode Reference node to insert before.
981
+ */
982
+ goog.dom.insertSiblingBefore = function(newNode, refNode) {
983
+ if (refNode.parentNode) {
984
+ refNode.parentNode.insertBefore(newNode, refNode);
985
+ }
986
+ };
987
+
988
+
989
+ /**
990
+ * Inserts a new node after an existing reference node (i.e. as the next
991
+ * sibling). If the reference node has no parent, then does nothing.
992
+ * @param {Node} newNode Node to insert.
993
+ * @param {Node} refNode Reference node to insert after.
994
+ */
995
+ goog.dom.insertSiblingAfter = function(newNode, refNode) {
996
+ if (refNode.parentNode) {
997
+ refNode.parentNode.insertBefore(newNode, refNode.nextSibling);
998
+ }
999
+ };
1000
+
1001
+
1002
+ /**
1003
+ * Insert a child at a given index. If index is larger than the number of child
1004
+ * nodes that the parent currently has, the node is inserted as the last child
1005
+ * node.
1006
+ * @param {Element} parent The element into which to insert the child.
1007
+ * @param {Node} child The element to insert.
1008
+ * @param {number} index The index at which to insert the new child node. Must
1009
+ * not be negative.
1010
+ */
1011
+ goog.dom.insertChildAt = function(parent, child, index) {
1012
+ // Note that if the second argument is null, insertBefore
1013
+ // will append the child at the end of the list of children.
1014
+ parent.insertBefore(child, parent.childNodes[index] || null);
1015
+ };
1016
+
1017
+
1018
+ /**
1019
+ * Removes a node from its parent.
1020
+ * @param {Node} node The node to remove.
1021
+ * @return {Node} The node removed if removed; else, null.
1022
+ */
1023
+ goog.dom.removeNode = function(node) {
1024
+ return node && node.parentNode ? node.parentNode.removeChild(node) : null;
1025
+ };
1026
+
1027
+
1028
+ /**
1029
+ * Replaces a node in the DOM tree. Will do nothing if {@code oldNode} has no
1030
+ * parent.
1031
+ * @param {Node} newNode Node to insert.
1032
+ * @param {Node} oldNode Node to replace.
1033
+ */
1034
+ goog.dom.replaceNode = function(newNode, oldNode) {
1035
+ var parent = oldNode.parentNode;
1036
+ if (parent) {
1037
+ parent.replaceChild(newNode, oldNode);
1038
+ }
1039
+ };
1040
+
1041
+
1042
+ /**
1043
+ * Flattens an element. That is, removes it and replace it with its children.
1044
+ * Does nothing if the element is not in the document.
1045
+ * @param {Element} element The element to flatten.
1046
+ * @return {Element|undefined} The original element, detached from the document
1047
+ * tree, sans children; or undefined, if the element was not in the
1048
+ * document to begin with.
1049
+ */
1050
+ goog.dom.flattenElement = function(element) {
1051
+ var child, parent = element.parentNode;
1052
+ if (parent && parent.nodeType != goog.dom.NodeType.DOCUMENT_FRAGMENT) {
1053
+ // Use IE DOM method (supported by Opera too) if available
1054
+ if (element.removeNode) {
1055
+ return /** @type {Element} */ (element.removeNode(false));
1056
+ } else {
1057
+ // Move all children of the original node up one level.
1058
+ while ((child = element.firstChild)) {
1059
+ parent.insertBefore(child, element);
1060
+ }
1061
+
1062
+ // Detach the original element.
1063
+ return /** @type {Element} */ (goog.dom.removeNode(element));
1064
+ }
1065
+ }
1066
+ };
1067
+
1068
+
1069
+ /**
1070
+ * Returns an array containing just the element children of the given element.
1071
+ * @param {Element} element The element whose element children we want.
1072
+ * @return {Array|NodeList} An array or array-like list of just the element
1073
+ * children of the given element.
1074
+ */
1075
+ goog.dom.getChildren = function(element) {
1076
+ // We check if the children attribute is supported for child elements
1077
+ // since IE8 misuses the attribute by also including comments.
1078
+ if (goog.dom.BrowserFeature.CAN_USE_CHILDREN_ATTRIBUTE &&
1079
+ element.children != undefined) {
1080
+ return element.children;
1081
+ }
1082
+ // Fall back to manually filtering the element's child nodes.
1083
+ return goog.array.filter(element.childNodes, function(node) {
1084
+ return node.nodeType == goog.dom.NodeType.ELEMENT;
1085
+ });
1086
+ };
1087
+
1088
+
1089
+ /**
1090
+ * Returns the first child node that is an element.
1091
+ * @param {Node} node The node to get the first child element of.
1092
+ * @return {Element} The first child node of {@code node} that is an element.
1093
+ */
1094
+ goog.dom.getFirstElementChild = function(node) {
1095
+ if (node.firstElementChild != undefined) {
1096
+ return /** @type {Element} */(node).firstElementChild;
1097
+ }
1098
+ return goog.dom.getNextElementNode_(node.firstChild, true);
1099
+ };
1100
+
1101
+
1102
+ /**
1103
+ * Returns the last child node that is an element.
1104
+ * @param {Node} node The node to get the last child element of.
1105
+ * @return {Element} The last child node of {@code node} that is an element.
1106
+ */
1107
+ goog.dom.getLastElementChild = function(node) {
1108
+ if (node.lastElementChild != undefined) {
1109
+ return /** @type {Element} */(node).lastElementChild;
1110
+ }
1111
+ return goog.dom.getNextElementNode_(node.lastChild, false);
1112
+ };
1113
+
1114
+
1115
+ /**
1116
+ * Returns the first next sibling that is an element.
1117
+ * @param {Node} node The node to get the next sibling element of.
1118
+ * @return {Element} The next sibling of {@code node} that is an element.
1119
+ */
1120
+ goog.dom.getNextElementSibling = function(node) {
1121
+ if (node.nextElementSibling != undefined) {
1122
+ return /** @type {Element} */(node).nextElementSibling;
1123
+ }
1124
+ return goog.dom.getNextElementNode_(node.nextSibling, true);
1125
+ };
1126
+
1127
+
1128
+ /**
1129
+ * Returns the first previous sibling that is an element.
1130
+ * @param {Node} node The node to get the previous sibling element of.
1131
+ * @return {Element} The first previous sibling of {@code node} that is
1132
+ * an element.
1133
+ */
1134
+ goog.dom.getPreviousElementSibling = function(node) {
1135
+ if (node.previousElementSibling != undefined) {
1136
+ return /** @type {Element} */(node).previousElementSibling;
1137
+ }
1138
+ return goog.dom.getNextElementNode_(node.previousSibling, false);
1139
+ };
1140
+
1141
+
1142
+ /**
1143
+ * Returns the first node that is an element in the specified direction,
1144
+ * starting with {@code node}.
1145
+ * @param {Node} node The node to get the next element from.
1146
+ * @param {boolean} forward Whether to look forwards or backwards.
1147
+ * @return {Element} The first element.
1148
+ * @private
1149
+ */
1150
+ goog.dom.getNextElementNode_ = function(node, forward) {
1151
+ while (node && node.nodeType != goog.dom.NodeType.ELEMENT) {
1152
+ node = forward ? node.nextSibling : node.previousSibling;
1153
+ }
1154
+
1155
+ return /** @type {Element} */ (node);
1156
+ };
1157
+
1158
+
1159
+ /**
1160
+ * Returns the next node in source order from the given node.
1161
+ * @param {Node} node The node.
1162
+ * @return {Node} The next node in the DOM tree, or null if this was the last
1163
+ * node.
1164
+ */
1165
+ goog.dom.getNextNode = function(node) {
1166
+ if (!node) {
1167
+ return null;
1168
+ }
1169
+
1170
+ if (node.firstChild) {
1171
+ return node.firstChild;
1172
+ }
1173
+
1174
+ while (node && !node.nextSibling) {
1175
+ node = node.parentNode;
1176
+ }
1177
+
1178
+ return node ? node.nextSibling : null;
1179
+ };
1180
+
1181
+
1182
+ /**
1183
+ * Returns the previous node in source order from the given node.
1184
+ * @param {Node} node The node.
1185
+ * @return {Node} The previous node in the DOM tree, or null if this was the
1186
+ * first node.
1187
+ */
1188
+ goog.dom.getPreviousNode = function(node) {
1189
+ if (!node) {
1190
+ return null;
1191
+ }
1192
+
1193
+ if (!node.previousSibling) {
1194
+ return node.parentNode;
1195
+ }
1196
+
1197
+ node = node.previousSibling;
1198
+ while (node && node.lastChild) {
1199
+ node = node.lastChild;
1200
+ }
1201
+
1202
+ return node;
1203
+ };
1204
+
1205
+
1206
+ /**
1207
+ * Whether the object looks like a DOM node.
1208
+ * @param {*} obj The object being tested for node likeness.
1209
+ * @return {boolean} Whether the object looks like a DOM node.
1210
+ */
1211
+ goog.dom.isNodeLike = function(obj) {
1212
+ return goog.isObject(obj) && obj.nodeType > 0;
1213
+ };
1214
+
1215
+
1216
+ /**
1217
+ * Whether the object looks like an Element.
1218
+ * @param {*} obj The object being tested for Element likeness.
1219
+ * @return {boolean} Whether the object looks like an Element.
1220
+ */
1221
+ goog.dom.isElement = function(obj) {
1222
+ return goog.isObject(obj) && obj.nodeType == goog.dom.NodeType.ELEMENT;
1223
+ };
1224
+
1225
+
1226
+ /**
1227
+ * Returns true if the specified value is a Window object. This includes the
1228
+ * global window for HTML pages, and iframe windows.
1229
+ * @param {*} obj Variable to test.
1230
+ * @return {boolean} Whether the variable is a window.
1231
+ */
1232
+ goog.dom.isWindow = function(obj) {
1233
+ return goog.isObject(obj) && obj['window'] == obj;
1234
+ };
1235
+
1236
+
1237
+ /**
1238
+ * Returns an element's parent, if it's an Element.
1239
+ * @param {Element} element The DOM element.
1240
+ * @return {Element} The parent, or null if not an Element.
1241
+ */
1242
+ goog.dom.getParentElement = function(element) {
1243
+ if (goog.dom.BrowserFeature.CAN_USE_PARENT_ELEMENT_PROPERTY) {
1244
+ return element.parentElement;
1245
+ }
1246
+ var parent = element.parentNode;
1247
+ return goog.dom.isElement(parent) ? (/** @type {!Element} */ parent) : null;
1248
+ };
1249
+
1250
+
1251
+ /**
1252
+ * Whether a node contains another node.
1253
+ * @param {Node} parent The node that should contain the other node.
1254
+ * @param {Node} descendant The node to test presence of.
1255
+ * @return {boolean} Whether the parent node contains the descendent node.
1256
+ */
1257
+ goog.dom.contains = function(parent, descendant) {
1258
+ // We use browser specific methods for this if available since it is faster
1259
+ // that way.
1260
+
1261
+ // IE DOM
1262
+ if (parent.contains && descendant.nodeType == goog.dom.NodeType.ELEMENT) {
1263
+ return parent == descendant || parent.contains(descendant);
1264
+ }
1265
+
1266
+ // W3C DOM Level 3
1267
+ if (typeof parent.compareDocumentPosition != 'undefined') {
1268
+ return parent == descendant ||
1269
+ Boolean(parent.compareDocumentPosition(descendant) & 16);
1270
+ }
1271
+
1272
+ // W3C DOM Level 1
1273
+ while (descendant && parent != descendant) {
1274
+ descendant = descendant.parentNode;
1275
+ }
1276
+ return descendant == parent;
1277
+ };
1278
+
1279
+
1280
+ /**
1281
+ * Compares the document order of two nodes, returning 0 if they are the same
1282
+ * node, a negative number if node1 is before node2, and a positive number if
1283
+ * node2 is before node1. Note that we compare the order the tags appear in the
1284
+ * document so in the tree <b><i>text</i></b> the B node is considered to be
1285
+ * before the I node.
1286
+ *
1287
+ * @param {Node} node1 The first node to compare.
1288
+ * @param {Node} node2 The second node to compare.
1289
+ * @return {number} 0 if the nodes are the same node, a negative number if node1
1290
+ * is before node2, and a positive number if node2 is before node1.
1291
+ */
1292
+ goog.dom.compareNodeOrder = function(node1, node2) {
1293
+ // Fall out quickly for equality.
1294
+ if (node1 == node2) {
1295
+ return 0;
1296
+ }
1297
+
1298
+ // Use compareDocumentPosition where available
1299
+ if (node1.compareDocumentPosition) {
1300
+ // 4 is the bitmask for FOLLOWS.
1301
+ return node1.compareDocumentPosition(node2) & 2 ? 1 : -1;
1302
+ }
1303
+
1304
+ // Process in IE using sourceIndex - we check to see if the first node has
1305
+ // a source index or if its parent has one.
1306
+ if ('sourceIndex' in node1 ||
1307
+ (node1.parentNode && 'sourceIndex' in node1.parentNode)) {
1308
+ var isElement1 = node1.nodeType == goog.dom.NodeType.ELEMENT;
1309
+ var isElement2 = node2.nodeType == goog.dom.NodeType.ELEMENT;
1310
+
1311
+ if (isElement1 && isElement2) {
1312
+ return node1.sourceIndex - node2.sourceIndex;
1313
+ } else {
1314
+ var parent1 = node1.parentNode;
1315
+ var parent2 = node2.parentNode;
1316
+
1317
+ if (parent1 == parent2) {
1318
+ return goog.dom.compareSiblingOrder_(node1, node2);
1319
+ }
1320
+
1321
+ if (!isElement1 && goog.dom.contains(parent1, node2)) {
1322
+ return -1 * goog.dom.compareParentsDescendantNodeIe_(node1, node2);
1323
+ }
1324
+
1325
+
1326
+ if (!isElement2 && goog.dom.contains(parent2, node1)) {
1327
+ return goog.dom.compareParentsDescendantNodeIe_(node2, node1);
1328
+ }
1329
+
1330
+ return (isElement1 ? node1.sourceIndex : parent1.sourceIndex) -
1331
+ (isElement2 ? node2.sourceIndex : parent2.sourceIndex);
1332
+ }
1333
+ }
1334
+
1335
+ // For Safari, we compare ranges.
1336
+ var doc = goog.dom.getOwnerDocument(node1);
1337
+
1338
+ var range1, range2;
1339
+ range1 = doc.createRange();
1340
+ range1.selectNode(node1);
1341
+ range1.collapse(true);
1342
+
1343
+ range2 = doc.createRange();
1344
+ range2.selectNode(node2);
1345
+ range2.collapse(true);
1346
+
1347
+ return range1.compareBoundaryPoints(goog.global['Range'].START_TO_END,
1348
+ range2);
1349
+ };
1350
+
1351
+
1352
+ /**
1353
+ * Utility function to compare the position of two nodes, when
1354
+ * {@code textNode}'s parent is an ancestor of {@code node}. If this entry
1355
+ * condition is not met, this function will attempt to reference a null object.
1356
+ * @param {Node} textNode The textNode to compare.
1357
+ * @param {Node} node The node to compare.
1358
+ * @return {number} -1 if node is before textNode, +1 otherwise.
1359
+ * @private
1360
+ */
1361
+ goog.dom.compareParentsDescendantNodeIe_ = function(textNode, node) {
1362
+ var parent = textNode.parentNode;
1363
+ if (parent == node) {
1364
+ // If textNode is a child of node, then node comes first.
1365
+ return -1;
1366
+ }
1367
+ var sibling = node;
1368
+ while (sibling.parentNode != parent) {
1369
+ sibling = sibling.parentNode;
1370
+ }
1371
+ return goog.dom.compareSiblingOrder_(sibling, textNode);
1372
+ };
1373
+
1374
+
1375
+ /**
1376
+ * Utility function to compare the position of two nodes known to be non-equal
1377
+ * siblings.
1378
+ * @param {Node} node1 The first node to compare.
1379
+ * @param {Node} node2 The second node to compare.
1380
+ * @return {number} -1 if node1 is before node2, +1 otherwise.
1381
+ * @private
1382
+ */
1383
+ goog.dom.compareSiblingOrder_ = function(node1, node2) {
1384
+ var s = node2;
1385
+ while ((s = s.previousSibling)) {
1386
+ if (s == node1) {
1387
+ // We just found node1 before node2.
1388
+ return -1;
1389
+ }
1390
+ }
1391
+
1392
+ // Since we didn't find it, node1 must be after node2.
1393
+ return 1;
1394
+ };
1395
+
1396
+
1397
+ /**
1398
+ * Find the deepest common ancestor of the given nodes.
1399
+ * @param {...Node} var_args The nodes to find a common ancestor of.
1400
+ * @return {Node} The common ancestor of the nodes, or null if there is none.
1401
+ * null will only be returned if two or more of the nodes are from different
1402
+ * documents.
1403
+ */
1404
+ goog.dom.findCommonAncestor = function(var_args) {
1405
+ var i, count = arguments.length;
1406
+ if (!count) {
1407
+ return null;
1408
+ } else if (count == 1) {
1409
+ return arguments[0];
1410
+ }
1411
+
1412
+ var paths = [];
1413
+ var minLength = Infinity;
1414
+ for (i = 0; i < count; i++) {
1415
+ // Compute the list of ancestors.
1416
+ var ancestors = [];
1417
+ var node = arguments[i];
1418
+ while (node) {
1419
+ ancestors.unshift(node);
1420
+ node = node.parentNode;
1421
+ }
1422
+
1423
+ // Save the list for comparison.
1424
+ paths.push(ancestors);
1425
+ minLength = Math.min(minLength, ancestors.length);
1426
+ }
1427
+ var output = null;
1428
+ for (i = 0; i < minLength; i++) {
1429
+ var first = paths[0][i];
1430
+ for (var j = 1; j < count; j++) {
1431
+ if (first != paths[j][i]) {
1432
+ return output;
1433
+ }
1434
+ }
1435
+ output = first;
1436
+ }
1437
+ return output;
1438
+ };
1439
+
1440
+
1441
+ /**
1442
+ * Returns the owner document for a node.
1443
+ * @param {Node|Window} node The node to get the document for.
1444
+ * @return {!Document} The document owning the node.
1445
+ */
1446
+ goog.dom.getOwnerDocument = function(node) {
1447
+ // TODO(arv): Remove IE5 code.
1448
+ // IE5 uses document instead of ownerDocument
1449
+ return /** @type {!Document} */ (
1450
+ node.nodeType == goog.dom.NodeType.DOCUMENT ? node :
1451
+ node.ownerDocument || node.document);
1452
+ };
1453
+
1454
+
1455
+ /**
1456
+ * Cross-browser function for getting the document element of a frame or iframe.
1457
+ * @param {Element} frame Frame element.
1458
+ * @return {!Document} The frame content document.
1459
+ */
1460
+ goog.dom.getFrameContentDocument = function(frame) {
1461
+ var doc = frame.contentDocument || frame.contentWindow.document;
1462
+ return doc;
1463
+ };
1464
+
1465
+
1466
+ /**
1467
+ * Cross-browser function for getting the window of a frame or iframe.
1468
+ * @param {HTMLIFrameElement|HTMLFrameElement} frame Frame element.
1469
+ * @return {Window} The window associated with the given frame.
1470
+ */
1471
+ goog.dom.getFrameContentWindow = function(frame) {
1472
+ return frame.contentWindow ||
1473
+ goog.dom.getWindow_(goog.dom.getFrameContentDocument(frame));
1474
+ };
1475
+
1476
+
1477
+ /**
1478
+ * Cross-browser function for setting the text content of an element.
1479
+ * @param {Element} element The element to change the text content of.
1480
+ * @param {string} text The string that should replace the current element
1481
+ * content.
1482
+ */
1483
+ goog.dom.setTextContent = function(element, text) {
1484
+ if ('textContent' in element) {
1485
+ element.textContent = text;
1486
+ } else if (element.firstChild &&
1487
+ element.firstChild.nodeType == goog.dom.NodeType.TEXT) {
1488
+ // If the first child is a text node we just change its data and remove the
1489
+ // rest of the children.
1490
+ while (element.lastChild != element.firstChild) {
1491
+ element.removeChild(element.lastChild);
1492
+ }
1493
+ element.firstChild.data = text;
1494
+ } else {
1495
+ goog.dom.removeChildren(element);
1496
+ var doc = goog.dom.getOwnerDocument(element);
1497
+ element.appendChild(doc.createTextNode(text));
1498
+ }
1499
+ };
1500
+
1501
+
1502
+ /**
1503
+ * Gets the outerHTML of a node, which islike innerHTML, except that it
1504
+ * actually contains the HTML of the node itself.
1505
+ * @param {Element} element The element to get the HTML of.
1506
+ * @return {string} The outerHTML of the given element.
1507
+ */
1508
+ goog.dom.getOuterHtml = function(element) {
1509
+ // IE, Opera and WebKit all have outerHTML.
1510
+ if ('outerHTML' in element) {
1511
+ return element.outerHTML;
1512
+ } else {
1513
+ var doc = goog.dom.getOwnerDocument(element);
1514
+ var div = doc.createElement('div');
1515
+ div.appendChild(element.cloneNode(true));
1516
+ return div.innerHTML;
1517
+ }
1518
+ };
1519
+
1520
+
1521
+ /**
1522
+ * Finds the first descendant node that matches the filter function, using
1523
+ * a depth first search. This function offers the most general purpose way
1524
+ * of finding a matching element. You may also wish to consider
1525
+ * {@code goog.dom.query} which can express many matching criteria using
1526
+ * CSS selector expressions. These expressions often result in a more
1527
+ * compact representation of the desired result.
1528
+ * @see goog.dom.query
1529
+ *
1530
+ * @param {Node} root The root of the tree to search.
1531
+ * @param {function(Node) : boolean} p The filter function.
1532
+ * @return {Node|undefined} The found node or undefined if none is found.
1533
+ */
1534
+ goog.dom.findNode = function(root, p) {
1535
+ var rv = [];
1536
+ var found = goog.dom.findNodes_(root, p, rv, true);
1537
+ return found ? rv[0] : undefined;
1538
+ };
1539
+
1540
+
1541
+ /**
1542
+ * Finds all the descendant nodes that match the filter function, using a
1543
+ * a depth first search. This function offers the most general-purpose way
1544
+ * of finding a set of matching elements. You may also wish to consider
1545
+ * {@code goog.dom.query} which can express many matching criteria using
1546
+ * CSS selector expressions. These expressions often result in a more
1547
+ * compact representation of the desired result.
1548
+
1549
+ * @param {Node} root The root of the tree to search.
1550
+ * @param {function(Node) : boolean} p The filter function.
1551
+ * @return {Array.<Node>} The found nodes or an empty array if none are found.
1552
+ */
1553
+ goog.dom.findNodes = function(root, p) {
1554
+ var rv = [];
1555
+ goog.dom.findNodes_(root, p, rv, false);
1556
+ return rv;
1557
+ };
1558
+
1559
+
1560
+ /**
1561
+ * Finds the first or all the descendant nodes that match the filter function,
1562
+ * using a depth first search.
1563
+ * @param {Node} root The root of the tree to search.
1564
+ * @param {function(Node) : boolean} p The filter function.
1565
+ * @param {Array.<Node>} rv The found nodes are added to this array.
1566
+ * @param {boolean} findOne If true we exit after the first found node.
1567
+ * @return {boolean} Whether the search is complete or not. True in case findOne
1568
+ * is true and the node is found. False otherwise.
1569
+ * @private
1570
+ */
1571
+ goog.dom.findNodes_ = function(root, p, rv, findOne) {
1572
+ if (root != null) {
1573
+ var child = root.firstChild;
1574
+ while (child) {
1575
+ if (p(child)) {
1576
+ rv.push(child);
1577
+ if (findOne) {
1578
+ return true;
1579
+ }
1580
+ }
1581
+ if (goog.dom.findNodes_(child, p, rv, findOne)) {
1582
+ return true;
1583
+ }
1584
+ child = child.nextSibling;
1585
+ }
1586
+ }
1587
+ return false;
1588
+ };
1589
+
1590
+
1591
+ /**
1592
+ * Map of tags whose content to ignore when calculating text length.
1593
+ * @type {Object}
1594
+ * @private
1595
+ */
1596
+ goog.dom.TAGS_TO_IGNORE_ = {
1597
+ 'SCRIPT': 1,
1598
+ 'STYLE': 1,
1599
+ 'HEAD': 1,
1600
+ 'IFRAME': 1,
1601
+ 'OBJECT': 1
1602
+ };
1603
+
1604
+
1605
+ /**
1606
+ * Map of tags which have predefined values with regard to whitespace.
1607
+ * @type {Object}
1608
+ * @private
1609
+ */
1610
+ goog.dom.PREDEFINED_TAG_VALUES_ = {'IMG': ' ', 'BR': '\n'};
1611
+
1612
+
1613
+ /**
1614
+ * Returns true if the element has a tab index that allows it to receive
1615
+ * keyboard focus (tabIndex >= 0), false otherwise. Note that form elements
1616
+ * natively support keyboard focus, even if they have no tab index.
1617
+ * @param {Element} element Element to check.
1618
+ * @return {boolean} Whether the element has a tab index that allows keyboard
1619
+ * focus.
1620
+ * @see http://fluidproject.org/blog/2008/01/09/getting-setting-and-removing-tabindex-values-with-javascript/
1621
+ */
1622
+ goog.dom.isFocusableTabIndex = function(element) {
1623
+ // IE returns 0 for an unset tabIndex, so we must use getAttributeNode(),
1624
+ // which returns an object with a 'specified' property if tabIndex is
1625
+ // specified. This works on other browsers, too.
1626
+ var attrNode = element.getAttributeNode('tabindex'); // Must be lowercase!
1627
+ if (attrNode && attrNode.specified) {
1628
+ var index = element.tabIndex;
1629
+ // NOTE: IE9 puts tabIndex in 16-bit int, e.g. -2 is 65534.
1630
+ return goog.isNumber(index) && index >= 0 && index < 32768;
1631
+ }
1632
+ return false;
1633
+ };
1634
+
1635
+
1636
+ /**
1637
+ * Enables or disables keyboard focus support on the element via its tab index.
1638
+ * Only elements for which {@link goog.dom.isFocusableTabIndex} returns true
1639
+ * (or elements that natively support keyboard focus, like form elements) can
1640
+ * receive keyboard focus. See http://go/tabindex for more info.
1641
+ * @param {Element} element Element whose tab index is to be changed.
1642
+ * @param {boolean} enable Whether to set or remove a tab index on the element
1643
+ * that supports keyboard focus.
1644
+ */
1645
+ goog.dom.setFocusableTabIndex = function(element, enable) {
1646
+ if (enable) {
1647
+ element.tabIndex = 0;
1648
+ } else {
1649
+ // Set tabIndex to -1 first, then remove it. This is a workaround for
1650
+ // Safari (confirmed in version 4 on Windows). When removing the attribute
1651
+ // without setting it to -1 first, the element remains keyboard focusable
1652
+ // despite not having a tabIndex attribute anymore.
1653
+ element.tabIndex = -1;
1654
+ element.removeAttribute('tabIndex'); // Must be camelCase!
1655
+ }
1656
+ };
1657
+
1658
+
1659
+ /**
1660
+ * Returns the text content of the current node, without markup and invisible
1661
+ * symbols. New lines are stripped and whitespace is collapsed,
1662
+ * such that each character would be visible.
1663
+ *
1664
+ * In browsers that support it, innerText is used. Other browsers attempt to
1665
+ * simulate it via node traversal. Line breaks are canonicalized in IE.
1666
+ *
1667
+ * @param {Node} node The node from which we are getting content.
1668
+ * @return {string} The text content.
1669
+ */
1670
+ goog.dom.getTextContent = function(node) {
1671
+ var textContent;
1672
+ // Note(arv): IE9, Opera, and Safari 3 support innerText but they include
1673
+ // text nodes in script tags. So we revert to use a user agent test here.
1674
+ if (goog.dom.BrowserFeature.CAN_USE_INNER_TEXT && ('innerText' in node)) {
1675
+ textContent = goog.string.canonicalizeNewlines(node.innerText);
1676
+ // Unfortunately .innerText() returns text with &shy; symbols
1677
+ // We need to filter it out and then remove duplicate whitespaces
1678
+ } else {
1679
+ var buf = [];
1680
+ goog.dom.getTextContent_(node, buf, true);
1681
+ textContent = buf.join('');
1682
+ }
1683
+
1684
+ // Strip &shy; entities. goog.format.insertWordBreaks inserts them in Opera.
1685
+ textContent = textContent.replace(/ \xAD /g, ' ').replace(/\xAD/g, '');
1686
+ // Strip &#8203; entities. goog.format.insertWordBreaks inserts them in IE8.
1687
+ textContent = textContent.replace(/\u200B/g, '');
1688
+
1689
+ // Skip this replacement on old browsers with working innerText, which
1690
+ // automatically turns &nbsp; into ' ' and / +/ into ' ' when reading
1691
+ // innerText.
1692
+ if (!goog.dom.BrowserFeature.CAN_USE_INNER_TEXT) {
1693
+ textContent = textContent.replace(/ +/g, ' ');
1694
+ }
1695
+ if (textContent != ' ') {
1696
+ textContent = textContent.replace(/^\s*/, '');
1697
+ }
1698
+
1699
+ return textContent;
1700
+ };
1701
+
1702
+
1703
+ /**
1704
+ * Returns the text content of the current node, without markup.
1705
+ *
1706
+ * Unlike {@code getTextContent} this method does not collapse whitespaces
1707
+ * or normalize lines breaks.
1708
+ *
1709
+ * @param {Node} node The node from which we are getting content.
1710
+ * @return {string} The raw text content.
1711
+ */
1712
+ goog.dom.getRawTextContent = function(node) {
1713
+ var buf = [];
1714
+ goog.dom.getTextContent_(node, buf, false);
1715
+
1716
+ return buf.join('');
1717
+ };
1718
+
1719
+
1720
+ /**
1721
+ * Recursive support function for text content retrieval.
1722
+ *
1723
+ * @param {Node} node The node from which we are getting content.
1724
+ * @param {Array} buf string buffer.
1725
+ * @param {boolean} normalizeWhitespace Whether to normalize whitespace.
1726
+ * @private
1727
+ */
1728
+ goog.dom.getTextContent_ = function(node, buf, normalizeWhitespace) {
1729
+ if (node.nodeName in goog.dom.TAGS_TO_IGNORE_) {
1730
+ // ignore certain tags
1731
+ } else if (node.nodeType == goog.dom.NodeType.TEXT) {
1732
+ if (normalizeWhitespace) {
1733
+ buf.push(String(node.nodeValue).replace(/(\r\n|\r|\n)/g, ''));
1734
+ } else {
1735
+ buf.push(node.nodeValue);
1736
+ }
1737
+ } else if (node.nodeName in goog.dom.PREDEFINED_TAG_VALUES_) {
1738
+ buf.push(goog.dom.PREDEFINED_TAG_VALUES_[node.nodeName]);
1739
+ } else {
1740
+ var child = node.firstChild;
1741
+ while (child) {
1742
+ goog.dom.getTextContent_(child, buf, normalizeWhitespace);
1743
+ child = child.nextSibling;
1744
+ }
1745
+ }
1746
+ };
1747
+
1748
+
1749
+ /**
1750
+ * Returns the text length of the text contained in a node, without markup. This
1751
+ * is equivalent to the selection length if the node was selected, or the number
1752
+ * of cursor movements to traverse the node. Images & BRs take one space. New
1753
+ * lines are ignored.
1754
+ *
1755
+ * @param {Node} node The node whose text content length is being calculated.
1756
+ * @return {number} The length of {@code node}'s text content.
1757
+ */
1758
+ goog.dom.getNodeTextLength = function(node) {
1759
+ return goog.dom.getTextContent(node).length;
1760
+ };
1761
+
1762
+
1763
+ /**
1764
+ * Returns the text offset of a node relative to one of its ancestors. The text
1765
+ * length is the same as the length calculated by goog.dom.getNodeTextLength.
1766
+ *
1767
+ * @param {Node} node The node whose offset is being calculated.
1768
+ * @param {Node=} opt_offsetParent The node relative to which the offset will
1769
+ * be calculated. Defaults to the node's owner document's body.
1770
+ * @return {number} The text offset.
1771
+ */
1772
+ goog.dom.getNodeTextOffset = function(node, opt_offsetParent) {
1773
+ var root = opt_offsetParent || goog.dom.getOwnerDocument(node).body;
1774
+ var buf = [];
1775
+ while (node && node != root) {
1776
+ var cur = node;
1777
+ while ((cur = cur.previousSibling)) {
1778
+ buf.unshift(goog.dom.getTextContent(cur));
1779
+ }
1780
+ node = node.parentNode;
1781
+ }
1782
+ // Trim left to deal with FF cases when there might be line breaks and empty
1783
+ // nodes at the front of the text
1784
+ return goog.string.trimLeft(buf.join('')).replace(/ +/g, ' ').length;
1785
+ };
1786
+
1787
+
1788
+ /**
1789
+ * Returns the node at a given offset in a parent node. If an object is
1790
+ * provided for the optional third parameter, the node and the remainder of the
1791
+ * offset will stored as properties of this object.
1792
+ * @param {Node} parent The parent node.
1793
+ * @param {number} offset The offset into the parent node.
1794
+ * @param {Object=} opt_result Object to be used to store the return value. The
1795
+ * return value will be stored in the form {node: Node, remainder: number}
1796
+ * if this object is provided.
1797
+ * @return {Node} The node at the given offset.
1798
+ */
1799
+ goog.dom.getNodeAtOffset = function(parent, offset, opt_result) {
1800
+ var stack = [parent], pos = 0, cur;
1801
+ while (stack.length > 0 && pos < offset) {
1802
+ cur = stack.pop();
1803
+ if (cur.nodeName in goog.dom.TAGS_TO_IGNORE_) {
1804
+ // ignore certain tags
1805
+ } else if (cur.nodeType == goog.dom.NodeType.TEXT) {
1806
+ var text = cur.nodeValue.replace(/(\r\n|\r|\n)/g, '').replace(/ +/g, ' ');
1807
+ pos += text.length;
1808
+ } else if (cur.nodeName in goog.dom.PREDEFINED_TAG_VALUES_) {
1809
+ pos += goog.dom.PREDEFINED_TAG_VALUES_[cur.nodeName].length;
1810
+ } else {
1811
+ for (var i = cur.childNodes.length - 1; i >= 0; i--) {
1812
+ stack.push(cur.childNodes[i]);
1813
+ }
1814
+ }
1815
+ }
1816
+ if (goog.isObject(opt_result)) {
1817
+ opt_result.remainder = cur ? cur.nodeValue.length + offset - pos - 1 : 0;
1818
+ opt_result.node = cur;
1819
+ }
1820
+
1821
+ return cur;
1822
+ };
1823
+
1824
+
1825
+ /**
1826
+ * Returns true if the object is a {@code NodeList}. To qualify as a NodeList,
1827
+ * the object must have a numeric length property and an item function (which
1828
+ * has type 'string' on IE for some reason).
1829
+ * @param {Object} val Object to test.
1830
+ * @return {boolean} Whether the object is a NodeList.
1831
+ */
1832
+ goog.dom.isNodeList = function(val) {
1833
+ // TODO(attila): Now the isNodeList is part of goog.dom we can use
1834
+ // goog.userAgent to make this simpler.
1835
+ // A NodeList must have a length property of type 'number' on all platforms.
1836
+ if (val && typeof val.length == 'number') {
1837
+ // A NodeList is an object everywhere except Safari, where it's a function.
1838
+ if (goog.isObject(val)) {
1839
+ // A NodeList must have an item function (on non-IE platforms) or an item
1840
+ // property of type 'string' (on IE).
1841
+ return typeof val.item == 'function' || typeof val.item == 'string';
1842
+ } else if (goog.isFunction(val)) {
1843
+ // On Safari, a NodeList is a function with an item property that is also
1844
+ // a function.
1845
+ return typeof val.item == 'function';
1846
+ }
1847
+ }
1848
+
1849
+ // Not a NodeList.
1850
+ return false;
1851
+ };
1852
+
1853
+
1854
+ /**
1855
+ * Walks up the DOM hierarchy returning the first ancestor that has the passed
1856
+ * tag name and/or class name. If the passed element matches the specified
1857
+ * criteria, the element itself is returned.
1858
+ * @param {Node} element The DOM node to start with.
1859
+ * @param {?string=} opt_tag The tag name to match (or null/undefined to match
1860
+ * any node regardless of tag name). Must be uppercase (goog.dom.TagName).
1861
+ * @param {?string=} opt_class The class name to match (or null/undefined to
1862
+ * match any node regardless of class name).
1863
+ * @return {Node} The first ancestor that matches the passed criteria, or
1864
+ * null if none match.
1865
+ */
1866
+ goog.dom.getAncestorByTagNameAndClass = function(element, opt_tag, opt_class) {
1867
+ var tagName = opt_tag ? opt_tag.toUpperCase() : null;
1868
+ return goog.dom.getAncestor(element,
1869
+ function(node) {
1870
+ return (!tagName || node.nodeName == tagName) &&
1871
+ (!opt_class || goog.dom.classes.has(node, opt_class));
1872
+ }, true);
1873
+ };
1874
+
1875
+
1876
+ /**
1877
+ * Walks up the DOM hierarchy returning the first ancestor that has the passed
1878
+ * class name. If the passed element matches the specified criteria, the
1879
+ * element itself is returned.
1880
+ * @param {Node} element The DOM node to start with.
1881
+ * @param {?string=} opt_class The class name to match (or null/undefined to
1882
+ * match any node regardless of class name).
1883
+ * @return {Node} The first ancestor that matches the passed criteria, or
1884
+ * null if none match.
1885
+ */
1886
+ goog.dom.getAncestorByClass = function(element, opt_class) {
1887
+ return goog.dom.getAncestorByTagNameAndClass(element, null, opt_class);
1888
+ };
1889
+
1890
+
1891
+ /**
1892
+ * Walks up the DOM hierarchy returning the first ancestor that passes the
1893
+ * matcher function.
1894
+ * @param {Node} element The DOM node to start with.
1895
+ * @param {function(Node) : boolean} matcher A function that returns true if the
1896
+ * passed node matches the desired criteria.
1897
+ * @param {boolean=} opt_includeNode If true, the node itself is included in
1898
+ * the search (the first call to the matcher will pass startElement as
1899
+ * the node to test).
1900
+ * @param {number=} opt_maxSearchSteps Maximum number of levels to search up the
1901
+ * dom.
1902
+ * @return {Node} DOM node that matched the matcher, or null if there was
1903
+ * no match.
1904
+ */
1905
+ goog.dom.getAncestor = function(
1906
+ element, matcher, opt_includeNode, opt_maxSearchSteps) {
1907
+ if (!opt_includeNode) {
1908
+ element = element.parentNode;
1909
+ }
1910
+ var ignoreSearchSteps = opt_maxSearchSteps == null;
1911
+ var steps = 0;
1912
+ while (element && (ignoreSearchSteps || steps <= opt_maxSearchSteps)) {
1913
+ if (matcher(element)) {
1914
+ return element;
1915
+ }
1916
+ element = element.parentNode;
1917
+ steps++;
1918
+ }
1919
+ // Reached the root of the DOM without a match
1920
+ return null;
1921
+ };
1922
+
1923
+
1924
+ /**
1925
+ * Determines the active element in the given document.
1926
+ * @param {Document} doc The document to look in.
1927
+ * @return {Element} The active element.
1928
+ */
1929
+ goog.dom.getActiveElement = function(doc) {
1930
+ try {
1931
+ return doc && doc.activeElement;
1932
+ } catch (e) {
1933
+ // NOTE(nicksantos): Sometimes, evaluating document.activeElement in IE
1934
+ // throws an exception. I'm not 100% sure why, but I suspect it chokes
1935
+ // on document.activeElement if the activeElement has been recently
1936
+ // removed from the DOM by a JS operation.
1937
+ //
1938
+ // We assume that an exception here simply means
1939
+ // "there is no active element."
1940
+ }
1941
+
1942
+ return null;
1943
+ };
1944
+
1945
+
1946
+
1947
+ /**
1948
+ * Create an instance of a DOM helper with a new document object.
1949
+ * @param {Document=} opt_document Document object to associate with this
1950
+ * DOM helper.
1951
+ * @constructor
1952
+ */
1953
+ goog.dom.DomHelper = function(opt_document) {
1954
+ /**
1955
+ * Reference to the document object to use
1956
+ * @type {!Document}
1957
+ * @private
1958
+ */
1959
+ this.document_ = opt_document || goog.global.document || document;
1960
+ };
1961
+
1962
+
1963
+ /**
1964
+ * Gets the dom helper object for the document where the element resides.
1965
+ * @param {Node=} opt_node If present, gets the DomHelper for this node.
1966
+ * @return {!goog.dom.DomHelper} The DomHelper.
1967
+ */
1968
+ goog.dom.DomHelper.prototype.getDomHelper = goog.dom.getDomHelper;
1969
+
1970
+
1971
+ /**
1972
+ * Sets the document object.
1973
+ * @param {!Document} document Document object.
1974
+ */
1975
+ goog.dom.DomHelper.prototype.setDocument = function(document) {
1976
+ this.document_ = document;
1977
+ };
1978
+
1979
+
1980
+ /**
1981
+ * Gets the document object being used by the dom library.
1982
+ * @return {!Document} Document object.
1983
+ */
1984
+ goog.dom.DomHelper.prototype.getDocument = function() {
1985
+ return this.document_;
1986
+ };
1987
+
1988
+
1989
+ /**
1990
+ * Alias for {@code getElementById}. If a DOM node is passed in then we just
1991
+ * return that.
1992
+ * @param {string|Element} element Element ID or a DOM node.
1993
+ * @return {Element} The element with the given ID, or the node passed in.
1994
+ */
1995
+ goog.dom.DomHelper.prototype.getElement = function(element) {
1996
+ if (goog.isString(element)) {
1997
+ return this.document_.getElementById(element);
1998
+ } else {
1999
+ return element;
2000
+ }
2001
+ };
2002
+
2003
+
2004
+ /**
2005
+ * Alias for {@code getElement}.
2006
+ * @param {string|Element} element Element ID or a DOM node.
2007
+ * @return {Element} The element with the given ID, or the node passed in.
2008
+ * @deprecated Use {@link goog.dom.DomHelper.prototype.getElement} instead.
2009
+ */
2010
+ goog.dom.DomHelper.prototype.$ = goog.dom.DomHelper.prototype.getElement;
2011
+
2012
+
2013
+ /**
2014
+ * Looks up elements by both tag and class name, using browser native functions
2015
+ * ({@code querySelectorAll}, {@code getElementsByTagName} or
2016
+ * {@code getElementsByClassName}) where possible. The returned array is a live
2017
+ * NodeList or a static list depending on the code path taken.
2018
+ *
2019
+ * @see goog.dom.query
2020
+ *
2021
+ * @param {?string=} opt_tag Element tag name or * for all tags.
2022
+ * @param {?string=} opt_class Optional class name.
2023
+ * @param {Document|Element=} opt_el Optional element to look in.
2024
+ * @return { {length: number} } Array-like list of elements (only a length
2025
+ * property and numerical indices are guaranteed to exist).
2026
+ */
2027
+ goog.dom.DomHelper.prototype.getElementsByTagNameAndClass = function(opt_tag,
2028
+ opt_class,
2029
+ opt_el) {
2030
+ return goog.dom.getElementsByTagNameAndClass_(this.document_, opt_tag,
2031
+ opt_class, opt_el);
2032
+ };
2033
+
2034
+
2035
+ /**
2036
+ * Returns an array of all the elements with the provided className.
2037
+ * @see {goog.dom.query}
2038
+ * @param {string} className the name of the class to look for.
2039
+ * @param {Element|Document=} opt_el Optional element to look in.
2040
+ * @return { {length: number} } The items found with the class name provided.
2041
+ */
2042
+ goog.dom.DomHelper.prototype.getElementsByClass = function(className, opt_el) {
2043
+ var doc = opt_el || this.document_;
2044
+ return goog.dom.getElementsByClass(className, doc);
2045
+ };
2046
+
2047
+
2048
+ /**
2049
+ * Returns the first element we find matching the provided class name.
2050
+ * @see {goog.dom.query}
2051
+ * @param {string} className the name of the class to look for.
2052
+ * @param {Element|Document=} opt_el Optional element to look in.
2053
+ * @return {Element} The first item found with the class name provided.
2054
+ */
2055
+ goog.dom.DomHelper.prototype.getElementByClass = function(className, opt_el) {
2056
+ var doc = opt_el || this.document_;
2057
+ return goog.dom.getElementByClass(className, doc);
2058
+ };
2059
+
2060
+
2061
+ /**
2062
+ * Alias for {@code getElementsByTagNameAndClass}.
2063
+ * @deprecated Use DomHelper getElementsByTagNameAndClass.
2064
+ * @see goog.dom.query
2065
+ *
2066
+ * @param {?string=} opt_tag Element tag name.
2067
+ * @param {?string=} opt_class Optional class name.
2068
+ * @param {Element=} opt_el Optional element to look in.
2069
+ * @return { {length: number} } Array-like list of elements (only a length
2070
+ * property and numerical indices are guaranteed to exist).
2071
+ */
2072
+ goog.dom.DomHelper.prototype.$$ =
2073
+ goog.dom.DomHelper.prototype.getElementsByTagNameAndClass;
2074
+
2075
+
2076
+ /**
2077
+ * Sets a number of properties on a node.
2078
+ * @param {Element} element DOM node to set properties on.
2079
+ * @param {Object} properties Hash of property:value pairs.
2080
+ */
2081
+ goog.dom.DomHelper.prototype.setProperties = goog.dom.setProperties;
2082
+
2083
+
2084
+ /**
2085
+ * Gets the dimensions of the viewport.
2086
+ * @param {Window=} opt_window Optional window element to test. Defaults to
2087
+ * the window of the Dom Helper.
2088
+ * @return {!goog.math.Size} Object with values 'width' and 'height'.
2089
+ */
2090
+ goog.dom.DomHelper.prototype.getViewportSize = function(opt_window) {
2091
+ // TODO(arv): This should not take an argument. That breaks the rule of a
2092
+ // a DomHelper representing a single frame/window/document.
2093
+ return goog.dom.getViewportSize(opt_window || this.getWindow());
2094
+ };
2095
+
2096
+
2097
+ /**
2098
+ * Calculates the height of the document.
2099
+ *
2100
+ * @return {number} The height of the document.
2101
+ */
2102
+ goog.dom.DomHelper.prototype.getDocumentHeight = function() {
2103
+ return goog.dom.getDocumentHeight_(this.getWindow());
2104
+ };
2105
+
2106
+
2107
+ /**
2108
+ * Typedef for use with goog.dom.createDom and goog.dom.append.
2109
+ * @typedef {Object|string|Array|NodeList}
2110
+ */
2111
+ goog.dom.Appendable;
2112
+
2113
+
2114
+ /**
2115
+ * Returns a dom node with a set of attributes. This function accepts varargs
2116
+ * for subsequent nodes to be added. Subsequent nodes will be added to the
2117
+ * first node as childNodes.
2118
+ *
2119
+ * So:
2120
+ * <code>createDom('div', null, createDom('p'), createDom('p'));</code>
2121
+ * would return a div with two child paragraphs
2122
+ *
2123
+ * An easy way to move all child nodes of an existing element to a new parent
2124
+ * element is:
2125
+ * <code>createDom('div', null, oldElement.childNodes);</code>
2126
+ * which will remove all child nodes from the old element and add them as
2127
+ * child nodes of the new DIV.
2128
+ *
2129
+ * @param {string} tagName Tag to create.
2130
+ * @param {Object|string=} opt_attributes If object, then a map of name-value
2131
+ * pairs for attributes. If a string, then this is the className of the new
2132
+ * element.
2133
+ * @param {...goog.dom.Appendable} var_args Further DOM nodes or
2134
+ * strings for text nodes. If one of the var_args is an array or
2135
+ * NodeList, its elements will be added as childNodes instead.
2136
+ * @return {!Element} Reference to a DOM node.
2137
+ */
2138
+ goog.dom.DomHelper.prototype.createDom = function(tagName,
2139
+ opt_attributes,
2140
+ var_args) {
2141
+ return goog.dom.createDom_(this.document_, arguments);
2142
+ };
2143
+
2144
+
2145
+ /**
2146
+ * Alias for {@code createDom}.
2147
+ * @param {string} tagName Tag to create.
2148
+ * @param {Object|string=} opt_attributes If object, then a map of name-value
2149
+ * pairs for attributes. If a string, then this is the className of the new
2150
+ * element.
2151
+ * @param {...goog.dom.Appendable} var_args Further DOM nodes or strings for
2152
+ * text nodes. If one of the var_args is an array, its children will be
2153
+ * added as childNodes instead.
2154
+ * @return {!Element} Reference to a DOM node.
2155
+ * @deprecated Use {@link goog.dom.DomHelper.prototype.createDom} instead.
2156
+ */
2157
+ goog.dom.DomHelper.prototype.$dom = goog.dom.DomHelper.prototype.createDom;
2158
+
2159
+
2160
+ /**
2161
+ * Creates a new element.
2162
+ * @param {string} name Tag name.
2163
+ * @return {!Element} The new element.
2164
+ */
2165
+ goog.dom.DomHelper.prototype.createElement = function(name) {
2166
+ return this.document_.createElement(name);
2167
+ };
2168
+
2169
+
2170
+ /**
2171
+ * Creates a new text node.
2172
+ * @param {string} content Content.
2173
+ * @return {!Text} The new text node.
2174
+ */
2175
+ goog.dom.DomHelper.prototype.createTextNode = function(content) {
2176
+ return this.document_.createTextNode(content);
2177
+ };
2178
+
2179
+
2180
+ /**
2181
+ * Create a table.
2182
+ * @param {number} rows The number of rows in the table. Must be >= 1.
2183
+ * @param {number} columns The number of columns in the table. Must be >= 1.
2184
+ * @param {boolean=} opt_fillWithNbsp If true, fills table entries with nsbps.
2185
+ * @return {!Element} The created table.
2186
+ */
2187
+ goog.dom.DomHelper.prototype.createTable = function(rows, columns,
2188
+ opt_fillWithNbsp) {
2189
+ return goog.dom.createTable_(this.document_, rows, columns,
2190
+ !!opt_fillWithNbsp);
2191
+ };
2192
+
2193
+
2194
+ /**
2195
+ * Converts an HTML string into a node or a document fragment. A single Node
2196
+ * is used if the {@code htmlString} only generates a single node. If the
2197
+ * {@code htmlString} generates multiple nodes then these are put inside a
2198
+ * {@code DocumentFragment}.
2199
+ *
2200
+ * @param {string} htmlString The HTML string to convert.
2201
+ * @return {!Node} The resulting node.
2202
+ */
2203
+ goog.dom.DomHelper.prototype.htmlToDocumentFragment = function(htmlString) {
2204
+ return goog.dom.htmlToDocumentFragment_(this.document_, htmlString);
2205
+ };
2206
+
2207
+
2208
+ /**
2209
+ * Returns the compatMode of the document.
2210
+ * @return {string} The result is either CSS1Compat or BackCompat.
2211
+ * @deprecated use goog.dom.DomHelper.prototype.isCss1CompatMode instead.
2212
+ */
2213
+ goog.dom.DomHelper.prototype.getCompatMode = function() {
2214
+ return this.isCss1CompatMode() ? 'CSS1Compat' : 'BackCompat';
2215
+ };
2216
+
2217
+
2218
+ /**
2219
+ * Returns true if the browser is in "CSS1-compatible" (standards-compliant)
2220
+ * mode, false otherwise.
2221
+ * @return {boolean} True if in CSS1-compatible mode.
2222
+ */
2223
+ goog.dom.DomHelper.prototype.isCss1CompatMode = function() {
2224
+ return goog.dom.isCss1CompatMode_(this.document_);
2225
+ };
2226
+
2227
+
2228
+ /**
2229
+ * Gets the window object associated with the document.
2230
+ * @return {!Window} The window associated with the given document.
2231
+ */
2232
+ goog.dom.DomHelper.prototype.getWindow = function() {
2233
+ return goog.dom.getWindow_(this.document_);
2234
+ };
2235
+
2236
+
2237
+ /**
2238
+ * Gets the document scroll element.
2239
+ * @return {Element} Scrolling element.
2240
+ */
2241
+ goog.dom.DomHelper.prototype.getDocumentScrollElement = function() {
2242
+ return goog.dom.getDocumentScrollElement_(this.document_);
2243
+ };
2244
+
2245
+
2246
+ /**
2247
+ * Gets the document scroll distance as a coordinate object.
2248
+ * @return {!goog.math.Coordinate} Object with properties 'x' and 'y'.
2249
+ */
2250
+ goog.dom.DomHelper.prototype.getDocumentScroll = function() {
2251
+ return goog.dom.getDocumentScroll_(this.document_);
2252
+ };
2253
+
2254
+
2255
+ /**
2256
+ * Appends a child to a node.
2257
+ * @param {Node} parent Parent.
2258
+ * @param {Node} child Child.
2259
+ */
2260
+ goog.dom.DomHelper.prototype.appendChild = goog.dom.appendChild;
2261
+
2262
+
2263
+ /**
2264
+ * Appends a node with text or other nodes.
2265
+ * @param {!Node} parent The node to append nodes to.
2266
+ * @param {...goog.dom.Appendable} var_args The things to append to the node.
2267
+ * If this is a Node it is appended as is.
2268
+ * If this is a string then a text node is appended.
2269
+ * If this is an array like object then fields 0 to length - 1 are appended.
2270
+ */
2271
+ goog.dom.DomHelper.prototype.append = goog.dom.append;
2272
+
2273
+
2274
+ /**
2275
+ * Removes all the child nodes on a DOM node.
2276
+ * @param {Node} node Node to remove children from.
2277
+ */
2278
+ goog.dom.DomHelper.prototype.removeChildren = goog.dom.removeChildren;
2279
+
2280
+
2281
+ /**
2282
+ * Inserts a new node before an existing reference node (i.e., as the previous
2283
+ * sibling). If the reference node has no parent, then does nothing.
2284
+ * @param {Node} newNode Node to insert.
2285
+ * @param {Node} refNode Reference node to insert before.
2286
+ */
2287
+ goog.dom.DomHelper.prototype.insertSiblingBefore = goog.dom.insertSiblingBefore;
2288
+
2289
+
2290
+ /**
2291
+ * Inserts a new node after an existing reference node (i.e., as the next
2292
+ * sibling). If the reference node has no parent, then does nothing.
2293
+ * @param {Node} newNode Node to insert.
2294
+ * @param {Node} refNode Reference node to insert after.
2295
+ */
2296
+ goog.dom.DomHelper.prototype.insertSiblingAfter = goog.dom.insertSiblingAfter;
2297
+
2298
+
2299
+ /**
2300
+ * Removes a node from its parent.
2301
+ * @param {Node} node The node to remove.
2302
+ * @return {Node} The node removed if removed; else, null.
2303
+ */
2304
+ goog.dom.DomHelper.prototype.removeNode = goog.dom.removeNode;
2305
+
2306
+
2307
+ /**
2308
+ * Replaces a node in the DOM tree. Will do nothing if {@code oldNode} has no
2309
+ * parent.
2310
+ * @param {Node} newNode Node to insert.
2311
+ * @param {Node} oldNode Node to replace.
2312
+ */
2313
+ goog.dom.DomHelper.prototype.replaceNode = goog.dom.replaceNode;
2314
+
2315
+
2316
+ /**
2317
+ * Flattens an element. That is, removes it and replace it with its children.
2318
+ * @param {Element} element The element to flatten.
2319
+ * @return {Element|undefined} The original element, detached from the document
2320
+ * tree, sans children, or undefined if the element was already not in the
2321
+ * document.
2322
+ */
2323
+ goog.dom.DomHelper.prototype.flattenElement = goog.dom.flattenElement;
2324
+
2325
+
2326
+ /**
2327
+ * Returns the first child node that is an element.
2328
+ * @param {Node} node The node to get the first child element of.
2329
+ * @return {Element} The first child node of {@code node} that is an element.
2330
+ */
2331
+ goog.dom.DomHelper.prototype.getFirstElementChild =
2332
+ goog.dom.getFirstElementChild;
2333
+
2334
+
2335
+ /**
2336
+ * Returns the last child node that is an element.
2337
+ * @param {Node} node The node to get the last child element of.
2338
+ * @return {Element} The last child node of {@code node} that is an element.
2339
+ */
2340
+ goog.dom.DomHelper.prototype.getLastElementChild = goog.dom.getLastElementChild;
2341
+
2342
+
2343
+ /**
2344
+ * Returns the first next sibling that is an element.
2345
+ * @param {Node} node The node to get the next sibling element of.
2346
+ * @return {Element} The next sibling of {@code node} that is an element.
2347
+ */
2348
+ goog.dom.DomHelper.prototype.getNextElementSibling =
2349
+ goog.dom.getNextElementSibling;
2350
+
2351
+
2352
+ /**
2353
+ * Returns the first previous sibling that is an element.
2354
+ * @param {Node} node The node to get the previous sibling element of.
2355
+ * @return {Element} The first previous sibling of {@code node} that is
2356
+ * an element.
2357
+ */
2358
+ goog.dom.DomHelper.prototype.getPreviousElementSibling =
2359
+ goog.dom.getPreviousElementSibling;
2360
+
2361
+
2362
+ /**
2363
+ * Returns the next node in source order from the given node.
2364
+ * @param {Node} node The node.
2365
+ * @return {Node} The next node in the DOM tree, or null if this was the last
2366
+ * node.
2367
+ */
2368
+ goog.dom.DomHelper.prototype.getNextNode =
2369
+ goog.dom.getNextNode;
2370
+
2371
+
2372
+ /**
2373
+ * Returns the previous node in source order from the given node.
2374
+ * @param {Node} node The node.
2375
+ * @return {Node} The previous node in the DOM tree, or null if this was the
2376
+ * first node.
2377
+ */
2378
+ goog.dom.DomHelper.prototype.getPreviousNode =
2379
+ goog.dom.getPreviousNode;
2380
+
2381
+
2382
+ /**
2383
+ * Whether the object looks like a DOM node.
2384
+ * @param {*} obj The object being tested for node likeness.
2385
+ * @return {boolean} Whether the object looks like a DOM node.
2386
+ */
2387
+ goog.dom.DomHelper.prototype.isNodeLike = goog.dom.isNodeLike;
2388
+
2389
+
2390
+ /**
2391
+ * Whether a node contains another node.
2392
+ * @param {Node} parent The node that should contain the other node.
2393
+ * @param {Node} descendant The node to test presence of.
2394
+ * @return {boolean} Whether the parent node contains the descendent node.
2395
+ */
2396
+ goog.dom.DomHelper.prototype.contains = goog.dom.contains;
2397
+
2398
+
2399
+ /**
2400
+ * Returns the owner document for a node.
2401
+ * @param {Node} node The node to get the document for.
2402
+ * @return {!Document} The document owning the node.
2403
+ */
2404
+ goog.dom.DomHelper.prototype.getOwnerDocument = goog.dom.getOwnerDocument;
2405
+
2406
+
2407
+ /**
2408
+ * Cross browser function for getting the document element of an iframe.
2409
+ * @param {HTMLIFrameElement|HTMLFrameElement} iframe Iframe element.
2410
+ * @return {!HTMLDocument} The frame content document.
2411
+ */
2412
+ goog.dom.DomHelper.prototype.getFrameContentDocument =
2413
+ goog.dom.getFrameContentDocument;
2414
+
2415
+
2416
+ /**
2417
+ * Cross browser function for getting the window of a frame or iframe.
2418
+ * @param {HTMLIFrameElement|HTMLFrameElement} frame Frame element.
2419
+ * @return {Window} The window associated with the given frame.
2420
+ */
2421
+ goog.dom.DomHelper.prototype.getFrameContentWindow =
2422
+ goog.dom.getFrameContentWindow;
2423
+
2424
+
2425
+ /**
2426
+ * Cross browser function for setting the text content of an element.
2427
+ * @param {Element} element The element to change the text content of.
2428
+ * @param {string} text The string that should replace the current element
2429
+ * content with.
2430
+ */
2431
+ goog.dom.DomHelper.prototype.setTextContent = goog.dom.setTextContent;
2432
+
2433
+
2434
+ /**
2435
+ * Finds the first descendant node that matches the filter function. This does
2436
+ * a depth first search.
2437
+ * @param {Node} root The root of the tree to search.
2438
+ * @param {function(Node) : boolean} p The filter function.
2439
+ * @return {(Node, undefined)} The found node or undefined if none is found.
2440
+ */
2441
+ goog.dom.DomHelper.prototype.findNode = goog.dom.findNode;
2442
+
2443
+
2444
+ /**
2445
+ * Finds all the descendant nodes that matches the filter function. This does a
2446
+ * depth first search.
2447
+ * @param {Node} root The root of the tree to search.
2448
+ * @param {function(Node) : boolean} p The filter function.
2449
+ * @return {Array.<Node>} The found nodes or an empty array if none are found.
2450
+ */
2451
+ goog.dom.DomHelper.prototype.findNodes = goog.dom.findNodes;
2452
+
2453
+
2454
+ /**
2455
+ * Returns the text contents of the current node, without markup. New lines are
2456
+ * stripped and whitespace is collapsed, such that each character would be
2457
+ * visible.
2458
+ *
2459
+ * In browsers that support it, innerText is used. Other browsers attempt to
2460
+ * simulate it via node traversal. Line breaks are canonicalized in IE.
2461
+ *
2462
+ * @param {Node} node The node from which we are getting content.
2463
+ * @return {string} The text content.
2464
+ */
2465
+ goog.dom.DomHelper.prototype.getTextContent = goog.dom.getTextContent;
2466
+
2467
+
2468
+ /**
2469
+ * Returns the text length of the text contained in a node, without markup. This
2470
+ * is equivalent to the selection length if the node was selected, or the number
2471
+ * of cursor movements to traverse the node. Images & BRs take one space. New
2472
+ * lines are ignored.
2473
+ *
2474
+ * @param {Node} node The node whose text content length is being calculated.
2475
+ * @return {number} The length of {@code node}'s text content.
2476
+ */
2477
+ goog.dom.DomHelper.prototype.getNodeTextLength = goog.dom.getNodeTextLength;
2478
+
2479
+
2480
+ /**
2481
+ * Returns the text offset of a node relative to one of its ancestors. The text
2482
+ * length is the same as the length calculated by
2483
+ * {@code goog.dom.getNodeTextLength}.
2484
+ *
2485
+ * @param {Node} node The node whose offset is being calculated.
2486
+ * @param {Node=} opt_offsetParent Defaults to the node's owner document's body.
2487
+ * @return {number} The text offset.
2488
+ */
2489
+ goog.dom.DomHelper.prototype.getNodeTextOffset = goog.dom.getNodeTextOffset;
2490
+
2491
+
2492
+ /**
2493
+ * Walks up the DOM hierarchy returning the first ancestor that has the passed
2494
+ * tag name and/or class name. If the passed element matches the specified
2495
+ * criteria, the element itself is returned.
2496
+ * @param {Node} element The DOM node to start with.
2497
+ * @param {?string=} opt_tag The tag name to match (or null/undefined to match
2498
+ * any node regardless of tag name). Must be uppercase (goog.dom.TagName).
2499
+ * @param {?string=} opt_class The class name to match (or null/undefined to
2500
+ * match any node regardless of class name).
2501
+ * @return {Node} The first ancestor that matches the passed criteria, or
2502
+ * null if none match.
2503
+ */
2504
+ goog.dom.DomHelper.prototype.getAncestorByTagNameAndClass =
2505
+ goog.dom.getAncestorByTagNameAndClass;
2506
+
2507
+
2508
+ /**
2509
+ * Walks up the DOM hierarchy returning the first ancestor that has the passed
2510
+ * class name. If the passed element matches the specified criteria, the
2511
+ * element itself is returned.
2512
+ * @param {Node} element The DOM node to start with.
2513
+ * @param {?string=} opt_class The class name to match (or null/undefined to
2514
+ * match any node regardless of class name).
2515
+ * @return {Node} The first ancestor that matches the passed criteria, or
2516
+ * null if none match.
2517
+ */
2518
+ goog.dom.DomHelper.prototype.getAncestorByClass =
2519
+ goog.dom.getAncestorByClass;
2520
+
2521
+
2522
+ /**
2523
+ * Walks up the DOM hierarchy returning the first ancestor that passes the
2524
+ * matcher function.
2525
+ * @param {Node} element The DOM node to start with.
2526
+ * @param {function(Node) : boolean} matcher A function that returns true if the
2527
+ * passed node matches the desired criteria.
2528
+ * @param {boolean=} opt_includeNode If true, the node itself is included in
2529
+ * the search (the first call to the matcher will pass startElement as
2530
+ * the node to test).
2531
+ * @param {number=} opt_maxSearchSteps Maximum number of levels to search up the
2532
+ * dom.
2533
+ * @return {Node} DOM node that matched the matcher, or null if there was
2534
+ * no match.
2535
+ */
2536
+ goog.dom.DomHelper.prototype.getAncestor = goog.dom.getAncestor;
2537
+ ;
2538
+ FI"
2539
+ F"%b1e7a396ba0ab89f2859bcdc0af66f56