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.
- data/README.rst +148 -0
- data/Rakefile +37 -0
- data/lib/tasks/yellow-brick-road_tasks.rake +4 -0
- data/lib/yellow-brick-road.rb +6 -0
- data/lib/yellow-brick-road/config.rb +15 -0
- data/lib/yellow-brick-road/directive_processor.rb +68 -0
- data/lib/yellow-brick-road/engine.rb +22 -0
- data/lib/yellow-brick-road/soy_processor.rb +56 -0
- data/lib/yellow-brick-road/utils.rb +38 -0
- data/lib/yellow-brick-road/version.rb +3 -0
- data/test/dummy/Rakefile +7 -0
- data/test/dummy/app/assets/javascripts/application.js +12 -0
- data/test/dummy/app/assets/javascripts/closure-deps.js +4 -0
- data/test/dummy/app/assets/javascripts/my-closure/simple.js.soy +14 -0
- data/test/dummy/app/assets/javascripts/my-closure/start.js +25 -0
- data/test/dummy/app/assets/stylesheets/application.css +7 -0
- data/test/dummy/app/controllers/application_controller.rb +7 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/views/application/index.html.erb +0 -0
- data/test/dummy/app/views/layouts/application.html.erb +18 -0
- data/test/dummy/config.ru +4 -0
- data/test/dummy/config/application.rb +51 -0
- data/test/dummy/config/boot.rb +10 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +30 -0
- data/test/dummy/config/environments/production.rb +60 -0
- data/test/dummy/config/environments/test.rb +39 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/inflections.rb +10 -0
- data/test/dummy/config/initializers/mime_types.rb +5 -0
- data/test/dummy/config/initializers/secret_token.rb +7 -0
- data/test/dummy/config/initializers/session_store.rb +8 -0
- data/test/dummy/config/initializers/wrap_parameters.rb +10 -0
- data/test/dummy/config/initializers/yellow_brick_road.rb +2 -0
- data/test/dummy/config/locales/en.yml +5 -0
- data/test/dummy/config/routes.rb +58 -0
- data/test/dummy/log/development.log +13924 -0
- data/test/dummy/log/test.log +0 -0
- data/test/dummy/public/404.html +26 -0
- data/test/dummy/public/422.html +26 -0
- data/test/dummy/public/500.html +26 -0
- data/test/dummy/public/favicon.ico +0 -0
- data/test/dummy/script/rails +6 -0
- data/test/dummy/tmp/cache/assets/BE0/120/sprockets%2F751842b8c6750008c7310d182600d173 +142 -0
- data/test/dummy/tmp/cache/assets/C19/950/sprockets%2F142437f1d8d9424007b4a882a5429333 +1372 -0
- data/test/dummy/tmp/cache/assets/C4B/D00/sprockets%2F400f22c081529179ce3d079457de3009 +806 -0
- data/test/dummy/tmp/cache/assets/C5A/EE0/sprockets%2Fb711429ed948c503b718d077037780f8 +0 -0
- data/test/dummy/tmp/cache/assets/C5B/A80/sprockets%2F508569b474262724c61a461f7777dab7 +1118 -0
- data/test/dummy/tmp/cache/assets/C6C/660/sprockets%2F2c298b3b02232a21527680685a3efc30 +9289 -0
- data/test/dummy/tmp/cache/assets/C72/8A0/sprockets%2F167b265129e30d87d253c406db305c60 +293 -0
- data/test/dummy/tmp/cache/assets/C7E/9F0/sprockets%2F89862076204c62c4593ac20de32da909 +9 -0
- data/test/dummy/tmp/cache/assets/C8B/5F0/sprockets%2Fc6a4470b5c21e285e829a99365839b24 +0 -0
- data/test/dummy/tmp/cache/assets/C92/D20/sprockets%2F2e618f7805f445889aec94885a500f03 +457 -0
- data/test/dummy/tmp/cache/assets/C98/FD0/sprockets%2Fb11442af041f96e87a43a1dc11231745 +283 -0
- data/test/dummy/tmp/cache/assets/CA3/520/sprockets%2F5379d7143c6c52b11b88dc0ab5436133 +277 -0
- data/test/dummy/tmp/cache/assets/CA5/450/sprockets%2F6bb727c9312a749134ad67323a317f0d +73 -0
- data/test/dummy/tmp/cache/assets/CA5/4F0/sprockets%2Feeb7de7771527700af194c0441d29101 +709 -0
- data/test/dummy/tmp/cache/assets/CA6/E90/sprockets%2F611f68180f43c4181f06ae5c5f8201e2 +1546 -0
- data/test/dummy/tmp/cache/assets/CA7/310/sprockets%2F45664cf816315200b574e029fde6f10a +0 -0
- data/test/dummy/tmp/cache/assets/CA9/9D0/sprockets%2F2672e32464cf7267c4ba3d028f54b153 +224 -0
- data/test/dummy/tmp/cache/assets/CAB/5A0/sprockets%2F7f50e0289f150c8636ac9253129bc13c +2556 -0
- data/test/dummy/tmp/cache/assets/CB5/7E0/sprockets%2F42ff6672683b2029233a800e7539eeee +474 -0
- data/test/dummy/tmp/cache/assets/CB6/DC0/sprockets%2F2f9882155bb2d4d3ab5d708951857c60 +494 -0
- data/test/dummy/tmp/cache/assets/CBB/680/sprockets%2F1dc336d96fb52df34b458185559922b5 +1018 -0
- data/test/dummy/tmp/cache/assets/CBC/640/sprockets%2F67d2e0d9e5129d237e575d2780c64b47 +1260 -0
- data/test/dummy/tmp/cache/assets/CBE/550/sprockets%2Fd680cac830e0b3408ba910f0b0421147 +25 -0
- data/test/dummy/tmp/cache/assets/CC7/790/sprockets%2F69941f32a12e4f99d4a57f65386d870d +608 -0
- data/test/dummy/tmp/cache/assets/CCB/F80/sprockets%2Fa865701ef2ec41155e524772c31a1a2b +1088 -0
- data/test/dummy/tmp/cache/assets/CCE/580/sprockets%2F879411ed27ed1c557d57853d8f579b56 +0 -0
- data/test/dummy/tmp/cache/assets/CD0/070/sprockets%2F6748fe8481965f260d9c56b7f9f508a2 +1530 -0
- data/test/dummy/tmp/cache/assets/CD4/750/sprockets%2Faf3505141ecb3169ce41ce519d136924 +0 -0
- data/test/dummy/tmp/cache/assets/CD6/A90/sprockets%2F3f55ac75b9fb8426312116bcb940a580 +2539 -0
- data/test/dummy/tmp/cache/assets/CD9/6C0/sprockets%2F5e2458bc52da90ba349a66035e3b6752 +0 -0
- data/test/dummy/tmp/cache/assets/CD9/F50/sprockets%2F9008bf696500cfae1d61f045f209181e +256 -0
- data/test/dummy/tmp/cache/assets/CDB/B40/sprockets%2F53529a22c994570a0df4742c0bfe61f4 +0 -0
- data/test/dummy/tmp/cache/assets/CDC/D10/sprockets%2F1e775b4ff06b4401c07503ce95a839b5 +75 -0
- data/test/dummy/tmp/cache/assets/CDE/CD0/sprockets%2F141066798b4acf07053f7e3a6cb4e555 +1613 -0
- data/test/dummy/tmp/cache/assets/CE1/760/sprockets%2F245fe11803630fe30d0cf8a869886ab5 +357 -0
- data/test/dummy/tmp/cache/assets/CE1/FA0/sprockets%2F863d1650ef066e4a2168bc57c7c0e096 +0 -0
- data/test/dummy/tmp/cache/assets/CE2/310/sprockets%2Fd38075d5592ecaf82e43526c03b467c4 +1545 -0
- data/test/dummy/tmp/cache/assets/CE2/420/sprockets%2F30f2e8f30477e80cf6416dfe27307c07 +204 -0
- data/test/dummy/tmp/cache/assets/CE2/D70/sprockets%2F467c44b63644e0f44dd06a585f36f1a6 +522 -0
- data/test/dummy/tmp/cache/assets/CE3/670/sprockets%2Fddd11860b444cd0f9996be0c46762318 +796 -0
- data/test/dummy/tmp/cache/assets/CE4/0B0/sprockets%2Ff11535d785c0d34349c64d673bd4b28f +1105 -0
- data/test/dummy/tmp/cache/assets/CE4/590/sprockets%2Fd68edcc87a4ae302794093081b45b819 +10 -0
- data/test/dummy/tmp/cache/assets/CE4/D60/sprockets%2F61772e4a60b616f74da91b838a2f4f82 +0 -0
- data/test/dummy/tmp/cache/assets/CE6/DD0/sprockets%2F2fe073024a2bf26bd98458388b57af37 +1355 -0
- data/test/dummy/tmp/cache/assets/CE7/160/sprockets%2Fba1f9939f031b4356ec1869d40fc2747 +1261 -0
- data/test/dummy/tmp/cache/assets/CE8/C70/sprockets%2Fc6c9ba3c677b5e2af8520395192c9445 +173 -0
- data/test/dummy/tmp/cache/assets/CEA/8B0/sprockets%2Ff77e549cb6d37604105f35d4e67d8c21 +511 -0
- data/test/dummy/tmp/cache/assets/CEA/AA0/sprockets%2F5f816982c86d2e6b72b2f5f65c51d070 +1529 -0
- data/test/dummy/tmp/cache/assets/CEA/C10/sprockets%2F9dcd541e67c299ab076a44a2183872f8 +0 -0
- data/test/dummy/tmp/cache/assets/CF1/5A0/sprockets%2F1a697695edf2bb7b49a2896904218bc7 +115 -0
- data/test/dummy/tmp/cache/assets/CF4/480/sprockets%2F5e99c77e93f4a522c84357e62b25e0f7 +43 -0
- data/test/dummy/tmp/cache/assets/CF7/460/sprockets%2Fd12ea9733fe3c92456f57f9145569b9c +0 -0
- data/test/dummy/tmp/cache/assets/CF7/470/sprockets%2F2897897a166ca3369fecb88f83f211b5 +435 -0
- data/test/dummy/tmp/cache/assets/CFA/760/sprockets%2Fa9024adba14091e0635c6874d1db4e22 +0 -0
- data/test/dummy/tmp/cache/assets/D01/F30/sprockets%2Fe639a15b6e0cca37d12443b408e1166f +25 -0
- data/test/dummy/tmp/cache/assets/D02/9F0/sprockets%2Fb99eae308897fe88cb9414b96824098b +0 -0
- data/test/dummy/tmp/cache/assets/D03/330/sprockets%2F1e003cdb3e7dcc9307e84090ad457127 +454 -0
- data/test/dummy/tmp/cache/assets/D03/8B0/sprockets%2Ff74632bddf2c0b2018ca7b736309380e +365 -0
- data/test/dummy/tmp/cache/assets/D05/0A0/sprockets%2Fe57658220260db13eb5577aef42cb61b +257 -0
- data/test/dummy/tmp/cache/assets/D05/920/sprockets%2F909507434dcc270db4853e4c147f0aac +31 -0
- data/test/dummy/tmp/cache/assets/D08/510/sprockets%2Fa567be6cb6d7310096f1739b25a5a3f0 +50 -0
- data/test/dummy/tmp/cache/assets/D08/9F0/sprockets%2F1d61d2c89ca50957066bacc5b69011f5 +1424 -0
- data/test/dummy/tmp/cache/assets/D0A/790/sprockets%2Feb16913e6504c9b0d3be431de39e4751 +300 -0
- data/test/dummy/tmp/cache/assets/D0C/C50/sprockets%2Fd6613bee8b40d50459af6b52a7084f34 +796 -0
- data/test/dummy/tmp/cache/assets/D0D/030/sprockets%2Fba3f13b4a444679e8bc2549226ec743b +21 -0
- data/test/dummy/tmp/cache/assets/D0D/350/sprockets%2F2670bce036d485e15d059c0f1e98f24a +207 -0
- data/test/dummy/tmp/cache/assets/D13/270/sprockets%2F497cb163e6317e3fc1565d832f406cfb +348 -0
- data/test/dummy/tmp/cache/assets/D13/380/sprockets%2F786d003c9c7fb759dd26c1030c087cf6 +813 -0
- data/test/dummy/tmp/cache/assets/D13/7C0/sprockets%2F528d22310a9ab8e6fba08d82844ce795 +0 -0
- data/test/dummy/tmp/cache/assets/D15/F60/sprockets%2Fa28394e3f80365b5bc86794dd46daa22 +0 -0
- data/test/dummy/tmp/cache/assets/D18/500/sprockets%2Fd22c2d97d1db2154f2f7592906e957ea +1001 -0
- data/test/dummy/tmp/cache/assets/D1B/C70/sprockets%2F1c04848b1e1d6e8e33b8581f8c8128ff +223 -0
- data/test/dummy/tmp/cache/assets/D1C/600/sprockets%2Fa0601f99147f59ddd6266e6aff077e14 +0 -0
- data/test/dummy/tmp/cache/assets/D1E/470/sprockets%2Fb3933e694547b78bf6fb15f44a4623fe +53 -0
- data/test/dummy/tmp/cache/assets/D1F/480/sprockets%2F545164168becf42b289efc8708f1db68 +1278 -0
- data/test/dummy/tmp/cache/assets/D21/630/sprockets%2F0a14114729c14637f2e8f122acabd1ab +823 -0
- data/test/dummy/tmp/cache/assets/D24/A90/sprockets%2F21c8a0d48c51b8585ba03bc112ecd153 +692 -0
- data/test/dummy/tmp/cache/assets/D25/D30/sprockets%2F3b937f4c16d2aa0a28fc02a0e922712b +1441 -0
- data/test/dummy/tmp/cache/assets/D25/F30/sprockets%2Fe251527c0c018f4b4ad44b2ad7064fb0 +2255 -0
- data/test/dummy/tmp/cache/assets/D26/DE0/sprockets%2F761ac1bdb0739cc56fa0c1224b137b09 +0 -0
- data/test/dummy/tmp/cache/assets/D2A/C30/sprockets%2F978fe7268754f451c59b9ccb323ffa56 +0 -0
- data/test/dummy/tmp/cache/assets/D2B/0F0/sprockets%2Fbf223af03335cfc08a71e549e10c0e27 +2539 -0
- data/test/dummy/tmp/cache/assets/D2E/6D0/sprockets%2F9e7fda3affb389313cc50223f01c0324 +0 -0
- data/test/dummy/tmp/cache/assets/D2F/110/sprockets%2F3730ffe2ed34c1219d580bd1615ea7b0 +2239 -0
- data/test/dummy/tmp/cache/assets/D2F/F10/sprockets%2F9cac242f0bf5ec00079ea3a463e26552 +153 -0
- data/test/dummy/tmp/cache/assets/D30/1D0/sprockets%2Fd690a26a0b0ae138c1c48d257a7674bc +589 -0
- data/test/dummy/tmp/cache/assets/D32/A10/sprockets%2F13fe41fee1fe35b49d145bcc06610705 +0 -0
- data/test/dummy/tmp/cache/assets/D36/940/sprockets%2Fb4ba462ac9f4aac561c36e60367c1378 +572 -0
- data/test/dummy/tmp/cache/assets/D37/090/sprockets%2F2f2e71ebbc240014ebf648f0917e854a +798 -0
- data/test/dummy/tmp/cache/assets/D3A/440/sprockets%2F6d332243647c841dea36ed822cdfc23e +165 -0
- data/test/dummy/tmp/cache/assets/D3B/5A0/sprockets%2F82e05cf8c88ba8cb1c87e50d51372a03 +0 -0
- data/test/dummy/tmp/cache/assets/D41/B30/sprockets%2Fba1b93913dd01d83ac9a96df334456f8 +0 -0
- data/test/dummy/tmp/cache/assets/D43/D90/sprockets%2F04f3a5926d5a7672456dc7afa73d0c7f +38 -0
- data/test/dummy/tmp/cache/assets/D45/1F0/sprockets%2F1bfa855d9c10ff2431a9a548958cc5e4 +83 -0
- data/test/dummy/tmp/cache/assets/D45/B10/sprockets%2F643f3d9b216e48df952b7f150a46a7ec +60 -0
- data/test/dummy/tmp/cache/assets/D46/040/sprockets%2F199546844e5939721a5afcbcce4ea43a +143 -0
- data/test/dummy/tmp/cache/assets/D49/320/sprockets%2Fd8123eb229e34c9f507f7b1876f1c4ee +506 -0
- data/test/dummy/tmp/cache/assets/D49/750/sprockets%2Ff41ef1c0832b7294fb15588ff5c783cb +0 -0
- data/test/dummy/tmp/cache/assets/D49/930/sprockets%2F1071926698fa55abc71e1b6fd966dfe1 +72 -0
- data/test/dummy/tmp/cache/assets/D49/D10/sprockets%2Fb67b8d7cc7579352a91694f9ae10cdb0 +127 -0
- data/test/dummy/tmp/cache/assets/D4A/160/sprockets%2Fcbd8988b5e18d153eff6230d72e9046a +516 -0
- data/test/dummy/tmp/cache/assets/D4A/2B0/sprockets%2Ffcdf002d38fd938bf7226b46b76706c8 +1101 -0
- data/test/dummy/tmp/cache/assets/D4E/2F0/sprockets%2F77cb87ae57f0d8c25cdaa74181a64997 +1441 -0
- data/test/dummy/tmp/cache/assets/D4F/060/sprockets%2Fa9e66b39ada7394a29bf44f3a682f665 +511 -0
- data/test/dummy/tmp/cache/assets/D50/BD0/sprockets%2F88650dd57ef64075462e6dae757dbe2c +0 -0
- data/test/dummy/tmp/cache/assets/D53/CA0/sprockets%2Fa85032e82709a043fbb2ec00e04f2bbd +0 -0
- data/test/dummy/tmp/cache/assets/D54/BF0/sprockets%2F91970514ff528e8d2bfd81f1ec83c9c8 +240 -0
- data/test/dummy/tmp/cache/assets/D54/ED0/sprockets%2F71c9fa01091d432b131da3bb73faf3d4 +10 -0
- data/test/dummy/tmp/cache/assets/D56/500/sprockets%2F75ebadd035f1324b194034af92eac3a0 +0 -0
- data/test/dummy/tmp/cache/assets/D5A/900/sprockets%2F5729f77d97fdee53b1942cf17f6f05e4 +474 -0
- data/test/dummy/tmp/cache/assets/D5A/D50/sprockets%2F81bace3db2c2f1241175ff3c5009d08c +0 -0
- data/test/dummy/tmp/cache/assets/D5B/C70/sprockets%2Fe64119a9db7017b7ab3b0da1b6076f0d +44 -0
- data/test/dummy/tmp/cache/assets/D5E/4D0/sprockets%2F756f25e4ad861a050cdc41bb8414e4ab +126 -0
- data/test/dummy/tmp/cache/assets/D5E/730/sprockets%2Fd30caf70b387c4604326bdda32aeb549 +0 -0
- data/test/dummy/tmp/cache/assets/D5E/AA0/sprockets%2F790dd07caaaacb30ceb0174664e90817 +136 -0
- data/test/dummy/tmp/cache/assets/D64/D30/sprockets%2Fe4bd8f60a3a446274c3cb2f5d16563ac +590 -0
- data/test/dummy/tmp/cache/assets/D66/400/sprockets%2Fecb7401d20daf25605a7afad3a793778 +0 -0
- data/test/dummy/tmp/cache/assets/D67/250/sprockets%2F8422eb26855ca9c953a3bf1aeade8004 +0 -0
- data/test/dummy/tmp/cache/assets/D69/900/sprockets%2F28ae64f52cb18f77b1a1bd9c51293da4 +0 -0
- data/test/dummy/tmp/cache/assets/D6A/870/sprockets%2F1ece6a51b42f9280f4cee9020c94b72e +197 -0
- data/test/dummy/tmp/cache/assets/D6B/DD0/sprockets%2F9a0abc735e27fc61dab019788518eba4 +164 -0
- data/test/dummy/tmp/cache/assets/D6E/CC0/sprockets%2F8b4fbcec2475e7203b8859baf8e310a3 +0 -0
- data/test/dummy/tmp/cache/assets/D72/470/sprockets%2Fbc0c6d850e7c769e83d58eb3e6061b2a +2556 -0
- data/test/dummy/tmp/cache/assets/D72/9D0/sprockets%2F9402daf6da6f7eed331a9d78216cb761 +0 -0
- data/test/dummy/tmp/cache/assets/D74/070/sprockets%2F91ae68f698a39bd7d0d11150fc46e4cd +418 -0
- data/test/dummy/tmp/cache/assets/D74/7F0/sprockets%2F22dad1069ea6a8f29ae757c44e18ec83 +0 -0
- data/test/dummy/tmp/cache/assets/D75/3C0/sprockets%2F44e2ae51b97853d8ecda7b264a267c7d +1630 -0
- data/test/dummy/tmp/cache/assets/D77/780/sprockets%2Ffa2abdfb0b62867c346f79187b8e42d5 +823 -0
- data/test/dummy/tmp/cache/assets/D79/850/sprockets%2F4d540586ed379fd9a70fa0b7ce4f5b27 +306 -0
- data/test/dummy/tmp/cache/assets/D7A/B60/sprockets%2F5925c92f48caff053d08853dc69e5bba +0 -0
- data/test/dummy/tmp/cache/assets/D81/C90/sprockets%2F19aca5c835efce74878c23e581e2a89e +2239 -0
- data/test/dummy/tmp/cache/assets/D84/210/sprockets%2Fabd0103ccec2b428ac62c94e4c40b384 +11228 -0
- data/test/dummy/tmp/cache/assets/D8B/B10/sprockets%2Fbb7a2305a8f2832631863bbababb67be +301 -0
- data/test/dummy/tmp/cache/assets/D92/3B0/sprockets%2F93dc71f411faf72ec8ef0b5d217b51a0 +505 -0
- data/test/dummy/tmp/cache/assets/D92/EE0/sprockets%2Ffd051ea0abf6be6970754aa732e40c8b +289 -0
- data/test/dummy/tmp/cache/assets/D93/810/sprockets%2F7dc1ac5e1b83ba5144e34cf156d48a8a +471 -0
- data/test/dummy/tmp/cache/assets/D94/020/sprockets%2F35cb4ce22488fb5bdd260508ca47a3fb +239 -0
- data/test/dummy/tmp/cache/assets/D94/EC0/sprockets%2F8b900fba18f8e046f5f12e6d1f0ac15b +74 -0
- data/test/dummy/tmp/cache/assets/D95/470/sprockets%2F5be5d361a52575c433831b5e8cccbace +9273 -0
- data/test/dummy/tmp/cache/assets/D97/9D0/sprockets%2F524a2a94a88c1d392dabcba654e365fc +103 -0
- data/test/dummy/tmp/cache/assets/D9D/800/sprockets%2Ff8387220eb4dda9b29a37b5deb9f2a40 +54 -0
- data/test/dummy/tmp/cache/assets/D9E/DD0/sprockets%2Fb6b4ffad6485eda046b327a196e8c44a +0 -0
- data/test/dummy/tmp/cache/assets/DA1/0A0/sprockets%2F359c94aee6ba2e15fc9666aa4de311c8 +166 -0
- data/test/dummy/tmp/cache/assets/DA1/0D0/sprockets%2F798de3c28452e164a7fc1a3a6e8cdb4b +652 -0
- data/test/dummy/tmp/cache/assets/DA3/CB0/sprockets%2Fdfad6bb1857a8af91f98ba137c18504e +815 -0
- data/test/dummy/tmp/cache/assets/DA4/E10/sprockets%2Fbda7705734ace95f12a7d79df9c14af2 +1354 -0
- data/test/dummy/tmp/cache/assets/DA5/E30/sprockets%2F2e6707a119e94a784efbadca18d7e8d1 +129 -0
- data/test/dummy/tmp/cache/assets/DA6/4B0/sprockets%2F4f7a708b6aa2d42eb577493dea7cef55 +115 -0
- data/test/dummy/tmp/cache/assets/DA8/A10/sprockets%2F0ae5bbb4ca200e74db3b64d72c2be120 +0 -0
- data/test/dummy/tmp/cache/assets/DA9/3F0/sprockets%2Fdd0af9cfd65fb7d02fe778c8f6015361 +499 -0
- data/test/dummy/tmp/cache/assets/DA9/780/sprockets%2Fe3559a635e8f92dec7727a3db2ae7b4c +813 -0
- data/test/dummy/tmp/cache/assets/DAC/0E0/sprockets%2F04cef8243a9ef7321846debc977ea8db +607 -0
- data/test/dummy/tmp/cache/assets/DB1/260/sprockets%2Fb879ff7530bf8cc175e0c7e70dc4e79b +0 -0
- data/test/dummy/tmp/cache/assets/DB4/4C0/sprockets%2F30a76cbd98dedb56742cd8ecaf858757 +0 -0
- data/test/dummy/tmp/cache/assets/DBD/7E0/sprockets%2F236fb1ac5ed9f2e3fe32203acc3a842d +0 -0
- data/test/dummy/tmp/cache/assets/DC2/DF0/sprockets%2Fe233bcaba8beff15626c35cb5e0e0936 +845 -0
- data/test/dummy/tmp/cache/assets/DC4/600/sprockets%2F59c96aa6c1cbebc61bff6c050a5351d7 +225 -0
- data/test/dummy/tmp/cache/assets/DC5/030/sprockets%2Fcb2fe864a0177eec19017c43ce9aa4ad +187 -0
- data/test/dummy/tmp/cache/assets/DC8/A60/sprockets%2F9fcb9f5c3f679ce749ee4c3f93869ba6 +74 -0
- data/test/dummy/tmp/cache/assets/DCB/D40/sprockets%2Ffd5542bfb7660a8d8cac3a2e46fb01f8 +828 -0
- data/test/dummy/tmp/cache/assets/DCC/020/sprockets%2F0c129c5f2784c96fa1dc3da6c19c1efc +128 -0
- data/test/dummy/tmp/cache/assets/DCF/520/sprockets%2F975162fb97a3ec7033db5f2fdba4fcd3 +70 -0
- data/test/dummy/tmp/cache/assets/DD0/000/sprockets%2F0d5d631cce6f0b078bb29cf98b3d78bb +276 -0
- data/test/dummy/tmp/cache/assets/DD0/C90/sprockets%2F84eeca4ef8f5eb29b13f7e1ea3c71324 +591 -0
- data/test/dummy/tmp/cache/assets/DD6/710/sprockets%2F9e83f2ebbe3e349efca2de390e15d635 +204 -0
- data/test/dummy/tmp/cache/assets/DE1/830/sprockets%2Fd23d3cafef2e2a9055bc103dffc1a023 +208 -0
- data/test/dummy/tmp/cache/assets/DEB/470/sprockets%2Fae2de0fcbc2214b71d3f2a875d2bc8c3 +2255 -0
- data/test/dummy/tmp/cache/assets/DEE/690/sprockets%2F5d824bcadef29060cc13eed71af4b4d0 +550 -0
- data/test/dummy/tmp/cache/assets/DF0/5D0/sprockets%2F25a875eb0a5d4c49bcc3fa39ea0adb26 +567 -0
- data/test/dummy/tmp/cache/assets/DF3/E20/sprockets%2Fada770c49a5d6b38d1cc43ed4d51c1fc +0 -0
- data/test/dummy/tmp/cache/assets/DF5/C80/sprockets%2F03ad7d7b3a8feba9ed9be82ae2083b20 +340 -0
- data/test/dummy/tmp/cache/assets/DF6/560/sprockets%2Fafceb2b55eb108efe4446dfb02b519b8 +0 -0
- data/test/dummy/tmp/cache/assets/DFC/030/sprockets%2Fcbdd3679eaba219b4e1ff20677af1c6f +1371 -0
- data/test/dummy/tmp/cache/assets/E00/970/sprockets%2F8bc6dde8d05c382dd07cbcec70f54e09 +378 -0
- data/test/dummy/tmp/cache/assets/E04/890/sprockets%2F2f5173deea6c795b8fdde723bb4b63af +0 -0
- data/test/dummy/tmp/cache/assets/E12/2F0/sprockets%2F5c45062ecfbe512dcba176ba6ddacc00 +523 -0
- data/test/dummy/tmp/cache/assets/E21/5D0/sprockets%2Fd4fe6d611bfb0e63c04de5dbfb128a8a +0 -0
- data/test/dummy/tmp/cache/assets/E22/770/sprockets%2F2ccebcabfbb9b59159e4b73b3b30a74d +208 -0
- data/test/dummy/tmp/cache/assets/E31/A90/sprockets%2F26b98bbbc15eee9b370c7dcde88d16aa +394 -0
- data/test/dummy/tmp/cache/assets/E31/E30/sprockets%2Ffbdafcc4e2f44bb9fc653bcf9625b750 +284 -0
- data/test/dummy/tmp/cache/assets/E37/1C0/sprockets%2Fa5eb45fc86dafe2bee4e4fa3f53c4785 +0 -0
- data/test/dummy/tmp/cache/assets/E4C/960/sprockets%2F13fd6e4bfdf031acb3dedca8d0f407b4 +1277 -0
- data/test/dummy/tmp/cache/assets/E88/330/sprockets%2Fe02d0adb5529ffafcea1b9e5ccf69a5b +669 -0
- data/test/dummy/tmp/pids/server.pid +1 -0
- data/test/dummy/tmp/simple.js.js +20 -0
- data/test/dummy/tmp/soy-1325274421.js +20 -0
- data/test/dummy/tmp/soy-1325286267.js +20 -0
- data/test/test_helper.rb +10 -0
- data/test/yellow-brick-road_test.rb +7 -0
- metadata +538 -0
@@ -0,0 +1,1354 @@
|
|
1
|
+
o: ActiveSupport::Cache::Entry :@compressedF:@expires_in0:@created_atf1325458072.2967958:@value{ I"length:EFi��I"digest;
|
2
|
+
F"%76437b1cb251e11f06c726ab3c5118ceI"source;
|
3
|
+
FI"��// 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 arrays.
|
19
|
+
*
|
20
|
+
*/
|
21
|
+
|
22
|
+
|
23
|
+
|
24
|
+
goog.provide('goog.array');
|
25
|
+
goog.provide('goog.array.ArrayLike');
|
26
|
+
|
27
|
+
goog.require('goog.asserts');
|
28
|
+
|
29
|
+
|
30
|
+
/**
|
31
|
+
* @define {boolean} NATIVE_ARRAY_PROTOTYPES indicates whether the code should
|
32
|
+
* rely on Array.prototype functions, if available.
|
33
|
+
*
|
34
|
+
* The Array.prototype functions can be defined by external libraries like
|
35
|
+
* Prototype and setting this flag to false forces closure to use its own
|
36
|
+
* goog.array implementation.
|
37
|
+
*
|
38
|
+
* If your javascript can be loaded by a third party site and you are wary about
|
39
|
+
* relying on the prototype functions, specify
|
40
|
+
* "--define goog.NATIVE_ARRAY_PROTOTYPES=false" to the JSCompiler.
|
41
|
+
*/
|
42
|
+
goog.NATIVE_ARRAY_PROTOTYPES = true;
|
43
|
+
|
44
|
+
|
45
|
+
/**
|
46
|
+
* @typedef {Array|NodeList|Arguments|{length: number}}
|
47
|
+
*/
|
48
|
+
goog.array.ArrayLike;
|
49
|
+
|
50
|
+
|
51
|
+
/**
|
52
|
+
* Returns the last element in an array without removing it.
|
53
|
+
* @param {goog.array.ArrayLike} array The array.
|
54
|
+
* @return {*} Last item in array.
|
55
|
+
*/
|
56
|
+
goog.array.peek = function(array) {
|
57
|
+
return array[array.length - 1];
|
58
|
+
};
|
59
|
+
|
60
|
+
|
61
|
+
/**
|
62
|
+
* Reference to the original {@code Array.prototype}.
|
63
|
+
* @private
|
64
|
+
*/
|
65
|
+
goog.array.ARRAY_PROTOTYPE_ = Array.prototype;
|
66
|
+
|
67
|
+
|
68
|
+
// NOTE(arv): Since most of the array functions are generic it allows you to
|
69
|
+
// pass an array-like object. Strings have a length and are considered array-
|
70
|
+
// like. However, the 'in' operator does not work on strings so we cannot just
|
71
|
+
// use the array path even if the browser supports indexing into strings. We
|
72
|
+
// therefore end up splitting the string.
|
73
|
+
|
74
|
+
|
75
|
+
/**
|
76
|
+
* Returns the index of the first element of an array with a specified
|
77
|
+
* value, or -1 if the element is not present in the array.
|
78
|
+
*
|
79
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-indexof}
|
80
|
+
*
|
81
|
+
* @param {goog.array.ArrayLike} arr The array to be searched.
|
82
|
+
* @param {*} obj The object for which we are searching.
|
83
|
+
* @param {number=} opt_fromIndex The index at which to start the search. If
|
84
|
+
* omitted the search starts at index 0.
|
85
|
+
* @return {number} The index of the first matching array element.
|
86
|
+
*/
|
87
|
+
goog.array.indexOf = goog.NATIVE_ARRAY_PROTOTYPES &&
|
88
|
+
goog.array.ARRAY_PROTOTYPE_.indexOf ?
|
89
|
+
function(arr, obj, opt_fromIndex) {
|
90
|
+
goog.asserts.assert(arr.length != null);
|
91
|
+
|
92
|
+
return goog.array.ARRAY_PROTOTYPE_.indexOf.call(arr, obj, opt_fromIndex);
|
93
|
+
} :
|
94
|
+
function(arr, obj, opt_fromIndex) {
|
95
|
+
var fromIndex = opt_fromIndex == null ?
|
96
|
+
0 : (opt_fromIndex < 0 ?
|
97
|
+
Math.max(0, arr.length + opt_fromIndex) : opt_fromIndex);
|
98
|
+
|
99
|
+
if (goog.isString(arr)) {
|
100
|
+
// Array.prototype.indexOf uses === so only strings should be found.
|
101
|
+
if (!goog.isString(obj) || obj.length != 1) {
|
102
|
+
return -1;
|
103
|
+
}
|
104
|
+
return arr.indexOf(obj, fromIndex);
|
105
|
+
}
|
106
|
+
|
107
|
+
for (var i = fromIndex; i < arr.length; i++) {
|
108
|
+
if (i in arr && arr[i] === obj)
|
109
|
+
return i;
|
110
|
+
}
|
111
|
+
return -1;
|
112
|
+
};
|
113
|
+
|
114
|
+
|
115
|
+
/**
|
116
|
+
* Returns the index of the last element of an array with a specified value, or
|
117
|
+
* -1 if the element is not present in the array.
|
118
|
+
*
|
119
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-lastindexof}
|
120
|
+
*
|
121
|
+
* @param {goog.array.ArrayLike} arr The array to be searched.
|
122
|
+
* @param {*} obj The object for which we are searching.
|
123
|
+
* @param {?number=} opt_fromIndex The index at which to start the search. If
|
124
|
+
* omitted the search starts at the end of the array.
|
125
|
+
* @return {number} The index of the last matching array element.
|
126
|
+
*/
|
127
|
+
goog.array.lastIndexOf = goog.NATIVE_ARRAY_PROTOTYPES &&
|
128
|
+
goog.array.ARRAY_PROTOTYPE_.lastIndexOf ?
|
129
|
+
function(arr, obj, opt_fromIndex) {
|
130
|
+
goog.asserts.assert(arr.length != null);
|
131
|
+
|
132
|
+
// Firefox treats undefined and null as 0 in the fromIndex argument which
|
133
|
+
// leads it to always return -1
|
134
|
+
var fromIndex = opt_fromIndex == null ? arr.length - 1 : opt_fromIndex;
|
135
|
+
return goog.array.ARRAY_PROTOTYPE_.lastIndexOf.call(arr, obj, fromIndex);
|
136
|
+
} :
|
137
|
+
function(arr, obj, opt_fromIndex) {
|
138
|
+
var fromIndex = opt_fromIndex == null ? arr.length - 1 : opt_fromIndex;
|
139
|
+
|
140
|
+
if (fromIndex < 0) {
|
141
|
+
fromIndex = Math.max(0, arr.length + fromIndex);
|
142
|
+
}
|
143
|
+
|
144
|
+
if (goog.isString(arr)) {
|
145
|
+
// Array.prototype.lastIndexOf uses === so only strings should be found.
|
146
|
+
if (!goog.isString(obj) || obj.length != 1) {
|
147
|
+
return -1;
|
148
|
+
}
|
149
|
+
return arr.lastIndexOf(obj, fromIndex);
|
150
|
+
}
|
151
|
+
|
152
|
+
for (var i = fromIndex; i >= 0; i--) {
|
153
|
+
if (i in arr && arr[i] === obj)
|
154
|
+
return i;
|
155
|
+
}
|
156
|
+
return -1;
|
157
|
+
};
|
158
|
+
|
159
|
+
|
160
|
+
/**
|
161
|
+
* Calls a function for each element in an array.
|
162
|
+
*
|
163
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-foreach}
|
164
|
+
*
|
165
|
+
* @param {goog.array.ArrayLike} arr Array or array like object over
|
166
|
+
* which to iterate.
|
167
|
+
* @param {?function(this: T, ...)} f The function to call for every element.
|
168
|
+
* This function takes 3 arguments (the element, the index and the array).
|
169
|
+
* The return value is ignored. The function is called only for indexes of
|
170
|
+
* the array which have assigned values; it is not called for indexes which
|
171
|
+
* have been deleted or which have never been assigned values.
|
172
|
+
* @param {T=} opt_obj The object to be used as the value of 'this'
|
173
|
+
* within f.
|
174
|
+
* @template T
|
175
|
+
*/
|
176
|
+
goog.array.forEach = goog.NATIVE_ARRAY_PROTOTYPES &&
|
177
|
+
goog.array.ARRAY_PROTOTYPE_.forEach ?
|
178
|
+
function(arr, f, opt_obj) {
|
179
|
+
goog.asserts.assert(arr.length != null);
|
180
|
+
|
181
|
+
goog.array.ARRAY_PROTOTYPE_.forEach.call(arr, f, opt_obj);
|
182
|
+
} :
|
183
|
+
function(arr, f, opt_obj) {
|
184
|
+
var l = arr.length; // must be fixed during loop... see docs
|
185
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
186
|
+
for (var i = 0; i < l; i++) {
|
187
|
+
if (i in arr2) {
|
188
|
+
f.call(opt_obj, arr2[i], i, arr);
|
189
|
+
}
|
190
|
+
}
|
191
|
+
};
|
192
|
+
|
193
|
+
|
194
|
+
/**
|
195
|
+
* Calls a function for each element in an array, starting from the last
|
196
|
+
* element rather than the first.
|
197
|
+
*
|
198
|
+
* @param {goog.array.ArrayLike} arr The array over which to iterate.
|
199
|
+
* @param {Function} f The function to call for every element. This function
|
200
|
+
* takes 3 arguments (the element, the index and the array). The return
|
201
|
+
* value is ignored.
|
202
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
203
|
+
* within f.
|
204
|
+
*/
|
205
|
+
goog.array.forEachRight = function(arr, f, opt_obj) {
|
206
|
+
var l = arr.length; // must be fixed during loop... see docs
|
207
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
208
|
+
for (var i = l - 1; i >= 0; --i) {
|
209
|
+
if (i in arr2) {
|
210
|
+
f.call(opt_obj, arr2[i], i, arr);
|
211
|
+
}
|
212
|
+
}
|
213
|
+
};
|
214
|
+
|
215
|
+
|
216
|
+
/**
|
217
|
+
* Calls a function for each element in an array, and if the function returns
|
218
|
+
* true adds the element to a new array.
|
219
|
+
*
|
220
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-filter}
|
221
|
+
*
|
222
|
+
* @param {goog.array.ArrayLike} arr The array over which to iterate.
|
223
|
+
* @param {Function} f The function to call for every element. This function
|
224
|
+
* takes 3 arguments (the element, the index and the array) and must
|
225
|
+
* return a Boolean. If the return value is true the element is added to the
|
226
|
+
* result array. If it is false the element is not included.
|
227
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
228
|
+
* within f.
|
229
|
+
* @return {!Array} a new array in which only elements that passed the test are
|
230
|
+
* present.
|
231
|
+
*/
|
232
|
+
goog.array.filter = goog.NATIVE_ARRAY_PROTOTYPES &&
|
233
|
+
goog.array.ARRAY_PROTOTYPE_.filter ?
|
234
|
+
function(arr, f, opt_obj) {
|
235
|
+
goog.asserts.assert(arr.length != null);
|
236
|
+
|
237
|
+
return goog.array.ARRAY_PROTOTYPE_.filter.call(arr, f, opt_obj);
|
238
|
+
} :
|
239
|
+
function(arr, f, opt_obj) {
|
240
|
+
var l = arr.length; // must be fixed during loop... see docs
|
241
|
+
var res = [];
|
242
|
+
var resLength = 0;
|
243
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
244
|
+
for (var i = 0; i < l; i++) {
|
245
|
+
if (i in arr2) {
|
246
|
+
var val = arr2[i]; // in case f mutates arr2
|
247
|
+
if (f.call(opt_obj, val, i, arr)) {
|
248
|
+
res[resLength++] = val;
|
249
|
+
}
|
250
|
+
}
|
251
|
+
}
|
252
|
+
return res;
|
253
|
+
};
|
254
|
+
|
255
|
+
|
256
|
+
/**
|
257
|
+
* Calls a function for each element in an array and inserts the result into a
|
258
|
+
* new array.
|
259
|
+
*
|
260
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-map}
|
261
|
+
*
|
262
|
+
* @param {goog.array.ArrayLike} arr The array over which to iterate.
|
263
|
+
* @param {Function} f The function to call for every element. This function
|
264
|
+
* takes 3 arguments (the element, the index and the array) and should
|
265
|
+
* return something. The result will be inserted into a new array.
|
266
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
267
|
+
* within f.
|
268
|
+
* @return {!Array} a new array with the results from f.
|
269
|
+
*/
|
270
|
+
goog.array.map = goog.NATIVE_ARRAY_PROTOTYPES &&
|
271
|
+
goog.array.ARRAY_PROTOTYPE_.map ?
|
272
|
+
function(arr, f, opt_obj) {
|
273
|
+
goog.asserts.assert(arr.length != null);
|
274
|
+
|
275
|
+
return goog.array.ARRAY_PROTOTYPE_.map.call(arr, f, opt_obj);
|
276
|
+
} :
|
277
|
+
function(arr, f, opt_obj) {
|
278
|
+
var l = arr.length; // must be fixed during loop... see docs
|
279
|
+
var res = new Array(l);
|
280
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
281
|
+
for (var i = 0; i < l; i++) {
|
282
|
+
if (i in arr2) {
|
283
|
+
res[i] = f.call(opt_obj, arr2[i], i, arr);
|
284
|
+
}
|
285
|
+
}
|
286
|
+
return res;
|
287
|
+
};
|
288
|
+
|
289
|
+
|
290
|
+
/**
|
291
|
+
* Passes every element of an array into a function and accumulates the result.
|
292
|
+
*
|
293
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-reduce}
|
294
|
+
*
|
295
|
+
* For example:
|
296
|
+
* var a = [1, 2, 3, 4];
|
297
|
+
* goog.array.reduce(a, function(r, v, i, arr) {return r + v;}, 0);
|
298
|
+
* returns 10
|
299
|
+
*
|
300
|
+
* @param {goog.array.ArrayLike} arr The array over which to iterate.
|
301
|
+
* @param {Function} f The function to call for every element. This function
|
302
|
+
* takes 4 arguments (the function's previous result or the initial value,
|
303
|
+
* the value of the current array element, the current array index, and the
|
304
|
+
* array itself)
|
305
|
+
* function(previousValue, currentValue, index, array).
|
306
|
+
* @param {*} val The initial value to pass into the function on the first call.
|
307
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
308
|
+
* within f.
|
309
|
+
* @return {*} Result of evaluating f repeatedly across the values of the array.
|
310
|
+
*/
|
311
|
+
goog.array.reduce = function(arr, f, val, opt_obj) {
|
312
|
+
if (arr.reduce) {
|
313
|
+
if (opt_obj) {
|
314
|
+
return arr.reduce(goog.bind(f, opt_obj), val);
|
315
|
+
} else {
|
316
|
+
return arr.reduce(f, val);
|
317
|
+
}
|
318
|
+
}
|
319
|
+
var rval = val;
|
320
|
+
goog.array.forEach(arr, function(val, index) {
|
321
|
+
rval = f.call(opt_obj, rval, val, index, arr);
|
322
|
+
});
|
323
|
+
return rval;
|
324
|
+
};
|
325
|
+
|
326
|
+
|
327
|
+
/**
|
328
|
+
* Passes every element of an array into a function and accumulates the result,
|
329
|
+
* starting from the last element and working towards the first.
|
330
|
+
*
|
331
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-reduceright}
|
332
|
+
*
|
333
|
+
* For example:
|
334
|
+
* var a = ['a', 'b', 'c'];
|
335
|
+
* goog.array.reduceRight(a, function(r, v, i, arr) {return r + v;}, '');
|
336
|
+
* returns 'cba'
|
337
|
+
*
|
338
|
+
* @param {goog.array.ArrayLike} arr The array over which to iterate.
|
339
|
+
* @param {Function} f The function to call for every element. This function
|
340
|
+
* takes 4 arguments (the function's previous result or the initial value,
|
341
|
+
* the value of the current array element, the current array index, and the
|
342
|
+
* array itself)
|
343
|
+
* function(previousValue, currentValue, index, array).
|
344
|
+
* @param {*} val The initial value to pass into the function on the first call.
|
345
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
346
|
+
* within f.
|
347
|
+
* @return {*} Object returned as a result of evaluating f repeatedly across the
|
348
|
+
* values of the array.
|
349
|
+
*/
|
350
|
+
goog.array.reduceRight = function(arr, f, val, opt_obj) {
|
351
|
+
if (arr.reduceRight) {
|
352
|
+
if (opt_obj) {
|
353
|
+
return arr.reduceRight(goog.bind(f, opt_obj), val);
|
354
|
+
} else {
|
355
|
+
return arr.reduceRight(f, val);
|
356
|
+
}
|
357
|
+
}
|
358
|
+
var rval = val;
|
359
|
+
goog.array.forEachRight(arr, function(val, index) {
|
360
|
+
rval = f.call(opt_obj, rval, val, index, arr);
|
361
|
+
});
|
362
|
+
return rval;
|
363
|
+
};
|
364
|
+
|
365
|
+
|
366
|
+
/**
|
367
|
+
* Calls f for each element of an array. If any call returns true, some()
|
368
|
+
* returns true (without checking the remaining elements). If all calls
|
369
|
+
* return false, some() returns false.
|
370
|
+
*
|
371
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-some}
|
372
|
+
*
|
373
|
+
* @param {goog.array.ArrayLike} arr The array to check.
|
374
|
+
* @param {Function} f The function to call for every element. This function
|
375
|
+
* takes 3 arguments (the element, the index and the array) and must
|
376
|
+
* return a Boolean.
|
377
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
378
|
+
* within f.
|
379
|
+
* @return {boolean} true if any element passes the test.
|
380
|
+
*/
|
381
|
+
goog.array.some = goog.NATIVE_ARRAY_PROTOTYPES &&
|
382
|
+
goog.array.ARRAY_PROTOTYPE_.some ?
|
383
|
+
function(arr, f, opt_obj) {
|
384
|
+
goog.asserts.assert(arr.length != null);
|
385
|
+
|
386
|
+
return goog.array.ARRAY_PROTOTYPE_.some.call(arr, f, opt_obj);
|
387
|
+
} :
|
388
|
+
function(arr, f, opt_obj) {
|
389
|
+
var l = arr.length; // must be fixed during loop... see docs
|
390
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
391
|
+
for (var i = 0; i < l; i++) {
|
392
|
+
if (i in arr2 && f.call(opt_obj, arr2[i], i, arr)) {
|
393
|
+
return true;
|
394
|
+
}
|
395
|
+
}
|
396
|
+
return false;
|
397
|
+
};
|
398
|
+
|
399
|
+
|
400
|
+
/**
|
401
|
+
* Call f for each element of an array. If all calls return true, every()
|
402
|
+
* returns true. If any call returns false, every() returns false and
|
403
|
+
* does not continue to check the remaining elements.
|
404
|
+
*
|
405
|
+
* See {@link http://tinyurl.com/developer-mozilla-org-array-every}
|
406
|
+
*
|
407
|
+
* @param {goog.array.ArrayLike} arr The array to check.
|
408
|
+
* @param {Function} f The function to call for every element. This function
|
409
|
+
* takes 3 arguments (the element, the index and the array) and must
|
410
|
+
* return a Boolean.
|
411
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
412
|
+
* within f.
|
413
|
+
* @return {boolean} false if any element fails the test.
|
414
|
+
*/
|
415
|
+
goog.array.every = goog.NATIVE_ARRAY_PROTOTYPES &&
|
416
|
+
goog.array.ARRAY_PROTOTYPE_.every ?
|
417
|
+
function(arr, f, opt_obj) {
|
418
|
+
goog.asserts.assert(arr.length != null);
|
419
|
+
|
420
|
+
return goog.array.ARRAY_PROTOTYPE_.every.call(arr, f, opt_obj);
|
421
|
+
} :
|
422
|
+
function(arr, f, opt_obj) {
|
423
|
+
var l = arr.length; // must be fixed during loop... see docs
|
424
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
425
|
+
for (var i = 0; i < l; i++) {
|
426
|
+
if (i in arr2 && !f.call(opt_obj, arr2[i], i, arr)) {
|
427
|
+
return false;
|
428
|
+
}
|
429
|
+
}
|
430
|
+
return true;
|
431
|
+
};
|
432
|
+
|
433
|
+
|
434
|
+
/**
|
435
|
+
* Search an array for the first element that satisfies a given condition and
|
436
|
+
* return that element.
|
437
|
+
* @param {goog.array.ArrayLike} arr The array to search.
|
438
|
+
* @param {Function} f The function to call for every element. This function
|
439
|
+
* takes 3 arguments (the element, the index and the array) and should
|
440
|
+
* return a boolean.
|
441
|
+
* @param {Object=} opt_obj An optional "this" context for the function.
|
442
|
+
* @return {*} The first array element that passes the test, or null if no
|
443
|
+
* element is found.
|
444
|
+
*/
|
445
|
+
goog.array.find = function(arr, f, opt_obj) {
|
446
|
+
var i = goog.array.findIndex(arr, f, opt_obj);
|
447
|
+
return i < 0 ? null : goog.isString(arr) ? arr.charAt(i) : arr[i];
|
448
|
+
};
|
449
|
+
|
450
|
+
|
451
|
+
/**
|
452
|
+
* Search an array for the first element that satisfies a given condition and
|
453
|
+
* return its index.
|
454
|
+
* @param {goog.array.ArrayLike} arr The array to search.
|
455
|
+
* @param {Function} f The function to call for every element. This function
|
456
|
+
* takes 3 arguments (the element, the index and the array) and should
|
457
|
+
* return a boolean.
|
458
|
+
* @param {Object=} opt_obj An optional "this" context for the function.
|
459
|
+
* @return {number} The index of the first array element that passes the test,
|
460
|
+
* or -1 if no element is found.
|
461
|
+
*/
|
462
|
+
goog.array.findIndex = function(arr, f, opt_obj) {
|
463
|
+
var l = arr.length; // must be fixed during loop... see docs
|
464
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
465
|
+
for (var i = 0; i < l; i++) {
|
466
|
+
if (i in arr2 && f.call(opt_obj, arr2[i], i, arr)) {
|
467
|
+
return i;
|
468
|
+
}
|
469
|
+
}
|
470
|
+
return -1;
|
471
|
+
};
|
472
|
+
|
473
|
+
|
474
|
+
/**
|
475
|
+
* Search an array (in reverse order) for the last element that satisfies a
|
476
|
+
* given condition and return that element.
|
477
|
+
* @param {goog.array.ArrayLike} arr The array to search.
|
478
|
+
* @param {Function} f The function to call for every element. This function
|
479
|
+
* takes 3 arguments (the element, the index and the array) and should
|
480
|
+
* return a boolean.
|
481
|
+
* @param {Object=} opt_obj An optional "this" context for the function.
|
482
|
+
* @return {*} The last array element that passes the test, or null if no
|
483
|
+
* element is found.
|
484
|
+
*/
|
485
|
+
goog.array.findRight = function(arr, f, opt_obj) {
|
486
|
+
var i = goog.array.findIndexRight(arr, f, opt_obj);
|
487
|
+
return i < 0 ? null : goog.isString(arr) ? arr.charAt(i) : arr[i];
|
488
|
+
};
|
489
|
+
|
490
|
+
|
491
|
+
/**
|
492
|
+
* Search an array (in reverse order) for the last element that satisfies a
|
493
|
+
* given condition and return its index.
|
494
|
+
* @param {goog.array.ArrayLike} arr The array to search.
|
495
|
+
* @param {Function} f The function to call for every element. This function
|
496
|
+
* takes 3 arguments (the element, the index and the array) and should
|
497
|
+
* return a boolean.
|
498
|
+
* @param {Object=} opt_obj An optional "this" context for the function.
|
499
|
+
* @return {number} The index of the last array element that passes the test,
|
500
|
+
* or -1 if no element is found.
|
501
|
+
*/
|
502
|
+
goog.array.findIndexRight = function(arr, f, opt_obj) {
|
503
|
+
var l = arr.length; // must be fixed during loop... see docs
|
504
|
+
var arr2 = goog.isString(arr) ? arr.split('') : arr;
|
505
|
+
for (var i = l - 1; i >= 0; i--) {
|
506
|
+
if (i in arr2 && f.call(opt_obj, arr2[i], i, arr)) {
|
507
|
+
return i;
|
508
|
+
}
|
509
|
+
}
|
510
|
+
return -1;
|
511
|
+
};
|
512
|
+
|
513
|
+
|
514
|
+
/**
|
515
|
+
* Whether the array contains the given object.
|
516
|
+
* @param {goog.array.ArrayLike} arr The array to test for the presence of the
|
517
|
+
* element.
|
518
|
+
* @param {*} obj The object for which to test.
|
519
|
+
* @return {boolean} true if obj is present.
|
520
|
+
*/
|
521
|
+
goog.array.contains = function(arr, obj) {
|
522
|
+
return goog.array.indexOf(arr, obj) >= 0;
|
523
|
+
};
|
524
|
+
|
525
|
+
|
526
|
+
/**
|
527
|
+
* Whether the array is empty.
|
528
|
+
* @param {goog.array.ArrayLike} arr The array to test.
|
529
|
+
* @return {boolean} true if empty.
|
530
|
+
*/
|
531
|
+
goog.array.isEmpty = function(arr) {
|
532
|
+
return arr.length == 0;
|
533
|
+
};
|
534
|
+
|
535
|
+
|
536
|
+
/**
|
537
|
+
* Clears the array.
|
538
|
+
* @param {goog.array.ArrayLike} arr Array or array like object to clear.
|
539
|
+
*/
|
540
|
+
goog.array.clear = function(arr) {
|
541
|
+
// For non real arrays we don't have the magic length so we delete the
|
542
|
+
// indices.
|
543
|
+
if (!goog.isArray(arr)) {
|
544
|
+
for (var i = arr.length - 1; i >= 0; i--) {
|
545
|
+
delete arr[i];
|
546
|
+
}
|
547
|
+
}
|
548
|
+
arr.length = 0;
|
549
|
+
};
|
550
|
+
|
551
|
+
|
552
|
+
/**
|
553
|
+
* Pushes an item into an array, if it's not already in the array.
|
554
|
+
* @param {Array} arr Array into which to insert the item.
|
555
|
+
* @param {*} obj Value to add.
|
556
|
+
*/
|
557
|
+
goog.array.insert = function(arr, obj) {
|
558
|
+
if (!goog.array.contains(arr, obj)) {
|
559
|
+
arr.push(obj);
|
560
|
+
}
|
561
|
+
};
|
562
|
+
|
563
|
+
|
564
|
+
/**
|
565
|
+
* Inserts an object at the given index of the array.
|
566
|
+
* @param {goog.array.ArrayLike} arr The array to modify.
|
567
|
+
* @param {*} obj The object to insert.
|
568
|
+
* @param {number=} opt_i The index at which to insert the object. If omitted,
|
569
|
+
* treated as 0. A negative index is counted from the end of the array.
|
570
|
+
*/
|
571
|
+
goog.array.insertAt = function(arr, obj, opt_i) {
|
572
|
+
goog.array.splice(arr, opt_i, 0, obj);
|
573
|
+
};
|
574
|
+
|
575
|
+
|
576
|
+
/**
|
577
|
+
* Inserts at the given index of the array, all elements of another array.
|
578
|
+
* @param {goog.array.ArrayLike} arr The array to modify.
|
579
|
+
* @param {goog.array.ArrayLike} elementsToAdd The array of elements to add.
|
580
|
+
* @param {number=} opt_i The index at which to insert the object. If omitted,
|
581
|
+
* treated as 0. A negative index is counted from the end of the array.
|
582
|
+
*/
|
583
|
+
goog.array.insertArrayAt = function(arr, elementsToAdd, opt_i) {
|
584
|
+
goog.partial(goog.array.splice, arr, opt_i, 0).apply(null, elementsToAdd);
|
585
|
+
};
|
586
|
+
|
587
|
+
|
588
|
+
/**
|
589
|
+
* Inserts an object into an array before a specified object.
|
590
|
+
* @param {Array} arr The array to modify.
|
591
|
+
* @param {*} obj The object to insert.
|
592
|
+
* @param {*=} opt_obj2 The object before which obj should be inserted. If obj2
|
593
|
+
* is omitted or not found, obj is inserted at the end of the array.
|
594
|
+
*/
|
595
|
+
goog.array.insertBefore = function(arr, obj, opt_obj2) {
|
596
|
+
var i;
|
597
|
+
if (arguments.length == 2 || (i = goog.array.indexOf(arr, opt_obj2)) < 0) {
|
598
|
+
arr.push(obj);
|
599
|
+
} else {
|
600
|
+
goog.array.insertAt(arr, obj, i);
|
601
|
+
}
|
602
|
+
};
|
603
|
+
|
604
|
+
|
605
|
+
/**
|
606
|
+
* Removes the first occurrence of a particular value from an array.
|
607
|
+
* @param {goog.array.ArrayLike} arr Array from which to remove value.
|
608
|
+
* @param {*} obj Object to remove.
|
609
|
+
* @return {boolean} True if an element was removed.
|
610
|
+
*/
|
611
|
+
goog.array.remove = function(arr, obj) {
|
612
|
+
var i = goog.array.indexOf(arr, obj);
|
613
|
+
var rv;
|
614
|
+
if ((rv = i >= 0)) {
|
615
|
+
goog.array.removeAt(arr, i);
|
616
|
+
}
|
617
|
+
return rv;
|
618
|
+
};
|
619
|
+
|
620
|
+
|
621
|
+
/**
|
622
|
+
* Removes from an array the element at index i
|
623
|
+
* @param {goog.array.ArrayLike} arr Array or array like object from which to
|
624
|
+
* remove value.
|
625
|
+
* @param {number} i The index to remove.
|
626
|
+
* @return {boolean} True if an element was removed.
|
627
|
+
*/
|
628
|
+
goog.array.removeAt = function(arr, i) {
|
629
|
+
goog.asserts.assert(arr.length != null);
|
630
|
+
|
631
|
+
// use generic form of splice
|
632
|
+
// splice returns the removed items and if successful the length of that
|
633
|
+
// will be 1
|
634
|
+
return goog.array.ARRAY_PROTOTYPE_.splice.call(arr, i, 1).length == 1;
|
635
|
+
};
|
636
|
+
|
637
|
+
|
638
|
+
/**
|
639
|
+
* Removes the first value that satisfies the given condition.
|
640
|
+
* @param {goog.array.ArrayLike} arr Array from which to remove value.
|
641
|
+
* @param {Function} f The function to call for every element. This function
|
642
|
+
* takes 3 arguments (the element, the index and the array) and should
|
643
|
+
* return a boolean.
|
644
|
+
* @param {Object=} opt_obj An optional "this" context for the function.
|
645
|
+
* @return {boolean} True if an element was removed.
|
646
|
+
*/
|
647
|
+
goog.array.removeIf = function(arr, f, opt_obj) {
|
648
|
+
var i = goog.array.findIndex(arr, f, opt_obj);
|
649
|
+
if (i >= 0) {
|
650
|
+
goog.array.removeAt(arr, i);
|
651
|
+
return true;
|
652
|
+
}
|
653
|
+
return false;
|
654
|
+
};
|
655
|
+
|
656
|
+
|
657
|
+
/**
|
658
|
+
* Returns a new array that is the result of joining the arguments. If arrays
|
659
|
+
* are passed then their items are added, however, if non-arrays are passed they
|
660
|
+
* will be added to the return array as is.
|
661
|
+
*
|
662
|
+
* Note that ArrayLike objects will be added as is, rather than having their
|
663
|
+
* items added.
|
664
|
+
*
|
665
|
+
* goog.array.concat([1, 2], [3, 4]) -> [1, 2, 3, 4]
|
666
|
+
* goog.array.concat(0, [1, 2]) -> [0, 1, 2]
|
667
|
+
* goog.array.concat([1, 2], null) -> [1, 2, null]
|
668
|
+
*
|
669
|
+
* There is bug in all current versions of IE (6, 7 and 8) where arrays created
|
670
|
+
* in an iframe become corrupted soon (not immediately) after the iframe is
|
671
|
+
* destroyed. This is common if loading data via goog.net.IframeIo, for example.
|
672
|
+
* This corruption only affects the concat method which will start throwing
|
673
|
+
* Catastrophic Errors (#-2147418113).
|
674
|
+
*
|
675
|
+
* See http://endoflow.com/scratch/corrupted-arrays.html for a test case.
|
676
|
+
*
|
677
|
+
* Internally goog.array should use this, so that all methods will continue to
|
678
|
+
* work on these broken array objects.
|
679
|
+
*
|
680
|
+
* @param {...*} var_args Items to concatenate. Arrays will have each item
|
681
|
+
* added, while primitives and objects will be added as is.
|
682
|
+
* @return {!Array} The new resultant array.
|
683
|
+
*/
|
684
|
+
goog.array.concat = function(var_args) {
|
685
|
+
return goog.array.ARRAY_PROTOTYPE_.concat.apply(
|
686
|
+
goog.array.ARRAY_PROTOTYPE_, arguments);
|
687
|
+
};
|
688
|
+
|
689
|
+
|
690
|
+
/**
|
691
|
+
* Does a shallow copy of an array.
|
692
|
+
* @param {goog.array.ArrayLike} arr Array or array-like object to clone.
|
693
|
+
* @return {!Array} Clone of the input array.
|
694
|
+
*/
|
695
|
+
goog.array.clone = function(arr) {
|
696
|
+
if (goog.isArray(arr)) {
|
697
|
+
return goog.array.concat(/** @type {!Array} */ (arr));
|
698
|
+
} else { // array like
|
699
|
+
// Concat does not work with non arrays.
|
700
|
+
var rv = [];
|
701
|
+
for (var i = 0, len = arr.length; i < len; i++) {
|
702
|
+
rv[i] = arr[i];
|
703
|
+
}
|
704
|
+
return rv;
|
705
|
+
}
|
706
|
+
};
|
707
|
+
|
708
|
+
|
709
|
+
/**
|
710
|
+
* Converts an object to an array.
|
711
|
+
* @param {goog.array.ArrayLike} object The object to convert to an array.
|
712
|
+
* @return {!Array} The object converted into an array. If object has a
|
713
|
+
* length property, every property indexed with a non-negative number
|
714
|
+
* less than length will be included in the result. If object does not
|
715
|
+
* have a length property, an empty array will be returned.
|
716
|
+
*/
|
717
|
+
goog.array.toArray = function(object) {
|
718
|
+
if (goog.isArray(object)) {
|
719
|
+
// This fixes the JS compiler warning and forces the Object to an Array type
|
720
|
+
return goog.array.concat(/** @type {!Array} */ (object));
|
721
|
+
}
|
722
|
+
// Clone what we hope to be an array-like object to an array.
|
723
|
+
// We could check isArrayLike() first, but no check we perform would be as
|
724
|
+
// reliable as simply making the call.
|
725
|
+
return goog.array.clone(/** @type {Array} */ (object));
|
726
|
+
};
|
727
|
+
|
728
|
+
|
729
|
+
/**
|
730
|
+
* Extends an array with another array, element, or "array like" object.
|
731
|
+
* This function operates 'in-place', it does not create a new Array.
|
732
|
+
*
|
733
|
+
* Example:
|
734
|
+
* var a = [];
|
735
|
+
* goog.array.extend(a, [0, 1]);
|
736
|
+
* a; // [0, 1]
|
737
|
+
* goog.array.extend(a, 2);
|
738
|
+
* a; // [0, 1, 2]
|
739
|
+
*
|
740
|
+
* @param {Array} arr1 The array to modify.
|
741
|
+
* @param {...*} var_args The elements or arrays of elements to add to arr1.
|
742
|
+
*/
|
743
|
+
goog.array.extend = function(arr1, var_args) {
|
744
|
+
for (var i = 1; i < arguments.length; i++) {
|
745
|
+
var arr2 = arguments[i];
|
746
|
+
// If we have an Array or an Arguments object we can just call push
|
747
|
+
// directly.
|
748
|
+
var isArrayLike;
|
749
|
+
if (goog.isArray(arr2) ||
|
750
|
+
// Detect Arguments. ES5 says that the [[Class]] of an Arguments object
|
751
|
+
// is "Arguments" but only V8 and JSC/Safari gets this right. We instead
|
752
|
+
// detect Arguments by checking for array like and presence of "callee".
|
753
|
+
(isArrayLike = goog.isArrayLike(arr2)) &&
|
754
|
+
// The getter for callee throws an exception in strict mode
|
755
|
+
// according to section 10.6 in ES5 so check for presence instead.
|
756
|
+
arr2.hasOwnProperty('callee')) {
|
757
|
+
arr1.push.apply(arr1, arr2);
|
758
|
+
|
759
|
+
} else if (isArrayLike) {
|
760
|
+
// Otherwise loop over arr2 to prevent copying the object.
|
761
|
+
var len1 = arr1.length;
|
762
|
+
var len2 = arr2.length;
|
763
|
+
for (var j = 0; j < len2; j++) {
|
764
|
+
arr1[len1 + j] = arr2[j];
|
765
|
+
}
|
766
|
+
} else {
|
767
|
+
arr1.push(arr2);
|
768
|
+
}
|
769
|
+
}
|
770
|
+
};
|
771
|
+
|
772
|
+
|
773
|
+
/**
|
774
|
+
* Adds or removes elements from an array. This is a generic version of Array
|
775
|
+
* splice. This means that it might work on other objects similar to arrays,
|
776
|
+
* such as the arguments object.
|
777
|
+
*
|
778
|
+
* @param {goog.array.ArrayLike} arr The array to modify.
|
779
|
+
* @param {number|undefined} index The index at which to start changing the
|
780
|
+
* array. If not defined, treated as 0.
|
781
|
+
* @param {number} howMany How many elements to remove (0 means no removal. A
|
782
|
+
* value below 0 is treated as zero and so is any other non number. Numbers
|
783
|
+
* are floored).
|
784
|
+
* @param {...*} var_args Optional, additional elements to insert into the
|
785
|
+
* array.
|
786
|
+
* @return {!Array} the removed elements.
|
787
|
+
*/
|
788
|
+
goog.array.splice = function(arr, index, howMany, var_args) {
|
789
|
+
goog.asserts.assert(arr.length != null);
|
790
|
+
|
791
|
+
return goog.array.ARRAY_PROTOTYPE_.splice.apply(
|
792
|
+
arr, goog.array.slice(arguments, 1));
|
793
|
+
};
|
794
|
+
|
795
|
+
|
796
|
+
/**
|
797
|
+
* Returns a new array from a segment of an array. This is a generic version of
|
798
|
+
* Array slice. This means that it might work on other objects similar to
|
799
|
+
* arrays, such as the arguments object.
|
800
|
+
*
|
801
|
+
* @param {goog.array.ArrayLike} arr The array from which to copy a segment.
|
802
|
+
* @param {number} start The index of the first element to copy.
|
803
|
+
* @param {number=} opt_end The index after the last element to copy.
|
804
|
+
* @return {!Array} A new array containing the specified segment of the original
|
805
|
+
* array.
|
806
|
+
*/
|
807
|
+
goog.array.slice = function(arr, start, opt_end) {
|
808
|
+
goog.asserts.assert(arr.length != null);
|
809
|
+
|
810
|
+
// passing 1 arg to slice is not the same as passing 2 where the second is
|
811
|
+
// null or undefined (in that case the second argument is treated as 0).
|
812
|
+
// we could use slice on the arguments object and then use apply instead of
|
813
|
+
// testing the length
|
814
|
+
if (arguments.length <= 2) {
|
815
|
+
return goog.array.ARRAY_PROTOTYPE_.slice.call(arr, start);
|
816
|
+
} else {
|
817
|
+
return goog.array.ARRAY_PROTOTYPE_.slice.call(arr, start, opt_end);
|
818
|
+
}
|
819
|
+
};
|
820
|
+
|
821
|
+
|
822
|
+
/**
|
823
|
+
* Removes all duplicates from an array (retaining only the first
|
824
|
+
* occurrence of each array element). This function modifies the
|
825
|
+
* array in place and doesn't change the order of the non-duplicate items.
|
826
|
+
*
|
827
|
+
* For objects, duplicates are identified as having the same unique ID as
|
828
|
+
* defined by {@link goog.getUid}.
|
829
|
+
*
|
830
|
+
* Runtime: N,
|
831
|
+
* Worstcase space: 2N (no dupes)
|
832
|
+
*
|
833
|
+
* @param {goog.array.ArrayLike} arr The array from which to remove duplicates.
|
834
|
+
* @param {Array=} opt_rv An optional array in which to return the results,
|
835
|
+
* instead of performing the removal inplace. If specified, the original
|
836
|
+
* array will remain unchanged.
|
837
|
+
*/
|
838
|
+
goog.array.removeDuplicates = function(arr, opt_rv) {
|
839
|
+
var returnArray = opt_rv || arr;
|
840
|
+
|
841
|
+
var seen = {}, cursorInsert = 0, cursorRead = 0;
|
842
|
+
while (cursorRead < arr.length) {
|
843
|
+
var current = arr[cursorRead++];
|
844
|
+
|
845
|
+
// Prefix each type with a single character representing the type to
|
846
|
+
// prevent conflicting keys (e.g. true and 'true').
|
847
|
+
var key = goog.isObject(current) ?
|
848
|
+
'o' + goog.getUid(current) :
|
849
|
+
(typeof current).charAt(0) + current;
|
850
|
+
|
851
|
+
if (!Object.prototype.hasOwnProperty.call(seen, key)) {
|
852
|
+
seen[key] = true;
|
853
|
+
returnArray[cursorInsert++] = current;
|
854
|
+
}
|
855
|
+
}
|
856
|
+
returnArray.length = cursorInsert;
|
857
|
+
};
|
858
|
+
|
859
|
+
|
860
|
+
/**
|
861
|
+
* Searches the specified array for the specified target using the binary
|
862
|
+
* search algorithm. If no opt_compareFn is specified, elements are compared
|
863
|
+
* using <code>goog.array.defaultCompare</code>, which compares the elements
|
864
|
+
* using the built in < and > operators. This will produce the expected
|
865
|
+
* behavior for homogeneous arrays of String(s) and Number(s). The array
|
866
|
+
* specified <b>must</b> be sorted in ascending order (as defined by the
|
867
|
+
* comparison function). If the array is not sorted, results are undefined.
|
868
|
+
* If the array contains multiple instances of the specified target value, any
|
869
|
+
* of these instances may be found.
|
870
|
+
*
|
871
|
+
* Runtime: O(log n)
|
872
|
+
*
|
873
|
+
* @param {goog.array.ArrayLike} arr The array to be searched.
|
874
|
+
* @param {*} target The sought value.
|
875
|
+
* @param {Function=} opt_compareFn Optional comparison function by which the
|
876
|
+
* array is ordered. Should take 2 arguments to compare, and return a
|
877
|
+
* negative number, zero, or a positive number depending on whether the
|
878
|
+
* first argument is less than, equal to, or greater than the second.
|
879
|
+
* @return {number} Lowest index of the target value if found, otherwise
|
880
|
+
* (-(insertion point) - 1). The insertion point is where the value should
|
881
|
+
* be inserted into arr to preserve the sorted property. Return value >= 0
|
882
|
+
* iff target is found.
|
883
|
+
*/
|
884
|
+
goog.array.binarySearch = function(arr, target, opt_compareFn) {
|
885
|
+
return goog.array.binarySearch_(arr,
|
886
|
+
opt_compareFn || goog.array.defaultCompare, false /* isEvaluator */,
|
887
|
+
target);
|
888
|
+
};
|
889
|
+
|
890
|
+
|
891
|
+
/**
|
892
|
+
* Selects an index in the specified array using the binary search algorithm.
|
893
|
+
* The evaluator receives an element and determines whether the desired index
|
894
|
+
* is before, at, or after it. The evaluator must be consistent (formally,
|
895
|
+
* goog.array.map(goog.array.map(arr, evaluator, opt_obj), goog.math.sign)
|
896
|
+
* must be monotonically non-increasing).
|
897
|
+
*
|
898
|
+
* Runtime: O(log n)
|
899
|
+
*
|
900
|
+
* @param {goog.array.ArrayLike} arr The array to be searched.
|
901
|
+
* @param {Function} evaluator Evaluator function that receives 3 arguments
|
902
|
+
* (the element, the index and the array). Should return a negative number,
|
903
|
+
* zero, or a positive number depending on whether the desired index is
|
904
|
+
* before, at, or after the element passed to it.
|
905
|
+
* @param {Object=} opt_obj The object to be used as the value of 'this'
|
906
|
+
* within evaluator.
|
907
|
+
* @return {number} Index of the leftmost element matched by the evaluator, if
|
908
|
+
* such exists; otherwise (-(insertion point) - 1). The insertion point is
|
909
|
+
* the index of the first element for which the evaluator returns negative,
|
910
|
+
* or arr.length if no such element exists. The return value is non-negative
|
911
|
+
* iff a match is found.
|
912
|
+
*/
|
913
|
+
goog.array.binarySelect = function(arr, evaluator, opt_obj) {
|
914
|
+
return goog.array.binarySearch_(arr, evaluator, true /* isEvaluator */,
|
915
|
+
undefined /* opt_target */, opt_obj);
|
916
|
+
};
|
917
|
+
|
918
|
+
|
919
|
+
/**
|
920
|
+
* Implementation of a binary search algorithm which knows how to use both
|
921
|
+
* comparison functions and evaluators. If an evaluator is provided, will call
|
922
|
+
* the evaluator with the given optional data object, conforming to the
|
923
|
+
* interface defined in binarySelect. Otherwise, if a comparison function is
|
924
|
+
* provided, will call the comparison function against the given data object.
|
925
|
+
*
|
926
|
+
* This implementation purposefully does not use goog.bind or goog.partial for
|
927
|
+
* performance reasons.
|
928
|
+
*
|
929
|
+
* Runtime: O(log n)
|
930
|
+
*
|
931
|
+
* @param {goog.array.ArrayLike} arr The array to be searched.
|
932
|
+
* @param {Function} compareFn Either an evaluator or a comparison function,
|
933
|
+
* as defined by binarySearch and binarySelect above.
|
934
|
+
* @param {boolean} isEvaluator Whether the function is an evaluator or a
|
935
|
+
* comparison function.
|
936
|
+
* @param {*=} opt_target If the function is a comparison function, then this is
|
937
|
+
* the target to binary search for.
|
938
|
+
* @param {Object=} opt_selfObj If the function is an evaluator, this is an
|
939
|
+
* optional this object for the evaluator.
|
940
|
+
* @return {number} Lowest index of the target value if found, otherwise
|
941
|
+
* (-(insertion point) - 1). The insertion point is where the value should
|
942
|
+
* be inserted into arr to preserve the sorted property. Return value >= 0
|
943
|
+
* iff target is found.
|
944
|
+
* @private
|
945
|
+
*/
|
946
|
+
goog.array.binarySearch_ = function(arr, compareFn, isEvaluator, opt_target,
|
947
|
+
opt_selfObj) {
|
948
|
+
var left = 0; // inclusive
|
949
|
+
var right = arr.length; // exclusive
|
950
|
+
var found;
|
951
|
+
while (left < right) {
|
952
|
+
var middle = (left + right) >> 1;
|
953
|
+
var compareResult;
|
954
|
+
if (isEvaluator) {
|
955
|
+
compareResult = compareFn.call(opt_selfObj, arr[middle], middle, arr);
|
956
|
+
} else {
|
957
|
+
compareResult = compareFn(opt_target, arr[middle]);
|
958
|
+
}
|
959
|
+
if (compareResult > 0) {
|
960
|
+
left = middle + 1;
|
961
|
+
} else {
|
962
|
+
right = middle;
|
963
|
+
// We are looking for the lowest index so we can't return immediately.
|
964
|
+
found = !compareResult;
|
965
|
+
}
|
966
|
+
}
|
967
|
+
// left is the index if found, or the insertion point otherwise.
|
968
|
+
// ~left is a shorthand for -left - 1.
|
969
|
+
return found ? left : ~left;
|
970
|
+
};
|
971
|
+
|
972
|
+
|
973
|
+
/**
|
974
|
+
* Sorts the specified array into ascending order. If no opt_compareFn is
|
975
|
+
* specified, elements are compared using
|
976
|
+
* <code>goog.array.defaultCompare</code>, which compares the elements using
|
977
|
+
* the built in < and > operators. This will produce the expected behavior
|
978
|
+
* for homogeneous arrays of String(s) and Number(s), unlike the native sort,
|
979
|
+
* but will give unpredictable results for heterogenous lists of strings and
|
980
|
+
* numbers with different numbers of digits.
|
981
|
+
*
|
982
|
+
* This sort is not guaranteed to be stable.
|
983
|
+
*
|
984
|
+
* Runtime: Same as <code>Array.prototype.sort</code>
|
985
|
+
*
|
986
|
+
* @param {Array} arr The array to be sorted.
|
987
|
+
* @param {Function=} opt_compareFn Optional comparison function by which the
|
988
|
+
* array is to be ordered. Should take 2 arguments to compare, and return a
|
989
|
+
* negative number, zero, or a positive number depending on whether the
|
990
|
+
* first argument is less than, equal to, or greater than the second.
|
991
|
+
*/
|
992
|
+
goog.array.sort = function(arr, opt_compareFn) {
|
993
|
+
// TODO(arv): Update type annotation since null is not accepted.
|
994
|
+
goog.asserts.assert(arr.length != null);
|
995
|
+
|
996
|
+
goog.array.ARRAY_PROTOTYPE_.sort.call(
|
997
|
+
arr, opt_compareFn || goog.array.defaultCompare);
|
998
|
+
};
|
999
|
+
|
1000
|
+
|
1001
|
+
/**
|
1002
|
+
* Sorts the specified array into ascending order in a stable way. If no
|
1003
|
+
* opt_compareFn is specified, elements are compared using
|
1004
|
+
* <code>goog.array.defaultCompare</code>, which compares the elements using
|
1005
|
+
* the built in < and > operators. This will produce the expected behavior
|
1006
|
+
* for homogeneous arrays of String(s) and Number(s).
|
1007
|
+
*
|
1008
|
+
* Runtime: Same as <code>Array.prototype.sort</code>, plus an additional
|
1009
|
+
* O(n) overhead of copying the array twice.
|
1010
|
+
*
|
1011
|
+
* @param {Array} arr The array to be sorted.
|
1012
|
+
* @param {function(*, *): number=} opt_compareFn Optional comparison function
|
1013
|
+
* by which the array is to be ordered. Should take 2 arguments to compare,
|
1014
|
+
* and return a negative number, zero, or a positive number depending on
|
1015
|
+
* whether the first argument is less than, equal to, or greater than the
|
1016
|
+
* second.
|
1017
|
+
*/
|
1018
|
+
goog.array.stableSort = function(arr, opt_compareFn) {
|
1019
|
+
for (var i = 0; i < arr.length; i++) {
|
1020
|
+
arr[i] = {index: i, value: arr[i]};
|
1021
|
+
}
|
1022
|
+
var valueCompareFn = opt_compareFn || goog.array.defaultCompare;
|
1023
|
+
function stableCompareFn(obj1, obj2) {
|
1024
|
+
return valueCompareFn(obj1.value, obj2.value) || obj1.index - obj2.index;
|
1025
|
+
};
|
1026
|
+
goog.array.sort(arr, stableCompareFn);
|
1027
|
+
for (var i = 0; i < arr.length; i++) {
|
1028
|
+
arr[i] = arr[i].value;
|
1029
|
+
}
|
1030
|
+
};
|
1031
|
+
|
1032
|
+
|
1033
|
+
/**
|
1034
|
+
* Sorts an array of objects by the specified object key and compare
|
1035
|
+
* function. If no compare function is provided, the key values are
|
1036
|
+
* compared in ascending order using <code>goog.array.defaultCompare</code>.
|
1037
|
+
* This won't work for keys that get renamed by the compiler. So use
|
1038
|
+
* {'foo': 1, 'bar': 2} rather than {foo: 1, bar: 2}.
|
1039
|
+
* @param {Array.<Object>} arr An array of objects to sort.
|
1040
|
+
* @param {string} key The object key to sort by.
|
1041
|
+
* @param {Function=} opt_compareFn The function to use to compare key
|
1042
|
+
* values.
|
1043
|
+
*/
|
1044
|
+
goog.array.sortObjectsByKey = function(arr, key, opt_compareFn) {
|
1045
|
+
var compare = opt_compareFn || goog.array.defaultCompare;
|
1046
|
+
goog.array.sort(arr, function(a, b) {
|
1047
|
+
return compare(a[key], b[key]);
|
1048
|
+
});
|
1049
|
+
};
|
1050
|
+
|
1051
|
+
|
1052
|
+
/**
|
1053
|
+
* Tells if the array is sorted.
|
1054
|
+
* @param {!Array} arr The array.
|
1055
|
+
* @param {Function=} opt_compareFn Function to compare the array elements.
|
1056
|
+
* Should take 2 arguments to compare, and return a negative number, zero,
|
1057
|
+
* or a positive number depending on whether the first argument is less
|
1058
|
+
* than, equal to, or greater than the second.
|
1059
|
+
* @param {boolean=} opt_strict If true no equal elements are allowed.
|
1060
|
+
* @return {boolean} Whether the array is sorted.
|
1061
|
+
*/
|
1062
|
+
goog.array.isSorted = function(arr, opt_compareFn, opt_strict) {
|
1063
|
+
var compare = opt_compareFn || goog.array.defaultCompare;
|
1064
|
+
for (var i = 1; i < arr.length; i++) {
|
1065
|
+
var compareResult = compare(arr[i - 1], arr[i]);
|
1066
|
+
if (compareResult > 0 || compareResult == 0 && opt_strict) {
|
1067
|
+
return false;
|
1068
|
+
}
|
1069
|
+
}
|
1070
|
+
return true;
|
1071
|
+
};
|
1072
|
+
|
1073
|
+
|
1074
|
+
/**
|
1075
|
+
* Compares two arrays for equality. Two arrays are considered equal if they
|
1076
|
+
* have the same length and their corresponding elements are equal according to
|
1077
|
+
* the comparison function.
|
1078
|
+
*
|
1079
|
+
* @param {goog.array.ArrayLike} arr1 The first array to compare.
|
1080
|
+
* @param {goog.array.ArrayLike} arr2 The second array to compare.
|
1081
|
+
* @param {Function=} opt_equalsFn Optional comparison function.
|
1082
|
+
* Should take 2 arguments to compare, and return true if the arguments
|
1083
|
+
* are equal. Defaults to {@link goog.array.defaultCompareEquality} which
|
1084
|
+
* compares the elements using the built-in '===' operator.
|
1085
|
+
* @return {boolean} Whether the two arrays are equal.
|
1086
|
+
*/
|
1087
|
+
goog.array.equals = function(arr1, arr2, opt_equalsFn) {
|
1088
|
+
if (!goog.isArrayLike(arr1) || !goog.isArrayLike(arr2) ||
|
1089
|
+
arr1.length != arr2.length) {
|
1090
|
+
return false;
|
1091
|
+
}
|
1092
|
+
var l = arr1.length;
|
1093
|
+
var equalsFn = opt_equalsFn || goog.array.defaultCompareEquality;
|
1094
|
+
for (var i = 0; i < l; i++) {
|
1095
|
+
if (!equalsFn(arr1[i], arr2[i])) {
|
1096
|
+
return false;
|
1097
|
+
}
|
1098
|
+
}
|
1099
|
+
return true;
|
1100
|
+
};
|
1101
|
+
|
1102
|
+
|
1103
|
+
/**
|
1104
|
+
* @deprecated Use {@link goog.array.equals}.
|
1105
|
+
* @param {goog.array.ArrayLike} arr1 See {@link goog.array.equals}.
|
1106
|
+
* @param {goog.array.ArrayLike} arr2 See {@link goog.array.equals}.
|
1107
|
+
* @param {Function=} opt_equalsFn See {@link goog.array.equals}.
|
1108
|
+
* @return {boolean} See {@link goog.array.equals}.
|
1109
|
+
*/
|
1110
|
+
goog.array.compare = function(arr1, arr2, opt_equalsFn) {
|
1111
|
+
return goog.array.equals(arr1, arr2, opt_equalsFn);
|
1112
|
+
};
|
1113
|
+
|
1114
|
+
|
1115
|
+
/**
|
1116
|
+
* 3-way array compare function.
|
1117
|
+
* @param {!goog.array.ArrayLike} arr1 The first array to compare.
|
1118
|
+
* @param {!goog.array.ArrayLike} arr2 The second array to compare.
|
1119
|
+
* @param {(function(*, *): number)=} opt_compareFn Optional comparison function
|
1120
|
+
* by which the array is to be ordered. Should take 2 arguments to compare,
|
1121
|
+
* and return a negative number, zero, or a positive number depending on
|
1122
|
+
* whether the first argument is less than, equal to, or greater than the
|
1123
|
+
* second.
|
1124
|
+
* @return {number} Negative number, zero, or a positive number depending on
|
1125
|
+
* whether the first argument is less than, equal to, or greater than the
|
1126
|
+
* second.
|
1127
|
+
*/
|
1128
|
+
goog.array.compare3 = function(arr1, arr2, opt_compareFn) {
|
1129
|
+
var compare = opt_compareFn || goog.array.defaultCompare;
|
1130
|
+
var l = Math.min(arr1.length, arr2.length);
|
1131
|
+
for (var i = 0; i < l; i++) {
|
1132
|
+
var result = compare(arr1[i], arr2[i]);
|
1133
|
+
if (result != 0) {
|
1134
|
+
return result;
|
1135
|
+
}
|
1136
|
+
}
|
1137
|
+
return goog.array.defaultCompare(arr1.length, arr2.length);
|
1138
|
+
};
|
1139
|
+
|
1140
|
+
|
1141
|
+
/**
|
1142
|
+
* Compares its two arguments for order, using the built in < and >
|
1143
|
+
* operators.
|
1144
|
+
* @param {*} a The first object to be compared.
|
1145
|
+
* @param {*} b The second object to be compared.
|
1146
|
+
* @return {number} A negative number, zero, or a positive number as the first
|
1147
|
+
* argument is less than, equal to, or greater than the second.
|
1148
|
+
*/
|
1149
|
+
goog.array.defaultCompare = function(a, b) {
|
1150
|
+
return a > b ? 1 : a < b ? -1 : 0;
|
1151
|
+
};
|
1152
|
+
|
1153
|
+
|
1154
|
+
/**
|
1155
|
+
* Compares its two arguments for equality, using the built in === operator.
|
1156
|
+
* @param {*} a The first object to compare.
|
1157
|
+
* @param {*} b The second object to compare.
|
1158
|
+
* @return {boolean} True if the two arguments are equal, false otherwise.
|
1159
|
+
*/
|
1160
|
+
goog.array.defaultCompareEquality = function(a, b) {
|
1161
|
+
return a === b;
|
1162
|
+
};
|
1163
|
+
|
1164
|
+
|
1165
|
+
/**
|
1166
|
+
* Inserts a value into a sorted array. The array is not modified if the
|
1167
|
+
* value is already present.
|
1168
|
+
* @param {Array} array The array to modify.
|
1169
|
+
* @param {*} value The object to insert.
|
1170
|
+
* @param {Function=} opt_compareFn Optional comparison function by which the
|
1171
|
+
* array is ordered. Should take 2 arguments to compare, and return a
|
1172
|
+
* negative number, zero, or a positive number depending on whether the
|
1173
|
+
* first argument is less than, equal to, or greater than the second.
|
1174
|
+
* @return {boolean} True if an element was inserted.
|
1175
|
+
*/
|
1176
|
+
goog.array.binaryInsert = function(array, value, opt_compareFn) {
|
1177
|
+
var index = goog.array.binarySearch(array, value, opt_compareFn);
|
1178
|
+
if (index < 0) {
|
1179
|
+
goog.array.insertAt(array, value, -(index + 1));
|
1180
|
+
return true;
|
1181
|
+
}
|
1182
|
+
return false;
|
1183
|
+
};
|
1184
|
+
|
1185
|
+
|
1186
|
+
/**
|
1187
|
+
* Removes a value from a sorted array.
|
1188
|
+
* @param {Array} array The array to modify.
|
1189
|
+
* @param {*} value The object to remove.
|
1190
|
+
* @param {Function=} opt_compareFn Optional comparison function by which the
|
1191
|
+
* array is ordered. Should take 2 arguments to compare, and return a
|
1192
|
+
* negative number, zero, or a positive number depending on whether the
|
1193
|
+
* first argument is less than, equal to, or greater than the second.
|
1194
|
+
* @return {boolean} True if an element was removed.
|
1195
|
+
*/
|
1196
|
+
goog.array.binaryRemove = function(array, value, opt_compareFn) {
|
1197
|
+
var index = goog.array.binarySearch(array, value, opt_compareFn);
|
1198
|
+
return (index >= 0) ? goog.array.removeAt(array, index) : false;
|
1199
|
+
};
|
1200
|
+
|
1201
|
+
|
1202
|
+
/**
|
1203
|
+
* Splits an array into disjoint buckets according to a splitting function.
|
1204
|
+
* @param {Array} array The array.
|
1205
|
+
* @param {Function} sorter Function to call for every element. This
|
1206
|
+
* takes 3 arguments (the element, the index and the array) and must
|
1207
|
+
* return a valid object key (a string, number, etc), or undefined, if
|
1208
|
+
* that object should not be placed in a bucket.
|
1209
|
+
* @return {!Object} An object, with keys being all of the unique return values
|
1210
|
+
* of sorter, and values being arrays containing the items for
|
1211
|
+
* which the splitter returned that key.
|
1212
|
+
*/
|
1213
|
+
goog.array.bucket = function(array, sorter) {
|
1214
|
+
var buckets = {};
|
1215
|
+
|
1216
|
+
for (var i = 0; i < array.length; i++) {
|
1217
|
+
var value = array[i];
|
1218
|
+
var key = sorter(value, i, array);
|
1219
|
+
if (goog.isDef(key)) {
|
1220
|
+
// Push the value to the right bucket, creating it if necessary.
|
1221
|
+
var bucket = buckets[key] || (buckets[key] = []);
|
1222
|
+
bucket.push(value);
|
1223
|
+
}
|
1224
|
+
}
|
1225
|
+
|
1226
|
+
return buckets;
|
1227
|
+
};
|
1228
|
+
|
1229
|
+
|
1230
|
+
/**
|
1231
|
+
* Returns an array consisting of the given value repeated N times.
|
1232
|
+
*
|
1233
|
+
* @param {*} value The value to repeat.
|
1234
|
+
* @param {number} n The repeat count.
|
1235
|
+
* @return {!Array.<*>} An array with the repeated value.
|
1236
|
+
*/
|
1237
|
+
goog.array.repeat = function(value, n) {
|
1238
|
+
var array = [];
|
1239
|
+
for (var i = 0; i < n; i++) {
|
1240
|
+
array[i] = value;
|
1241
|
+
}
|
1242
|
+
return array;
|
1243
|
+
};
|
1244
|
+
|
1245
|
+
|
1246
|
+
/**
|
1247
|
+
* Returns an array consisting of every argument with all arrays
|
1248
|
+
* expanded in-place recursively.
|
1249
|
+
*
|
1250
|
+
* @param {...*} var_args The values to flatten.
|
1251
|
+
* @return {!Array.<*>} An array containing the flattened values.
|
1252
|
+
*/
|
1253
|
+
goog.array.flatten = function(var_args) {
|
1254
|
+
var result = [];
|
1255
|
+
for (var i = 0; i < arguments.length; i++) {
|
1256
|
+
var element = arguments[i];
|
1257
|
+
if (goog.isArray(element)) {
|
1258
|
+
result.push.apply(result, goog.array.flatten.apply(null, element));
|
1259
|
+
} else {
|
1260
|
+
result.push(element);
|
1261
|
+
}
|
1262
|
+
}
|
1263
|
+
return result;
|
1264
|
+
};
|
1265
|
+
|
1266
|
+
|
1267
|
+
/**
|
1268
|
+
* Rotates an array in-place. After calling this method, the element at
|
1269
|
+
* index i will be the element previously at index (i - n) %
|
1270
|
+
* array.length, for all values of i between 0 and array.length - 1,
|
1271
|
+
* inclusive.
|
1272
|
+
*
|
1273
|
+
* For example, suppose list comprises [t, a, n, k, s]. After invoking
|
1274
|
+
* rotate(array, 1) (or rotate(array, -4)), array will comprise [s, t, a, n, k].
|
1275
|
+
*
|
1276
|
+
* @param {!Array.<*>} array The array to rotate.
|
1277
|
+
* @param {number} n The amount to rotate.
|
1278
|
+
* @return {!Array.<*>} The array.
|
1279
|
+
*/
|
1280
|
+
goog.array.rotate = function(array, n) {
|
1281
|
+
goog.asserts.assert(array.length != null);
|
1282
|
+
|
1283
|
+
if (array.length) {
|
1284
|
+
n %= array.length;
|
1285
|
+
if (n > 0) {
|
1286
|
+
goog.array.ARRAY_PROTOTYPE_.unshift.apply(array, array.splice(-n, n));
|
1287
|
+
} else if (n < 0) {
|
1288
|
+
goog.array.ARRAY_PROTOTYPE_.push.apply(array, array.splice(0, -n));
|
1289
|
+
}
|
1290
|
+
}
|
1291
|
+
return array;
|
1292
|
+
};
|
1293
|
+
|
1294
|
+
|
1295
|
+
/**
|
1296
|
+
* Creates a new array for which the element at position i is an array of the
|
1297
|
+
* ith element of the provided arrays. The returned array will only be as long
|
1298
|
+
* as the shortest array provided; additional values are ignored. For example,
|
1299
|
+
* the result of zipping [1, 2] and [3, 4, 5] is [[1,3], [2, 4]].
|
1300
|
+
*
|
1301
|
+
* This is similar to the zip() function in Python. See {@link
|
1302
|
+
* http://docs.python.org/library/functions.html#zip}
|
1303
|
+
*
|
1304
|
+
* @param {...!goog.array.ArrayLike} var_args Arrays to be combined.
|
1305
|
+
* @return {!Array.<!Array>} A new array of arrays created from provided arrays.
|
1306
|
+
*/
|
1307
|
+
goog.array.zip = function(var_args) {
|
1308
|
+
if (!arguments.length) {
|
1309
|
+
return [];
|
1310
|
+
}
|
1311
|
+
var result = [];
|
1312
|
+
for (var i = 0; true; i++) {
|
1313
|
+
var value = [];
|
1314
|
+
for (var j = 0; j < arguments.length; j++) {
|
1315
|
+
var arr = arguments[j];
|
1316
|
+
// If i is larger than the array length, this is the shortest array.
|
1317
|
+
if (i >= arr.length) {
|
1318
|
+
return result;
|
1319
|
+
}
|
1320
|
+
value.push(arr[i]);
|
1321
|
+
}
|
1322
|
+
result.push(value);
|
1323
|
+
}
|
1324
|
+
};
|
1325
|
+
|
1326
|
+
|
1327
|
+
/**
|
1328
|
+
* Shuffles the values in the specified array using the Fisher-Yates in-place
|
1329
|
+
* shuffle (also known as the Knuth Shuffle). By default, calls Math.random()
|
1330
|
+
* and so resets the state of that random number generator. Similarly, may reset
|
1331
|
+
* the state of the any other specified random number generator.
|
1332
|
+
*
|
1333
|
+
* Runtime: O(n)
|
1334
|
+
*
|
1335
|
+
* @param {!Array} arr The array to be shuffled.
|
1336
|
+
* @param {Function=} opt_randFn Optional random function to use for shuffling.
|
1337
|
+
* Takes no arguments, and returns a random number on the interval [0, 1).
|
1338
|
+
* Defaults to Math.random() using JavaScript's built-in Math library.
|
1339
|
+
*/
|
1340
|
+
goog.array.shuffle = function(arr, opt_randFn) {
|
1341
|
+
var randFn = opt_randFn || Math.random;
|
1342
|
+
|
1343
|
+
for (var i = arr.length - 1; i > 0; i--) {
|
1344
|
+
// Choose a random array index in [0, i] (inclusive with i).
|
1345
|
+
var j = Math.floor(randFn() * (i + 1));
|
1346
|
+
|
1347
|
+
var tmp = arr[i];
|
1348
|
+
arr[i] = arr[j];
|
1349
|
+
arr[j] = tmp;
|
1350
|
+
}
|
1351
|
+
};
|
1352
|
+
;
|
1353
|
+
FI"
|
1354
|
+
F"%5358f5a6ea2494e1dedb86f86a078827
|