shoesgem 0.1424.0 → 0.1426.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (320) hide show
  1. data/README.rdoc +3 -0
  2. data/shoes/CHANGELOG.txt +21 -21
  3. data/shoes/COPYING.txt +30 -30
  4. data/shoes/VERSION.txt +1 -1
  5. data/shoes/freetype6.dll +0 -0
  6. data/shoes/lib/shoes/cache.rb +54 -54
  7. data/shoes/lib/shoes/data.rb +39 -39
  8. data/shoes/lib/shoes/help.rb +421 -421
  9. data/shoes/lib/shoes/image.rb +25 -25
  10. data/shoes/lib/shoes/inspect.rb +128 -128
  11. data/shoes/lib/shoes/log.rb +48 -48
  12. data/shoes/lib/shoes/minitar.rb +986 -986
  13. data/shoes/lib/shoes/override.rb +38 -38
  14. data/shoes/lib/shoes/pack.rb +503 -503
  15. data/shoes/lib/shoes/search.rb +46 -46
  16. data/shoes/lib/shoes/setup.rb +329 -329
  17. data/shoes/lib/shoes/shy.rb +131 -131
  18. data/shoes/lib/shoes/shybuilder.rb +44 -44
  19. data/shoes/lib/shoes.rb +522 -522
  20. data/shoes/libcairo-2.dll +0 -0
  21. data/shoes/libeay32.dll +0 -0
  22. data/shoes/libexpat-1.dll +0 -0
  23. data/shoes/libfontconfig-1.dll +0 -0
  24. data/shoes/libgio-2.0-0.dll +0 -0
  25. data/shoes/libglib-2.0-0.dll +0 -0
  26. data/shoes/libgmodule-2.0-0.dll +0 -0
  27. data/shoes/libgobject-2.0-0.dll +0 -0
  28. data/shoes/libgthread-2.0-0.dll +0 -0
  29. data/shoes/libiconv2.dll +0 -0
  30. data/shoes/libjpeg-8.dll +0 -0
  31. data/shoes/libpango-1.0-0.dll +0 -0
  32. data/shoes/libpangocairo-1.0-0.dll +0 -0
  33. data/shoes/libpangoft2-1.0-0.dll +0 -0
  34. data/shoes/libpangowin32-1.0-0.dll +0 -0
  35. data/shoes/libpng14-14.dll +0 -0
  36. data/shoes/libportaudio-2.dll +0 -0
  37. data/shoes/libshoes.dll +0 -0
  38. data/shoes/libssl32.dll +0 -0
  39. data/shoes/libungif4.dll +0 -0
  40. data/shoes/msvcrt-ruby191.dll +0 -0
  41. data/shoes/nsis/base.nsi +644 -0
  42. data/shoes/nsis/installer-1.bmp +0 -0
  43. data/shoes/nsis/installer-2.bmp +0 -0
  44. data/shoes/nsis/setup.ico +0 -0
  45. data/shoes/nsis/shoes.exe.manifest +17 -0
  46. data/shoes/nsis/shoes.ico +0 -0
  47. data/shoes/nsis/shoes.nsi +644 -0
  48. data/shoes/nsis/stub-inject.c +59 -0
  49. data/shoes/nsis/stub.c +271 -0
  50. data/shoes/nsis/stub32.h +14 -0
  51. data/shoes/nsis/stub32.rc +16 -0
  52. data/shoes/readline5.dll +0 -0
  53. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/fast_xs.so +0 -0
  54. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/blankslate.rb +63 -63
  55. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/builder.rb +216 -216
  56. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/elements.rb +510 -510
  57. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/htmlinfo.rb +691 -691
  58. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/inspect.rb +103 -103
  59. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/modules.rb +40 -40
  60. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/parse.rb +38 -38
  61. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/tag.rb +202 -202
  62. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/tags.rb +164 -164
  63. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/traverse.rb +838 -838
  64. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot/xchar.rb +94 -94
  65. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot.rb +26 -26
  66. data/shoes/ruby/gems/1.9.1/gems/hpricot-0.8.1/lib/hpricot_scan.so +0 -0
  67. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/add/core.rb +135 -135
  68. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/add/rails.rb +58 -58
  69. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/common.rb +354 -354
  70. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/ext/generator.so +0 -0
  71. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/ext/parser.so +0 -0
  72. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/ext.rb +13 -13
  73. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json/version.rb +9 -9
  74. data/shoes/ruby/gems/1.9.1/gems/json-shoes-1.1.3/lib/json.rb +8 -8
  75. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/constants.rb +49 -49
  76. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/database.rb +721 -721
  77. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/driver/dl/api.rb +152 -152
  78. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/driver/dl/driver.rb +307 -307
  79. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/driver/native/driver.rb +219 -219
  80. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/errors.rb +68 -68
  81. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/pragmas.rb +271 -271
  82. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/resultset.rb +180 -180
  83. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/statement.rb +231 -231
  84. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/translator.rb +109 -109
  85. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/value.rb +57 -57
  86. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3/version.rb +16 -16
  87. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3.rb +1 -1
  88. data/shoes/ruby/gems/1.9.1/gems/sqlite3-ruby-1.2.5-x86-mswin32/lib/sqlite3_api.so +0 -0
  89. data/shoes/ruby/gems/1.9.1/specifications/hpricot-0.8.1.gemspec +32 -32
  90. data/shoes/ruby/gems/1.9.1/specifications/json-shoes-1.1.3.gemspec +34 -34
  91. data/shoes/ruby/gems/1.9.1/specifications/sqlite3-ruby-1.2.5-x86-mswin32.gemspec +46 -46
  92. data/shoes/ruby/lib/ftsearch/analysis/analyzer.rb +16 -16
  93. data/shoes/ruby/lib/ftsearch/analysis/simple_identifier_analyzer.rb +23 -23
  94. data/shoes/ruby/lib/ftsearch/analysis/whitespace_analyzer.rb +22 -22
  95. data/shoes/ruby/lib/ftsearch/document_map_reader.rb +106 -106
  96. data/shoes/ruby/lib/ftsearch/document_map_writer.rb +46 -46
  97. data/shoes/ruby/lib/ftsearch/field_infos.rb +46 -46
  98. data/shoes/ruby/lib/ftsearch/fragment_writer.rb +114 -114
  99. data/shoes/ruby/lib/ftsearch/fulltext_reader.rb +52 -52
  100. data/shoes/ruby/lib/ftsearch/fulltext_writer.rb +75 -75
  101. data/shoes/ruby/lib/ftsearch/suffix_array_reader.rb +275 -275
  102. data/shoes/ruby/lib/ftsearch/suffix_array_writer.rb +99 -99
  103. data/shoes/ruby/lib/ftsearch/util.rb +21 -21
  104. data/shoes/ruby/lib/i386-mingw32/bigdecimal.so +0 -0
  105. data/shoes/ruby/lib/i386-mingw32/binject.so +0 -0
  106. data/shoes/ruby/lib/i386-mingw32/bloops.so +0 -0
  107. data/shoes/ruby/lib/i386-mingw32/continuation.so +0 -0
  108. data/shoes/ruby/lib/i386-mingw32/coverage.so +0 -0
  109. data/shoes/ruby/lib/i386-mingw32/curses.so +0 -0
  110. data/shoes/ruby/lib/i386-mingw32/digest/bubblebabble.so +0 -0
  111. data/shoes/ruby/lib/i386-mingw32/digest/md5.so +0 -0
  112. data/shoes/ruby/lib/i386-mingw32/digest/rmd160.so +0 -0
  113. data/shoes/ruby/lib/i386-mingw32/digest/sha1.so +0 -0
  114. data/shoes/ruby/lib/i386-mingw32/digest/sha2.so +0 -0
  115. data/shoes/ruby/lib/i386-mingw32/digest.so +0 -0
  116. data/shoes/ruby/lib/i386-mingw32/dl.so +0 -0
  117. data/shoes/ruby/lib/i386-mingw32/enc/big5.so +0 -0
  118. data/shoes/ruby/lib/i386-mingw32/enc/cp949.so +0 -0
  119. data/shoes/ruby/lib/i386-mingw32/enc/emacs_mule.so +0 -0
  120. data/shoes/ruby/lib/i386-mingw32/enc/encdb.so +0 -0
  121. data/shoes/ruby/lib/i386-mingw32/enc/euc_jp.so +0 -0
  122. data/shoes/ruby/lib/i386-mingw32/enc/euc_kr.so +0 -0
  123. data/shoes/ruby/lib/i386-mingw32/enc/euc_tw.so +0 -0
  124. data/shoes/ruby/lib/i386-mingw32/enc/gb18030.so +0 -0
  125. data/shoes/ruby/lib/i386-mingw32/enc/gb2312.so +0 -0
  126. data/shoes/ruby/lib/i386-mingw32/enc/gbk.so +0 -0
  127. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_1.so +0 -0
  128. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_10.so +0 -0
  129. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_11.so +0 -0
  130. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_13.so +0 -0
  131. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_14.so +0 -0
  132. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_15.so +0 -0
  133. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_16.so +0 -0
  134. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_2.so +0 -0
  135. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_3.so +0 -0
  136. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_4.so +0 -0
  137. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_5.so +0 -0
  138. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_6.so +0 -0
  139. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_7.so +0 -0
  140. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_8.so +0 -0
  141. data/shoes/ruby/lib/i386-mingw32/enc/iso_8859_9.so +0 -0
  142. data/shoes/ruby/lib/i386-mingw32/enc/koi8_r.so +0 -0
  143. data/shoes/ruby/lib/i386-mingw32/enc/koi8_u.so +0 -0
  144. data/shoes/ruby/lib/i386-mingw32/enc/shift_jis.so +0 -0
  145. data/shoes/ruby/lib/i386-mingw32/enc/trans/big5.so +0 -0
  146. data/shoes/ruby/lib/i386-mingw32/enc/trans/chinese.so +0 -0
  147. data/shoes/ruby/lib/i386-mingw32/enc/trans/escape.so +0 -0
  148. data/shoes/ruby/lib/i386-mingw32/enc/trans/gb18030.so +0 -0
  149. data/shoes/ruby/lib/i386-mingw32/enc/trans/gbk.so +0 -0
  150. data/shoes/ruby/lib/i386-mingw32/enc/trans/iso2022.so +0 -0
  151. data/shoes/ruby/lib/i386-mingw32/enc/trans/japanese.so +0 -0
  152. data/shoes/ruby/lib/i386-mingw32/enc/trans/japanese_euc.so +0 -0
  153. data/shoes/ruby/lib/i386-mingw32/enc/trans/japanese_sjis.so +0 -0
  154. data/shoes/ruby/lib/i386-mingw32/enc/trans/korean.so +0 -0
  155. data/shoes/ruby/lib/i386-mingw32/enc/trans/single_byte.so +0 -0
  156. data/shoes/ruby/lib/i386-mingw32/enc/trans/transdb.so +0 -0
  157. data/shoes/ruby/lib/i386-mingw32/enc/trans/utf_16_32.so +0 -0
  158. data/shoes/ruby/lib/i386-mingw32/enc/utf_16be.so +0 -0
  159. data/shoes/ruby/lib/i386-mingw32/enc/utf_16le.so +0 -0
  160. data/shoes/ruby/lib/i386-mingw32/enc/utf_32be.so +0 -0
  161. data/shoes/ruby/lib/i386-mingw32/enc/utf_32le.so +0 -0
  162. data/shoes/ruby/lib/i386-mingw32/enc/windows_1251.so +0 -0
  163. data/shoes/ruby/lib/i386-mingw32/etc.so +0 -0
  164. data/shoes/ruby/lib/i386-mingw32/fcntl.so +0 -0
  165. data/shoes/ruby/lib/i386-mingw32/fiber.so +0 -0
  166. data/shoes/ruby/lib/i386-mingw32/ftsearchrt.so +0 -0
  167. data/shoes/ruby/lib/i386-mingw32/gdbm.so +0 -0
  168. data/shoes/ruby/lib/i386-mingw32/iconv.so +0 -0
  169. data/shoes/ruby/lib/i386-mingw32/io/wait.so +0 -0
  170. data/shoes/ruby/lib/i386-mingw32/json/ext/generator.so +0 -0
  171. data/shoes/ruby/lib/i386-mingw32/json/ext/parser.so +0 -0
  172. data/shoes/ruby/lib/i386-mingw32/mathn/complex.so +0 -0
  173. data/shoes/ruby/lib/i386-mingw32/mathn/rational.so +0 -0
  174. data/shoes/ruby/lib/i386-mingw32/nkf.so +0 -0
  175. data/shoes/ruby/lib/i386-mingw32/openssl.so +0 -0
  176. data/shoes/ruby/lib/i386-mingw32/racc/cparse.so +0 -0
  177. data/shoes/ruby/lib/i386-mingw32/rbconfig.rb +2 -2
  178. data/shoes/ruby/lib/i386-mingw32/readline.so +0 -0
  179. data/shoes/ruby/lib/i386-mingw32/ripper.so +0 -0
  180. data/shoes/ruby/lib/i386-mingw32/sdbm.so +0 -0
  181. data/shoes/ruby/lib/i386-mingw32/socket.so +0 -0
  182. data/shoes/ruby/lib/i386-mingw32/stringio.so +0 -0
  183. data/shoes/ruby/lib/i386-mingw32/strscan.so +0 -0
  184. data/shoes/ruby/lib/i386-mingw32/syck.so +0 -0
  185. data/shoes/ruby/lib/i386-mingw32/win32ole.so +0 -0
  186. data/shoes/ruby/lib/i386-mingw32/zlib.so +0 -0
  187. data/shoes/ruby/lib/rbconfig/datadir.rb +24 -24
  188. data/shoes/ruby/lib/rubygems/builder.rb +88 -88
  189. data/shoes/ruby/lib/rubygems/command.rb +406 -406
  190. data/shoes/ruby/lib/rubygems/command_manager.rb +146 -146
  191. data/shoes/ruby/lib/rubygems/commands/build_command.rb +53 -53
  192. data/shoes/ruby/lib/rubygems/commands/cert_command.rb +86 -86
  193. data/shoes/ruby/lib/rubygems/commands/check_command.rb +75 -75
  194. data/shoes/ruby/lib/rubygems/commands/cleanup_command.rb +91 -91
  195. data/shoes/ruby/lib/rubygems/commands/contents_command.rb +74 -74
  196. data/shoes/ruby/lib/rubygems/commands/dependency_command.rb +188 -188
  197. data/shoes/ruby/lib/rubygems/commands/environment_command.rb +128 -128
  198. data/shoes/ruby/lib/rubygems/commands/fetch_command.rb +62 -62
  199. data/shoes/ruby/lib/rubygems/commands/generate_index_command.rb +57 -57
  200. data/shoes/ruby/lib/rubygems/commands/help_command.rb +172 -172
  201. data/shoes/ruby/lib/rubygems/commands/install_command.rb +148 -148
  202. data/shoes/ruby/lib/rubygems/commands/list_command.rb +35 -35
  203. data/shoes/ruby/lib/rubygems/commands/lock_command.rb +110 -110
  204. data/shoes/ruby/lib/rubygems/commands/mirror_command.rb +111 -111
  205. data/shoes/ruby/lib/rubygems/commands/outdated_command.rb +33 -33
  206. data/shoes/ruby/lib/rubygems/commands/pristine_command.rb +93 -93
  207. data/shoes/ruby/lib/rubygems/commands/query_command.rb +233 -233
  208. data/shoes/ruby/lib/rubygems/commands/rdoc_command.rb +82 -82
  209. data/shoes/ruby/lib/rubygems/commands/search_command.rb +37 -37
  210. data/shoes/ruby/lib/rubygems/commands/server_command.rb +48 -48
  211. data/shoes/ruby/lib/rubygems/commands/sources_command.rb +152 -152
  212. data/shoes/ruby/lib/rubygems/commands/specification_command.rb +77 -77
  213. data/shoes/ruby/lib/rubygems/commands/stale_command.rb +27 -27
  214. data/shoes/ruby/lib/rubygems/commands/uninstall_command.rb +73 -73
  215. data/shoes/ruby/lib/rubygems/commands/unpack_command.rb +95 -95
  216. data/shoes/ruby/lib/rubygems/commands/update_command.rb +181 -181
  217. data/shoes/ruby/lib/rubygems/commands/which_command.rb +87 -87
  218. data/shoes/ruby/lib/rubygems/config_file.rb +266 -266
  219. data/shoes/ruby/lib/rubygems/custom_require.rb +46 -46
  220. data/shoes/ruby/lib/rubygems/defaults.rb +89 -89
  221. data/shoes/ruby/lib/rubygems/dependency.rb +119 -119
  222. data/shoes/ruby/lib/rubygems/dependency_installer.rb +258 -258
  223. data/shoes/ruby/lib/rubygems/dependency_list.rb +165 -165
  224. data/shoes/ruby/lib/rubygems/digest/digest_adapter.rb +39 -39
  225. data/shoes/ruby/lib/rubygems/digest/md5.rb +23 -23
  226. data/shoes/ruby/lib/rubygems/digest/sha1.rb +16 -16
  227. data/shoes/ruby/lib/rubygems/digest/sha2.rb +17 -17
  228. data/shoes/ruby/lib/rubygems/doc_manager.rb +214 -214
  229. data/shoes/ruby/lib/rubygems/exceptions.rb +84 -84
  230. data/shoes/ruby/lib/rubygems/ext/builder.rb +56 -56
  231. data/shoes/ruby/lib/rubygems/ext/configure_builder.rb +24 -24
  232. data/shoes/ruby/lib/rubygems/ext/ext_conf_builder.rb +23 -23
  233. data/shoes/ruby/lib/rubygems/ext/rake_builder.rb +27 -27
  234. data/shoes/ruby/lib/rubygems/ext.rb +18 -18
  235. data/shoes/ruby/lib/rubygems/format.rb +87 -87
  236. data/shoes/ruby/lib/rubygems/gem_openssl.rb +83 -83
  237. data/shoes/ruby/lib/rubygems/gem_path_searcher.rb +100 -100
  238. data/shoes/ruby/lib/rubygems/gem_runner.rb +58 -58
  239. data/shoes/ruby/lib/rubygems/indexer.rb +370 -370
  240. data/shoes/ruby/lib/rubygems/install_update_options.rb +113 -113
  241. data/shoes/ruby/lib/rubygems/installer.rb +578 -578
  242. data/shoes/ruby/lib/rubygems/local_remote_options.rb +134 -134
  243. data/shoes/ruby/lib/rubygems/old_format.rb +148 -148
  244. data/shoes/ruby/lib/rubygems/package/f_sync_dir.rb +24 -24
  245. data/shoes/ruby/lib/rubygems/package/tar_header.rb +245 -245
  246. data/shoes/ruby/lib/rubygems/package/tar_input.rb +219 -219
  247. data/shoes/ruby/lib/rubygems/package/tar_output.rb +143 -143
  248. data/shoes/ruby/lib/rubygems/package/tar_reader/entry.rb +99 -99
  249. data/shoes/ruby/lib/rubygems/package/tar_reader.rb +86 -86
  250. data/shoes/ruby/lib/rubygems/package/tar_writer.rb +180 -180
  251. data/shoes/ruby/lib/rubygems/package.rb +95 -95
  252. data/shoes/ruby/lib/rubygems/platform.rb +178 -178
  253. data/shoes/ruby/lib/rubygems/remote_fetcher.rb +344 -344
  254. data/shoes/ruby/lib/rubygems/require_paths_builder.rb +14 -14
  255. data/shoes/ruby/lib/rubygems/requirement.rb +163 -163
  256. data/shoes/ruby/lib/rubygems/rubygems_version.rb +6 -6
  257. data/shoes/ruby/lib/rubygems/security.rb +786 -786
  258. data/shoes/ruby/lib/rubygems/server.rb +629 -629
  259. data/shoes/ruby/lib/rubygems/source_index.rb +559 -559
  260. data/shoes/ruby/lib/rubygems/source_info_cache.rb +393 -393
  261. data/shoes/ruby/lib/rubygems/source_info_cache_entry.rb +56 -56
  262. data/shoes/ruby/lib/rubygems/spec_fetcher.rb +249 -249
  263. data/shoes/ruby/lib/rubygems/specification.rb +1262 -1262
  264. data/shoes/ruby/lib/rubygems/test_utilities.rb +131 -131
  265. data/shoes/ruby/lib/rubygems/timer.rb +25 -25
  266. data/shoes/ruby/lib/rubygems/uninstaller.rb +242 -242
  267. data/shoes/ruby/lib/rubygems/user_interaction.rb +360 -360
  268. data/shoes/ruby/lib/rubygems/validator.rb +208 -208
  269. data/shoes/ruby/lib/rubygems/version.rb +167 -167
  270. data/shoes/ruby/lib/rubygems/version_option.rb +48 -48
  271. data/shoes/ruby/lib/rubygems.rb +888 -888
  272. data/shoes/ruby/lib/ubygems.rb +10 -10
  273. data/shoes/samples/class-book.rb +43 -43
  274. data/shoes/samples/class-book.yaml +387 -387
  275. data/shoes/samples/expert-definr.rb +23 -23
  276. data/shoes/samples/expert-funnies.rb +51 -51
  277. data/shoes/samples/expert-irb.rb +112 -112
  278. data/shoes/samples/expert-minesweeper.rb +267 -267
  279. data/shoes/samples/expert-othello.rb +319 -319
  280. data/shoes/samples/expert-pong.rb +62 -62
  281. data/shoes/samples/expert-tankspank.rb +385 -385
  282. data/shoes/samples/good-arc.rb +37 -37
  283. data/shoes/samples/good-clock.rb +51 -51
  284. data/shoes/samples/good-follow.rb +26 -26
  285. data/shoes/samples/good-reminder.rb +174 -174
  286. data/shoes/samples/good-vjot.rb +56 -56
  287. data/shoes/samples/simple-accordion.rb +75 -75
  288. data/shoes/samples/simple-anim-shapes.rb +17 -17
  289. data/shoes/samples/simple-anim-text.rb +13 -13
  290. data/shoes/samples/simple-arc.rb +23 -23
  291. data/shoes/samples/simple-bounce.rb +24 -24
  292. data/shoes/samples/simple-calc.rb +70 -70
  293. data/shoes/samples/simple-control-sizes.rb +24 -24
  294. data/shoes/samples/simple-curve.rb +26 -26
  295. data/shoes/samples/simple-dialogs.rb +29 -29
  296. data/shoes/samples/simple-downloader.rb +27 -27
  297. data/shoes/samples/simple-draw.rb +13 -13
  298. data/shoes/samples/simple-editor.rb +28 -28
  299. data/shoes/samples/simple-form.rb +28 -28
  300. data/shoes/samples/simple-mask.rb +21 -21
  301. data/shoes/samples/simple-menu.rb +31 -31
  302. data/shoes/samples/simple-menu1.rb +35 -35
  303. data/shoes/samples/simple-rubygems.rb +29 -29
  304. data/shoes/samples/simple-slide.rb +45 -45
  305. data/shoes/samples/simple-sphere.rb +28 -28
  306. data/shoes/samples/simple-timer.rb +13 -13
  307. data/shoes/samples/simple-video.rb +13 -13
  308. data/shoes/shoes.exe +0 -0
  309. data/shoes/sqlite3.dll +0 -0
  310. data/shoes/static/code_highlighter.js +188 -188
  311. data/shoes/static/code_highlighter_ruby.js +26 -26
  312. data/shoes/static/manual-en.txt +2783 -2783
  313. data/shoes/static/manual-ja.txt +2780 -2780
  314. data/shoes/static/manual.css +167 -167
  315. data/shoes/static/stubs/blank.run +375 -375
  316. data/shoes/static/stubs/sh-install +48 -48
  317. data/shoes/zlib.dll +0 -0
  318. data/shoes/zlib1.dll +0 -0
  319. metadata +15 -5
  320. data/shoes/static/Thumbs.db +0 -0
