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,204 @@
|
|
1
|
+
o: ActiveSupport::Cache::Entry :@compressedF:@expires_in0:@created_atf1325458171.512854:@value{I"
|
2
|
+
class:EFI"BundledAsset;
|
3
|
+
FI"id;
|
4
|
+
F"%6064b522903278398ca8fcdd52a5c177I"logical_path;
|
5
|
+
FI"'closure/goog/i18n/graphemebreak.js;
|
6
|
+
TI"
|
7
|
+
F"m/Volumes/Development/dev-web/yellow-brick-road/vendor/closure-library/closure/goog/i18n/graphemebreak.jsI"content_type;
|
8
|
+
FI"application/javascript;
|
9
|
+
FI"
|
10
|
+
mtime;
|
11
|
+
FI"2011-12-30T02:16:24+00:00;
|
12
|
+
FI" body;
|
13
|
+
FI"�// Copyright 2006 The Closure Library Authors. All Rights Reserved.
|
14
|
+
//
|
15
|
+
// Licensed under the Apache License, Version 2.0 (the "License");
|
16
|
+
// you may not use this file except in compliance with the License.
|
17
|
+
// You may obtain a copy of the License at
|
18
|
+
//
|
19
|
+
// http://www.apache.org/licenses/LICENSE-2.0
|
20
|
+
//
|
21
|
+
// Unless required by applicable law or agreed to in writing, software
|
22
|
+
// distributed under the License is distributed on an "AS-IS" BASIS,
|
23
|
+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
24
|
+
// See the License for the specific language governing permissions and
|
25
|
+
// limitations under the License.
|
26
|
+
|
27
|
+
/**
|
28
|
+
* @fileoverview Detect Grapheme Cluster Break in a pair of codepoints. Follows
|
29
|
+
* Unicode 5.1 UAX#29.
|
30
|
+
*
|
31
|
+
*/
|
32
|
+
|
33
|
+
|
34
|
+
goog.provide('goog.i18n.GraphemeBreak');
|
35
|
+
goog.require('goog.structs.InversionMap');
|
36
|
+
|
37
|
+
|
38
|
+
/**
|
39
|
+
* Enum for all Grapheme Cluster Break properties.
|
40
|
+
* These enums directly corresponds to Grapheme_Cluster_Break property values
|
41
|
+
* mentioned in http://unicode.org/reports/tr29 table 2.
|
42
|
+
*
|
43
|
+
* CR and LF are moved to the bottom of the list because they occur only once
|
44
|
+
* and so good candidates to take 2 decimal digit values.
|
45
|
+
* @enum {number}
|
46
|
+
* @protected
|
47
|
+
*/
|
48
|
+
goog.i18n.GraphemeBreak.property = {
|
49
|
+
ANY: 0,
|
50
|
+
CONTROL: 1,
|
51
|
+
EXTEND: 2,
|
52
|
+
PREPEND: 3,
|
53
|
+
SPACING_MARK: 4,
|
54
|
+
L: 5,
|
55
|
+
V: 6,
|
56
|
+
T: 7,
|
57
|
+
LV: 8,
|
58
|
+
LVT: 9,
|
59
|
+
CR: 10,
|
60
|
+
LF: 11
|
61
|
+
};
|
62
|
+
|
63
|
+
|
64
|
+
/**
|
65
|
+
* Grapheme Cluster Break property values for all codepoints as inversion map.
|
66
|
+
* Constructed lazily.
|
67
|
+
*
|
68
|
+
* @type {goog.structs.InversionMap}
|
69
|
+
* @private
|
70
|
+
*/
|
71
|
+
goog.i18n.GraphemeBreak.inversions_ = null;
|
72
|
+
|
73
|
+
|
74
|
+
/**
|
75
|
+
* There are two kinds of grapheme clusters: 1) Legacy 2)Extended. This method
|
76
|
+
* is to check for legacy rules.
|
77
|
+
*
|
78
|
+
* @param {number} prop_a The property enum value of the first character.
|
79
|
+
* @param {number} prop_b The property enum value of the second character.
|
80
|
+
* @return {boolean} True if a & b do not form a cluster; False otherwise.
|
81
|
+
* @private
|
82
|
+
*/
|
83
|
+
goog.i18n.GraphemeBreak.applyLegacyBreakRules_ = function(prop_a, prop_b) {
|
84
|
+
|
85
|
+
var prop = goog.i18n.GraphemeBreak.property;
|
86
|
+
|
87
|
+
if (prop_a == prop.CR && prop_b == prop.LF) {
|
88
|
+
return false;
|
89
|
+
}
|
90
|
+
if (prop_a == prop.CONTROL || prop_a == prop.CR || prop_a == prop.LF) {
|
91
|
+
return true;
|
92
|
+
}
|
93
|
+
if (prop_b == prop.CONTROL || prop_b == prop.CR || prop_b == prop.LF) {
|
94
|
+
return true;
|
95
|
+
}
|
96
|
+
if ((prop_a == prop.L) &&
|
97
|
+
(prop_b == prop.L || prop_b == prop.V ||
|
98
|
+
prop_b == prop.LV || prop_b == prop.LVT)) {
|
99
|
+
return false;
|
100
|
+
}
|
101
|
+
if ((prop_a == prop.LV || prop_a == prop.V) &&
|
102
|
+
(prop_b == prop.V || prop_b == prop.T)) {
|
103
|
+
return false;
|
104
|
+
}
|
105
|
+
if ((prop_a == prop.LVT || prop_a == prop.T) && (prop_b == prop.T)) {
|
106
|
+
return false;
|
107
|
+
}
|
108
|
+
if (prop_b == prop.EXTEND) {
|
109
|
+
return false;
|
110
|
+
}
|
111
|
+
return true;
|
112
|
+
};
|
113
|
+
|
114
|
+
|
115
|
+
/**
|
116
|
+
* Method to return property enum value of the codepoint. If it is Hangul LV or
|
117
|
+
* LVT, then it is computed; for the rest it is picked from the inversion map.
|
118
|
+
* @param {number} acode The code point value of the character.
|
119
|
+
* @return {number} Property enum value of codepoint.
|
120
|
+
* @private
|
121
|
+
*/
|
122
|
+
goog.i18n.GraphemeBreak.getBreakProp_ = function(acode) {
|
123
|
+
if (0xAC00 <= acode && acode <= 0xD7A3) {
|
124
|
+
var prop = goog.i18n.GraphemeBreak.property;
|
125
|
+
if (acode % 0x1C == 0x10) {
|
126
|
+
return prop.LV;
|
127
|
+
}
|
128
|
+
return prop.LVT;
|
129
|
+
} else {
|
130
|
+
if (!goog.i18n.GraphemeBreak.inversions_) {
|
131
|
+
goog.i18n.GraphemeBreak.inversions_ = new goog.structs.InversionMap(
|
132
|
+
[0, 10, 1, 2, 1, 18, 95, 33, 13, 1, 594, 112, 275, 7, 263, 45, 1, 1,
|
133
|
+
1, 2, 1, 2, 1, 1, 56, 4, 12, 11, 48, 20, 17, 1, 101, 7, 1, 7, 2, 2,
|
134
|
+
1, 4, 33, 1, 1, 1, 30, 27, 91, 11, 58, 9, 269, 2, 1, 56, 1, 1, 3, 8,
|
135
|
+
4, 1, 3, 4, 13, 2, 29, 1, 2, 56, 1, 1, 1, 2, 6, 6, 1, 9, 1, 10, 2,
|
136
|
+
29, 2, 1, 56, 2, 3, 17, 30, 2, 3, 14, 1, 56, 1, 1, 3, 8, 4, 1, 20,
|
137
|
+
2, 29, 1, 2, 56, 1, 1, 2, 1, 6, 6, 11, 10, 2, 30, 1, 59, 1, 1, 1,
|
138
|
+
12, 1, 9, 1, 41, 3, 58, 3, 5, 17, 11, 2, 30, 2, 56, 1, 1, 1, 1, 2,
|
139
|
+
1, 3, 1, 5, 11, 11, 2, 30, 2, 58, 1, 2, 5, 7, 11, 10, 2, 30, 2, 70,
|
140
|
+
6, 2, 6, 7, 19, 2, 60, 11, 5, 5, 1, 1, 8, 97, 13, 3, 5, 3, 6, 74, 2,
|
141
|
+
27, 1, 1, 1, 1, 1, 4, 2, 49, 14, 1, 5, 1, 2, 8, 45, 9, 1, 100, 2, 4,
|
142
|
+
1, 6, 1, 2, 2, 2, 23, 2, 2, 4, 3, 1, 3, 2, 7, 3, 4, 13, 1, 2, 2, 6,
|
143
|
+
1, 1, 1, 112, 96, 72, 82, 357, 1, 946, 3, 29, 3, 29, 2, 30, 2, 64,
|
144
|
+
2, 1, 7, 8, 1, 2, 11, 9, 1, 45, 3, 155, 1, 118, 3, 4, 2, 9, 1, 6, 3,
|
145
|
+
116, 17, 7, 2, 77, 2, 3, 228, 4, 1, 47, 1, 1, 5, 1, 1, 5, 1, 2, 38,
|
146
|
+
9, 12, 2, 1, 30, 1, 4, 2, 2, 1, 121, 8, 8, 2, 2, 392, 64, 523, 1, 2,
|
147
|
+
2, 24, 7, 49, 16, 96, 33, 3311, 32, 554, 6, 105, 2, 30164, 4, 9, 2,
|
148
|
+
388, 1, 3, 1, 4, 1, 23, 2, 2, 1, 88, 2, 50, 16, 1, 97, 8, 25, 11, 2,
|
149
|
+
213, 6, 2, 2, 2, 2, 12, 1, 8, 1, 1, 434, 11172, 1116, 1024, 6942, 1,
|
150
|
+
737, 16, 16, 7, 216, 1, 158, 2, 89, 3, 513, 1, 2051, 15, 40, 8,
|
151
|
+
50981, 1, 1, 3, 3, 1, 5, 8, 8, 2, 7, 30, 4, 148, 3, 798140, 255],
|
152
|
+
[1, 11, 1, 10, 1, 0, 1, 0, 1, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0,
|
153
|
+
2, 0, 1, 0, 2, 0, 2, 0, 2, 0, 2, 1, 2, 0, 2, 0, 2, 0, 1, 0, 2, 0, 2,
|
154
|
+
0, 2, 0, 2, 0, 2, 4, 0, 2, 0, 4, 2, 4, 2, 0, 2, 0, 2, 0, 2, 4, 0, 2,
|
155
|
+
0, 2, 4, 2, 4, 2, 0, 2, 0, 2, 0, 2, 4, 0, 2, 4, 2, 0, 2, 0, 2, 4, 0,
|
156
|
+
2, 0, 4, 2, 4, 2, 0, 2, 0, 2, 4, 0, 2, 0, 2, 4, 2, 4, 2, 0, 2, 0, 2,
|
157
|
+
0, 2, 4, 2, 4, 2, 0, 2, 0, 4, 0, 2, 4, 2, 0, 2, 0, 4, 0, 2, 0, 4, 2,
|
158
|
+
4, 2, 4, 2, 4, 2, 0, 2, 0, 4, 0, 2, 4, 2, 4, 2, 0, 2, 0, 4, 0, 2, 4,
|
159
|
+
2, 4, 2, 4, 0, 2, 0, 3, 2, 0, 2, 0, 2, 0, 3, 0, 2, 0, 2, 0, 2, 0, 2,
|
160
|
+
0, 2, 0, 4, 0, 2, 4, 2, 0, 2, 0, 2, 0, 2, 0, 4, 2, 4, 2, 4, 2, 4, 2,
|
161
|
+
0, 4, 2, 0, 2, 0, 4, 0, 4, 0, 2, 0, 2, 4, 2, 4, 2, 0, 4, 0, 5, 6, 7,
|
162
|
+
0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 1, 4, 2, 4, 2, 4, 2, 0, 2, 0, 2, 0,
|
163
|
+
2, 0, 2, 4, 2, 4, 2, 4, 2, 0, 4, 0, 4, 0, 2, 4, 0, 2, 4, 0, 2, 4, 2,
|
164
|
+
4, 2, 4, 2, 4, 0, 2, 0, 2, 4, 0, 4, 2, 4, 2, 4, 0, 4, 2, 4, 2, 0, 2,
|
165
|
+
0, 1, 2, 1, 0, 1, 0, 1, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0,
|
166
|
+
2, 0, 2, 0, 4, 2, 4, 0, 4, 0, 4, 2, 0, 2, 0, 2, 4, 0, 2, 4, 2, 4, 2,
|
167
|
+
0, 2, 0, 2, 4, 0, 9, 0, 2, 0, 2, 0, 2, 0, 2, 0, 1, 0, 2, 0, 1, 0, 2,
|
168
|
+
0, 2, 0, 2, 0, 2, 4, 2, 0, 4, 2, 1, 2, 0, 2, 0, 2, 0, 2, 0, 1, 2],
|
169
|
+
true);
|
170
|
+
}
|
171
|
+
return (/** @type {number} */
|
172
|
+
goog.i18n.GraphemeBreak.inversions_.at(acode));
|
173
|
+
}
|
174
|
+
};
|
175
|
+
|
176
|
+
|
177
|
+
/**
|
178
|
+
* There are two kinds of grapheme clusters: 1) Legacy 2)Extended. This method
|
179
|
+
* is to check for both using a boolean flag to switch between them.
|
180
|
+
* @param {number} a The code point value of the first character.
|
181
|
+
* @param {number} b The code point value of the second character.
|
182
|
+
* @param {boolean=} opt_extended If true, indicates extended grapheme cluster;
|
183
|
+
* If false, indicates legacy cluster.
|
184
|
+
* @return {boolean} True if a & b do not form a cluster; False otherwise.
|
185
|
+
*/
|
186
|
+
goog.i18n.GraphemeBreak.hasGraphemeBreak = function(a, b, opt_extended) {
|
187
|
+
|
188
|
+
var prop_a = goog.i18n.GraphemeBreak.getBreakProp_(a);
|
189
|
+
var prop_b = goog.i18n.GraphemeBreak.getBreakProp_(b);
|
190
|
+
var prop = goog.i18n.GraphemeBreak.property;
|
191
|
+
|
192
|
+
return goog.i18n.GraphemeBreak.applyLegacyBreakRules_(prop_a, prop_b) &&
|
193
|
+
!(opt_extended && (prop_a == prop.PREPEND || prop_b == prop.SPACING_MARK));
|
194
|
+
};
|
195
|
+
;
|
196
|
+
FI"asset_paths;
|
197
|
+
F["m/Volumes/Development/dev-web/yellow-brick-road/vendor/closure-library/closure/goog/i18n/graphemebreak.jsI"dependency_paths;
|
198
|
+
F[{I" path;
|
199
|
+
F"m/Volumes/Development/dev-web/yellow-brick-road/vendor/closure-library/closure/goog/i18n/graphemebreak.jsI"
|
200
|
+
mtime;
|
201
|
+
FIu: Time
|
202
|
+
T:offsetiI"hexdigest;
|
203
|
+
F"%2643825de66578a8fd08c8563f3a2184I"
|
204
|
+
F"%46dde6621c301f4928e3b34efee9e3b5
|
@@ -0,0 +1,208 @@
|
|
1
|
+
o: ActiveSupport::Cache::Entry :@compressedF:@expires_in0:@created_atf1325183043.4630148:@value{ I"length:EFi[I"digest;
|
2
|
+
F"%b79e4f4b1499a4c7ac0173a21b695d07I"source;
|
3
|
+
FI"[// Copyright 2007 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 A utility class for representing two-dimensional sizes.
|
19
|
+
*/
|
20
|
+
|
21
|
+
|
22
|
+
|
23
|
+
|
24
|
+
goog.provide('goog.math.Size');
|
25
|
+
|
26
|
+
|
27
|
+
|
28
|
+
/**
|
29
|
+
* Class for representing sizes consisting of a width and height. Undefined
|
30
|
+
* width and height support is deprecated and results in compiler warning.
|
31
|
+
* @param {number} width Width.
|
32
|
+
* @param {number} height Height.
|
33
|
+
* @constructor
|
34
|
+
*/
|
35
|
+
goog.math.Size = function(width, height) {
|
36
|
+
/**
|
37
|
+
* Width
|
38
|
+
* @type {number}
|
39
|
+
*/
|
40
|
+
this.width = width;
|
41
|
+
|
42
|
+
/**
|
43
|
+
* Height
|
44
|
+
* @type {number}
|
45
|
+
*/
|
46
|
+
this.height = height;
|
47
|
+
};
|
48
|
+
|
49
|
+
|
50
|
+
/**
|
51
|
+
* Compares sizes for equality.
|
52
|
+
* @param {goog.math.Size} a A Size.
|
53
|
+
* @param {goog.math.Size} b A Size.
|
54
|
+
* @return {boolean} True iff the sizes have equal widths and equal
|
55
|
+
* heights, or if both are null.
|
56
|
+
*/
|
57
|
+
goog.math.Size.equals = function(a, b) {
|
58
|
+
if (a == b) {
|
59
|
+
return true;
|
60
|
+
}
|
61
|
+
if (!a || !b) {
|
62
|
+
return false;
|
63
|
+
}
|
64
|
+
return a.width == b.width && a.height == b.height;
|
65
|
+
};
|
66
|
+
|
67
|
+
|
68
|
+
/**
|
69
|
+
* @return {!goog.math.Size} A new copy of the Size.
|
70
|
+
*/
|
71
|
+
goog.math.Size.prototype.clone = function() {
|
72
|
+
return new goog.math.Size(this.width, this.height);
|
73
|
+
};
|
74
|
+
|
75
|
+
|
76
|
+
if (goog.DEBUG) {
|
77
|
+
/**
|
78
|
+
* Returns a nice string representing size.
|
79
|
+
* @return {string} In the form (50 x 73).
|
80
|
+
*/
|
81
|
+
goog.math.Size.prototype.toString = function() {
|
82
|
+
return '(' + this.width + ' x ' + this.height + ')';
|
83
|
+
};
|
84
|
+
}
|
85
|
+
|
86
|
+
|
87
|
+
/**
|
88
|
+
* @return {number} The longer of the two dimensions in the size.
|
89
|
+
*/
|
90
|
+
goog.math.Size.prototype.getLongest = function() {
|
91
|
+
return Math.max(this.width, this.height);
|
92
|
+
};
|
93
|
+
|
94
|
+
|
95
|
+
/**
|
96
|
+
* @return {number} The shorter of the two dimensions in the size.
|
97
|
+
*/
|
98
|
+
goog.math.Size.prototype.getShortest = function() {
|
99
|
+
return Math.min(this.width, this.height);
|
100
|
+
};
|
101
|
+
|
102
|
+
|
103
|
+
/**
|
104
|
+
* @return {number} The area of the size (width * height).
|
105
|
+
*/
|
106
|
+
goog.math.Size.prototype.area = function() {
|
107
|
+
return this.width * this.height;
|
108
|
+
};
|
109
|
+
|
110
|
+
|
111
|
+
/**
|
112
|
+
* @return {number} The perimeter of the size (width + height) * 2.
|
113
|
+
*/
|
114
|
+
goog.math.Size.prototype.perimeter = function() {
|
115
|
+
return (this.width + this.height) * 2;
|
116
|
+
};
|
117
|
+
|
118
|
+
|
119
|
+
/**
|
120
|
+
* @return {number} The ratio of the size's width to its height.
|
121
|
+
*/
|
122
|
+
goog.math.Size.prototype.aspectRatio = function() {
|
123
|
+
return this.width / this.height;
|
124
|
+
};
|
125
|
+
|
126
|
+
|
127
|
+
/**
|
128
|
+
* @return {boolean} True if the size has zero area, false if both dimensions
|
129
|
+
* are non-zero numbers.
|
130
|
+
*/
|
131
|
+
goog.math.Size.prototype.isEmpty = function() {
|
132
|
+
return !this.area();
|
133
|
+
};
|
134
|
+
|
135
|
+
|
136
|
+
/**
|
137
|
+
* Clamps the width and height parameters upward to integer values.
|
138
|
+
* @return {!goog.math.Size} This size with ceil'd components.
|
139
|
+
*/
|
140
|
+
goog.math.Size.prototype.ceil = function() {
|
141
|
+
this.width = Math.ceil(this.width);
|
142
|
+
this.height = Math.ceil(this.height);
|
143
|
+
return this;
|
144
|
+
};
|
145
|
+
|
146
|
+
|
147
|
+
/**
|
148
|
+
* @param {!goog.math.Size} target The target size.
|
149
|
+
* @return {boolean} True if this Size is the same size or smaller than the
|
150
|
+
* target size in both dimensions.
|
151
|
+
*/
|
152
|
+
goog.math.Size.prototype.fitsInside = function(target) {
|
153
|
+
return this.width <= target.width && this.height <= target.height;
|
154
|
+
};
|
155
|
+
|
156
|
+
|
157
|
+
/**
|
158
|
+
* Clamps the width and height parameters downward to integer values.
|
159
|
+
* @return {!goog.math.Size} This size with floored components.
|
160
|
+
*/
|
161
|
+
goog.math.Size.prototype.floor = function() {
|
162
|
+
this.width = Math.floor(this.width);
|
163
|
+
this.height = Math.floor(this.height);
|
164
|
+
return this;
|
165
|
+
};
|
166
|
+
|
167
|
+
|
168
|
+
/**
|
169
|
+
* Rounds the width and height parameters to integer values.
|
170
|
+
* @return {!goog.math.Size} This size with rounded components.
|
171
|
+
*/
|
172
|
+
goog.math.Size.prototype.round = function() {
|
173
|
+
this.width = Math.round(this.width);
|
174
|
+
this.height = Math.round(this.height);
|
175
|
+
return this;
|
176
|
+
};
|
177
|
+
|
178
|
+
|
179
|
+
/**
|
180
|
+
* Scales the size uniformly by a factor.
|
181
|
+
* @param {number} s The scale factor.
|
182
|
+
* @return {!goog.math.Size} This Size object after scaling.
|
183
|
+
*/
|
184
|
+
goog.math.Size.prototype.scale = function(s) {
|
185
|
+
this.width *= s;
|
186
|
+
this.height *= s;
|
187
|
+
return this;
|
188
|
+
};
|
189
|
+
|
190
|
+
|
191
|
+
/**
|
192
|
+
* Uniformly scales the size to fit inside the dimensions of a given size. The
|
193
|
+
* original aspect ratio will be preserved.
|
194
|
+
*
|
195
|
+
* This function assumes that both Sizes contain strictly positive dimensions.
|
196
|
+
* @param {!goog.math.Size} target The target size.
|
197
|
+
* @return {!goog.math.Size} This Size object, after optional scaling.
|
198
|
+
*/
|
199
|
+
goog.math.Size.prototype.scaleToFit = function(target) {
|
200
|
+
var s = this.aspectRatio() > target.aspectRatio() ?
|
201
|
+
target.width / this.width :
|
202
|
+
target.height / this.height;
|
203
|
+
|
204
|
+
return this.scale(s);
|
205
|
+
};
|
206
|
+
;
|
207
|
+
FI"
|
208
|
+
F"%596dd22ddf7fd76fa86fd4f6175aa891
|
@@ -0,0 +1,2255 @@
|
|
1
|
+
o: ActiveSupport::Cache::Entry :@compressedF:@expires_in0:@created_atf1325456067.6155698:@value{I"
|
2
|
+
class:EFI"BundledAsset;
|
3
|
+
FI"id;
|
4
|
+
F"%61f4a2a192d6d1ed545564c1cfc631e4I"logical_path;
|
5
|
+
F"soyutils.jsI"
|
6
|
+
F"^/Volumes/Development/dev-web/yellow-brick-road/vendor/assets/closure-soyutils/soyutils.jsI"content_type;
|
7
|
+
FI"application/javascript;
|
8
|
+
FI"
|
9
|
+
mtime;
|
10
|
+
FI"2011-12-22T12:03:48+00:00;
|
11
|
+
FI" body;
|
12
|
+
FI"�/*
|
13
|
+
* Copyright 2008 Google Inc.
|
14
|
+
*
|
15
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
16
|
+
* you may not use this file except in compliance with the License.
|
17
|
+
* You may obtain a copy of the License at
|
18
|
+
*
|
19
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
20
|
+
*
|
21
|
+
* Unless required by applicable law or agreed to in writing, software
|
22
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
23
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
24
|
+
* See the License for the specific language governing permissions and
|
25
|
+
* limitations under the License.
|
26
|
+
*/
|
27
|
+
|
28
|
+
/**
|
29
|
+
* @fileoverview
|
30
|
+
* Utility functions and classes for Soy.
|
31
|
+
*
|
32
|
+
* <p>
|
33
|
+
* The top portion of this file contains utilities for Soy users:<ul>
|
34
|
+
* <li> soy.StringBuilder: Compatible with the 'stringbuilder' code style.
|
35
|
+
* <li> soy.renderElement: Render template and set as innerHTML of an element.
|
36
|
+
* <li> soy.renderAsFragment: Render template and return as HTML fragment.
|
37
|
+
* </ul>
|
38
|
+
*
|
39
|
+
* <p>
|
40
|
+
* The bottom portion of this file contains utilities that should only be called
|
41
|
+
* by Soy-generated JS code. Please do not use these functions directly from
|
42
|
+
* your hand-writen code. Their names all start with '$$'.
|
43
|
+
*
|
44
|
+
*/
|
45
|
+
|
46
|
+
|
47
|
+
// COPIED FROM nogoog_shim.js
|
48
|
+
|
49
|
+
// Create closure namespaces.
|
50
|
+
var goog = goog || {};
|
51
|
+
|
52
|
+
|
53
|
+
goog.inherits = function(childCtor, parentCtor) {
|
54
|
+
/** @constructor */
|
55
|
+
function tempCtor() {}
|
56
|
+
tempCtor.prototype = parentCtor.prototype;
|
57
|
+
childCtor.superClass_ = parentCtor.prototype;
|
58
|
+
childCtor.prototype = new tempCtor();
|
59
|
+
childCtor.prototype.constructor = childCtor;
|
60
|
+
};
|
61
|
+
|
62
|
+
|
63
|
+
// Just enough browser detection for this file.
|
64
|
+
if (!goog.userAgent) {
|
65
|
+
goog.userAgent = (function() {
|
66
|
+
var userAgent = "";
|
67
|
+
if ("undefined" !== typeof navigator && navigator
|
68
|
+
&& "string" == typeof navigator.userAgent) {
|
69
|
+
userAgent = navigator.userAgent;
|
70
|
+
}
|
71
|
+
var isOpera = userAgent.indexOf('Opera') == 0;
|
72
|
+
return {
|
73
|
+
/**
|
74
|
+
* @type {boolean}
|
75
|
+
*/
|
76
|
+
HAS_JSCRIPT: typeof 'ScriptEngine' in this,
|
77
|
+
/**
|
78
|
+
* @type {boolean}
|
79
|
+
*/
|
80
|
+
IS_OPERA: isOpera,
|
81
|
+
/**
|
82
|
+
* @type {boolean}
|
83
|
+
*/
|
84
|
+
IS_IE: !isOpera && userAgent.indexOf('MSIE') != -1,
|
85
|
+
/**
|
86
|
+
* @type {boolean}
|
87
|
+
*/
|
88
|
+
IS_WEBKIT: !isOpera && userAgent.indexOf('WebKit') != -1
|
89
|
+
};
|
90
|
+
})();
|
91
|
+
}
|
92
|
+
|
93
|
+
if (!goog.asserts) {
|
94
|
+
goog.asserts = {
|
95
|
+
/**
|
96
|
+
* @param {...*} var_args
|
97
|
+
*/
|
98
|
+
fail: function (var_args) {}
|
99
|
+
};
|
100
|
+
}
|
101
|
+
|
102
|
+
|
103
|
+
// Stub out the document wrapper used by renderAs*.
|
104
|
+
if (!goog.dom) {
|
105
|
+
goog.dom = {};
|
106
|
+
/**
|
107
|
+
* @param {Document=} d
|
108
|
+
* @constructor
|
109
|
+
*/
|
110
|
+
goog.dom.DomHelper = function(d) {
|
111
|
+
this.document_ = d || document;
|
112
|
+
};
|
113
|
+
/**
|
114
|
+
* @return {!Document}
|
115
|
+
*/
|
116
|
+
goog.dom.DomHelper.prototype.getDocument = function() {
|
117
|
+
return this.document_;
|
118
|
+
};
|
119
|
+
/**
|
120
|
+
* Creates a new element.
|
121
|
+
* @param {string} name Tag name.
|
122
|
+
* @return {!Element}
|
123
|
+
*/
|
124
|
+
goog.dom.DomHelper.prototype.createElement = function(name) {
|
125
|
+
return this.document_.createElement(name);
|
126
|
+
};
|
127
|
+
/**
|
128
|
+
* Creates a new document fragment.
|
129
|
+
* @return {!DocumentFragment}
|
130
|
+
*/
|
131
|
+
goog.dom.DomHelper.prototype.createDocumentFragment = function() {
|
132
|
+
return this.document_.createDocumentFragment();
|
133
|
+
};
|
134
|
+
}
|
135
|
+
|
136
|
+
|
137
|
+
if (!goog.format) {
|
138
|
+
goog.format = {
|
139
|
+
insertWordBreaks: function(str, maxCharsBetweenWordBreaks) {
|
140
|
+
str = String(str);
|
141
|
+
|
142
|
+
var resultArr = [];
|
143
|
+
var resultArrLen = 0;
|
144
|
+
|
145
|
+
// These variables keep track of important state inside str.
|
146
|
+
var isInTag = false; // whether we're inside an HTML tag
|
147
|
+
var isMaybeInEntity = false; // whether we might be inside an HTML entity
|
148
|
+
var numCharsWithoutBreak = 0; // number of chars since last word break
|
149
|
+
var flushIndex = 0; // index of first char not yet flushed to resultArr
|
150
|
+
|
151
|
+
for (var i = 0, n = str.length; i < n; ++i) {
|
152
|
+
var charCode = str.charCodeAt(i);
|
153
|
+
|
154
|
+
// If hit maxCharsBetweenWordBreaks, and not space next, then add <wbr>.
|
155
|
+
if (numCharsWithoutBreak >= maxCharsBetweenWordBreaks &&
|
156
|
+
// space
|
157
|
+
charCode != 32) {
|
158
|
+
resultArr[resultArrLen++] = str.substring(flushIndex, i);
|
159
|
+
flushIndex = i;
|
160
|
+
resultArr[resultArrLen++] = goog.format.WORD_BREAK;
|
161
|
+
numCharsWithoutBreak = 0;
|
162
|
+
}
|
163
|
+
|
164
|
+
if (isInTag) {
|
165
|
+
// If inside an HTML tag and we see '>', it's the end of the tag.
|
166
|
+
if (charCode == 62) {
|
167
|
+
isInTag = false;
|
168
|
+
}
|
169
|
+
|
170
|
+
} else if (isMaybeInEntity) {
|
171
|
+
switch (charCode) {
|
172
|
+
// Inside an entity, a ';' is the end of the entity.
|
173
|
+
// The entity that just ended counts as one char, so increment
|
174
|
+
// numCharsWithoutBreak.
|
175
|
+
case 59: // ';'
|
176
|
+
isMaybeInEntity = false;
|
177
|
+
++numCharsWithoutBreak;
|
178
|
+
break;
|
179
|
+
// If maybe inside an entity and we see '<', we weren't actually in
|
180
|
+
// an entity. But now we're inside and HTML tag.
|
181
|
+
case 60: // '<'
|
182
|
+
isMaybeInEntity = false;
|
183
|
+
isInTag = true;
|
184
|
+
break;
|
185
|
+
// If maybe inside an entity and we see ' ', we weren't actually in
|
186
|
+
// an entity. Just correct the state and reset the
|
187
|
+
// numCharsWithoutBreak since we just saw a space.
|
188
|
+
case 32: // ' '
|
189
|
+
isMaybeInEntity = false;
|
190
|
+
numCharsWithoutBreak = 0;
|
191
|
+
break;
|
192
|
+
}
|
193
|
+
|
194
|
+
} else { // !isInTag && !isInEntity
|
195
|
+
switch (charCode) {
|
196
|
+
// When not within a tag or an entity and we see '<', we're now
|
197
|
+
// inside an HTML tag.
|
198
|
+
case 60: // '<'
|
199
|
+
isInTag = true;
|
200
|
+
break;
|
201
|
+
// When not within a tag or an entity and we see '&', we might be
|
202
|
+
// inside an entity.
|
203
|
+
case 38: // '&'
|
204
|
+
isMaybeInEntity = true;
|
205
|
+
break;
|
206
|
+
// When we see a space, reset the numCharsWithoutBreak count.
|
207
|
+
case 32: // ' '
|
208
|
+
numCharsWithoutBreak = 0;
|
209
|
+
break;
|
210
|
+
// When we see a non-space, increment the numCharsWithoutBreak.
|
211
|
+
default:
|
212
|
+
++numCharsWithoutBreak;
|
213
|
+
break;
|
214
|
+
}
|
215
|
+
}
|
216
|
+
}
|
217
|
+
|
218
|
+
// Flush the remaining chars at the end of the string.
|
219
|
+
resultArr[resultArrLen++] = str.substring(flushIndex);
|
220
|
+
|
221
|
+
return resultArr.join('');
|
222
|
+
},
|
223
|
+
/**
|
224
|
+
* String inserted as a word break by insertWordBreaks(). Safari requires
|
225
|
+
* <wbr></wbr>, Opera needs the 'shy' entity, though this will give a
|
226
|
+
* visible hyphen at breaks. Other browsers just use <wbr>.
|
227
|
+
* @type {string}
|
228
|
+
* @private
|
229
|
+
*/
|
230
|
+
WORD_BREAK: goog.userAgent.IS_WEBKIT
|
231
|
+
? '<wbr></wbr>' : goog.userAgent.IS_OPERA ? '­' : '<wbr>'
|
232
|
+
};
|
233
|
+
}
|
234
|
+
|
235
|
+
|
236
|
+
if (!goog.i18n) {
|
237
|
+
goog.i18n = {
|
238
|
+
bidi: {
|
239
|
+
/**
|
240
|
+
* Check the directionality of a piece of text, return true if the piece
|
241
|
+
* of text should be laid out in RTL direction.
|
242
|
+
* @param {string} text The piece of text that need to be detected.
|
243
|
+
* @param {boolean=} opt_isHtml Whether {@code text} is HTML/HTML-escaped.
|
244
|
+
* Default: false.
|
245
|
+
* @return {boolean}
|
246
|
+
* @private
|
247
|
+
*/
|
248
|
+
detectRtlDirectionality: function(text, opt_isHtml) {
|
249
|
+
text = soyshim.$$bidiStripHtmlIfNecessary_(text, opt_isHtml);
|
250
|
+
return soyshim.$$bidiRtlWordRatio_(text)
|
251
|
+
> soyshim.$$bidiRtlDetectionThreshold_;
|
252
|
+
}
|
253
|
+
}
|
254
|
+
};
|
255
|
+
}
|
256
|
+
|
257
|
+
/**
|
258
|
+
* Directionality enum.
|
259
|
+
* @enum {number}
|
260
|
+
*/
|
261
|
+
goog.i18n.bidi.Dir = {
|
262
|
+
RTL: -1,
|
263
|
+
UNKNOWN: 0,
|
264
|
+
LTR: 1
|
265
|
+
};
|
266
|
+
|
267
|
+
|
268
|
+
/**
|
269
|
+
* Convert a directionality given in various formats to a goog.i18n.bidi.Dir
|
270
|
+
* constant. Useful for interaction with different standards of directionality
|
271
|
+
* representation.
|
272
|
+
*
|
273
|
+
* @param {goog.i18n.bidi.Dir|number|boolean} givenDir Directionality given in
|
274
|
+
* one of the following formats:
|
275
|
+
* 1. A goog.i18n.bidi.Dir constant.
|
276
|
+
* 2. A number (positive = LRT, negative = RTL, 0 = unknown).
|
277
|
+
* 3. A boolean (true = RTL, false = LTR).
|
278
|
+
* @return {goog.i18n.bidi.Dir} A goog.i18n.bidi.Dir constant matching the given
|
279
|
+
* directionality.
|
280
|
+
*/
|
281
|
+
goog.i18n.bidi.toDir = function(givenDir) {
|
282
|
+
if (typeof givenDir == 'number') {
|
283
|
+
return givenDir > 0 ? goog.i18n.bidi.Dir.LTR :
|
284
|
+
givenDir < 0 ? goog.i18n.bidi.Dir.RTL : goog.i18n.bidi.Dir.UNKNOWN;
|
285
|
+
} else {
|
286
|
+
return givenDir ? goog.i18n.bidi.Dir.RTL : goog.i18n.bidi.Dir.LTR;
|
287
|
+
}
|
288
|
+
};
|
289
|
+
|
290
|
+
|
291
|
+
/**
|
292
|
+
* Utility class for formatting text for display in a potentially
|
293
|
+
* opposite-directionality context without garbling. Provides the following
|
294
|
+
* functionality:
|
295
|
+
*
|
296
|
+
* @param {goog.i18n.bidi.Dir|number|boolean} dir The context
|
297
|
+
* directionality as a number
|
298
|
+
* (positive = LRT, negative = RTL, 0 = unknown).
|
299
|
+
* @constructor
|
300
|
+
*/
|
301
|
+
goog.i18n.BidiFormatter = function(dir) {
|
302
|
+
this.dir_ = goog.i18n.bidi.toDir(dir);
|
303
|
+
};
|
304
|
+
|
305
|
+
|
306
|
+
/**
|
307
|
+
* Returns "dir=ltr" or "dir=rtl", depending on {@code text}'s estimated
|
308
|
+
* directionality, if it is not the same as the context directionality.
|
309
|
+
* Otherwise, returns the empty string.
|
310
|
+
*
|
311
|
+
* @param {string} text Text whose directionality is to be estimated.
|
312
|
+
* @param {boolean=} opt_isHtml Whether {@code text} is HTML / HTML-escaped.
|
313
|
+
* Default: false.
|
314
|
+
* @return {string} "dir=rtl" for RTL text in non-RTL context; "dir=ltr" for LTR
|
315
|
+
* text in non-LTR context; else, the empty string.
|
316
|
+
*/
|
317
|
+
goog.i18n.BidiFormatter.prototype.dirAttr = function (text, opt_isHtml) {
|
318
|
+
var dir = soy.$$bidiTextDir(text, opt_isHtml);
|
319
|
+
return dir && dir != this.dir_ ? dir < 0 ? 'dir=rtl' : 'dir=ltr' : '';
|
320
|
+
};
|
321
|
+
|
322
|
+
/**
|
323
|
+
* Returns the trailing horizontal edge, i.e. "right" or "left", depending on
|
324
|
+
* the global bidi directionality.
|
325
|
+
* @return {string} "left" for RTL context and "right" otherwise.
|
326
|
+
*/
|
327
|
+
goog.i18n.BidiFormatter.prototype.endEdge = function () {
|
328
|
+
return this.dir_ < 0 ? 'left' : 'right';
|
329
|
+
};
|
330
|
+
|
331
|
+
/**
|
332
|
+
* Returns the Unicode BiDi mark matching the context directionality (LRM for
|
333
|
+
* LTR context directionality, RLM for RTL context directionality), or the
|
334
|
+
* empty string for neutral / unknown context directionality.
|
335
|
+
*
|
336
|
+
* @return {string} LRM for LTR context directionality and RLM for RTL context
|
337
|
+
* directionality.
|
338
|
+
*/
|
339
|
+
goog.i18n.BidiFormatter.prototype.mark = function () {
|
340
|
+
return (
|
341
|
+
(this.dir_ > 0) ? '\u200E' /*LRM*/ :
|
342
|
+
(this.dir_ < 0) ? '\u200F' /*RLM*/ :
|
343
|
+
'');
|
344
|
+
};
|
345
|
+
|
346
|
+
/**
|
347
|
+
* Returns a Unicode BiDi mark matching the context directionality (LRM or RLM)
|
348
|
+
* if the directionality or the exit directionality of {@code text} are opposite
|
349
|
+
* to the context directionality. Otherwise returns the empty string.
|
350
|
+
*
|
351
|
+
* @param {string} text The input text.
|
352
|
+
* @param {boolean=} opt_isHtml Whether {@code text} is HTML / HTML-escaped.
|
353
|
+
* Default: false.
|
354
|
+
* @return {string} A Unicode bidi mark matching the global directionality or
|
355
|
+
* the empty string.
|
356
|
+
*/
|
357
|
+
goog.i18n.BidiFormatter.prototype.markAfter = function (text, opt_isHtml) {
|
358
|
+
var dir = soy.$$bidiTextDir(text, opt_isHtml);
|
359
|
+
return soyshim.$$bidiMarkAfterKnownDir_(this.dir_, dir, text, opt_isHtml);
|
360
|
+
};
|
361
|
+
|
362
|
+
/**
|
363
|
+
* Formats a string of unknown directionality for use in HTML output of the
|
364
|
+
* context directionality, so an opposite-directionality string is neither
|
365
|
+
* garbled nor garbles what follows it.
|
366
|
+
*
|
367
|
+
* @param {string} str The input text.
|
368
|
+
* @param {boolean=} placeholder This argument exists for consistency with the
|
369
|
+
* Closure Library. Specifying it has no effect.
|
370
|
+
* @return {string} Input text after applying the above processing.
|
371
|
+
*/
|
372
|
+
goog.i18n.BidiFormatter.prototype.spanWrap = function(str, placeholder) {
|
373
|
+
str = String(str);
|
374
|
+
var textDir = soy.$$bidiTextDir(str, true);
|
375
|
+
var reset = soyshim.$$bidiMarkAfterKnownDir_(this.dir_, textDir, str, true);
|
376
|
+
if (textDir > 0 && this.dir_ <= 0) {
|
377
|
+
str = '<span dir=ltr>' + str + '</span>';
|
378
|
+
} else if (textDir < 0 && this.dir_ >= 0) {
|
379
|
+
str = '<span dir=rtl>' + str + '</span>';
|
380
|
+
}
|
381
|
+
return str + reset;
|
382
|
+
};
|
383
|
+
|
384
|
+
/**
|
385
|
+
* Returns the leading horizontal edge, i.e. "left" or "right", depending on
|
386
|
+
* the global bidi directionality.
|
387
|
+
* @return {string} "right" for RTL context and "left" otherwise.
|
388
|
+
*/
|
389
|
+
goog.i18n.BidiFormatter.prototype.startEdge = function () {
|
390
|
+
return this.dir_ < 0 ? 'right' : 'left';
|
391
|
+
};
|
392
|
+
|
393
|
+
/**
|
394
|
+
* Formats a string of unknown directionality for use in plain-text output of
|
395
|
+
* the context directionality, so an opposite-directionality string is neither
|
396
|
+
* garbled nor garbles what follows it.
|
397
|
+
* As opposed to {@link #spanWrap}, this makes use of unicode BiDi formatting
|
398
|
+
* characters. In HTML, its *only* valid use is inside of elements that do not
|
399
|
+
* allow mark-up, e.g. an 'option' tag.
|
400
|
+
*
|
401
|
+
* @param {string} str The input text.
|
402
|
+
* @param {boolean=} placeholder This argument exists for consistency with the
|
403
|
+
* Closure Library. Specifying it has no effect.
|
404
|
+
* @return {string} Input text after applying the above processing.
|
405
|
+
*/
|
406
|
+
goog.i18n.BidiFormatter.prototype.unicodeWrap = function(str, placeholder) {
|
407
|
+
str = String(str);
|
408
|
+
var textDir = soy.$$bidiTextDir(str, true);
|
409
|
+
var reset = soyshim.$$bidiMarkAfterKnownDir_(this.dir_, textDir, str, true);
|
410
|
+
if (textDir > 0 && this.dir_ <= 0) {
|
411
|
+
str = '\u202A' + str + '\u202C';
|
412
|
+
} else if (textDir < 0 && this.dir_ >= 0) {
|
413
|
+
str = '\u202B' + str + '\u202C';
|
414
|
+
}
|
415
|
+
return str + reset;
|
416
|
+
};
|
417
|
+
|
418
|
+
|
419
|
+
goog.string = {
|
420
|
+
|
421
|
+
/**
|
422
|
+
* Converts \r\n, \r, and \n to <br>s
|
423
|
+
* @param {*} str The string in which to convert newlines.
|
424
|
+
* @param {boolean=} opt_xml Whether to use XML compatible tags.
|
425
|
+
* @return {string} A copy of {@code str} with converted newlines.
|
426
|
+
*/
|
427
|
+
newLineToBr: function(str, opt_xml) {
|
428
|
+
|
429
|
+
str = String(str);
|
430
|
+
|
431
|
+
// This quick test helps in the case when there are no chars to replace,
|
432
|
+
// in the worst case this makes barely a difference to the time taken.
|
433
|
+
if (!goog.string.NEWLINE_TO_BR_RE_.test(str)) {
|
434
|
+
return str;
|
435
|
+
}
|
436
|
+
|
437
|
+
return str.replace(/(\r\n|\r|\n)/g, opt_xml ? '<br />' : '<br>');
|
438
|
+
},
|
439
|
+
urlEncode: encodeURIComponent,
|
440
|
+
/**
|
441
|
+
* Regular expression used within newlineToBr().
|
442
|
+
* @type {RegExp}
|
443
|
+
* @private
|
444
|
+
*/
|
445
|
+
NEWLINE_TO_BR_RE_: /[\r\n]/
|
446
|
+
};
|
447
|
+
|
448
|
+
|
449
|
+
/**
|
450
|
+
* Utility class to facilitate much faster string concatenation in IE,
|
451
|
+
* using Array.join() rather than the '+' operator. For other browsers
|
452
|
+
* we simply use the '+' operator.
|
453
|
+
*
|
454
|
+
* @param {Object|number|string|boolean=} opt_a1 Optional first initial item
|
455
|
+
* to append.
|
456
|
+
* @param {...Object|number|string|boolean} var_args Other initial items to
|
457
|
+
* append, e.g., new goog.string.StringBuffer('foo', 'bar').
|
458
|
+
* @constructor
|
459
|
+
*/
|
460
|
+
goog.string.StringBuffer = function(opt_a1, var_args) {
|
461
|
+
/**
|
462
|
+
* Internal buffer for the string to be concatenated.
|
463
|
+
* @type {string|Array}
|
464
|
+
* @private
|
465
|
+
*/
|
466
|
+
this.buffer_ = goog.userAgent.HAS_JSCRIPT ? [] : '';
|
467
|
+
|
468
|
+
if (opt_a1 != null) {
|
469
|
+
this.append.apply(this, arguments);
|
470
|
+
}
|
471
|
+
};
|
472
|
+
|
473
|
+
|
474
|
+
/**
|
475
|
+
* Length of internal buffer (faster than calling buffer_.length).
|
476
|
+
* Only used for IE.
|
477
|
+
* @type {number}
|
478
|
+
* @private
|
479
|
+
*/
|
480
|
+
goog.string.StringBuffer.prototype.bufferLength_ = 0;
|
481
|
+
|
482
|
+
/**
|
483
|
+
* Appends one or more items to the string.
|
484
|
+
*
|
485
|
+
* Calling this with null, undefined, or empty arguments is an error.
|
486
|
+
*
|
487
|
+
* @param {Object|number|string|boolean} a1 Required first string.
|
488
|
+
* @param {Object|number|string|boolean=} opt_a2 Optional second string.
|
489
|
+
* @param {...Object|number|string|boolean} var_args Other items to append,
|
490
|
+
* e.g., sb.append('foo', 'bar', 'baz').
|
491
|
+
* @return {goog.string.StringBuffer} This same StringBuilder object.
|
492
|
+
*/
|
493
|
+
goog.string.StringBuffer.prototype.append = function(a1, opt_a2, var_args) {
|
494
|
+
|
495
|
+
if (goog.userAgent.HAS_JSCRIPT) {
|
496
|
+
if (opt_a2 == null) { // no second argument (note: undefined == null)
|
497
|
+
// Array assignment is 2x faster than Array push. Also, use a1
|
498
|
+
// directly to avoid arguments instantiation, another 2x improvement.
|
499
|
+
this.buffer_[this.bufferLength_++] = a1;
|
500
|
+
} else {
|
501
|
+
var arr = /**@type {Array.<number|string|boolean>}*/this.buffer_;
|
502
|
+
arr.push.apply(arr, arguments);
|
503
|
+
this.bufferLength_ = this.buffer_.length;
|
504
|
+
}
|
505
|
+
|
506
|
+
} else {
|
507
|
+
|
508
|
+
// Use a1 directly to avoid arguments instantiation for single-arg case.
|
509
|
+
this.buffer_ += a1;
|
510
|
+
if (opt_a2 != null) { // no second argument (note: undefined == null)
|
511
|
+
for (var i = 1; i < arguments.length; i++) {
|
512
|
+
this.buffer_ += arguments[i];
|
513
|
+
}
|
514
|
+
}
|
515
|
+
}
|
516
|
+
|
517
|
+
return this;
|
518
|
+
};
|
519
|
+
|
520
|
+
|
521
|
+
/**
|
522
|
+
* Clears the string.
|
523
|
+
*/
|
524
|
+
goog.string.StringBuffer.prototype.clear = function() {
|
525
|
+
|
526
|
+
if (goog.userAgent.HAS_JSCRIPT) {
|
527
|
+
this.buffer_.length = 0; // reuse array to avoid creating new object
|
528
|
+
this.bufferLength_ = 0;
|
529
|
+
|
530
|
+
} else {
|
531
|
+
this.buffer_ = '';
|
532
|
+
}
|
533
|
+
};
|
534
|
+
|
535
|
+
|
536
|
+
/**
|
537
|
+
* Returns the concatenated string.
|
538
|
+
*
|
539
|
+
* @return {string} The concatenated string.
|
540
|
+
*/
|
541
|
+
goog.string.StringBuffer.prototype.toString = function() {
|
542
|
+
|
543
|
+
if (goog.userAgent.HAS_JSCRIPT) {
|
544
|
+
var str = this.buffer_.join('');
|
545
|
+
// Given a string with the entire contents, simplify the StringBuilder by
|
546
|
+
// setting its contents to only be this string, rather than many fragments.
|
547
|
+
this.clear();
|
548
|
+
if (str) {
|
549
|
+
this.append(str);
|
550
|
+
}
|
551
|
+
return str;
|
552
|
+
|
553
|
+
} else {
|
554
|
+
return /** @type {string} */ (this.buffer_);
|
555
|
+
}
|
556
|
+
};
|
557
|
+
|
558
|
+
|
559
|
+
if (!goog.soy) goog.soy = {
|
560
|
+
/**
|
561
|
+
* Helper function to render a Soy template and then set the
|
562
|
+
* output string as the innerHTML of an element. It is recommended
|
563
|
+
* to use this helper function instead of directly setting
|
564
|
+
* innerHTML in your hand-written code, so that it will be easier
|
565
|
+
* to audit the code for cross-site scripting vulnerabilities.
|
566
|
+
*
|
567
|
+
* @param {Function} template The Soy template defining element's content.
|
568
|
+
* @param {Object=} opt_templateData The data for the template.
|
569
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
570
|
+
* @param {(goog.dom.DomHelper|Document)=} opt_dom The context in which DOM
|
571
|
+
* nodes will be created.
|
572
|
+
*/
|
573
|
+
renderAsElement: function(
|
574
|
+
template, opt_templateData, opt_injectedData, opt_dom) {
|
575
|
+
return /** @type {!Element} */ (soyshim.$$renderWithWrapper_(
|
576
|
+
template, opt_templateData, opt_dom, true /* asElement */,
|
577
|
+
opt_injectedData));
|
578
|
+
},
|
579
|
+
/**
|
580
|
+
* Helper function to render a Soy template into a single node or
|
581
|
+
* a document fragment. If the rendered HTML string represents a
|
582
|
+
* single node, then that node is returned (note that this is
|
583
|
+
* *not* a fragment, despite them name of the method). Otherwise a
|
584
|
+
* document fragment is returned containing the rendered nodes.
|
585
|
+
*
|
586
|
+
* @param {Function} template The Soy template defining element's content.
|
587
|
+
* @param {Object=} opt_templateData The data for the template.
|
588
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
589
|
+
* @param {(goog.dom.DomHelper|Document)=} opt_dom The context in which DOM
|
590
|
+
* nodes will be created.
|
591
|
+
* @return {!Node} The resulting node or document fragment.
|
592
|
+
*/
|
593
|
+
renderAsFragment: function(
|
594
|
+
template, opt_templateData, opt_injectedData, opt_dom) {
|
595
|
+
return soyshim.$$renderWithWrapper_(
|
596
|
+
template, opt_templateData, opt_dom, false /* asElement */,
|
597
|
+
opt_injectedData);
|
598
|
+
},
|
599
|
+
/**
|
600
|
+
* Helper function to render a Soy template and then set the output string as
|
601
|
+
* the innerHTML of an element. It is recommended to use this helper function
|
602
|
+
* instead of directly setting innerHTML in your hand-written code, so that it
|
603
|
+
* will be easier to audit the code for cross-site scripting vulnerabilities.
|
604
|
+
*
|
605
|
+
* NOTE: New code should consider using goog.soy.renderElement instead.
|
606
|
+
*
|
607
|
+
* @param {Element} element The element whose content we are rendering.
|
608
|
+
* @param {Function} template The Soy template defining the element's content.
|
609
|
+
* @param {Object=} opt_templateData The data for the template.
|
610
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
611
|
+
*/
|
612
|
+
renderElement: function(
|
613
|
+
element, template, opt_templateData, opt_injectedData) {
|
614
|
+
element.innerHTML = template(opt_templateData, null, opt_injectedData);
|
615
|
+
}
|
616
|
+
};
|
617
|
+
|
618
|
+
|
619
|
+
var soy = { esc: {} };
|
620
|
+
var soydata = {};
|
621
|
+
var soyshim = { $$DEFAULT_TEMPLATE_DATA_: {} };
|
622
|
+
/**
|
623
|
+
* Helper function to render a Soy template into a single node or a document
|
624
|
+
* fragment. If the rendered HTML string represents a single node, then that
|
625
|
+
* node is returned. Otherwise a document fragment is created and returned
|
626
|
+
* (wrapped in a DIV element if #opt_singleNode is true).
|
627
|
+
*
|
628
|
+
* @param {Function} template The Soy template defining the element's content.
|
629
|
+
* @param {Object=} opt_templateData The data for the template.
|
630
|
+
* @param {(goog.dom.DomHelper|Document)=} opt_dom The context in which DOM
|
631
|
+
* nodes will be created.
|
632
|
+
* @param {boolean=} opt_asElement Whether to wrap the fragment in an
|
633
|
+
* element if the template does not render a single element. If true,
|
634
|
+
* result is always an Element.
|
635
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
636
|
+
* @return {!Node} The resulting node or document fragment.
|
637
|
+
* @private
|
638
|
+
*/
|
639
|
+
soyshim.$$renderWithWrapper_ = function(
|
640
|
+
template, opt_templateData, opt_dom, opt_asElement, opt_injectedData) {
|
641
|
+
|
642
|
+
var dom = opt_dom || document;
|
643
|
+
var wrapper = dom.createElement('div');
|
644
|
+
wrapper.innerHTML = template(
|
645
|
+
opt_templateData || soyshim.$$DEFAULT_TEMPLATE_DATA_, undefined,
|
646
|
+
opt_injectedData);
|
647
|
+
|
648
|
+
// If the template renders as a single element, return it.
|
649
|
+
if (wrapper.childNodes.length == 1) {
|
650
|
+
var firstChild = wrapper.firstChild;
|
651
|
+
if (!opt_asElement || firstChild.nodeType == 1 /* Element */) {
|
652
|
+
return /** @type {!Node} */ (firstChild);
|
653
|
+
}
|
654
|
+
}
|
655
|
+
|
656
|
+
// If we're forcing it to be a single element, return the wrapper DIV.
|
657
|
+
if (opt_asElement) {
|
658
|
+
return wrapper;
|
659
|
+
}
|
660
|
+
|
661
|
+
// Otherwise, create and return a fragment.
|
662
|
+
var fragment = dom.createDocumentFragment();
|
663
|
+
while (wrapper.firstChild) {
|
664
|
+
fragment.appendChild(wrapper.firstChild);
|
665
|
+
}
|
666
|
+
return fragment;
|
667
|
+
};
|
668
|
+
|
669
|
+
|
670
|
+
/**
|
671
|
+
* Returns a Unicode BiDi mark matching bidiGlobalDir (LRM or RLM) if the
|
672
|
+
* directionality or the exit directionality of text are opposite to
|
673
|
+
* bidiGlobalDir. Otherwise returns the empty string.
|
674
|
+
* If opt_isHtml, makes sure to ignore the LTR nature of the mark-up and escapes
|
675
|
+
* in text, making the logic suitable for HTML and HTML-escaped text.
|
676
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
677
|
+
* if rtl, 0 if unknown.
|
678
|
+
* @param {number} dir text's directionality: 1 if ltr, -1 if rtl, 0 if unknown.
|
679
|
+
* @param {string} text The text whose directionality is to be estimated.
|
680
|
+
* @param {boolean=} opt_isHtml Whether text is HTML/HTML-escaped.
|
681
|
+
* Default: false.
|
682
|
+
* @return {string} A Unicode bidi mark matching bidiGlobalDir, or
|
683
|
+
* the empty string when text's overall and exit directionalities both match
|
684
|
+
* bidiGlobalDir, or bidiGlobalDir is 0 (unknown).
|
685
|
+
* @private
|
686
|
+
*/
|
687
|
+
soyshim.$$bidiMarkAfterKnownDir_ = function(
|
688
|
+
bidiGlobalDir, dir, text, opt_isHtml) {
|
689
|
+
return (
|
690
|
+
bidiGlobalDir > 0 && (dir < 0 ||
|
691
|
+
soyshim.$$bidiIsRtlExitText_(text, opt_isHtml)) ? '\u200E' : // LRM
|
692
|
+
bidiGlobalDir < 0 && (dir > 0 ||
|
693
|
+
soyshim.$$bidiIsLtrExitText_(text, opt_isHtml)) ? '\u200F' : // RLM
|
694
|
+
'');
|
695
|
+
};
|
696
|
+
|
697
|
+
|
698
|
+
/**
|
699
|
+
* Strips str of any HTML mark-up and escapes. Imprecise in several ways, but
|
700
|
+
* precision is not very important, since the result is only meant to be used
|
701
|
+
* for directionality detection.
|
702
|
+
* @param {string} str The string to be stripped.
|
703
|
+
* @param {boolean=} opt_isHtml Whether str is HTML / HTML-escaped.
|
704
|
+
* Default: false.
|
705
|
+
* @return {string} The stripped string.
|
706
|
+
* @private
|
707
|
+
*/
|
708
|
+
soyshim.$$bidiStripHtmlIfNecessary_ = function(str, opt_isHtml) {
|
709
|
+
return opt_isHtml ? str.replace(soyshim.$$BIDI_HTML_SKIP_RE_, ' ') : str;
|
710
|
+
};
|
711
|
+
|
712
|
+
|
713
|
+
/**
|
714
|
+
* Simplified regular expression for am HTML tag (opening or closing) or an HTML
|
715
|
+
* escape - the things we want to skip over in order to ignore their ltr
|
716
|
+
* characters.
|
717
|
+
* @type {RegExp}
|
718
|
+
* @private
|
719
|
+
*/
|
720
|
+
soyshim.$$BIDI_HTML_SKIP_RE_ = /<[^>]*>|&[^;]+;/g;
|
721
|
+
|
722
|
+
|
723
|
+
/**
|
724
|
+
* A practical pattern to identify strong LTR character. This pattern is not
|
725
|
+
* theoretically correct according to unicode standard. It is simplified for
|
726
|
+
* performance and small code size.
|
727
|
+
* @type {string}
|
728
|
+
* @private
|
729
|
+
*/
|
730
|
+
soyshim.$$bidiLtrChars_ =
|
731
|
+
'A-Za-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02B8\u0300-\u0590\u0800-\u1FFF' +
|
732
|
+
'\u2C00-\uFB1C\uFDFE-\uFE6F\uFEFD-\uFFFF';
|
733
|
+
|
734
|
+
|
735
|
+
/**
|
736
|
+
* A practical pattern to identify strong neutral and weak character. This
|
737
|
+
* pattern is not theoretically correct according to unicode standard. It is
|
738
|
+
* simplified for performance and small code size.
|
739
|
+
* @type {string}
|
740
|
+
* @private
|
741
|
+
*/
|
742
|
+
soyshim.$$bidiNeutralChars_ =
|
743
|
+
'\u0000-\u0020!-@[-`{-\u00BF\u00D7\u00F7\u02B9-\u02FF\u2000-\u2BFF';
|
744
|
+
|
745
|
+
|
746
|
+
/**
|
747
|
+
* A practical pattern to identify strong RTL character. This pattern is not
|
748
|
+
* theoretically correct according to unicode standard. It is simplified for
|
749
|
+
* performance and small code size.
|
750
|
+
* @type {string}
|
751
|
+
* @private
|
752
|
+
*/
|
753
|
+
soyshim.$$bidiRtlChars_ = '\u0591-\u07FF\uFB1D-\uFDFD\uFE70-\uFEFC';
|
754
|
+
|
755
|
+
|
756
|
+
/**
|
757
|
+
* Regular expressions to check if a piece of text is of RTL directionality
|
758
|
+
* on first character with strong directionality.
|
759
|
+
* @type {RegExp}
|
760
|
+
* @private
|
761
|
+
*/
|
762
|
+
soyshim.$$bidiRtlDirCheckRe_ = new RegExp(
|
763
|
+
'^[^' + soyshim.$$bidiLtrChars_ + ']*[' + soyshim.$$bidiRtlChars_ + ']');
|
764
|
+
|
765
|
+
|
766
|
+
/**
|
767
|
+
* Regular expressions to check if a piece of text is of neutral directionality.
|
768
|
+
* Url are considered as neutral.
|
769
|
+
* @type {RegExp}
|
770
|
+
* @private
|
771
|
+
*/
|
772
|
+
soyshim.$$bidiNeutralDirCheckRe_ = new RegExp(
|
773
|
+
'^[' + soyshim.$$bidiNeutralChars_ + ']*$|^http://');
|
774
|
+
|
775
|
+
|
776
|
+
/**
|
777
|
+
* Check the directionality of the a piece of text based on the first character
|
778
|
+
* with strong directionality.
|
779
|
+
* @param {string} str string being checked.
|
780
|
+
* @return {boolean} return true if rtl directionality is being detected.
|
781
|
+
* @private
|
782
|
+
*/
|
783
|
+
soyshim.$$bidiIsRtlText_ = function(str) {
|
784
|
+
return soyshim.$$bidiRtlDirCheckRe_.test(str);
|
785
|
+
};
|
786
|
+
|
787
|
+
|
788
|
+
/**
|
789
|
+
* Check the directionality of the a piece of text based on the first character
|
790
|
+
* with strong directionality.
|
791
|
+
* @param {string} str string being checked.
|
792
|
+
* @return {boolean} true if all characters have neutral directionality.
|
793
|
+
* @private
|
794
|
+
*/
|
795
|
+
soyshim.$$bidiIsNeutralText_ = function(str) {
|
796
|
+
return soyshim.$$bidiNeutralDirCheckRe_.test(str);
|
797
|
+
};
|
798
|
+
|
799
|
+
|
800
|
+
/**
|
801
|
+
* This constant controls threshold of rtl directionality.
|
802
|
+
* @type {number}
|
803
|
+
* @private
|
804
|
+
*/
|
805
|
+
soyshim.$$bidiRtlDetectionThreshold_ = 0.40;
|
806
|
+
|
807
|
+
|
808
|
+
/**
|
809
|
+
* Returns the RTL ratio based on word count.
|
810
|
+
* @param {string} str the string that need to be checked.
|
811
|
+
* @return {number} the ratio of RTL words among all words with directionality.
|
812
|
+
* @private
|
813
|
+
*/
|
814
|
+
soyshim.$$bidiRtlWordRatio_ = function(str) {
|
815
|
+
var rtlCount = 0;
|
816
|
+
var totalCount = 0;
|
817
|
+
var tokens = str.split(' ');
|
818
|
+
for (var i = 0; i < tokens.length; i++) {
|
819
|
+
if (soyshim.$$bidiIsRtlText_(tokens[i])) {
|
820
|
+
rtlCount++;
|
821
|
+
totalCount++;
|
822
|
+
} else if (!soyshim.$$bidiIsNeutralText_(tokens[i])) {
|
823
|
+
totalCount++;
|
824
|
+
}
|
825
|
+
}
|
826
|
+
|
827
|
+
return totalCount == 0 ? 0 : rtlCount / totalCount;
|
828
|
+
};
|
829
|
+
|
830
|
+
|
831
|
+
/**
|
832
|
+
* Regular expressions to check if the last strongly-directional character in a
|
833
|
+
* piece of text is LTR.
|
834
|
+
* @type {RegExp}
|
835
|
+
* @private
|
836
|
+
*/
|
837
|
+
soyshim.$$bidiLtrExitDirCheckRe_ = new RegExp(
|
838
|
+
'[' + soyshim.$$bidiLtrChars_ + '][^' + soyshim.$$bidiRtlChars_ + ']*$');
|
839
|
+
|
840
|
+
|
841
|
+
/**
|
842
|
+
* Regular expressions to check if the last strongly-directional character in a
|
843
|
+
* piece of text is RTL.
|
844
|
+
* @type {RegExp}
|
845
|
+
* @private
|
846
|
+
*/
|
847
|
+
soyshim.$$bidiRtlExitDirCheckRe_ = new RegExp(
|
848
|
+
'[' + soyshim.$$bidiRtlChars_ + '][^' + soyshim.$$bidiLtrChars_ + ']*$');
|
849
|
+
|
850
|
+
|
851
|
+
/**
|
852
|
+
* Check if the exit directionality a piece of text is LTR, i.e. if the last
|
853
|
+
* strongly-directional character in the string is LTR.
|
854
|
+
* @param {string} str string being checked.
|
855
|
+
* @param {boolean=} opt_isHtml Whether str is HTML / HTML-escaped.
|
856
|
+
* Default: false.
|
857
|
+
* @return {boolean} Whether LTR exit directionality was detected.
|
858
|
+
* @private
|
859
|
+
*/
|
860
|
+
soyshim.$$bidiIsLtrExitText_ = function(str, opt_isHtml) {
|
861
|
+
str = soyshim.$$bidiStripHtmlIfNecessary_(str, opt_isHtml);
|
862
|
+
return soyshim.$$bidiLtrExitDirCheckRe_.test(str);
|
863
|
+
};
|
864
|
+
|
865
|
+
|
866
|
+
/**
|
867
|
+
* Check if the exit directionality a piece of text is RTL, i.e. if the last
|
868
|
+
* strongly-directional character in the string is RTL.
|
869
|
+
* @param {string} str string being checked.
|
870
|
+
* @param {boolean=} opt_isHtml Whether str is HTML / HTML-escaped.
|
871
|
+
* Default: false.
|
872
|
+
* @return {boolean} Whether RTL exit directionality was detected.
|
873
|
+
* @private
|
874
|
+
*/
|
875
|
+
soyshim.$$bidiIsRtlExitText_ = function(str, opt_isHtml) {
|
876
|
+
str = soyshim.$$bidiStripHtmlIfNecessary_(str, opt_isHtml);
|
877
|
+
return soyshim.$$bidiRtlExitDirCheckRe_.test(str);
|
878
|
+
};
|
879
|
+
|
880
|
+
|
881
|
+
// =============================================================================
|
882
|
+
// COPIED FROM soyutils_usegoog.js
|
883
|
+
|
884
|
+
|
885
|
+
// -----------------------------------------------------------------------------
|
886
|
+
// StringBuilder (compatible with the 'stringbuilder' code style).
|
887
|
+
|
888
|
+
|
889
|
+
/**
|
890
|
+
* Utility class to facilitate much faster string concatenation in IE,
|
891
|
+
* using Array.join() rather than the '+' operator. For other browsers
|
892
|
+
* we simply use the '+' operator.
|
893
|
+
*
|
894
|
+
* @param {Object} var_args Initial items to append,
|
895
|
+
* e.g., new soy.StringBuilder('foo', 'bar').
|
896
|
+
* @constructor
|
897
|
+
*/
|
898
|
+
soy.StringBuilder = goog.string.StringBuffer;
|
899
|
+
|
900
|
+
|
901
|
+
// -----------------------------------------------------------------------------
|
902
|
+
// soydata: Defines typed strings, e.g. an HTML string {@code "a<b>c"} is
|
903
|
+
// semantically distinct from the plain text string {@code "a<b>c"} and smart
|
904
|
+
// templates can take that distinction into account.
|
905
|
+
|
906
|
+
/**
|
907
|
+
* A type of textual content.
|
908
|
+
* @enum {number}
|
909
|
+
*/
|
910
|
+
soydata.SanitizedContentKind = {
|
911
|
+
|
912
|
+
/**
|
913
|
+
* A snippet of HTML that does not start or end inside a tag, comment, entity,
|
914
|
+
* or DOCTYPE; and that does not contain any executable code
|
915
|
+
* (JS, {@code <object>}s, etc.) from a different trust domain.
|
916
|
+
*/
|
917
|
+
HTML: 0,
|
918
|
+
|
919
|
+
/**
|
920
|
+
* A sequence of code units that can appear between quotes (either kind) in a
|
921
|
+
* JS program without causing a parse error, and without causing any side
|
922
|
+
* effects.
|
923
|
+
* <p>
|
924
|
+
* The content should not contain unescaped quotes, newlines, or anything else
|
925
|
+
* that would cause parsing to fail or to cause a JS parser to finish the
|
926
|
+
* string its parsing inside the content.
|
927
|
+
* <p>
|
928
|
+
* The content must also not end inside an escape sequence ; no partial octal
|
929
|
+
* escape sequences or odd number of '{@code \}'s at the end.
|
930
|
+
*/
|
931
|
+
JS_STR_CHARS: 1,
|
932
|
+
|
933
|
+
/** A properly encoded portion of a URI. */
|
934
|
+
URI: 2,
|
935
|
+
|
936
|
+
/** An attribute name and value such as {@code dir="ltr"}. */
|
937
|
+
HTML_ATTRIBUTE: 3
|
938
|
+
};
|
939
|
+
|
940
|
+
|
941
|
+
/**
|
942
|
+
* A string-like object that carries a content-type.
|
943
|
+
* @param {string} content
|
944
|
+
* @constructor
|
945
|
+
* @private
|
946
|
+
*/
|
947
|
+
soydata.SanitizedContent = function(content) {
|
948
|
+
/**
|
949
|
+
* The textual content.
|
950
|
+
* @type {string}
|
951
|
+
*/
|
952
|
+
this.content = content;
|
953
|
+
};
|
954
|
+
|
955
|
+
/** @type {soydata.SanitizedContentKind} */
|
956
|
+
soydata.SanitizedContent.prototype.contentKind;
|
957
|
+
|
958
|
+
/** @override */
|
959
|
+
soydata.SanitizedContent.prototype.toString = function() {
|
960
|
+
return this.content;
|
961
|
+
};
|
962
|
+
|
963
|
+
|
964
|
+
/**
|
965
|
+
* Content of type {@link soydata.SanitizedContentKind.HTML}.
|
966
|
+
* @param {string} content A string of HTML that can safely be embedded in
|
967
|
+
* a PCDATA context in your app. If you would be surprised to find that an
|
968
|
+
* HTML sanitizer produced {@code s} (e.g. it runs code or fetches bad URLs)
|
969
|
+
* and you wouldn't write a template that produces {@code s} on security or
|
970
|
+
* privacy grounds, then don't pass {@code s} here.
|
971
|
+
* @constructor
|
972
|
+
* @extends {soydata.SanitizedContent}
|
973
|
+
*/
|
974
|
+
soydata.SanitizedHtml = function(content) {
|
975
|
+
soydata.SanitizedContent.call(this, content);
|
976
|
+
};
|
977
|
+
goog.inherits(soydata.SanitizedHtml, soydata.SanitizedContent);
|
978
|
+
|
979
|
+
/** @override */
|
980
|
+
soydata.SanitizedHtml.prototype.contentKind = soydata.SanitizedContentKind.HTML;
|
981
|
+
|
982
|
+
|
983
|
+
/**
|
984
|
+
* Content of type {@link soydata.SanitizedContentKind.JS_STR_CHARS}.
|
985
|
+
* @param {string} content A string of JS that when evaled, produces a
|
986
|
+
* value that does not depend on any sensitive data and has no side effects
|
987
|
+
* <b>OR</b> a string of JS that does not reference any variables or have
|
988
|
+
* any side effects not known statically to the app authors.
|
989
|
+
* @constructor
|
990
|
+
* @extends {soydata.SanitizedContent}
|
991
|
+
*/
|
992
|
+
soydata.SanitizedJsStrChars = function(content) {
|
993
|
+
soydata.SanitizedContent.call(this, content);
|
994
|
+
};
|
995
|
+
goog.inherits(soydata.SanitizedJsStrChars, soydata.SanitizedContent);
|
996
|
+
|
997
|
+
/** @override */
|
998
|
+
soydata.SanitizedJsStrChars.prototype.contentKind =
|
999
|
+
soydata.SanitizedContentKind.JS_STR_CHARS;
|
1000
|
+
|
1001
|
+
|
1002
|
+
/**
|
1003
|
+
* Content of type {@link soydata.SanitizedContentKind.URI}.
|
1004
|
+
* @param {string} content A chunk of URI that the caller knows is safe to
|
1005
|
+
* emit in a template.
|
1006
|
+
* @constructor
|
1007
|
+
* @extends {soydata.SanitizedContent}
|
1008
|
+
*/
|
1009
|
+
soydata.SanitizedUri = function(content) {
|
1010
|
+
soydata.SanitizedContent.call(this, content);
|
1011
|
+
};
|
1012
|
+
goog.inherits(soydata.SanitizedUri, soydata.SanitizedContent);
|
1013
|
+
|
1014
|
+
/** @override */
|
1015
|
+
soydata.SanitizedUri.prototype.contentKind = soydata.SanitizedContentKind.URI;
|
1016
|
+
|
1017
|
+
|
1018
|
+
/**
|
1019
|
+
* Content of type {@link soydata.SanitizedContentKind.HTML_ATTRIBUTE}.
|
1020
|
+
* @param {string} content An attribute name and value, such as
|
1021
|
+
* {@code dir="ltr"}.
|
1022
|
+
* @constructor
|
1023
|
+
* @extends {soydata.SanitizedContent}
|
1024
|
+
*/
|
1025
|
+
soydata.SanitizedHtmlAttribute = function(content) {
|
1026
|
+
soydata.SanitizedContent.call(this, content);
|
1027
|
+
};
|
1028
|
+
goog.inherits(soydata.SanitizedHtmlAttribute, soydata.SanitizedContent);
|
1029
|
+
|
1030
|
+
/** @override */
|
1031
|
+
soydata.SanitizedHtmlAttribute.prototype.contentKind =
|
1032
|
+
soydata.SanitizedContentKind.HTML_ATTRIBUTE;
|
1033
|
+
|
1034
|
+
|
1035
|
+
// -----------------------------------------------------------------------------
|
1036
|
+
// Public utilities.
|
1037
|
+
|
1038
|
+
|
1039
|
+
/**
|
1040
|
+
* Helper function to render a Soy template and then set the output string as
|
1041
|
+
* the innerHTML of an element. It is recommended to use this helper function
|
1042
|
+
* instead of directly setting innerHTML in your hand-written code, so that it
|
1043
|
+
* will be easier to audit the code for cross-site scripting vulnerabilities.
|
1044
|
+
*
|
1045
|
+
* NOTE: New code should consider using goog.soy.renderElement instead.
|
1046
|
+
*
|
1047
|
+
* @param {Element} element The element whose content we are rendering.
|
1048
|
+
* @param {Function} template The Soy template defining the element's content.
|
1049
|
+
* @param {Object=} opt_templateData The data for the template.
|
1050
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
1051
|
+
*/
|
1052
|
+
soy.renderElement = goog.soy.renderElement;
|
1053
|
+
|
1054
|
+
|
1055
|
+
/**
|
1056
|
+
* Helper function to render a Soy template into a single node or a document
|
1057
|
+
* fragment. If the rendered HTML string represents a single node, then that
|
1058
|
+
* node is returned (note that this is *not* a fragment, despite them name of
|
1059
|
+
* the method). Otherwise a document fragment is returned containing the
|
1060
|
+
* rendered nodes.
|
1061
|
+
*
|
1062
|
+
* NOTE: New code should consider using goog.soy.renderAsFragment
|
1063
|
+
* instead (note that the arguments are different).
|
1064
|
+
*
|
1065
|
+
* @param {Function} template The Soy template defining the element's content.
|
1066
|
+
* @param {Object=} opt_templateData The data for the template.
|
1067
|
+
* @param {Document=} opt_document The document used to create DOM nodes. If not
|
1068
|
+
* specified, global document object is used.
|
1069
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
1070
|
+
* @return {!Node} The resulting node or document fragment.
|
1071
|
+
*/
|
1072
|
+
soy.renderAsFragment = function(
|
1073
|
+
template, opt_templateData, opt_document, opt_injectedData) {
|
1074
|
+
return goog.soy.renderAsFragment(
|
1075
|
+
template, opt_templateData, opt_injectedData,
|
1076
|
+
new goog.dom.DomHelper(opt_document));
|
1077
|
+
};
|
1078
|
+
|
1079
|
+
|
1080
|
+
/**
|
1081
|
+
* Helper function to render a Soy template into a single node. If the rendered
|
1082
|
+
* HTML string represents a single node, then that node is returned. Otherwise,
|
1083
|
+
* a DIV element is returned containing the rendered nodes.
|
1084
|
+
*
|
1085
|
+
* NOTE: New code should consider using goog.soy.renderAsElement
|
1086
|
+
* instead (note that the arguments are different).
|
1087
|
+
*
|
1088
|
+
* @param {Function} template The Soy template defining the element's content.
|
1089
|
+
* @param {Object=} opt_templateData The data for the template.
|
1090
|
+
* @param {Document=} opt_document The document used to create DOM nodes. If not
|
1091
|
+
* specified, global document object is used.
|
1092
|
+
* @param {Object=} opt_injectedData The injected data for the template.
|
1093
|
+
* @return {!Element} Rendered template contents, wrapped in a parent DIV
|
1094
|
+
* element if necessary.
|
1095
|
+
*/
|
1096
|
+
soy.renderAsElement = function(
|
1097
|
+
template, opt_templateData, opt_document, opt_injectedData) {
|
1098
|
+
return goog.soy.renderAsElement(
|
1099
|
+
template, opt_templateData, opt_injectedData,
|
1100
|
+
new goog.dom.DomHelper(opt_document));
|
1101
|
+
};
|
1102
|
+
|
1103
|
+
|
1104
|
+
// -----------------------------------------------------------------------------
|
1105
|
+
// Below are private utilities to be used by Soy-generated code only.
|
1106
|
+
|
1107
|
+
|
1108
|
+
/**
|
1109
|
+
* Builds an augmented data object to be passed when a template calls another,
|
1110
|
+
* and needs to pass both original data and additional params. The returned
|
1111
|
+
* object will contain both the original data and the additional params. If the
|
1112
|
+
* same key appears in both, then the value from the additional params will be
|
1113
|
+
* visible, while the value from the original data will be hidden. The original
|
1114
|
+
* data object will be used, but not modified.
|
1115
|
+
*
|
1116
|
+
* @param {!Object} origData The original data to pass.
|
1117
|
+
* @param {Object} additionalParams The additional params to pass.
|
1118
|
+
* @return {Object} An augmented data object containing both the original data
|
1119
|
+
* and the additional params.
|
1120
|
+
*/
|
1121
|
+
soy.$$augmentData = function(origData, additionalParams) {
|
1122
|
+
|
1123
|
+
// Create a new object whose '__proto__' field is set to origData.
|
1124
|
+
/** @constructor */
|
1125
|
+
function TempCtor() {}
|
1126
|
+
TempCtor.prototype = origData;
|
1127
|
+
var newData = new TempCtor();
|
1128
|
+
|
1129
|
+
// Add the additional params to the new object.
|
1130
|
+
for (var key in additionalParams) {
|
1131
|
+
newData[key] = additionalParams[key];
|
1132
|
+
}
|
1133
|
+
|
1134
|
+
return newData;
|
1135
|
+
};
|
1136
|
+
|
1137
|
+
|
1138
|
+
/**
|
1139
|
+
* Gets the keys in a map as an array. There are no guarantees on the order.
|
1140
|
+
* @param {Object} map The map to get the keys of.
|
1141
|
+
* @return {Array.<string>} The array of keys in the given map.
|
1142
|
+
*/
|
1143
|
+
soy.$$getMapKeys = function(map) {
|
1144
|
+
var mapKeys = [];
|
1145
|
+
for (var key in map) {
|
1146
|
+
mapKeys.push(key);
|
1147
|
+
}
|
1148
|
+
return mapKeys;
|
1149
|
+
};
|
1150
|
+
|
1151
|
+
|
1152
|
+
/**
|
1153
|
+
* Gets a consistent unique id for the given delegate template name. Two calls
|
1154
|
+
* to this function will return the same id if and only if the input names are
|
1155
|
+
* the same.
|
1156
|
+
*
|
1157
|
+
* <p> Important: This function must always be called with a string constant.
|
1158
|
+
*
|
1159
|
+
* <p> If Closure Compiler is not being used, then this is just this identity
|
1160
|
+
* function. If Closure Compiler is being used, then each call to this function
|
1161
|
+
* will be replaced with a short string constant, which will be consistent per
|
1162
|
+
* input name.
|
1163
|
+
*
|
1164
|
+
* @param {string} delTemplateName The delegate template name for which to get a
|
1165
|
+
* consistent unique id.
|
1166
|
+
* @return {string} A unique id that is consistent per input name.
|
1167
|
+
*
|
1168
|
+
* @consistentIdGenerator
|
1169
|
+
*/
|
1170
|
+
soy.$$getDelegateId = function(delTemplateName) {
|
1171
|
+
return delTemplateName;
|
1172
|
+
};
|
1173
|
+
|
1174
|
+
|
1175
|
+
/**
|
1176
|
+
* Map from registered delegate template id/name to the priority of the
|
1177
|
+
* implementation.
|
1178
|
+
* @type {Object}
|
1179
|
+
* @private
|
1180
|
+
*/
|
1181
|
+
soy.$$DELEGATE_REGISTRY_PRIORITIES_ = {};
|
1182
|
+
|
1183
|
+
/**
|
1184
|
+
* Map from registered delegate template id/name to the implementation function.
|
1185
|
+
* @type {Object}
|
1186
|
+
* @private
|
1187
|
+
*/
|
1188
|
+
soy.$$DELEGATE_REGISTRY_FUNCTIONS_ = {};
|
1189
|
+
|
1190
|
+
|
1191
|
+
/**
|
1192
|
+
* Registers a delegate implementation. If the same delegate template id/name
|
1193
|
+
* has been registered previously, then priority values are compared and only
|
1194
|
+
* the higher priority implementation is stored (if priorities are equal, an
|
1195
|
+
* error is thrown).
|
1196
|
+
*
|
1197
|
+
* @param {string} delTemplateId The delegate template id/name to register.
|
1198
|
+
* @param {number} delPriority The implementation's priority value.
|
1199
|
+
* @param {Function} delFn The implementation function.
|
1200
|
+
*/
|
1201
|
+
soy.$$registerDelegateFn = function(delTemplateId, delPriority, delFn) {
|
1202
|
+
var mapKey = 'key_' + delTemplateId;
|
1203
|
+
var currPriority = soy.$$DELEGATE_REGISTRY_PRIORITIES_[mapKey];
|
1204
|
+
if (currPriority === undefined || delPriority > currPriority) {
|
1205
|
+
// Registering new or higher-priority function: replace registry entry.
|
1206
|
+
soy.$$DELEGATE_REGISTRY_PRIORITIES_[mapKey] = delPriority;
|
1207
|
+
soy.$$DELEGATE_REGISTRY_FUNCTIONS_[mapKey] = delFn;
|
1208
|
+
} else if (delPriority == currPriority) {
|
1209
|
+
// Registering same-priority function: error.
|
1210
|
+
throw Error(
|
1211
|
+
'Encountered two active delegates with same priority (id/name "' +
|
1212
|
+
delTemplateId + '").');
|
1213
|
+
} else {
|
1214
|
+
// Registering lower-priority function: do nothing.
|
1215
|
+
}
|
1216
|
+
};
|
1217
|
+
|
1218
|
+
|
1219
|
+
/**
|
1220
|
+
* Retrieves the (highest-priority) implementation that has been registered for
|
1221
|
+
* a given delegate template id/name. If no implementation has been registered
|
1222
|
+
* for the id/name, then returns an implementation that is equivalent to an
|
1223
|
+
* empty template (i.e. rendered output would be empty string).
|
1224
|
+
*
|
1225
|
+
* @param {string} delTemplateId The delegate template id/name to get.
|
1226
|
+
* @return {Function} The retrieved implementation function.
|
1227
|
+
*/
|
1228
|
+
soy.$$getDelegateFn = function(delTemplateId) {
|
1229
|
+
var delFn = soy.$$DELEGATE_REGISTRY_FUNCTIONS_['key_' + delTemplateId];
|
1230
|
+
return delFn ? delFn : soy.$$EMPTY_TEMPLATE_FN_;
|
1231
|
+
};
|
1232
|
+
|
1233
|
+
|
1234
|
+
/**
|
1235
|
+
* Private helper soy.$$getDelegateFn(). This is the empty template function
|
1236
|
+
* that is returned whenever there's no delegate implementation found.
|
1237
|
+
*
|
1238
|
+
* @param {Object.<string, *>=} opt_data
|
1239
|
+
* @param {soy.StringBuilder=} opt_sb
|
1240
|
+
* @param {Object.<string, *>=} opt_ijData
|
1241
|
+
* @return {string}
|
1242
|
+
* @private
|
1243
|
+
*/
|
1244
|
+
soy.$$EMPTY_TEMPLATE_FN_ = function(opt_data, opt_sb, opt_ijData) {
|
1245
|
+
return '';
|
1246
|
+
};
|
1247
|
+
|
1248
|
+
|
1249
|
+
// -----------------------------------------------------------------------------
|
1250
|
+
// Escape/filter/normalize.
|
1251
|
+
|
1252
|
+
|
1253
|
+
/**
|
1254
|
+
* Escapes HTML special characters in a string. Escapes double quote '"' in
|
1255
|
+
* addition to '&', '<', and '>' so that a string can be included in an HTML
|
1256
|
+
* tag attribute value within double quotes.
|
1257
|
+
* Will emit known safe HTML as-is.
|
1258
|
+
*
|
1259
|
+
* @param {*} value The string-like value to be escaped. May not be a string,
|
1260
|
+
* but the value will be coerced to a string.
|
1261
|
+
* @return {string} An escaped version of value.
|
1262
|
+
*/
|
1263
|
+
soy.$$escapeHtml = function(value) {
|
1264
|
+
if (typeof value === 'object' && value &&
|
1265
|
+
value.contentKind === soydata.SanitizedContentKind.HTML) {
|
1266
|
+
return value.content;
|
1267
|
+
}
|
1268
|
+
return soy.esc.$$escapeHtmlHelper(value);
|
1269
|
+
};
|
1270
|
+
|
1271
|
+
|
1272
|
+
/**
|
1273
|
+
* Escapes HTML special characters in a string so that it can be embedded in
|
1274
|
+
* RCDATA.
|
1275
|
+
* <p>
|
1276
|
+
* Escapes HTML special characters so that the value will not prematurely end
|
1277
|
+
* the body of a tag like {@code <textarea>} or {@code <title>}. RCDATA tags
|
1278
|
+
* cannot contain other HTML entities, so it is not strictly necessary to escape
|
1279
|
+
* HTML special characters except when part of that text looks like an HTML
|
1280
|
+
* entity or like a close tag : {@code </textarea>}.
|
1281
|
+
* <p>
|
1282
|
+
* Will normalize known safe HTML to make sure that sanitized HTML (which could
|
1283
|
+
* contain an innocuous {@code </textarea>} don't prematurely end an RCDATA
|
1284
|
+
* element.
|
1285
|
+
*
|
1286
|
+
* @param {*} value The string-like value to be escaped. May not be a string,
|
1287
|
+
* but the value will be coerced to a string.
|
1288
|
+
* @return {string} An escaped version of value.
|
1289
|
+
*/
|
1290
|
+
soy.$$escapeHtmlRcdata = function(value) {
|
1291
|
+
if (typeof value === 'object' && value &&
|
1292
|
+
value.contentKind === soydata.SanitizedContentKind.HTML) {
|
1293
|
+
return soy.esc.$$normalizeHtmlHelper(value.content);
|
1294
|
+
}
|
1295
|
+
return soy.esc.$$escapeHtmlHelper(value);
|
1296
|
+
};
|
1297
|
+
|
1298
|
+
|
1299
|
+
/**
|
1300
|
+
* Removes HTML tags from a string of known safe HTML so it can be used as an
|
1301
|
+
* attribute value.
|
1302
|
+
*
|
1303
|
+
* @param {*} value The HTML to be escaped. May not be a string, but the
|
1304
|
+
* value will be coerced to a string.
|
1305
|
+
* @return {string} A representation of value without tags, HTML comments, or
|
1306
|
+
* other content.
|
1307
|
+
*/
|
1308
|
+
soy.$$stripHtmlTags = function(value) {
|
1309
|
+
return String(value).replace(soy.esc.$$HTML_TAG_REGEX_, '');
|
1310
|
+
};
|
1311
|
+
|
1312
|
+
|
1313
|
+
/**
|
1314
|
+
* Escapes HTML special characters in an HTML attribute value.
|
1315
|
+
*
|
1316
|
+
* @param {*} value The HTML to be escaped. May not be a string, but the
|
1317
|
+
* value will be coerced to a string.
|
1318
|
+
* @return {string} An escaped version of value.
|
1319
|
+
*/
|
1320
|
+
soy.$$escapeHtmlAttribute = function(value) {
|
1321
|
+
if (typeof value === 'object' && value &&
|
1322
|
+
value.contentKind === soydata.SanitizedContentKind.HTML) {
|
1323
|
+
return soy.esc.$$normalizeHtmlHelper(soy.$$stripHtmlTags(value.content));
|
1324
|
+
}
|
1325
|
+
return soy.esc.$$escapeHtmlHelper(value);
|
1326
|
+
};
|
1327
|
+
|
1328
|
+
|
1329
|
+
/**
|
1330
|
+
* Escapes HTML special characters in a string including space and other
|
1331
|
+
* characters that can end an unquoted HTML attribute value.
|
1332
|
+
*
|
1333
|
+
* @param {*} value The HTML to be escaped. May not be a string, but the
|
1334
|
+
* value will be coerced to a string.
|
1335
|
+
* @return {string} An escaped version of value.
|
1336
|
+
*/
|
1337
|
+
soy.$$escapeHtmlAttributeNospace = function(value) {
|
1338
|
+
if (typeof value === 'object' && value &&
|
1339
|
+
value.contentKind === soydata.SanitizedContentKind.HTML) {
|
1340
|
+
return soy.esc.$$normalizeHtmlNospaceHelper(
|
1341
|
+
soy.$$stripHtmlTags(value.content));
|
1342
|
+
}
|
1343
|
+
return soy.esc.$$escapeHtmlNospaceHelper(value);
|
1344
|
+
};
|
1345
|
+
|
1346
|
+
|
1347
|
+
/**
|
1348
|
+
* Filters out strings that cannot be a substring of a valid HTML attribute.
|
1349
|
+
*
|
1350
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1351
|
+
* will be coerced to a string.
|
1352
|
+
* @return {string} A valid HTML attribute name part or name/value pair.
|
1353
|
+
* {@code "zSoyz"} if the input is invalid.
|
1354
|
+
*/
|
1355
|
+
soy.$$filterHtmlAttribute = function(value) {
|
1356
|
+
if (typeof value === 'object' && value &&
|
1357
|
+
value.contentKind === soydata.SanitizedContentKind.HTML_ATTRIBUTE) {
|
1358
|
+
return value.content.replace(/=([^"']*)$/, '="$1"');
|
1359
|
+
}
|
1360
|
+
return soy.esc.$$filterHtmlAttributeHelper(value);
|
1361
|
+
};
|
1362
|
+
|
1363
|
+
|
1364
|
+
/**
|
1365
|
+
* Filters out strings that cannot be a substring of a valid HTML element name.
|
1366
|
+
*
|
1367
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1368
|
+
* will be coerced to a string.
|
1369
|
+
* @return {string} A valid HTML element name part.
|
1370
|
+
* {@code "zSoyz"} if the input is invalid.
|
1371
|
+
*/
|
1372
|
+
soy.$$filterHtmlElementName = function(value) {
|
1373
|
+
return soy.esc.$$filterHtmlElementNameHelper(value);
|
1374
|
+
};
|
1375
|
+
|
1376
|
+
|
1377
|
+
/**
|
1378
|
+
* Escapes characters in the value to make it valid content for a JS string
|
1379
|
+
* literal.
|
1380
|
+
*
|
1381
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1382
|
+
* will be coerced to a string.
|
1383
|
+
* @return {string} An escaped version of value.
|
1384
|
+
* @deprecated
|
1385
|
+
*/
|
1386
|
+
soy.$$escapeJs = function(value) {
|
1387
|
+
return soy.$$escapeJsString(value);
|
1388
|
+
};
|
1389
|
+
|
1390
|
+
|
1391
|
+
/**
|
1392
|
+
* Escapes characters in the value to make it valid content for a JS string
|
1393
|
+
* literal.
|
1394
|
+
*
|
1395
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1396
|
+
* will be coerced to a string.
|
1397
|
+
* @return {string} An escaped version of value.
|
1398
|
+
*/
|
1399
|
+
soy.$$escapeJsString = function(value) {
|
1400
|
+
if (typeof value === 'object' &&
|
1401
|
+
value.contentKind === soydata.SanitizedContentKind.JS_STR_CHARS) {
|
1402
|
+
return value.content;
|
1403
|
+
}
|
1404
|
+
return soy.esc.$$escapeJsStringHelper(value);
|
1405
|
+
};
|
1406
|
+
|
1407
|
+
|
1408
|
+
/**
|
1409
|
+
* Encodes a value as a JavaScript literal.
|
1410
|
+
*
|
1411
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1412
|
+
* will be coerced to a string.
|
1413
|
+
* @return {string} A JavaScript code representation of the input.
|
1414
|
+
*/
|
1415
|
+
soy.$$escapeJsValue = function(value) {
|
1416
|
+
// We surround values with spaces so that they can't be interpolated into
|
1417
|
+
// identifiers by accident.
|
1418
|
+
// We could use parentheses but those might be interpreted as a function call.
|
1419
|
+
if (value == null) { // Intentionally matches undefined.
|
1420
|
+
// Java returns null from maps where there is no corresponding key while
|
1421
|
+
// JS returns undefined.
|
1422
|
+
// We always output null for compatibility with Java which does not have a
|
1423
|
+
// distinct undefined value.
|
1424
|
+
return ' null ';
|
1425
|
+
}
|
1426
|
+
switch (typeof value) {
|
1427
|
+
case 'boolean': case 'number':
|
1428
|
+
return ' ' + value + ' ';
|
1429
|
+
default:
|
1430
|
+
return "'" + soy.esc.$$escapeJsStringHelper(String(value)) + "'";
|
1431
|
+
}
|
1432
|
+
};
|
1433
|
+
|
1434
|
+
|
1435
|
+
/**
|
1436
|
+
* Escapes characters in the string to make it valid content for a JS regular
|
1437
|
+
* expression literal.
|
1438
|
+
*
|
1439
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1440
|
+
* will be coerced to a string.
|
1441
|
+
* @return {string} An escaped version of value.
|
1442
|
+
*/
|
1443
|
+
soy.$$escapeJsRegex = function(value) {
|
1444
|
+
return soy.esc.$$escapeJsRegexHelper(value);
|
1445
|
+
};
|
1446
|
+
|
1447
|
+
|
1448
|
+
/**
|
1449
|
+
* Matches all URI mark characters that conflict with HTML attribute delimiters
|
1450
|
+
* or that cannot appear in a CSS uri.
|
1451
|
+
* From <a href="http://www.w3.org/TR/CSS2/grammar.html">G.2: CSS grammar</a>
|
1452
|
+
* <pre>
|
1453
|
+
* url ([!#$%&*-~]|{nonascii}|{escape})*
|
1454
|
+
* </pre>
|
1455
|
+
*
|
1456
|
+
* @type {RegExp}
|
1457
|
+
* @private
|
1458
|
+
*/
|
1459
|
+
soy.$$problematicUriMarks_ = /['()]/g;
|
1460
|
+
|
1461
|
+
/**
|
1462
|
+
* @param {string} ch A single character in {@link soy.$$problematicUriMarks_}.
|
1463
|
+
* @return {string}
|
1464
|
+
* @private
|
1465
|
+
*/
|
1466
|
+
soy.$$pctEncode_ = function(ch) {
|
1467
|
+
return '%' + ch.charCodeAt(0).toString(16);
|
1468
|
+
};
|
1469
|
+
|
1470
|
+
/**
|
1471
|
+
* Escapes a string so that it can be safely included in a URI.
|
1472
|
+
*
|
1473
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1474
|
+
* will be coerced to a string.
|
1475
|
+
* @return {string} An escaped version of value.
|
1476
|
+
*/
|
1477
|
+
soy.$$escapeUri = function(value) {
|
1478
|
+
if (typeof value === 'object' &&
|
1479
|
+
value.contentKind === soydata.SanitizedContentKind.URI) {
|
1480
|
+
return soy.$$normalizeUri(value);
|
1481
|
+
}
|
1482
|
+
// Apostophes and parentheses are not matched by encodeURIComponent.
|
1483
|
+
// They are technically special in URIs, but only appear in the obsolete mark
|
1484
|
+
// production in Appendix D.2 of RFC 3986, so can be encoded without changing
|
1485
|
+
// semantics.
|
1486
|
+
var encoded = soy.esc.$$escapeUriHelper(value);
|
1487
|
+
soy.$$problematicUriMarks_.lastIndex = 0;
|
1488
|
+
if (soy.$$problematicUriMarks_.test(encoded)) {
|
1489
|
+
return encoded.replace(soy.$$problematicUriMarks_, soy.$$pctEncode_);
|
1490
|
+
}
|
1491
|
+
return encoded;
|
1492
|
+
};
|
1493
|
+
|
1494
|
+
|
1495
|
+
/**
|
1496
|
+
* Removes rough edges from a URI by escaping any raw HTML/JS string delimiters.
|
1497
|
+
*
|
1498
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1499
|
+
* will be coerced to a string.
|
1500
|
+
* @return {string} An escaped version of value.
|
1501
|
+
*/
|
1502
|
+
soy.$$normalizeUri = function(value) {
|
1503
|
+
return soy.esc.$$normalizeUriHelper(value);
|
1504
|
+
};
|
1505
|
+
|
1506
|
+
|
1507
|
+
/**
|
1508
|
+
* Vets a URI's protocol and removes rough edges from a URI by escaping
|
1509
|
+
* any raw HTML/JS string delimiters.
|
1510
|
+
*
|
1511
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1512
|
+
* will be coerced to a string.
|
1513
|
+
* @return {string} An escaped version of value.
|
1514
|
+
*/
|
1515
|
+
soy.$$filterNormalizeUri = function(value) {
|
1516
|
+
return soy.esc.$$filterNormalizeUriHelper(value);
|
1517
|
+
};
|
1518
|
+
|
1519
|
+
|
1520
|
+
/**
|
1521
|
+
* Escapes a string so it can safely be included inside a quoted CSS string.
|
1522
|
+
*
|
1523
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1524
|
+
* will be coerced to a string.
|
1525
|
+
* @return {string} An escaped version of value.
|
1526
|
+
*/
|
1527
|
+
soy.$$escapeCssString = function(value) {
|
1528
|
+
return soy.esc.$$escapeCssStringHelper(value);
|
1529
|
+
};
|
1530
|
+
|
1531
|
+
|
1532
|
+
/**
|
1533
|
+
* Encodes a value as a CSS identifier part, keyword, or quantity.
|
1534
|
+
*
|
1535
|
+
* @param {*} value The value to escape. May not be a string, but the value
|
1536
|
+
* will be coerced to a string.
|
1537
|
+
* @return {string} A safe CSS identifier part, keyword, or quanitity.
|
1538
|
+
*/
|
1539
|
+
soy.$$filterCssValue = function(value) {
|
1540
|
+
// Uses == to intentionally match null and undefined for Java compatibility.
|
1541
|
+
if (value == null) {
|
1542
|
+
return '';
|
1543
|
+
}
|
1544
|
+
return soy.esc.$$filterCssValueHelper(value);
|
1545
|
+
};
|
1546
|
+
|
1547
|
+
|
1548
|
+
// -----------------------------------------------------------------------------
|
1549
|
+
// Basic directives/functions.
|
1550
|
+
|
1551
|
+
|
1552
|
+
/**
|
1553
|
+
* Converts \r\n, \r, and \n to <br>s
|
1554
|
+
* @param {*} str The string in which to convert newlines.
|
1555
|
+
* @return {string} A copy of {@code str} with converted newlines.
|
1556
|
+
*/
|
1557
|
+
soy.$$changeNewlineToBr = function(str) {
|
1558
|
+
return goog.string.newLineToBr(String(str), false);
|
1559
|
+
};
|
1560
|
+
|
1561
|
+
|
1562
|
+
/**
|
1563
|
+
* Inserts word breaks ('wbr' tags) into a HTML string at a given interval. The
|
1564
|
+
* counter is reset if a space is encountered. Word breaks aren't inserted into
|
1565
|
+
* HTML tags or entities. Entites count towards the character count; HTML tags
|
1566
|
+
* do not.
|
1567
|
+
*
|
1568
|
+
* @param {*} str The HTML string to insert word breaks into. Can be other
|
1569
|
+
* types, but the value will be coerced to a string.
|
1570
|
+
* @param {number} maxCharsBetweenWordBreaks Maximum number of non-space
|
1571
|
+
* characters to allow before adding a word break.
|
1572
|
+
* @return {string} The string including word breaks.
|
1573
|
+
*/
|
1574
|
+
soy.$$insertWordBreaks = function(str, maxCharsBetweenWordBreaks) {
|
1575
|
+
return goog.format.insertWordBreaks(String(str), maxCharsBetweenWordBreaks);
|
1576
|
+
};
|
1577
|
+
|
1578
|
+
|
1579
|
+
/**
|
1580
|
+
* Truncates a string to a given max length (if it's currently longer),
|
1581
|
+
* optionally adding ellipsis at the end.
|
1582
|
+
*
|
1583
|
+
* @param {*} str The string to truncate. Can be other types, but the value will
|
1584
|
+
* be coerced to a string.
|
1585
|
+
* @param {number} maxLen The maximum length of the string after truncation
|
1586
|
+
* (including ellipsis, if applicable).
|
1587
|
+
* @param {boolean} doAddEllipsis Whether to add ellipsis if the string needs
|
1588
|
+
* truncation.
|
1589
|
+
* @return {string} The string after truncation.
|
1590
|
+
*/
|
1591
|
+
soy.$$truncate = function(str, maxLen, doAddEllipsis) {
|
1592
|
+
|
1593
|
+
str = String(str);
|
1594
|
+
if (str.length <= maxLen) {
|
1595
|
+
return str; // no need to truncate
|
1596
|
+
}
|
1597
|
+
|
1598
|
+
// If doAddEllipsis, either reduce maxLen to compensate, or else if maxLen is
|
1599
|
+
// too small, just turn off doAddEllipsis.
|
1600
|
+
if (doAddEllipsis) {
|
1601
|
+
if (maxLen > 3) {
|
1602
|
+
maxLen -= 3;
|
1603
|
+
} else {
|
1604
|
+
doAddEllipsis = false;
|
1605
|
+
}
|
1606
|
+
}
|
1607
|
+
|
1608
|
+
// Make sure truncating at maxLen doesn't cut up a unicode surrogate pair.
|
1609
|
+
if (soy.$$isHighSurrogate_(str.charAt(maxLen - 1)) &&
|
1610
|
+
soy.$$isLowSurrogate_(str.charAt(maxLen))) {
|
1611
|
+
maxLen -= 1;
|
1612
|
+
}
|
1613
|
+
|
1614
|
+
// Truncate.
|
1615
|
+
str = str.substring(0, maxLen);
|
1616
|
+
|
1617
|
+
// Add ellipsis.
|
1618
|
+
if (doAddEllipsis) {
|
1619
|
+
str += '...';
|
1620
|
+
}
|
1621
|
+
|
1622
|
+
return str;
|
1623
|
+
};
|
1624
|
+
|
1625
|
+
/**
|
1626
|
+
* Private helper for $$truncate() to check whether a char is a high surrogate.
|
1627
|
+
* @param {string} ch The char to check.
|
1628
|
+
* @return {boolean} Whether the given char is a unicode high surrogate.
|
1629
|
+
* @private
|
1630
|
+
*/
|
1631
|
+
soy.$$isHighSurrogate_ = function(ch) {
|
1632
|
+
return 0xD800 <= ch && ch <= 0xDBFF;
|
1633
|
+
};
|
1634
|
+
|
1635
|
+
/**
|
1636
|
+
* Private helper for $$truncate() to check whether a char is a low surrogate.
|
1637
|
+
* @param {string} ch The char to check.
|
1638
|
+
* @return {boolean} Whether the given char is a unicode low surrogate.
|
1639
|
+
* @private
|
1640
|
+
*/
|
1641
|
+
soy.$$isLowSurrogate_ = function(ch) {
|
1642
|
+
return 0xDC00 <= ch && ch <= 0xDFFF;
|
1643
|
+
};
|
1644
|
+
|
1645
|
+
|
1646
|
+
// -----------------------------------------------------------------------------
|
1647
|
+
// Bidi directives/functions.
|
1648
|
+
|
1649
|
+
|
1650
|
+
/**
|
1651
|
+
* Cache of bidi formatter by context directionality, so we don't keep on
|
1652
|
+
* creating new objects.
|
1653
|
+
* @type {!Object.<!goog.i18n.BidiFormatter>}
|
1654
|
+
* @private
|
1655
|
+
*/
|
1656
|
+
soy.$$bidiFormatterCache_ = {};
|
1657
|
+
|
1658
|
+
|
1659
|
+
/**
|
1660
|
+
* Returns cached bidi formatter for bidiGlobalDir, or creates a new one.
|
1661
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
1662
|
+
* if rtl, 0 if unknown.
|
1663
|
+
* @return {goog.i18n.BidiFormatter} A formatter for bidiGlobalDir.
|
1664
|
+
* @private
|
1665
|
+
*/
|
1666
|
+
soy.$$getBidiFormatterInstance_ = function(bidiGlobalDir) {
|
1667
|
+
return soy.$$bidiFormatterCache_[bidiGlobalDir] ||
|
1668
|
+
(soy.$$bidiFormatterCache_[bidiGlobalDir] =
|
1669
|
+
new goog.i18n.BidiFormatter(bidiGlobalDir));
|
1670
|
+
};
|
1671
|
+
|
1672
|
+
|
1673
|
+
/**
|
1674
|
+
* Estimate the overall directionality of text. If opt_isHtml, makes sure to
|
1675
|
+
* ignore the LTR nature of the mark-up and escapes in text, making the logic
|
1676
|
+
* suitable for HTML and HTML-escaped text.
|
1677
|
+
* @param {string} text The text whose directionality is to be estimated.
|
1678
|
+
* @param {boolean=} opt_isHtml Whether text is HTML/HTML-escaped.
|
1679
|
+
* Default: false.
|
1680
|
+
* @return {number} 1 if text is LTR, -1 if it is RTL, and 0 if it is neutral.
|
1681
|
+
*/
|
1682
|
+
soy.$$bidiTextDir = function(text, opt_isHtml) {
|
1683
|
+
if (!text) {
|
1684
|
+
return 0;
|
1685
|
+
}
|
1686
|
+
return goog.i18n.bidi.detectRtlDirectionality(text, opt_isHtml) ? -1 : 1;
|
1687
|
+
};
|
1688
|
+
|
1689
|
+
|
1690
|
+
/**
|
1691
|
+
* Returns "dir=ltr" or "dir=rtl", depending on text's estimated
|
1692
|
+
* directionality, if it is not the same as bidiGlobalDir.
|
1693
|
+
* Otherwise, returns the empty string.
|
1694
|
+
* If opt_isHtml, makes sure to ignore the LTR nature of the mark-up and escapes
|
1695
|
+
* in text, making the logic suitable for HTML and HTML-escaped text.
|
1696
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
1697
|
+
* if rtl, 0 if unknown.
|
1698
|
+
* @param {string} text The text whose directionality is to be estimated.
|
1699
|
+
* @param {boolean=} opt_isHtml Whether text is HTML/HTML-escaped.
|
1700
|
+
* Default: false.
|
1701
|
+
* @return {soydata.SanitizedHtmlAttribute} "dir=rtl" for RTL text in non-RTL
|
1702
|
+
* context; "dir=ltr" for LTR text in non-LTR context;
|
1703
|
+
* else, the empty string.
|
1704
|
+
*/
|
1705
|
+
soy.$$bidiDirAttr = function(bidiGlobalDir, text, opt_isHtml) {
|
1706
|
+
return new soydata.SanitizedHtmlAttribute(
|
1707
|
+
soy.$$getBidiFormatterInstance_(bidiGlobalDir).dirAttr(text, opt_isHtml));
|
1708
|
+
};
|
1709
|
+
|
1710
|
+
|
1711
|
+
/**
|
1712
|
+
* Returns a Unicode BiDi mark matching bidiGlobalDir (LRM or RLM) if the
|
1713
|
+
* directionality or the exit directionality of text are opposite to
|
1714
|
+
* bidiGlobalDir. Otherwise returns the empty string.
|
1715
|
+
* If opt_isHtml, makes sure to ignore the LTR nature of the mark-up and escapes
|
1716
|
+
* in text, making the logic suitable for HTML and HTML-escaped text.
|
1717
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
1718
|
+
* if rtl, 0 if unknown.
|
1719
|
+
* @param {string} text The text whose directionality is to be estimated.
|
1720
|
+
* @param {boolean=} opt_isHtml Whether text is HTML/HTML-escaped.
|
1721
|
+
* Default: false.
|
1722
|
+
* @return {string} A Unicode bidi mark matching bidiGlobalDir, or the empty
|
1723
|
+
* string when text's overall and exit directionalities both match
|
1724
|
+
* bidiGlobalDir, or bidiGlobalDir is 0 (unknown).
|
1725
|
+
*/
|
1726
|
+
soy.$$bidiMarkAfter = function(bidiGlobalDir, text, opt_isHtml) {
|
1727
|
+
var formatter = soy.$$getBidiFormatterInstance_(bidiGlobalDir);
|
1728
|
+
return formatter.markAfter(text, opt_isHtml);
|
1729
|
+
};
|
1730
|
+
|
1731
|
+
|
1732
|
+
/**
|
1733
|
+
* Returns str wrapped in a <span dir=ltr|rtl> according to its directionality -
|
1734
|
+
* but only if that is neither neutral nor the same as the global context.
|
1735
|
+
* Otherwise, returns str unchanged.
|
1736
|
+
* Always treats str as HTML/HTML-escaped, i.e. ignores mark-up and escapes when
|
1737
|
+
* estimating str's directionality.
|
1738
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
1739
|
+
* if rtl, 0 if unknown.
|
1740
|
+
* @param {*} str The string to be wrapped. Can be other types, but the value
|
1741
|
+
* will be coerced to a string.
|
1742
|
+
* @return {string} The wrapped string.
|
1743
|
+
*/
|
1744
|
+
soy.$$bidiSpanWrap = function(bidiGlobalDir, str) {
|
1745
|
+
var formatter = soy.$$getBidiFormatterInstance_(bidiGlobalDir);
|
1746
|
+
return formatter.spanWrap(str + '', true);
|
1747
|
+
};
|
1748
|
+
|
1749
|
+
|
1750
|
+
/**
|
1751
|
+
* Returns str wrapped in Unicode BiDi formatting characters according to its
|
1752
|
+
* directionality, i.e. either LRE or RLE at the beginning and PDF at the end -
|
1753
|
+
* but only if str's directionality is neither neutral nor the same as the
|
1754
|
+
* global context. Otherwise, returns str unchanged.
|
1755
|
+
* Always treats str as HTML/HTML-escaped, i.e. ignores mark-up and escapes when
|
1756
|
+
* estimating str's directionality.
|
1757
|
+
* @param {number} bidiGlobalDir The global directionality context: 1 if ltr, -1
|
1758
|
+
* if rtl, 0 if unknown.
|
1759
|
+
* @param {*} str The string to be wrapped. Can be other types, but the value
|
1760
|
+
* will be coerced to a string.
|
1761
|
+
* @return {string} The wrapped string.
|
1762
|
+
*/
|
1763
|
+
soy.$$bidiUnicodeWrap = function(bidiGlobalDir, str) {
|
1764
|
+
var formatter = soy.$$getBidiFormatterInstance_(bidiGlobalDir);
|
1765
|
+
return formatter.unicodeWrap(str + '', true);
|
1766
|
+
};
|
1767
|
+
|
1768
|
+
|
1769
|
+
// -----------------------------------------------------------------------------
|
1770
|
+
// Generated code.
|
1771
|
+
|
1772
|
+
|
1773
|
+
|
1774
|
+
|
1775
|
+
// START GENERATED CODE FOR ESCAPERS.
|
1776
|
+
|
1777
|
+
/**
|
1778
|
+
* @type {function (*) : string}
|
1779
|
+
*/
|
1780
|
+
soy.esc.$$escapeUriHelper = function(v) {
|
1781
|
+
return encodeURIComponent(String(v));
|
1782
|
+
};
|
1783
|
+
|
1784
|
+
/**
|
1785
|
+
* Maps charcters to the escaped versions for the named escape directives.
|
1786
|
+
* @type {Object.<string, string>}
|
1787
|
+
* @private
|
1788
|
+
*/
|
1789
|
+
soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_ = {
|
1790
|
+
'\x00': '\x26#0;',
|
1791
|
+
'\x22': '\x26quot;',
|
1792
|
+
'\x26': '\x26amp;',
|
1793
|
+
'\x27': '\x26#39;',
|
1794
|
+
'\x3c': '\x26lt;',
|
1795
|
+
'\x3e': '\x26gt;',
|
1796
|
+
'\x09': '\x26#9;',
|
1797
|
+
'\x0a': '\x26#10;',
|
1798
|
+
'\x0b': '\x26#11;',
|
1799
|
+
'\x0c': '\x26#12;',
|
1800
|
+
'\x0d': '\x26#13;',
|
1801
|
+
' ': '\x26#32;',
|
1802
|
+
'-': '\x26#45;',
|
1803
|
+
'\/': '\x26#47;',
|
1804
|
+
'\x3d': '\x26#61;',
|
1805
|
+
'`': '\x26#96;',
|
1806
|
+
'\x85': '\x26#133;',
|
1807
|
+
'\xa0': '\x26#160;',
|
1808
|
+
'\u2028': '\x26#8232;',
|
1809
|
+
'\u2029': '\x26#8233;'
|
1810
|
+
};
|
1811
|
+
|
1812
|
+
/**
|
1813
|
+
* A function that can be used with String.replace..
|
1814
|
+
* @param {string} ch A single character matched by a compatible matcher.
|
1815
|
+
* @return {string} A token in the output language.
|
1816
|
+
* @private
|
1817
|
+
*/
|
1818
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_ = function(ch) {
|
1819
|
+
return soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_[ch];
|
1820
|
+
};
|
1821
|
+
|
1822
|
+
/**
|
1823
|
+
* Maps charcters to the escaped versions for the named escape directives.
|
1824
|
+
* @type {Object.<string, string>}
|
1825
|
+
* @private
|
1826
|
+
*/
|
1827
|
+
soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_JS_STRING__AND__ESCAPE_JS_REGEX_ = {
|
1828
|
+
'\x00': '\\x00',
|
1829
|
+
'\x08': '\\x08',
|
1830
|
+
'\x09': '\\t',
|
1831
|
+
'\x0a': '\\n',
|
1832
|
+
'\x0b': '\\x0b',
|
1833
|
+
'\x0c': '\\f',
|
1834
|
+
'\x0d': '\\r',
|
1835
|
+
'\x22': '\\x22',
|
1836
|
+
'\x26': '\\x26',
|
1837
|
+
'\x27': '\\x27',
|
1838
|
+
'\/': '\\\/',
|
1839
|
+
'\x3c': '\\x3c',
|
1840
|
+
'\x3d': '\\x3d',
|
1841
|
+
'\x3e': '\\x3e',
|
1842
|
+
'\\': '\\\\',
|
1843
|
+
'\x85': '\\x85',
|
1844
|
+
'\u2028': '\\u2028',
|
1845
|
+
'\u2029': '\\u2029',
|
1846
|
+
'$': '\\x24',
|
1847
|
+
'(': '\\x28',
|
1848
|
+
')': '\\x29',
|
1849
|
+
'*': '\\x2a',
|
1850
|
+
'+': '\\x2b',
|
1851
|
+
',': '\\x2c',
|
1852
|
+
'-': '\\x2d',
|
1853
|
+
'.': '\\x2e',
|
1854
|
+
':': '\\x3a',
|
1855
|
+
'?': '\\x3f',
|
1856
|
+
'[': '\\x5b',
|
1857
|
+
']': '\\x5d',
|
1858
|
+
'^': '\\x5e',
|
1859
|
+
'{': '\\x7b',
|
1860
|
+
'|': '\\x7c',
|
1861
|
+
'}': '\\x7d'
|
1862
|
+
};
|
1863
|
+
|
1864
|
+
/**
|
1865
|
+
* A function that can be used with String.replace..
|
1866
|
+
* @param {string} ch A single character matched by a compatible matcher.
|
1867
|
+
* @return {string} A token in the output language.
|
1868
|
+
* @private
|
1869
|
+
*/
|
1870
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_JS_STRING__AND__ESCAPE_JS_REGEX_ = function(ch) {
|
1871
|
+
return soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_JS_STRING__AND__ESCAPE_JS_REGEX_[ch];
|
1872
|
+
};
|
1873
|
+
|
1874
|
+
/**
|
1875
|
+
* Maps charcters to the escaped versions for the named escape directives.
|
1876
|
+
* @type {Object.<string, string>}
|
1877
|
+
* @private
|
1878
|
+
*/
|
1879
|
+
soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_CSS_STRING_ = {
|
1880
|
+
'\x00': '\\0 ',
|
1881
|
+
'\x08': '\\8 ',
|
1882
|
+
'\x09': '\\9 ',
|
1883
|
+
'\x0a': '\\a ',
|
1884
|
+
'\x0b': '\\b ',
|
1885
|
+
'\x0c': '\\c ',
|
1886
|
+
'\x0d': '\\d ',
|
1887
|
+
'\x22': '\\22 ',
|
1888
|
+
'\x26': '\\26 ',
|
1889
|
+
'\x27': '\\27 ',
|
1890
|
+
'(': '\\28 ',
|
1891
|
+
')': '\\29 ',
|
1892
|
+
'*': '\\2a ',
|
1893
|
+
'\/': '\\2f ',
|
1894
|
+
':': '\\3a ',
|
1895
|
+
';': '\\3b ',
|
1896
|
+
'\x3c': '\\3c ',
|
1897
|
+
'\x3d': '\\3d ',
|
1898
|
+
'\x3e': '\\3e ',
|
1899
|
+
'@': '\\40 ',
|
1900
|
+
'\\': '\\5c ',
|
1901
|
+
'{': '\\7b ',
|
1902
|
+
'}': '\\7d ',
|
1903
|
+
'\x85': '\\85 ',
|
1904
|
+
'\xa0': '\\a0 ',
|
1905
|
+
'\u2028': '\\2028 ',
|
1906
|
+
'\u2029': '\\2029 '
|
1907
|
+
};
|
1908
|
+
|
1909
|
+
/**
|
1910
|
+
* A function that can be used with String.replace..
|
1911
|
+
* @param {string} ch A single character matched by a compatible matcher.
|
1912
|
+
* @return {string} A token in the output language.
|
1913
|
+
* @private
|
1914
|
+
*/
|
1915
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_CSS_STRING_ = function(ch) {
|
1916
|
+
return soy.esc.$$ESCAPE_MAP_FOR_ESCAPE_CSS_STRING_[ch];
|
1917
|
+
};
|
1918
|
+
|
1919
|
+
/**
|
1920
|
+
* Maps charcters to the escaped versions for the named escape directives.
|
1921
|
+
* @type {Object.<string, string>}
|
1922
|
+
* @private
|
1923
|
+
*/
|
1924
|
+
soy.esc.$$ESCAPE_MAP_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_ = {
|
1925
|
+
'\x00': '%00',
|
1926
|
+
'\x01': '%01',
|
1927
|
+
'\x02': '%02',
|
1928
|
+
'\x03': '%03',
|
1929
|
+
'\x04': '%04',
|
1930
|
+
'\x05': '%05',
|
1931
|
+
'\x06': '%06',
|
1932
|
+
'\x07': '%07',
|
1933
|
+
'\x08': '%08',
|
1934
|
+
'\x09': '%09',
|
1935
|
+
'\x0a': '%0A',
|
1936
|
+
'\x0b': '%0B',
|
1937
|
+
'\x0c': '%0C',
|
1938
|
+
'\x0d': '%0D',
|
1939
|
+
'\x0e': '%0E',
|
1940
|
+
'\x0f': '%0F',
|
1941
|
+
'\x10': '%10',
|
1942
|
+
'\x11': '%11',
|
1943
|
+
'\x12': '%12',
|
1944
|
+
'\x13': '%13',
|
1945
|
+
'\x14': '%14',
|
1946
|
+
'\x15': '%15',
|
1947
|
+
'\x16': '%16',
|
1948
|
+
'\x17': '%17',
|
1949
|
+
'\x18': '%18',
|
1950
|
+
'\x19': '%19',
|
1951
|
+
'\x1a': '%1A',
|
1952
|
+
'\x1b': '%1B',
|
1953
|
+
'\x1c': '%1C',
|
1954
|
+
'\x1d': '%1D',
|
1955
|
+
'\x1e': '%1E',
|
1956
|
+
'\x1f': '%1F',
|
1957
|
+
' ': '%20',
|
1958
|
+
'\x22': '%22',
|
1959
|
+
'\x27': '%27',
|
1960
|
+
'(': '%28',
|
1961
|
+
')': '%29',
|
1962
|
+
'\x3c': '%3C',
|
1963
|
+
'\x3e': '%3E',
|
1964
|
+
'\\': '%5C',
|
1965
|
+
'{': '%7B',
|
1966
|
+
'}': '%7D',
|
1967
|
+
'\x7f': '%7F',
|
1968
|
+
'\x85': '%C2%85',
|
1969
|
+
'\xa0': '%C2%A0',
|
1970
|
+
'\u2028': '%E2%80%A8',
|
1971
|
+
'\u2029': '%E2%80%A9',
|
1972
|
+
'\uff01': '%EF%BC%81',
|
1973
|
+
'\uff03': '%EF%BC%83',
|
1974
|
+
'\uff04': '%EF%BC%84',
|
1975
|
+
'\uff06': '%EF%BC%86',
|
1976
|
+
'\uff07': '%EF%BC%87',
|
1977
|
+
'\uff08': '%EF%BC%88',
|
1978
|
+
'\uff09': '%EF%BC%89',
|
1979
|
+
'\uff0a': '%EF%BC%8A',
|
1980
|
+
'\uff0b': '%EF%BC%8B',
|
1981
|
+
'\uff0c': '%EF%BC%8C',
|
1982
|
+
'\uff0f': '%EF%BC%8F',
|
1983
|
+
'\uff1a': '%EF%BC%9A',
|
1984
|
+
'\uff1b': '%EF%BC%9B',
|
1985
|
+
'\uff1d': '%EF%BC%9D',
|
1986
|
+
'\uff1f': '%EF%BC%9F',
|
1987
|
+
'\uff20': '%EF%BC%A0',
|
1988
|
+
'\uff3b': '%EF%BC%BB',
|
1989
|
+
'\uff3d': '%EF%BC%BD'
|
1990
|
+
};
|
1991
|
+
|
1992
|
+
/**
|
1993
|
+
* A function that can be used with String.replace..
|
1994
|
+
* @param {string} ch A single character matched by a compatible matcher.
|
1995
|
+
* @return {string} A token in the output language.
|
1996
|
+
* @private
|
1997
|
+
*/
|
1998
|
+
soy.esc.$$REPLACER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_ = function(ch) {
|
1999
|
+
return soy.esc.$$ESCAPE_MAP_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_[ch];
|
2000
|
+
};
|
2001
|
+
|
2002
|
+
/**
|
2003
|
+
* Matches characters that need to be escaped for the named directives.
|
2004
|
+
* @type RegExp
|
2005
|
+
* @private
|
2006
|
+
*/
|
2007
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_HTML_ = /[\x00\x22\x26\x27\x3c\x3e]/g;
|
2008
|
+
|
2009
|
+
/**
|
2010
|
+
* Matches characters that need to be escaped for the named directives.
|
2011
|
+
* @type RegExp
|
2012
|
+
* @private
|
2013
|
+
*/
|
2014
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_HTML_ = /[\x00\x22\x27\x3c\x3e]/g;
|
2015
|
+
|
2016
|
+
/**
|
2017
|
+
* Matches characters that need to be escaped for the named directives.
|
2018
|
+
* @type RegExp
|
2019
|
+
* @private
|
2020
|
+
*/
|
2021
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_HTML_NOSPACE_ = /[\x00\x09-\x0d \x22\x26\x27\x2d\/\x3c-\x3e`\x85\xa0\u2028\u2029]/g;
|
2022
|
+
|
2023
|
+
/**
|
2024
|
+
* Matches characters that need to be escaped for the named directives.
|
2025
|
+
* @type RegExp
|
2026
|
+
* @private
|
2027
|
+
*/
|
2028
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_HTML_NOSPACE_ = /[\x00\x09-\x0d \x22\x27\x2d\/\x3c-\x3e`\x85\xa0\u2028\u2029]/g;
|
2029
|
+
|
2030
|
+
/**
|
2031
|
+
* Matches characters that need to be escaped for the named directives.
|
2032
|
+
* @type RegExp
|
2033
|
+
* @private
|
2034
|
+
*/
|
2035
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_JS_STRING_ = /[\x00\x08-\x0d\x22\x26\x27\/\x3c-\x3e\\\x85\u2028\u2029]/g;
|
2036
|
+
|
2037
|
+
/**
|
2038
|
+
* Matches characters that need to be escaped for the named directives.
|
2039
|
+
* @type RegExp
|
2040
|
+
* @private
|
2041
|
+
*/
|
2042
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_JS_REGEX_ = /[\x00\x08-\x0d\x22\x24\x26-\/\x3a\x3c-\x3f\x5b-\x5e\x7b-\x7d\x85\u2028\u2029]/g;
|
2043
|
+
|
2044
|
+
/**
|
2045
|
+
* Matches characters that need to be escaped for the named directives.
|
2046
|
+
* @type RegExp
|
2047
|
+
* @private
|
2048
|
+
*/
|
2049
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_CSS_STRING_ = /[\x00\x08-\x0d\x22\x26-\x2a\/\x3a-\x3e@\\\x7b\x7d\x85\xa0\u2028\u2029]/g;
|
2050
|
+
|
2051
|
+
/**
|
2052
|
+
* Matches characters that need to be escaped for the named directives.
|
2053
|
+
* @type RegExp
|
2054
|
+
* @private
|
2055
|
+
*/
|
2056
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_ = /[\x00- \x22\x27-\x29\x3c\x3e\\\x7b\x7d\x7f\x85\xa0\u2028\u2029\uff01\uff03\uff04\uff06-\uff0c\uff0f\uff1a\uff1b\uff1d\uff1f\uff20\uff3b\uff3d]/g;
|
2057
|
+
|
2058
|
+
/**
|
2059
|
+
* A pattern that vets values produced by the named directives.
|
2060
|
+
* @type RegExp
|
2061
|
+
* @private
|
2062
|
+
*/
|
2063
|
+
soy.esc.$$FILTER_FOR_FILTER_CSS_VALUE_ = /^(?!-*(?:expression|(?:moz-)?binding))(?:[.#]?-?(?:[_a-z0-9-]+)(?:-[_a-z0-9-]+)*-?|-?(?:[0-9]+(?:\.[0-9]*)?|\.[0-9]+)(?:[a-z]{1,2}|%)?|!important|)$/i;
|
2064
|
+
|
2065
|
+
/**
|
2066
|
+
* A pattern that vets values produced by the named directives.
|
2067
|
+
* @type RegExp
|
2068
|
+
* @private
|
2069
|
+
*/
|
2070
|
+
soy.esc.$$FILTER_FOR_FILTER_NORMALIZE_URI_ = /^(?:(?:https?|mailto):|[^&:\/?#]*(?:[\/?#]|$))/i;
|
2071
|
+
|
2072
|
+
/**
|
2073
|
+
* A pattern that vets values produced by the named directives.
|
2074
|
+
* @type RegExp
|
2075
|
+
* @private
|
2076
|
+
*/
|
2077
|
+
soy.esc.$$FILTER_FOR_FILTER_HTML_ATTRIBUTE_ = /^(?!style|on|action|archive|background|cite|classid|codebase|data|dsync|href|longdesc|src|usemap)(?:[a-z0-9_$:-]*)$/i;
|
2078
|
+
|
2079
|
+
/**
|
2080
|
+
* A pattern that vets values produced by the named directives.
|
2081
|
+
* @type RegExp
|
2082
|
+
* @private
|
2083
|
+
*/
|
2084
|
+
soy.esc.$$FILTER_FOR_FILTER_HTML_ELEMENT_NAME_ = /^(?!script|style|title|textarea|xmp|no)[a-z0-9_$:-]*$/i;
|
2085
|
+
|
2086
|
+
/**
|
2087
|
+
* A helper for the Soy directive |escapeHtml
|
2088
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2089
|
+
* @return {string} The escaped text.
|
2090
|
+
*/
|
2091
|
+
soy.esc.$$escapeHtmlHelper = function(value) {
|
2092
|
+
var str = String(value);
|
2093
|
+
return str.replace(
|
2094
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_HTML_,
|
2095
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_);
|
2096
|
+
};
|
2097
|
+
|
2098
|
+
/**
|
2099
|
+
* A helper for the Soy directive |normalizeHtml
|
2100
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2101
|
+
* @return {string} The escaped text.
|
2102
|
+
*/
|
2103
|
+
soy.esc.$$normalizeHtmlHelper = function(value) {
|
2104
|
+
var str = String(value);
|
2105
|
+
return str.replace(
|
2106
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_HTML_,
|
2107
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_);
|
2108
|
+
};
|
2109
|
+
|
2110
|
+
/**
|
2111
|
+
* A helper for the Soy directive |escapeHtmlNospace
|
2112
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2113
|
+
* @return {string} The escaped text.
|
2114
|
+
*/
|
2115
|
+
soy.esc.$$escapeHtmlNospaceHelper = function(value) {
|
2116
|
+
var str = String(value);
|
2117
|
+
return str.replace(
|
2118
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_HTML_NOSPACE_,
|
2119
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_);
|
2120
|
+
};
|
2121
|
+
|
2122
|
+
/**
|
2123
|
+
* A helper for the Soy directive |normalizeHtmlNospace
|
2124
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2125
|
+
* @return {string} The escaped text.
|
2126
|
+
*/
|
2127
|
+
soy.esc.$$normalizeHtmlNospaceHelper = function(value) {
|
2128
|
+
var str = String(value);
|
2129
|
+
return str.replace(
|
2130
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_HTML_NOSPACE_,
|
2131
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_HTML__AND__NORMALIZE_HTML__AND__ESCAPE_HTML_NOSPACE__AND__NORMALIZE_HTML_NOSPACE_);
|
2132
|
+
};
|
2133
|
+
|
2134
|
+
/**
|
2135
|
+
* A helper for the Soy directive |escapeJsString
|
2136
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2137
|
+
* @return {string} The escaped text.
|
2138
|
+
*/
|
2139
|
+
soy.esc.$$escapeJsStringHelper = function(value) {
|
2140
|
+
var str = String(value);
|
2141
|
+
return str.replace(
|
2142
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_JS_STRING_,
|
2143
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_JS_STRING__AND__ESCAPE_JS_REGEX_);
|
2144
|
+
};
|
2145
|
+
|
2146
|
+
/**
|
2147
|
+
* A helper for the Soy directive |escapeJsRegex
|
2148
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2149
|
+
* @return {string} The escaped text.
|
2150
|
+
*/
|
2151
|
+
soy.esc.$$escapeJsRegexHelper = function(value) {
|
2152
|
+
var str = String(value);
|
2153
|
+
return str.replace(
|
2154
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_JS_REGEX_,
|
2155
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_JS_STRING__AND__ESCAPE_JS_REGEX_);
|
2156
|
+
};
|
2157
|
+
|
2158
|
+
/**
|
2159
|
+
* A helper for the Soy directive |escapeCssString
|
2160
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2161
|
+
* @return {string} The escaped text.
|
2162
|
+
*/
|
2163
|
+
soy.esc.$$escapeCssStringHelper = function(value) {
|
2164
|
+
var str = String(value);
|
2165
|
+
return str.replace(
|
2166
|
+
soy.esc.$$MATCHER_FOR_ESCAPE_CSS_STRING_,
|
2167
|
+
soy.esc.$$REPLACER_FOR_ESCAPE_CSS_STRING_);
|
2168
|
+
};
|
2169
|
+
|
2170
|
+
/**
|
2171
|
+
* A helper for the Soy directive |filterCssValue
|
2172
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2173
|
+
* @return {string} The escaped text.
|
2174
|
+
*/
|
2175
|
+
soy.esc.$$filterCssValueHelper = function(value) {
|
2176
|
+
var str = String(value);
|
2177
|
+
if (!soy.esc.$$FILTER_FOR_FILTER_CSS_VALUE_.test(str)) {
|
2178
|
+
return 'zSoyz';
|
2179
|
+
}
|
2180
|
+
return str;
|
2181
|
+
};
|
2182
|
+
|
2183
|
+
/**
|
2184
|
+
* A helper for the Soy directive |normalizeUri
|
2185
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2186
|
+
* @return {string} The escaped text.
|
2187
|
+
*/
|
2188
|
+
soy.esc.$$normalizeUriHelper = function(value) {
|
2189
|
+
var str = String(value);
|
2190
|
+
return str.replace(
|
2191
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_,
|
2192
|
+
soy.esc.$$REPLACER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_);
|
2193
|
+
};
|
2194
|
+
|
2195
|
+
/**
|
2196
|
+
* A helper for the Soy directive |filterNormalizeUri
|
2197
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2198
|
+
* @return {string} The escaped text.
|
2199
|
+
*/
|
2200
|
+
soy.esc.$$filterNormalizeUriHelper = function(value) {
|
2201
|
+
var str = String(value);
|
2202
|
+
if (!soy.esc.$$FILTER_FOR_FILTER_NORMALIZE_URI_.test(str)) {
|
2203
|
+
return 'zSoyz';
|
2204
|
+
}
|
2205
|
+
return str.replace(
|
2206
|
+
soy.esc.$$MATCHER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_,
|
2207
|
+
soy.esc.$$REPLACER_FOR_NORMALIZE_URI__AND__FILTER_NORMALIZE_URI_);
|
2208
|
+
};
|
2209
|
+
|
2210
|
+
/**
|
2211
|
+
* A helper for the Soy directive |filterHtmlAttribute
|
2212
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2213
|
+
* @return {string} The escaped text.
|
2214
|
+
*/
|
2215
|
+
soy.esc.$$filterHtmlAttributeHelper = function(value) {
|
2216
|
+
var str = String(value);
|
2217
|
+
if (!soy.esc.$$FILTER_FOR_FILTER_HTML_ATTRIBUTE_.test(str)) {
|
2218
|
+
return 'zSoyz';
|
2219
|
+
}
|
2220
|
+
return str;
|
2221
|
+
};
|
2222
|
+
|
2223
|
+
/**
|
2224
|
+
* A helper for the Soy directive |filterHtmlElementName
|
2225
|
+
* @param {*} value Can be of any type but will be coerced to a string.
|
2226
|
+
* @return {string} The escaped text.
|
2227
|
+
*/
|
2228
|
+
soy.esc.$$filterHtmlElementNameHelper = function(value) {
|
2229
|
+
var str = String(value);
|
2230
|
+
if (!soy.esc.$$FILTER_FOR_FILTER_HTML_ELEMENT_NAME_.test(str)) {
|
2231
|
+
return 'zSoyz';
|
2232
|
+
}
|
2233
|
+
return str;
|
2234
|
+
};
|
2235
|
+
|
2236
|
+
/**
|
2237
|
+
* Matches all tags, HTML comments, and DOCTYPEs in tag soup HTML.
|
2238
|
+
*
|
2239
|
+
* @type {RegExp}
|
2240
|
+
* @private
|
2241
|
+
*/
|
2242
|
+
soy.esc.$$HTML_TAG_REGEX_ = /<(?:!|\/?[a-zA-Z])(?:[^>'"]|"[^"]*"|'[^']*')*>/g;
|
2243
|
+
|
2244
|
+
// END GENERATED CODE
|
2245
|
+
;
|
2246
|
+
;
|
2247
|
+
FI"asset_paths;
|
2248
|
+
F["^/Volumes/Development/dev-web/yellow-brick-road/vendor/assets/closure-soyutils/soyutils.jsI"dependency_paths;
|
2249
|
+
F[{I" path;
|
2250
|
+
F"^/Volumes/Development/dev-web/yellow-brick-road/vendor/assets/closure-soyutils/soyutils.jsI"
|
2251
|
+
mtime;
|
2252
|
+
FIu: Time
|
2253
|
+
T:offsetiI"hexdigest;
|
2254
|
+
F"%5108961b05643b150b5065b1aa5919f8I"
|
2255
|
+
F"%46dde6621c301f4928e3b34efee9e3b5
|