@@ -1,838 +1,838 @@
1
- require 'hpricot/elements'
2
- require 'uri'
3
-
4
- module Hpricot
5
- module Traverse
6
- # Is this object the enclosing HTML or XML document?
7
- def doc?() Doc::Trav === self end
8
- # Is this object an HTML or XML element?
9
- def elem?() Elem::Trav === self end
10
- # Is this object an HTML text node?
11
- def text?() Text::Trav === self end
12
- # Is this object an XML declaration?
13
- def xmldecl?() XMLDecl::Trav === self end
14
- # Is this object a doctype tag?
15
- def doctype?() DocType::Trav === self end
16
- # Is this object an XML processing instruction?
17
- def procins?() ProcIns::Trav === self end
18
- # Is this object a comment?
19
- def comment?() Comment::Trav === self end
20
- # Is this object a stranded end tag?
21
- def bogusetag?() BogusETag::Trav === self end
22
-
23
- # Parses an HTML string, making an HTML fragment based on
24
- # the options used to create the container document.
25
- def make(input = nil, &blk)
26
- if parent and parent.respond_to? :make
27
- parent.make(input, &blk)
28
- else
29
- Hpricot.make(input, &blk).children
30
- end
31
- end
32
-
33
- # Builds an HTML string from this node and its contents.
34
- # If you need to write to a stream, try calling <tt>output(io)</tt>
35
- # as a method on this object.
36
- def to_html
37
- output("")
38
- end
39
- alias_method :to_s, :to_html
40
-
41
- # Attempts to preserve the original HTML of the document, only
42
- # outputing new tags for elements which have changed.
43
- def to_original_html
44
- output("", :preserve => true)
45
- end
46
-
47
- def index(name)
48
- i = 0
49
- return i if name == "*"
50
- children.each do |x|
51
- return i if (x.respond_to?(:name) and name == x.name) or
52
- (x.text? and name == "text()")
53
- i += 1
54
- end if children
55
- -1
56
- end
57
-
58
- # Puts together an array of neighboring nodes based on their proximity
59
- # to this node. So, for example, to get the next node, you could use
60
- # <tt>nodes_at(1). Or, to get the previous node, use <tt>nodes_at(1)</tt>.
61
- #
62
- # This method also accepts ranges and sets of numbers.
63
- #
64
- # ele.nodes_at(-3..-1, 1..3) # gets three nodes before and three after
65
- # ele.nodes_at(1, 5, 7) # gets three nodes at offsets below the current node
66
- # ele.nodes_at(0, 5..6) # the current node and two others
67
- def nodes_at(*pos)
68
- sib = parent.children
69
- i, si = 0, sib.index(self)
70
- pos.map! do |r|
71
- if r.is_a?(Range) and r.begin.is_a?(String)
72
- r = Range.new(parent.index(r.begin)-si, parent.index(r.end)-si, r.exclude_end?)
73
- end
74
- r
75
- end
76
- p pos
77
- Elements[*
78
- sib.select do |x|
79
- sel =
80
- case i - si when *pos
81
- true
82
- end
83
- i += 1
84
- sel
85
- end
86
- ]
87
- end
88
-
89
- # Returns the node neighboring this node to the south: just below it.
90
- # This method includes text nodes and comments and such.
91
- def next
92
- sib = parent.children
93
- sib[sib.index(self) + 1] if parent
94
- end
95
- alias_method :next_node, :next
96
-
97
- # Returns to node neighboring this node to the north: just above it.
98
- # This method includes text nodes and comments and such.
99
- def previous
100
- sib = parent.children
101
- x = sib.index(self) - 1
102
- sib[x] if sib and x >= 0
103
- end
104
- alias_method :previous_node, :previous
105
-
106
- # Find all preceding nodes.
107
- def preceding
108
- sibs = parent.children
109
- si = sibs.index(self)
110
- return Elements[*sibs[0...si]]
111
- end
112
-
113
- # Find all nodes which follow the current one.
114
- def following
115
- sibs = parent.children
116
- si = sibs.index(self) + 1
117
- return Elements[*sibs[si...sibs.length]]
118
- end
119
-
120
- # Adds elements immediately after this element, contained in the +html+ string.
121
- def after(html = nil, &blk)
122
- parent.insert_after(make(html, &blk), self)
123
- end
124
-
125
- # Adds elements immediately before this element, contained in the +html+ string.
126
- def before(html = nil, &blk)
127
- parent.insert_before(make(html, &blk), self)
128
- end
129
-
130
-
131
- # Replace this element and its contents with the nodes contained
132
- # in the +html+ string.
133
- def swap(html = nil, &blk)
134
- parent.altered!
135
- parent.replace_child(self, make(html, &blk))
136
- end
137
-
138
- def get_subnode(*indexes)
139
- n = self
140
- indexes.each {|index|
141
- n = n.get_subnode_internal(index)
142
- }
143
- n
144
- end
145
-
146
- # Builds a string from the text contained in this node. All
147
- # HTML elements are removed.
148
- def to_plain_text
149
- if respond_to?(:children) and children
150
- children.map { |x| x.to_plain_text }.join.strip.gsub(/\n{2,}/, "\n\n")
151
- else
152
- ""
153
- end
154
- end
155
-
156
- # Builds a string from the text contained in this node. All
157
- # HTML elements are removed.
158
- def inner_text
159
- if respond_to?(:children) and children
160
- children.map { |x| x.inner_text }.join
161
- else
162
- ""
163
- end
164
- end
165
- alias_method :innerText, :inner_text
166
-
167
- # Builds an HTML string from the contents of this node.
168
- def html(inner = nil, &blk)
169
- if inner or blk
170
- altered!
171
- case inner
172
- when Array
173
- self.children = inner
174
- else
175
- self.children = make(inner, &blk)
176
- end
177
- reparent self.children
178
- else
179
- if respond_to?(:children) and children
180
- children.map { |x| x.output("") }.join
181
- else
182
- ""
183
- end
184
- end
185
- end
186
- alias_method :inner_html, :html
187
- alias_method :innerHTML, :inner_html
188
-
189
- # Inserts new contents into the current node, based on
190
- # the HTML contained in string +inner+.
191
- def inner_html=(inner)
192
- html(inner || [])
193
- end
194
- alias_method :innerHTML=, :inner_html=
195
-
196
- def reparent(nodes)
197
- altered!
198
- [*nodes].each { |e| e.parent = self }
199
- end
200
- private :reparent
201
-
202
- def clean_path(path)
203
- path.gsub(/^\s+|\s+$/, '')
204
- end
205
-
206
- # Builds a unique XPath string for this node, from the
207
- # root of the document containing it.
208
- def xpath
209
- if elem? and has_attribute? 'id'
210
- "//#{self.name}[@id='#{get_attribute('id')}']"
211
- else
212
- sim, id = 0, 0, 0
213
- parent.children.each do |e|
214
- id = sim if e == self
215
- sim += 1 if e.pathname == self.pathname
216
- end if parent.children
217
- p = File.join(parent.xpath, self.pathname)
218
- p += "[#{id+1}]" if sim >= 2
219
- p
220
- end
221
- end
222
-
223
- # Builds a unique CSS string for this node, from the
224
- # root of the document containing it.
225
- def css_path
226
- if elem? and has_attribute? 'id'
227
- "##{get_attribute('id')}"
228
- else
229
- sim, i, id = 0, 0, 0
230
- parent.children.each do |e|
231
- id = sim if e == self
232
- sim += 1 if e.pathname == self.pathname
233
- end if parent.children
234
- p = parent.css_path
235
- p = p ? "#{p} > #{self.pathname}" : self.pathname
236
- p += ":nth(#{id})" if sim >= 2
237
- p
238
- end
239
- end
240
-
241
- def node_position
242
- parent.children.index(self)
243
- end
244
-
245
- def position
246
- parent.children_of_type(self.pathname).index(self)
247
- end
248
-
249
- # Searches this node for all elements matching
250
- # the CSS or XPath +expr+. Returns an Elements array
251
- # containing the matching nodes. If +blk+ is given, it
252
- # is used to iterate through the matching set.
253
- def search(expr, &blk)
254
- if Range === expr
255
- return Elements.expand(at(expr.begin), at(expr.end), expr.exclude_end?)
256
- end
257
- last = nil
258
- nodes = [self]
259
- done = []
260
- expr = expr.to_s
261
- hist = []
262
- until expr.empty?
263
- expr = clean_path(expr)
264
- expr.gsub!(%r!^//!, '')
265
-
266
- case expr
267
- when %r!^/?\.\.!
268
- last = expr = $'
269
- nodes.map! { |node| node.parent }
270
- when %r!^[>/]\s*!
271
- last = expr = $'
272
- nodes = Elements[*nodes.map { |node| node.children if node.respond_to? :children }.flatten.compact]
273
- when %r!^\+!
274
- last = expr = $'
275
- nodes.map! do |node|
276
- siblings = node.parent.children
277
- siblings[siblings.index(node)+1]
278
- end
279
- nodes.compact!
280
- when %r!^~!
281
- last = expr = $'
282
- nodes.map! do |node|
283
- siblings = node.parent.children
284
- siblings[(siblings.index(node)+1)..-1]
285
- end
286
- nodes.flatten!
287
- when %r!^[|,]!
288
- last = expr = " #$'"
289
- nodes.shift if nodes.first == self
290
- done += nodes
291
- nodes = [self]
292
- else
293
- m = expr.match(%r!^([#.]?)([a-z0-9\\*_-]*)!i).to_a
294
- after = $'
295
- mt = after[%r!:[a-z0-9\\*_-]+!i, 0]
296
- oop = false
297
- if mt and not (mt == ":not" or Traverse.method_defined? "filter[#{mt}]")
298
- after = $'
299
- m[2] += mt
300
- expr = after
301
- end
302
- if m[1] == '#'
303
- oid = get_element_by_id(m[2])
304
- nodes = oid ? [oid] : []
305
- expr = after
306
- else
307
- m[2] = "*" if after =~ /^\(\)/ || m[2] == "" || m[1] == "."
308
- ret = []
309
- nodes.each do |node|
310
- case m[2]
311
- when '*'
312
- node.traverse_element { |n| ret << n }
313
- else
314
- if node.respond_to? :get_elements_by_tag_name
315
- ret += [*node.get_elements_by_tag_name(m[2])] - [*(node unless last)]
316
- end
317
- end
318
- end
319
- nodes = ret
320
- end
321
- last = nil
322
- end
323
-
324
- hist << expr
325
- break if hist[-1] == hist[-2]
326
- nodes, expr = Elements.filter(nodes, expr)
327
- end
328
- nodes = done + nodes.flatten.uniq
329
- if blk
330
- nodes.each(&blk)
331
- self
332
- else
333
- Elements[*nodes]
334
- end
335
- end
336
- alias_method :/, :search
337
-
338
- # Find the first matching node for the CSS or XPath
339
- # +expr+ string.
340
- def at(expr)
341
- search(expr).first
342
- end
343
- alias_method :%, :at
344
-
345
- # +traverse_element+ traverses elements in the tree.
346
- # It yields elements in depth first order.
347
- #
348
- # If _names_ are empty, it yields all elements.
349
- # If non-empty _names_ are given, it should be list of universal names.
350
- #
351
- # A nested element is yielded in depth first order as follows.
352
- #
353
- # t = Hpricot('<a id=0><b><a id=1 /></b><c id=2 /></a>')
354
- # t.traverse_element("a", "c") {|e| p e}
355
- # # =>
356
- # {elem <a id="0"> {elem <b> {emptyelem <a id="1">} </b>} {emptyelem <c id="2">} </a>}
357
- # {emptyelem <a id="1">}
358
- # {emptyelem <c id="2">}
359
- #
360
- # Universal names are specified as follows.
361
- #
362
- # t = Hpricot(<<'End')
363
- # <html>
364
- # <meta name="robots" content="index,nofollow">
365
- # <meta name="author" content="Who am I?">
366
- # </html>
367
- # End
368
- # t.traverse_element("{http://www.w3.org/1999/xhtml}meta") {|e| p e}
369
- # # =>
370
- # {emptyelem <{http://www.w3.org/1999/xhtml}meta name="robots" content="index,nofollow">}
371
- # {emptyelem <{http://www.w3.org/1999/xhtml}meta name="author" content="Who am I?">}
372
- #
373
- def traverse_element(*names, &block) # :yields: element
374
- if names.empty?
375
- traverse_all_element(&block)
376
- else
377
- name_set = {}
378
- names.each {|n| name_set[n] = true }
379
- traverse_some_element(name_set, &block)
380
- end
381
- nil
382
- end
383
-
384
- # Find children of a given +tag_name+.
385
- #
386
- # ele.children_of_type('p')
387
- # #=> [...array of paragraphs...]
388
- #
389
- def children_of_type(tag_name)
390
- if respond_to? :children
391
- children.find_all do |x|
392
- x.respond_to?(:pathname) && x.pathname == tag_name
393
- end
394
- end
395
- end
396
-
397
- end
398
-
399
- module Container::Trav
400
- # Return all children of this node which can contain other
401
- # nodes. This is a good way to get all HTML elements which
402
- # aren't text, comment, doctype or processing instruction nodes.
403
- def containers
404
- children.grep(Container::Trav)
405
- end
406
-
407
- # Returns the container node neighboring this node to the south: just below it.
408
- # By "container" node, I mean: this method does not find text nodes or comments or cdata or any of that.
409
- # See Hpricot::Traverse#next_node if you need to hunt out all kinds of nodes.
410
- def next_sibling
411
- sib = parent.containers
412
- sib[sib.index(self) + 1] if parent
413
- end
414
-
415
- # Returns the container node neighboring this node to the north: just above it.
416
- # By "container" node, I mean: this method does not find text nodes or comments or cdata or any of that.
417
- # See Hpricot::Traverse#previous_node if you need to hunt out all kinds of nodes.
418
- def previous_sibling
419
- sib = parent.containers
420
- x = sib.index(self) - 1
421
- sib[x] if sib and x >= 0
422
- end
423
-
424
- # Find all preceding sibling elements. Like the other "sibling" methods, this weeds
425
- # out text and comment nodes.
426
- def preceding_siblings()
427
- sibs = parent.containers
428
- si = sibs.index(self)
429
- return Elements[*sibs[0...si]]
430
- end
431
-
432
- # Find sibling elements which follow the current one. Like the other "sibling" methods, this weeds
433
- # out text and comment nodes.
434
- def following_siblings()
435
- sibs = parent.containers
436
- si = sibs.index(self) + 1
437
- return Elements[*sibs[si...sibs.length]]
438
- end
439
-
440
- # Puts together an array of neighboring sibling elements based on their proximity
441
- # to this element.
442
- #
443
- # This method accepts ranges and sets of numbers.
444
- #
445
- # ele.siblings_at(-3..-1, 1..3) # gets three elements before and three after
446
- # ele.siblings_at(1, 5, 7) # gets three elements at offsets below the current element
447
- # ele.siblings_at(0, 5..6) # the current element and two others
448
- #
449
- # Like the other "sibling" methods, this doesn't find text and comment nodes.
450
- # Use nodes_at to include those nodes.
451
- def siblings_at(*pos)
452
- sib = parent.containers
453
- i, si = 0, sib.index(self)
454
- Elements[*
455
- sib.select do |x|
456
- sel = case i - si when *pos
457
- true
458
- end
459
- i += 1
460
- sel
461
- end
462
- ]
463
- end
464
-
465
- # Replace +old+, a child of the current node, with +new+ node.
466
- def replace_child(old, new)
467
- reparent new
468
- children[children.index(old), 1] = [*new]
469
- end
470
-
471
- # Insert +nodes+, an array of HTML elements or a single element,
472
- # before the node +ele+, a child of the current node.
473
- def insert_before(nodes, ele)
474
- case nodes
475
- when Array
476
- nodes.each { |n| insert_before(n, ele) }
477
- else
478
- reparent nodes
479
- children[children.index(ele) || 0, 0] = nodes
480
- end
481
- end
482
-
483
- # Insert +nodes+, an array of HTML elements or a single element,
484
- # after the node +ele+, a child of the current node.
485
- def insert_after(nodes, ele)
486
- case nodes
487
- when Array
488
- nodes.reverse_each { |n| insert_after(n, ele) }
489
- else
490
- reparent nodes
491
- idx = children.index(ele)
492
- children[idx ? idx + 1 : children.length, 0] = nodes
493
- end
494
- end
495
-
496
- # +each_child+ iterates over each child.
497
- def each_child(&block) # :yields: child_node
498
- children.each(&block) if children
499
- nil
500
- end
501
-
502
- # +each_child_with_index+ iterates over each child.
503
- def each_child_with_index(&block) # :yields: child_node, index
504
- children.each_with_index(&block) if children
505
- nil
506
- end
507
-
508
- # +find_element+ searches an element which universal name is specified by
509
- # the arguments.
510
- # It returns nil if not found.
511
- def find_element(*names)
512
- traverse_element(*names) {|e| return e }
513
- nil
514
- end
515
-
516
- # Returns a list of CSS classes to which this element belongs.
517
- def classes
518
- get_attribute('class').to_s.strip.split(/\s+/)
519
- end
520
-
521
- def get_element_by_id(id)
522
- traverse_all_element do |ele|
523
- if ele.elem? and eid = ele.get_attribute('id')
524
- return ele if eid.to_s == id
525
- end
526
- end
527
- nil
528
- end
529
-
530
- def get_elements_by_tag_name(*a)
531
- list = Elements[]
532
- a.delete("*")
533
- traverse_element(*a.map { |tag| [tag, "{http://www.w3.org/1999/xhtml}#{tag}"] }.flatten) do |e|
534
- list << e if e.elem?
535
- end
536
- list
537
- end
538
-
539
- def each_hyperlink_attribute
540
- traverse_element(
541
- '{http://www.w3.org/1999/xhtml}a',
542
- '{http://www.w3.org/1999/xhtml}area',
543
- '{http://www.w3.org/1999/xhtml}link',
544
- '{http://www.w3.org/1999/xhtml}img',
545
- '{http://www.w3.org/1999/xhtml}object',
546
- '{http://www.w3.org/1999/xhtml}q',
547
- '{http://www.w3.org/1999/xhtml}blockquote',
548
- '{http://www.w3.org/1999/xhtml}ins',
549
- '{http://www.w3.org/1999/xhtml}del',
550
- '{http://www.w3.org/1999/xhtml}form',
551
- '{http://www.w3.org/1999/xhtml}input',
552
- '{http://www.w3.org/1999/xhtml}head',
553
- '{http://www.w3.org/1999/xhtml}base',
554
- '{http://www.w3.org/1999/xhtml}script') {|elem|
555
- case elem.name
556
- when %r{\{http://www.w3.org/1999/xhtml\}(?:base|a|area|link)\z}i
557
- attrs = ['href']
558
- when %r{\{http://www.w3.org/1999/xhtml\}(?:img)\z}i
559
- attrs = ['src', 'longdesc', 'usemap']
560
- when %r{\{http://www.w3.org/1999/xhtml\}(?:object)\z}i
561
- attrs = ['classid', 'codebase', 'data', 'usemap']
562
- when %r{\{http://www.w3.org/1999/xhtml\}(?:q|blockquote|ins|del)\z}i
563
- attrs = ['cite']
564
- when %r{\{http://www.w3.org/1999/xhtml\}(?:form)\z}i
565
- attrs = ['action']
566
- when %r{\{http://www.w3.org/1999/xhtml\}(?:input)\z}i
567
- attrs = ['src', 'usemap']
568
- when %r{\{http://www.w3.org/1999/xhtml\}(?:head)\z}i
569
- attrs = ['profile']
570
- when %r{\{http://www.w3.org/1999/xhtml\}(?:script)\z}i
571
- attrs = ['src', 'for']
572
- end
573
- attrs.each {|attr|
574
- if hyperlink = elem.get_attribute(attr)
575
- yield elem, attr, hyperlink
576
- end
577
- }
578
- }
579
- end
580
- private :each_hyperlink_attribute
581
-
582
- # +each_hyperlink_uri+ traverses hyperlinks such as HTML href attribute
583
- # of A element.
584
- #
585
- # It yields Hpricot::Text and URI for each hyperlink.
586
- #
587
- # The URI objects are created with a base URI which is given by
588
- # HTML BASE element or the argument ((|base_uri|)).
589
- # +each_hyperlink_uri+ doesn't yields href of the BASE element.
590
- def each_hyperlink_uri(base_uri=nil) # :yields: hyperlink, uri
591
- base_uri = URI.parse(base_uri) if String === base_uri
592
- links = []
593
- each_hyperlink_attribute {|elem, attr, hyperlink|
594
- if %r{\{http://www.w3.org/1999/xhtml\}(?:base)\z}i =~ elem.name
595
- base_uri = URI.parse(hyperlink.to_s)
596
- else
597
- links << hyperlink
598
- end
599
- }
600
- if base_uri
601
- links.each {|hyperlink| yield hyperlink, base_uri + hyperlink.to_s }
602
- else
603
- links.each {|hyperlink| yield hyperlink, URI.parse(hyperlink.to_s) }
604
- end
605
- end
606
-
607
- # +each_hyperlink+ traverses hyperlinks such as HTML href attribute
608
- # of A element.
609
- #
610
- # It yields Hpricot::Text.
611
- #
612
- # Note that +each_hyperlink+ yields HTML href attribute of BASE element.
613
- def each_hyperlink # :yields: text
614
- links = []
615
- each_hyperlink_attribute {|elem, attr, hyperlink|
616
- yield hyperlink
617
- }
618
- end
619
-
620
- # +each_uri+ traverses hyperlinks such as HTML href attribute
621
- # of A element.
622
- #
623
- # It yields URI for each hyperlink.
624
- #
625
- # The URI objects are created with a base URI which is given by
626
- # HTML BASE element or the argument ((|base_uri|)).
627
- def each_uri(base_uri=nil) # :yields: URI
628
- each_hyperlink_uri(base_uri) {|hyperlink, uri| yield uri }
629
- end
630
- end
631
-
632
- # :stopdoc:
633
- module Doc::Trav
634
- def traverse_all_element(&block)
635
- children.each {|c| c.traverse_all_element(&block) } if children
636
- end
637
- def xpath
638
- "/"
639
- end
640
- def css_path
641
- nil
642
- end
643
- end
644
-
645
- module Elem::Trav
646
- def traverse_all_element(&block)
647
- yield self
648
- children.each {|c| c.traverse_all_element(&block) } if children
649
- end
650
- end
651
-
652
- module Leaf::Trav
653
- def traverse_all_element
654
- yield self
655
- end
656
- end
657
-
658
- module Doc::Trav
659
- def traverse_some_element(name_set, &block)
660
- children.each {|c| c.traverse_some_element(name_set, &block) } if children
661
- end
662
- end
663
-
664
- module Elem::Trav
665
- def traverse_some_element(name_set, &block)
666
- yield self if name_set.include? self.name
667
- children.each {|c| c.traverse_some_element(name_set, &block) } if children
668
- end
669
- end
670
-
671
- module Leaf::Trav
672
- def traverse_some_element(name_set)
673
- end
674
- end
675
- # :startdoc:
676
-
677
- module Traverse
678
- # +traverse_text+ traverses texts in the tree
679
- def traverse_text(&block) # :yields: text
680
- traverse_text_internal(&block)
681
- nil
682
- end
683
- end
684
-
685
- # :stopdoc:
686
- module Container::Trav
687
- def traverse_text_internal(&block)
688
- each_child {|c| c.traverse_text_internal(&block) }
689
- end
690
- end
691
-
692
- module Leaf::Trav
693
- def traverse_text_internal
694
- end
695
- end
696
-
697
- module Text::Trav
698
- def traverse_text_internal
699
- yield self
700
- end
701
- end
702
- # :startdoc:
703
-
704
- module Container::Trav
705
- # +filter+ rebuilds the tree without some components.
706
- #
707
- # node.filter {|descendant_node| predicate } -> node
708
- # loc.filter {|descendant_loc| predicate } -> node
709
- #
710
- # +filter+ yields each node except top node.
711
- # If given block returns false, corresponding node is dropped.
712
- # If given block returns true, corresponding node is retained and
713
- # inner nodes are examined.
714
- #
715
- # +filter+ returns an node.
716
- # It doesn't return location object even if self is location object.
717
- #
718
- def filter(&block)
719
- subst = {}
720
- each_child_with_index {|descendant, i|
721
- if yield descendant
722
- if descendant.elem?
723
- subst[i] = descendant.filter(&block)
724
- else
725
- subst[i] = descendant
726
- end
727
- else
728
- subst[i] = nil
729
- end
730
- }
731
- to_node.subst_subnode(subst)
732
- end
733
- end
734
-
735
- module Doc::Trav
736
- # +title+ searches title and return it as a text.
737
- # It returns nil if not found.
738
- #
739
- # +title+ searchs following information.
740
- #
741
- # - <title>...</title> in HTML
742
- # - <title>...</title> in RSS
743
- def title
744
- e = find_element('title',
745
- '{http://www.w3.org/1999/xhtml}title',
746
- '{http://purl.org/rss/1.0/}title',
747
- '{http://my.netscape.com/rdf/simple/0.9/}title')
748
- e && e.extract_text
749
- end
750
-
751
- # +author+ searches author and return it as a text.
752
- # It returns nil if not found.
753
- #
754
- # +author+ searchs following information.
755
- #
756
- # - <meta name="author" content="author-name"> in HTML
757
- # - <link rev="made" title="author-name"> in HTML
758
- # - <dc:creator>author-name</dc:creator> in RSS
759
- # - <dc:publisher>author-name</dc:publisher> in RSS
760
- def author
761
- traverse_element('meta',
762
- '{http://www.w3.org/1999/xhtml}meta') {|e|
763
- begin
764
- next unless e.fetch_attr('name').downcase == 'author'
765
- author = e.fetch_attribute('content').strip
766
- return author if !author.empty?
767
- rescue IndexError
768
- end
769
- }
770
-
771
- traverse_element('link',
772
- '{http://www.w3.org/1999/xhtml}link') {|e|
773
- begin
774
- next unless e.fetch_attr('rev').downcase == 'made'
775
- author = e.fetch_attribute('title').strip
776
- return author if !author.empty?
777
- rescue IndexError
778
- end
779
- }
780
-
781
- if channel = find_element('{http://purl.org/rss/1.0/}channel')
782
- channel.traverse_element('{http://purl.org/dc/elements/1.1/}creator') {|e|
783
- begin
784
- author = e.extract_text.strip
785
- return author if !author.empty?
786
- rescue IndexError
787
- end
788
- }
789
- channel.traverse_element('{http://purl.org/dc/elements/1.1/}publisher') {|e|
790
- begin
791
- author = e.extract_text.strip
792
- return author if !author.empty?
793
- rescue IndexError
794
- end
795
- }
796
- end
797
-
798
- nil
799
- end
800
-
801
- end
802
-
803
- module Doc::Trav
804
- def root
805
- es = []
806
- children.each {|c| es << c if c.elem? } if children
807
- raise Hpricot::Error, "no element" if es.empty?
808
- raise Hpricot::Error, "multiple top elements" if 1 < es.length
809
- es[0]
810
- end
811
- end
812
-
813
- module Elem::Trav
814
- def has_attribute?(name)
815
- self.raw_attributes && self.raw_attributes.has_key?(name.to_s)
816
- end
817
- def get_attribute(name)
818
- a = self.raw_attributes && self.raw_attributes[name.to_s]
819
- a = Hpricot.uxs(a) if a
820
- a
821
- end
822
- alias_method :[], :get_attribute
823
- def set_attribute(name, val)
824
- altered!
825
- self.raw_attributes ||= {}
826
- self.raw_attributes[name.to_s] = val.fast_xs
827
- end
828
- alias_method :[]=, :set_attribute
829
- def remove_attribute(name)
830
- name = name.to_s
831
- if has_attribute? name
832
- altered!
833
- self.raw_attributes.delete(name)
834
- end
835
- end
836
- end
837
-
838
- end
1
+ require 'hpricot/elements'
2
+ require 'uri'
3
+
4
+ module Hpricot
5
+ module Traverse
6
+ # Is this object the enclosing HTML or XML document?
7
+ def doc?() Doc::Trav === self end
8
+ # Is this object an HTML or XML element?
9
+ def elem?() Elem::Trav === self end
10
+ # Is this object an HTML text node?
11
+ def text?() Text::Trav === self end
12
+ # Is this object an XML declaration?
13
+ def xmldecl?() XMLDecl::Trav === self end
14
+ # Is this object a doctype tag?
15
+ def doctype?() DocType::Trav === self end
16
+ # Is this object an XML processing instruction?
17
+ def procins?() ProcIns::Trav === self end
18
+ # Is this object a comment?
19
+ def comment?() Comment::Trav === self end
20
+ # Is this object a stranded end tag?
21
+ def bogusetag?() BogusETag::Trav === self end
22
+
23
+ # Parses an HTML string, making an HTML fragment based on
24
+ # the options used to create the container document.
25
+ def make(input = nil, &blk)
26
+ if parent and parent.respond_to? :make
27
+ parent.make(input, &blk)
28
+ else
29
+ Hpricot.make(input, &blk).children
30
+ end
31
+ end
32
+
33
+ # Builds an HTML string from this node and its contents.
34
+ # If you need to write to a stream, try calling <tt>output(io)</tt>
35
+ # as a method on this object.
36
+ def to_html
37
+ output("")
38
+ end
39
+ alias_method :to_s, :to_html
40
+
41
+ # Attempts to preserve the original HTML of the document, only
42
+ # outputing new tags for elements which have changed.
43
+ def to_original_html
44
+ output("", :preserve => true)
45
+ end
46
+
47
+ def index(name)
48
+ i = 0
49
+ return i if name == "*"
50
+ children.each do |x|
51
+ return i if (x.respond_to?(:name) and name == x.name) or
52
+ (x.text? and name == "text()")
53
+ i += 1
54
+ end if children
55
+ -1
56
+ end
57
+
58
+ # Puts together an array of neighboring nodes based on their proximity
59
+ # to this node. So, for example, to get the next node, you could use
60
+ # <tt>nodes_at(1). Or, to get the previous node, use <tt>nodes_at(1)</tt>.
61
+ #
62
+ # This method also accepts ranges and sets of numbers.
63
+ #
64
+ # ele.nodes_at(-3..-1, 1..3) # gets three nodes before and three after
65
+ # ele.nodes_at(1, 5, 7) # gets three nodes at offsets below the current node
66
+ # ele.nodes_at(0, 5..6) # the current node and two others
67
+ def nodes_at(*pos)
68
+ sib = parent.children
69
+ i, si = 0, sib.index(self)
70
+ pos.map! do |r|
71
+ if r.is_a?(Range) and r.begin.is_a?(String)
72
+ r = Range.new(parent.index(r.begin)-si, parent.index(r.end)-si, r.exclude_end?)
73
+ end
74
+ r
75
+ end
76
+ p pos
77
+ Elements[*
78
+ sib.select do |x|
79
+ sel =
80
+ case i - si when *pos
81
+ true
82
+ end
83
+ i += 1
84
+ sel
85
+ end
86
+ ]
87
+ end
88
+
89
+ # Returns the node neighboring this node to the south: just below it.
90
+ # This method includes text nodes and comments and such.
91
+ def next
92
+ sib = parent.children
93
+ sib[sib.index(self) + 1] if parent
94
+ end
95
+ alias_method :next_node, :next
96
+
97
+ # Returns to node neighboring this node to the north: just above it.
98
+ # This method includes text nodes and comments and such.
99
+ def previous
100
+ sib = parent.children
101
+ x = sib.index(self) - 1
102
+ sib[x] if sib and x >= 0
103
+ end
104
+ alias_method :previous_node, :previous
105
+
106
+ # Find all preceding nodes.
107
+ def preceding
108
+ sibs = parent.children
109
+ si = sibs.index(self)
110
+ return Elements[*sibs[0...si]]
111
+ end
112
+
113
+ # Find all nodes which follow the current one.
114
+ def following
115
+ sibs = parent.children
116
+ si = sibs.index(self) + 1
117
+ return Elements[*sibs[si...sibs.length]]
118
+ end
119
+
120
+ # Adds elements immediately after this element, contained in the +html+ string.
121
+ def after(html = nil, &blk)
122
+ parent.insert_after(make(html, &blk), self)
123
+ end
124
+
125
+ # Adds elements immediately before this element, contained in the +html+ string.
126
+ def before(html = nil, &blk)
127
+ parent.insert_before(make(html, &blk), self)
128
+ end
129
+
130
+
131
+ # Replace this element and its contents with the nodes contained
132
+ # in the +html+ string.
133
+ def swap(html = nil, &blk)
134
+ parent.altered!
135
+ parent.replace_child(self, make(html, &blk))
136
+ end
137
+
138
+ def get_subnode(*indexes)
139
+ n = self
140
+ indexes.each {|index|
141
+ n = n.get_subnode_internal(index)
142
+ }
143
+ n
144
+ end
145
+
146
+ # Builds a string from the text contained in this node. All
147
+ # HTML elements are removed.
148
+ def to_plain_text
149
+ if respond_to?(:children) and children
150
+ children.map { |x| x.to_plain_text }.join.strip.gsub(/\n{2,}/, "\n\n")
151
+ else
152
+ ""
153
+ end
154
+ end
155
+
156
+ # Builds a string from the text contained in this node. All
157
+ # HTML elements are removed.
158
+ def inner_text
159
+ if respond_to?(:children) and children
160
+ children.map { |x| x.inner_text }.join
161
+ else
162
+ ""
163
+ end
164
+ end
165
+ alias_method :innerText, :inner_text
166
+
167
+ # Builds an HTML string from the contents of this node.
168
+ def html(inner = nil, &blk)
169
+ if inner or blk
170
+ altered!
171
+ case inner
172
+ when Array
173
+ self.children = inner
174
+ else
175
+ self.children = make(inner, &blk)
176
+ end
177
+ reparent self.children
178
+ else
179
+ if respond_to?(:children) and children
180
+ children.map { |x| x.output("") }.join
181
+ else
182
+ ""
183
+ end
184
+ end
185
+ end
186
+ alias_method :inner_html, :html
187
+ alias_method :innerHTML, :inner_html
188
+
189
+ # Inserts new contents into the current node, based on
190
+ # the HTML contained in string +inner+.
191
+ def inner_html=(inner)
192
+ html(inner || [])
193
+ end
194
+ alias_method :innerHTML=, :inner_html=
195
+
196
+ def reparent(nodes)
197
+ altered!
198
+ [*nodes].each { |e| e.parent = self }
199
+ end
200
+ private :reparent
201
+
202
+ def clean_path(path)
203
+ path.gsub(/^\s+|\s+$/, '')
204
+ end
205
+
206
+ # Builds a unique XPath string for this node, from the
207
+ # root of the document containing it.
208
+ def xpath
209
+ if elem? and has_attribute? 'id'
210
+ "//#{self.name}[@id='#{get_attribute('id')}']"
211
+ else
212
+ sim, id = 0, 0, 0
213
+ parent.children.each do |e|
214
+ id = sim if e == self
215
+ sim += 1 if e.pathname == self.pathname
216
+ end if parent.children
217
+ p = File.join(parent.xpath, self.pathname)
218
+ p += "[#{id+1}]" if sim >= 2
219
+ p
220
+ end
221
+ end
222
+
223
+ # Builds a unique CSS string for this node, from the
224
+ # root of the document containing it.
225
+ def css_path
226
+ if elem? and has_attribute? 'id'
227
+ "##{get_attribute('id')}"
228
+ else
229
+ sim, i, id = 0, 0, 0
230
+ parent.children.each do |e|
231
+ id = sim if e == self
232
+ sim += 1 if e.pathname == self.pathname
233
+ end if parent.children
234
+ p = parent.css_path
235
+ p = p ? "#{p} > #{self.pathname}" : self.pathname
236
+ p += ":nth(#{id})" if sim >= 2
237
+ p
238
+ end
239
+ end
240
+
241
+ def node_position
242
+ parent.children.index(self)
243
+ end
244
+
245
+ def position
246
+ parent.children_of_type(self.pathname).index(self)
247
+ end
248
+
249
+ # Searches this node for all elements matching
250
+ # the CSS or XPath +expr+. Returns an Elements array
251
+ # containing the matching nodes. If +blk+ is given, it
252
+ # is used to iterate through the matching set.
253
+ def search(expr, &blk)
254
+ if Range === expr
255
+ return Elements.expand(at(expr.begin), at(expr.end), expr.exclude_end?)
256
+ end
257
+ last = nil
258
+ nodes = [self]
259
+ done = []
260
+ expr = expr.to_s
261
+ hist = []
262
+ until expr.empty?
263
+ expr = clean_path(expr)
264
+ expr.gsub!(%r!^//!, '')
265
+
266
+ case expr
267
+ when %r!^/?\.\.!
268
+ last = expr = $'
269
+ nodes.map! { |node| node.parent }
270
+ when %r!^[>/]\s*!
271
+ last = expr = $'
272
+ nodes = Elements[*nodes.map { |node| node.children if node.respond_to? :children }.flatten.compact]
273
+ when %r!^\+!
274
+ last = expr = $'
275
+ nodes.map! do |node|
276
+ siblings = node.parent.children
277
+ siblings[siblings.index(node)+1]
278
+ end
279
+ nodes.compact!
280
+ when %r!^~!
281
+ last = expr = $'
282
+ nodes.map! do |node|
283
+ siblings = node.parent.children
284
+ siblings[(siblings.index(node)+1)..-1]
285
+ end
286
+ nodes.flatten!
287
+ when %r!^[|,]!
288
+ last = expr = " #$'"
289
+ nodes.shift if nodes.first == self
290
+ done += nodes
291
+ nodes = [self]
292
+ else
293
+ m = expr.match(%r!^([#.]?)([a-z0-9\\*_-]*)!i).to_a
294
+ after = $'
295
+ mt = after[%r!:[a-z0-9\\*_-]+!i, 0]
296
+ oop = false
297
+ if mt and not (mt == ":not" or Traverse.method_defined? "filter[#{mt}]")
298
+ after = $'
299
+ m[2] += mt
300
+ expr = after
301
+ end
302
+ if m[1] == '#'
303
+ oid = get_element_by_id(m[2])
304
+ nodes = oid ? [oid] : []
305
+ expr = after
306
+ else
307
+ m[2] = "*" if after =~ /^\(\)/ || m[2] == "" || m[1] == "."
308
+ ret = []
309
+ nodes.each do |node|
310
+ case m[2]
311
+ when '*'
312
+ node.traverse_element { |n| ret << n }
313
+ else
314
+ if node.respond_to? :get_elements_by_tag_name
315
+ ret += [*node.get_elements_by_tag_name(m[2])] - [*(node unless last)]
316
+ end
317
+ end
318
+ end
319
+ nodes = ret
320
+ end
321
+ last = nil
322
+ end
323
+
324
+ hist << expr
325
+ break if hist[-1] == hist[-2]
326
+ nodes, expr = Elements.filter(nodes, expr)
327
+ end
328
+ nodes = done + nodes.flatten.uniq
329
+ if blk
330
+ nodes.each(&blk)
331
+ self
332
+ else
333
+ Elements[*nodes]
334
+ end
335
+ end
336
+ alias_method :/, :search
337
+
338
+ # Find the first matching node for the CSS or XPath
339
+ # +expr+ string.
340
+ def at(expr)
341
+ search(expr).first
342
+ end
343
+ alias_method :%, :at
344
+
345
+ # +traverse_element+ traverses elements in the tree.
346
+ # It yields elements in depth first order.
347
+ #
348
+ # If _names_ are empty, it yields all elements.
349
+ # If non-empty _names_ are given, it should be list of universal names.
350
+ #
351
+ # A nested element is yielded in depth first order as follows.
352
+ #
353
+ # t = Hpricot('<a id=0><b><a id=1 /></b><c id=2 /></a>')
354
+ # t.traverse_element("a", "c") {|e| p e}
355
+ # # =>
356
+ # {elem <a id="0"> {elem <b> {emptyelem <a id="1">} </b>} {emptyelem <c id="2">} </a>}
357
+ # {emptyelem <a id="1">}
358
+ # {emptyelem <c id="2">}
359
+ #
360
+ # Universal names are specified as follows.
361
+ #
362
+ # t = Hpricot(<<'End')
363
+ # <html>
364
+ # <meta name="robots" content="index,nofollow">
365
+ # <meta name="author" content="Who am I?">
366
+ # </html>
367
+ # End
368
+ # t.traverse_element("{http://www.w3.org/1999/xhtml}meta") {|e| p e}
369
+ # # =>
370
+ # {emptyelem <{http://www.w3.org/1999/xhtml}meta name="robots" content="index,nofollow">}
371
+ # {emptyelem <{http://www.w3.org/1999/xhtml}meta name="author" content="Who am I?">}
372
+ #
373
+ def traverse_element(*names, &block) # :yields: element
374
+ if names.empty?
375
+ traverse_all_element(&block)
376
+ else
377
+ name_set = {}
378
+ names.each {|n| name_set[n] = true }
379
+ traverse_some_element(name_set, &block)
380
+ end
381
+ nil
382
+ end
383
+
384
+ # Find children of a given +tag_name+.
385
+ #
386
+ # ele.children_of_type('p')
387
+ # #=> [...array of paragraphs...]
388
+ #
389
+ def children_of_type(tag_name)
390
+ if respond_to? :children
391
+ children.find_all do |x|
392
+ x.respond_to?(:pathname) && x.pathname == tag_name
393
+ end
394
+ end
395
+ end
396
+
397
+ end
398
+
399
+ module Container::Trav
400
+ # Return all children of this node which can contain other
401
+ # nodes. This is a good way to get all HTML elements which
402
+ # aren't text, comment, doctype or processing instruction nodes.
403
+ def containers
404
+ children.grep(Container::Trav)
405
+ end
406
+
407
+ # Returns the container node neighboring this node to the south: just below it.
408
+ # By "container" node, I mean: this method does not find text nodes or comments or cdata or any of that.
409
+ # See Hpricot::Traverse#next_node if you need to hunt out all kinds of nodes.
410
+ def next_sibling
411
+ sib = parent.containers
412
+ sib[sib.index(self) + 1] if parent
413
+ end
414
+
415
+ # Returns the container node neighboring this node to the north: just above it.
416
+ # By "container" node, I mean: this method does not find text nodes or comments or cdata or any of that.
417
+ # See Hpricot::Traverse#previous_node if you need to hunt out all kinds of nodes.
418
+ def previous_sibling
419
+ sib = parent.containers
420
+ x = sib.index(self) - 1
421
+ sib[x] if sib and x >= 0
422
+ end
423
+
424
+ # Find all preceding sibling elements. Like the other "sibling" methods, this weeds
425
+ # out text and comment nodes.
426
+ def preceding_siblings()
427
+ sibs = parent.containers
428
+ si = sibs.index(self)
429
+ return Elements[*sibs[0...si]]
430
+ end
431
+
432
+ # Find sibling elements which follow the current one. Like the other "sibling" methods, this weeds
433
+ # out text and comment nodes.
434
+ def following_siblings()
435
+ sibs = parent.containers
436
+ si = sibs.index(self) + 1
437
+ return Elements[*sibs[si...sibs.length]]
438
+ end
439
+
440
+ # Puts together an array of neighboring sibling elements based on their proximity
441
+ # to this element.
442
+ #
443
+ # This method accepts ranges and sets of numbers.
444
+ #
445
+ # ele.siblings_at(-3..-1, 1..3) # gets three elements before and three after
446
+ # ele.siblings_at(1, 5, 7) # gets three elements at offsets below the current element
447
+ # ele.siblings_at(0, 5..6) # the current element and two others
448
+ #
449
+ # Like the other "sibling" methods, this doesn't find text and comment nodes.
450
+ # Use nodes_at to include those nodes.
451
+ def siblings_at(*pos)
452
+ sib = parent.containers
453
+ i, si = 0, sib.index(self)
454
+ Elements[*
455
+ sib.select do |x|
456
+ sel = case i - si when *pos
457
+ true
458
+ end
459
+ i += 1
460
+ sel
461
+ end
462
+ ]
463
+ end
464
+
465
+ # Replace +old+, a child of the current node, with +new+ node.
466
+ def replace_child(old, new)
467
+ reparent new
468
+ children[children.index(old), 1] = [*new]
469
+ end
470
+
471
+ # Insert +nodes+, an array of HTML elements or a single element,
472
+ # before the node +ele+, a child of the current node.
473
+ def insert_before(nodes, ele)
474
+ case nodes
475
+ when Array
476
+ nodes.each { |n| insert_before(n, ele) }
477
+ else
478
+ reparent nodes
479
+ children[children.index(ele) || 0, 0] = nodes
480
+ end
481
+ end
482
+
483
+ # Insert +nodes+, an array of HTML elements or a single element,
484
+ # after the node +ele+, a child of the current node.
485
+ def insert_after(nodes, ele)
486
+ case nodes
487
+ when Array
488
+ nodes.reverse_each { |n| insert_after(n, ele) }
489
+ else
490
+ reparent nodes
491
+ idx = children.index(ele)
492
+ children[idx ? idx + 1 : children.length, 0] = nodes
493
+ end
494
+ end
495
+
496
+ # +each_child+ iterates over each child.
497
+ def each_child(&block) # :yields: child_node
498
+ children.each(&block) if children
499
+ nil
500
+ end
501
+
502
+ # +each_child_with_index+ iterates over each child.
503
+ def each_child_with_index(&block) # :yields: child_node, index
504
+ children.each_with_index(&block) if children
505
+ nil
506
+ end
507
+
508
+ # +find_element+ searches an element which universal name is specified by
509
+ # the arguments.
510
+ # It returns nil if not found.
511
+ def find_element(*names)
512
+ traverse_element(*names) {|e| return e }
513
+ nil
514
+ end
515
+
516
+ # Returns a list of CSS classes to which this element belongs.
517
+ def classes
518
+ get_attribute('class').to_s.strip.split(/\s+/)
519
+ end
520
+
521
+ def get_element_by_id(id)
522
+ traverse_all_element do |ele|
523
+ if ele.elem? and eid = ele.get_attribute('id')
524
+ return ele if eid.to_s == id
525
+ end
526
+ end
527
+ nil
528
+ end
529
+
530
+ def get_elements_by_tag_name(*a)
531
+ list = Elements[]
532
+ a.delete("*")
533
+ traverse_element(*a.map { |tag| [tag, "{http://www.w3.org/1999/xhtml}#{tag}"] }.flatten) do |e|
534
+ list << e if e.elem?
535
+ end
536
+ list
537
+ end
538
+
539
+ def each_hyperlink_attribute
540
+ traverse_element(
541
+ '{http://www.w3.org/1999/xhtml}a',
542
+ '{http://www.w3.org/1999/xhtml}area',
543
+ '{http://www.w3.org/1999/xhtml}link',
544
+ '{http://www.w3.org/1999/xhtml}img',
545
+ '{http://www.w3.org/1999/xhtml}object',
546
+ '{http://www.w3.org/1999/xhtml}q',
547
+ '{http://www.w3.org/1999/xhtml}blockquote',
548
+ '{http://www.w3.org/1999/xhtml}ins',
549
+ '{http://www.w3.org/1999/xhtml}del',
550
+ '{http://www.w3.org/1999/xhtml}form',
551
+ '{http://www.w3.org/1999/xhtml}input',
552
+ '{http://www.w3.org/1999/xhtml}head',
553
+ '{http://www.w3.org/1999/xhtml}base',
554
+ '{http://www.w3.org/1999/xhtml}script') {|elem|
555
+ case elem.name
556
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:base|a|area|link)\z}i
557
+ attrs = ['href']
558
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:img)\z}i
559
+ attrs = ['src', 'longdesc', 'usemap']
560
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:object)\z}i
561
+ attrs = ['classid', 'codebase', 'data', 'usemap']
562
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:q|blockquote|ins|del)\z}i
563
+ attrs = ['cite']
564
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:form)\z}i
565
+ attrs = ['action']
566
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:input)\z}i
567
+ attrs = ['src', 'usemap']
568
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:head)\z}i
569
+ attrs = ['profile']
570
+ when %r{\{http://www.w3.org/1999/xhtml\}(?:script)\z}i
571
+ attrs = ['src', 'for']
572
+ end
573
+ attrs.each {|attr|
574
+ if hyperlink = elem.get_attribute(attr)
575
+ yield elem, attr, hyperlink
576
+ end
577
+ }
578
+ }
579
+ end
580
+ private :each_hyperlink_attribute
581
+
582
+ # +each_hyperlink_uri+ traverses hyperlinks such as HTML href attribute
583
+ # of A element.
584
+ #
585
+ # It yields Hpricot::Text and URI for each hyperlink.
586
+ #
587
+ # The URI objects are created with a base URI which is given by
588
+ # HTML BASE element or the argument ((|base_uri|)).
589
+ # +each_hyperlink_uri+ doesn't yields href of the BASE element.
590
+ def each_hyperlink_uri(base_uri=nil) # :yields: hyperlink, uri
591
+ base_uri = URI.parse(base_uri) if String === base_uri
592
+ links = []
593
+ each_hyperlink_attribute {|elem, attr, hyperlink|
594
+ if %r{\{http://www.w3.org/1999/xhtml\}(?:base)\z}i =~ elem.name
595
+ base_uri = URI.parse(hyperlink.to_s)
596
+ else
597
+ links << hyperlink
598
+ end
599
+ }
600
+ if base_uri
601
+ links.each {|hyperlink| yield hyperlink, base_uri + hyperlink.to_s }
602
+ else
603
+ links.each {|hyperlink| yield hyperlink, URI.parse(hyperlink.to_s) }
604
+ end
605
+ end
606
+
607
+ # +each_hyperlink+ traverses hyperlinks such as HTML href attribute
608
+ # of A element.
609
+ #
610
+ # It yields Hpricot::Text.
611
+ #
612
+ # Note that +each_hyperlink+ yields HTML href attribute of BASE element.
613
+ def each_hyperlink # :yields: text
614
+ links = []
615
+ each_hyperlink_attribute {|elem, attr, hyperlink|
616
+ yield hyperlink
617
+ }
618
+ end
619
+
620
+ # +each_uri+ traverses hyperlinks such as HTML href attribute
621
+ # of A element.
622
+ #
623
+ # It yields URI for each hyperlink.
624
+ #
625
+ # The URI objects are created with a base URI which is given by
626
+ # HTML BASE element or the argument ((|base_uri|)).
627
+ def each_uri(base_uri=nil) # :yields: URI
628
+ each_hyperlink_uri(base_uri) {|hyperlink, uri| yield uri }
629
+ end
630
+ end
631
+
632
+ # :stopdoc:
633
+ module Doc::Trav
634
+ def traverse_all_element(&block)
635
+ children.each {|c| c.traverse_all_element(&block) } if children
636
+ end
637
+ def xpath
638
+ "/"
639
+ end
640
+ def css_path
641
+ nil
642
+ end
643
+ end
644
+
645
+ module Elem::Trav
646
+ def traverse_all_element(&block)
647
+ yield self
648
+ children.each {|c| c.traverse_all_element(&block) } if children
649
+ end
650
+ end
651
+
652
+ module Leaf::Trav
653
+ def traverse_all_element
654
+ yield self
655
+ end
656
+ end
657
+
658
+ module Doc::Trav
659
+ def traverse_some_element(name_set, &block)
660
+ children.each {|c| c.traverse_some_element(name_set, &block) } if children
661
+ end
662
+ end
663
+
664
+ module Elem::Trav
665
+ def traverse_some_element(name_set, &block)
666
+ yield self if name_set.include? self.name
667
+ children.each {|c| c.traverse_some_element(name_set, &block) } if children
668
+ end
669
+ end
670
+
671
+ module Leaf::Trav
672
+ def traverse_some_element(name_set)
673
+ end
674
+ end
675
+ # :startdoc:
676
+
677
+ module Traverse
678
+ # +traverse_text+ traverses texts in the tree
679
+ def traverse_text(&block) # :yields: text
680
+ traverse_text_internal(&block)
681
+ nil
682
+ end
683
+ end
684
+
685
+ # :stopdoc:
686
+ module Container::Trav
687
+ def traverse_text_internal(&block)
688
+ each_child {|c| c.traverse_text_internal(&block) }
689
+ end
690
+ end
691
+
692
+ module Leaf::Trav
693
+ def traverse_text_internal
694
+ end
695
+ end
696
+
697
+ module Text::Trav
698
+ def traverse_text_internal
699
+ yield self
700
+ end
701
+ end
702
+ # :startdoc:
703
+
704
+ module Container::Trav
705
+ # +filter+ rebuilds the tree without some components.
706
+ #
707
+ # node.filter {|descendant_node| predicate } -> node
708
+ # loc.filter {|descendant_loc| predicate } -> node
709
+ #
710
+ # +filter+ yields each node except top node.
711
+ # If given block returns false, corresponding node is dropped.
712
+ # If given block returns true, corresponding node is retained and
713
+ # inner nodes are examined.
714
+ #
715
+ # +filter+ returns an node.
716
+ # It doesn't return location object even if self is location object.
717
+ #
718
+ def filter(&block)
719
+ subst = {}
720
+ each_child_with_index {|descendant, i|
721
+ if yield descendant
722
+ if descendant.elem?
723
+ subst[i] = descendant.filter(&block)
724
+ else
725
+ subst[i] = descendant
726
+ end
727
+ else
728
+ subst[i] = nil
729
+ end
730
+ }
731
+ to_node.subst_subnode(subst)
732
+ end
733
+ end
734
+
735
+ module Doc::Trav
736
+ # +title+ searches title and return it as a text.
737
+ # It returns nil if not found.
738
+ #
739
+ # +title+ searchs following information.
740
+ #
741
+ # - <title>...</title> in HTML
742
+ # - <title>...</title> in RSS
743
+ def title
744
+ e = find_element('title',
745
+ '{http://www.w3.org/1999/xhtml}title',
746
+ '{http://purl.org/rss/1.0/}title',
747
+ '{http://my.netscape.com/rdf/simple/0.9/}title')
748
+ e && e.extract_text
749
+ end
750
+
751
+ # +author+ searches author and return it as a text.
752
+ # It returns nil if not found.
753
+ #
754
+ # +author+ searchs following information.
755
+ #
756
+ # - <meta name="author" content="author-name"> in HTML
757
+ # - <link rev="made" title="author-name"> in HTML
758
+ # - <dc:creator>author-name</dc:creator> in RSS
759
+ # - <dc:publisher>author-name</dc:publisher> in RSS
760
+ def author
761
+ traverse_element('meta',
762
+ '{http://www.w3.org/1999/xhtml}meta') {|e|
763
+ begin
764
+ next unless e.fetch_attr('name').downcase == 'author'
765
+ author = e.fetch_attribute('content').strip
766
+ return author if !author.empty?
767
+ rescue IndexError
768
+ end
769
+ }
770
+
771
+ traverse_element('link',
772
+ '{http://www.w3.org/1999/xhtml}link') {|e|
773
+ begin
774
+ next unless e.fetch_attr('rev').downcase == 'made'
775
+ author = e.fetch_attribute('title').strip
776
+ return author if !author.empty?
777
+ rescue IndexError
778
+ end
779
+ }
780
+
781
+ if channel = find_element('{http://purl.org/rss/1.0/}channel')
782
+ channel.traverse_element('{http://purl.org/dc/elements/1.1/}creator') {|e|
783
+ begin
784
+ author = e.extract_text.strip
785
+ return author if !author.empty?
786
+ rescue IndexError
787
+ end
788
+ }
789
+ channel.traverse_element('{http://purl.org/dc/elements/1.1/}publisher') {|e|
790
+ begin
791
+ author = e.extract_text.strip
792
+ return author if !author.empty?
793
+ rescue IndexError
794
+ end
795
+ }
796
+ end
797
+
798
+ nil
799
+ end
800
+
801
+ end
802
+
803
+ module Doc::Trav
804
+ def root
805
+ es = []
806
+ children.each {|c| es << c if c.elem? } if children
807
+ raise Hpricot::Error, "no element" if es.empty?
808
+ raise Hpricot::Error, "multiple top elements" if 1 < es.length
809
+ es[0]
810
+ end
811
+ end
812
+
813
+ module Elem::Trav
814
+ def has_attribute?(name)
815
+ self.raw_attributes && self.raw_attributes.has_key?(name.to_s)
816
+ end
817
+ def get_attribute(name)
818
+ a = self.raw_attributes && self.raw_attributes[name.to_s]
819
+ a = Hpricot.uxs(a) if a
820
+ a
821
+ end
822
+ alias_method :[], :get_attribute
823
+ def set_attribute(name, val)
824
+ altered!
825
+ self.raw_attributes ||= {}
826
+ self.raw_attributes[name.to_s] = val.fast_xs
827
+ end
828
+ alias_method :[]=, :set_attribute
829
+ def remove_attribute(name)
830
+ name = name.to_s
831
+ if has_attribute? name
832
+ altered!
833
+ self.raw_attributes.delete(name)
834
+ end
835
+ end
836
+ end
837
+
838
+ end