The Tale Extended

Расширение базового функционала в браузерной игре Сказка

当前为 2015-03-07 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name The Tale Extended
  3. // @description Расширение базового функционала в браузерной игре Сказка
  4. // @author standy
  5. // @version 0.3.6.3
  6. // @include http://the-tale.org/game/
  7. // @run-at document-end
  8. // @license MIT License
  9. // @namespace the-tale-ext
  10. // ==/UserScript==
  11. // @require http://code.jquery.com/jquery-1.11.1.min.js
  12. // @grant GM_addStyle
  13.  
  14. (function (window, undefined) {
  15.  
  16. var css = ".sort{cursor:default;position:relative}.sort-down:after,.sort-up:after{content:'';display:inline-block;width:0;height:0;margin:8px -9px 0 1px;vertical-align:1px;border-right:4px solid transparent;border-left:4px solid transparent}.sort-up:after{border-top:4px solid #000}.sort-down:after{border-bottom:4px solid #000}.caption a,.caption a:visited,.caption a:hover{opacity:.5}.caption.new-messages a,.caption.new-messages a:link,.caption.new-messages a:visited,.caption.new-messages a:hover{opacity:1}.link-ajax{color:#08c;cursor:default;background-image:-webkit-linear-gradient(left,#08c,#08c 50%,transparent 50%);background-image:-moz-linear-gradient(left,#08c,#08c 50%,transparent 50%);background-image:linear-gradient(left,#08c,#08c 50%,transparent 50%);background-size:2px 1px;background-position:0 95%;background-repeat:repeat-x}.link-ajax:hover{color:#005580;background-image:-webkit-linear-gradient(left,#005580,#005580 50%,transparent 50%);background-image:-moz-linear-gradient(left,#005580,#005580 50%,transparent 50%);background-image:linear-gradient(left,#005580,#005580 50%,transparent 50%)}span[title],input[disabled]{cursor:default}#abilities-block .angel-ability{width:auto!important;display:inline-block}#abilities-block .angel-ability.pgf-ability-help{margin:0 50px}#abilities-block .dropdown{width:auto!important}.ext-controls{font-size:12px}.ext-control{margin-right:10px}.ext-control.link-ajax{background:0 0}.table-stats th,.table-stats td{padding:1px 10px 1px 0;cursor:default}.table-stats .stats-name{width:16px;padding-right:0}.table-stats .stats-sum,.table-stats .stats-count,.table-stats .stats-average{text-align:right}.table-stats .stats-bonus{text-align:left;padding-left:10px;padding-right:5px}.table-stats th.stats-bonus{text-align:center}.table-stats .stats-against{border-top:10px solid #f5fafa}.table-stats .stats-row-dmgSum td{font-weight:700}.stats-side{margin-bottom:10px}.stats-side.stats-loot{margin-top:20px;padding-top:10px;border-top:1px solid #ccc}.archive-log-list{margin-left:30px}.stats-archive{display:block}.stats-archive-act{margin-right:6px}.stats-archive-me{color:#005000}.stats-archive-enemy{color:darkred}.stats-log{margin-top:10px;max-height:500px;overflow-y:auto}.log-record-short{display:inline-block}.log-record-short .submessage{display:inline-block;cursor:default;padding:0 2px}.log-record-short .submessage:first-child{padding-left:4px}.log-record-short .submessage:last-child{padding-right:4px}.log-record-short:hover{background:#ddd}.stats-me .glyphicon{color:#005000}.stats-loot .glyphicon{color:darkgoldenrod}.stats-enemy .glyphicon{color:darkred}.stats .glyphicon{font-size:12px;opacity:.8}.stats .glyphicon.small{font-size:9px}.log-block .glyphicon{font-size:12px;opacity:.8}.log-block .glyphicon-heart{vertical-align:-1px}.log-block .glyphicon-heart-empty{vertical-align:-2px}.log-block .glyphicon.small{font-size:9px}.act.me{color:#000}.act.me .glyphicon{color:#005000}.act.enemy{color:#B90000}.act.enemy .glyphicon{color:darkred}.act-rest.me,.act-heal.me{color:#005000}.act-heal.enemy{color:darkred}.act-godheal.act .glyphicon,.act-godheal.act{color:darkmagenta;text-decoration:underline}.act-vamp.me .glyphicon,.act-vamp.me .vamp{color:#005000}.act-vamp.enemy .glyphicon,.act-vamp.enemy .vamp{color:darkred}.act-slow.me{color:#00008B}.act-coins.me .glyphicon,.act-coins.me,.act-coins.enemy{color:darkgoldenrod}.act-coins.enemy{text-decoration:line-through}.act-coins .glyphicon{font-size:9px}.act-pvpeff.me{color:dodgerblue}.act-pvpeff.enemy{color:dodgerblue;text-decoration:line-through}.act-pvpice.me,.act-pvpflame.enemy,.act-pvpfail.enemy{color:darkgreen}.act-pvpice.enemy,.act-pvpflame.me,.act-pvpfail.me{color:darkred}.act-pvpice .glyphicon,.act-pvpflame .glyphicon,.act-pvpfail .glyphicon{font-size:15px}.sub-icon .glyphicon{font-size:10px;vertical-align:sub;margin-left:-1px}.act.me .sub-icon .glyphicon{color:darkred}.act.enemy .sub-icon .glyphicon{color:#005000}.me .value{color:darkred}.enemy .value{color:red}.msg-dmg.me .value{color:darkred}.msg-dmg.enemy .value{color:red}.msg-godhit{color:darkmagenta}.msg-pickup .item{color:darkgoldenrod}.msg-empty,.msg-drop{color:darkgoldenrod;text-decoration:line-through}.submessage .energy{color:dodgerblue}.submessage .coins{color:darkgoldenrod}.submessage .level,.submessage .exp,.archive-content .level{color:darkviolet;text-decoration:underline}.group{position:relative;margin-bottom:2px}.group .group-toggle{cursor:default}.group .on-open{display:none}.group.open .on-open{display:block}.group.open .on-close{display:none}.group.open{margin:0}.group .glyphicon-time{font-size:9px}.group-time.average{color:#8a6d3b;text-decoration:underline}.group-time.bad{color:#a94442;text-decoration:underline}.group-title{cursor:default;margin-right:0;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.group-title:hover{background:#ddd}.action-icon .glyphicon{font-size:10px;opacity:.8}.action-icon.fight{color:darkred}.action-icon.fight-god{color:darkviolet}.action-icon.equip,.action-icon.trade{color:darkgoldenrod}.action-icon.quest,.action-icon.noeffect,.action-icon.idle,.action-icon.energy{color:dodgerblue}.action-icon.rest{color:#005000}.action-icon.dead{color:darkred}.action-icon.broken{color:red}.action-icon{display:inline-block;vertical-align:text-bottom;width:15px}.group-title.god .group-time{color:darkviolet;text-decoration:underline}.group-time{margin-right:3px}.group-controls{position:absolute;right:0;top:1px}.group-controls .glyphicon{position:absolute;right:0;top:0}.group-controls .glyphicon:hover{color:#000}.group-stats{margin-bottom:2px}.group-stats .stat{margin:0 0 0 4px}.group-stats .stat-me+.stat-enemy{margin-left:8px}.group-stats .stat-me{color:#005000}.group-stats .stat-enemy{color:darkred}.group-stats .stat-against{display:inline-block;width:35px;white-space:nowrap}.text-muted{color:#999}.text-primary{color:#428bca}.text-success{color:#3c763d}.text-info{color:#31708f}.text-warning{color:#8a6d3b}.text-danger{color:#a94442}.quest-icon-mini{float:none;display:inline-block;margin:0;vertical-align:bottom;width:16px;height:16px;background-size:32px 192px;background-image:url('http://static.the-tale.org/static/168/game/images/quests.png')}.quest-icon-mini.caravan{background-position:0 0}.quest-icon-mini.delivery{background-position:0 -16px}.quest-icon-mini.help{background-position:0 -32px}.quest-icon-mini.help_friend{background-position:0 -48px}.quest-icon-mini.hometown{background-position:0 -64px}.quest-icon-mini.hunt{background-position:0 -80px}.quest-icon-mini.interfere_enemy{background-position:0 -96px}.quest-icon-mini.collect_debt{background-position:0 -112px}.quest-icon-mini.spying{background-position:0 -128px}.quest-icon-mini.search_smith{background-position:0 -144px}.quest-icon-mini.no-quest{background-position:0 -160px}.quest-icon-mini.next-spending{background-position:0 -176px}.quest-icon-mini.pilgrimage{background-position:-16px -80px}#pgf-towns-container .reload{font-size:12px}body .table-noborder tbody,.table-noborder td,.table-noborder th{border:none}.table tbody tr.unhover:hover td,.table tbody tr.unhover:hover th{background:inherit}.table-hover-dark tbody tr:hover td,.table-hover-dark tbody tr:hover th{background:#ddd}.table-towns th.size,.table-towns td.size{padding-left:0;padding-right:0}.table-towns th.size{text-align:center}.table-towns td.size{text-align:right}.table-towns td{white-space:nowrap}#log-block-sets{height:auto}.sets-header{margin:10px 0 5px}.sets{margin:5px 0 0 20px}.sets-inline{display:inline-block}.note-block{color:#999}input.input-tiny{height:11px;width:60px;font-size:12px;margin:-2px 0 -3px;vertical-align:0}input.input-tiny-num{width:30px}.input-append .add-on{height:11px;line-height:11px;margin-top:-2px;margin-bottom:-3px;vertical-align:-1px}.sets label{display:inline;font-weight:400;margin-bottom:0}.sets .input-wrap-inline{display:inline-block}.sets .input-wrap{transition:opacity .2s}.sets .input-wrap.disabled{opacity:.7}@font-face{font-family:'Glyphicons Halflings';src:url('data:application/octet-stream;base64,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/zr+0wXdAAEAAAAAAAAAAAAAAAAAAACfAbgAKASwAAAEsAAABLAAAASwAGQEsAAABLAAAAKMAAAFGAAAAowAAAUYAAABsgAAAUYAAADZAAAA2QAAAKMAAAEEAAAASAAAAQQAAAFGAAAEsABkBLAAyAH0AAAEsP/yBLAAAASw//MEsAAABLAADgSwABcEsABkBLD/uASw/7gEsAAABLAAAASwAAAEsAAABLAAAASwAB0EsABqBLAAFwSwABcEsAAXBLAAZASwABoEsABkBLAAAQSwAGQEsAAEBLD/nASwAAAEsAABBLAABASwAAAEsAAEBLAAFwSwABcEsABkBLAAAASwAGQEsAAABLAAAASwAAAEsAAABLAAAASwAAAEsAAABLAAAQSwAGQEsADIBLAAAASwAAAEsAA1BLAAZASwAMgEsP+1BLAAIQSwAAAEsAAABLAAAASwAAAEsAAABLD/mwSwAAEEsAAABLAAAASwAJQEsAABBLAAdQSwAAAEsAAABLAAAASwAAAEsADIBLAAAASwAIgEsADIBLAAyASwAMgEsAAABLAAAASwASwEsABkBLAAuQSwARAEsAADBLAAAwSwAAMEsAADBLAAAwSwAAMEsAAABLAABASwAAQEsAAEBLAAAASwAAAEsADMBLAAaASwAAAEsAAABLAAIgSwABcEsAAABLAAAASwAG8EsP/DBLD/wwSw/58EsABkBLAAAASwAAAEsAAABLAAZASw/+IEsABGBLD/OgSwABIEsAAABLAAAQSwAS4EsAAABLAAAASw/5sEsABKBLAAFQSwAAAEsAAABLAACASw/5sEsABhBLAAAgSwAAUEsAAABLAABQSwAAUEsAAFBLAAAATEAAAEsABkAAAAAAAA/9gAUAA5AMgAAAEnAGQAAgACAAIAAgACAAIAAgAAAAAAAAAAAAAA2QAAAAAABAAAAAAAAAAAAAAAFwAAAAAAAAAAAAAAAAAAAGQAZAAAABAAAABkAGT/nP+c/5z/nP+c/5z/nP+cAAQAAP/y//IAZAB5AAAAKgAqACoAKgBmAKQApACkAKQApACkAKQApACkAKQApACkAKQApACkATABSAFQAYQBqgHOAf4CNgKYAswC7gMsA0wD9ARyBWQGDgYiBkQG4gdMB7AH6AiUCTAJYAmUCgoKRAqICu4LVguQC+YMPgywDRwNdg2qDjoOYA6KDtwP2BBMEIgQ1hEOESYRlBIUEmAS2hMWE44UDBRiFLoVJhWSFmIW2hdMF4YX6Bg4GIIYvhksGZgZ/BpKGnwathrWGuQbFBswG04bhhu2G+Ab9BwOHFocpBzuHTIdwh4QHooe7B9GH6AfwB/iIAYgKiBGIG4gkCDyIWohxCJCIsgjfiOuJBQkkiTmJRQliCWcJbIl7iZaJogmviboJw4naifOKDgoZCi4KTYp1CpuKuwraCuyK/osRCyYLb4t5i5mLpIu+i86MBgwtjEmMX4x0jIIMrgzYDOQNBo0ojUENWY1vjYWNlw2tDcKN2A3rDf8OFA4tDkYOVg5ljnUOhg6XjqGOtY7LDt0O948QDxoPNw9RD2oPhA+SD68Pu4/MD+aQAJAYEC0QSJBiEHwQoJC+EOIQ/ZEPERgRJ4AAQAAANsAmwARAAAAAAACAAEAAgAWAAABAAEBAAAAAAAAAA8AugABAAAAAAATABIAAAADAAEECQAAAGoAEgADAAEECQABACgAfAADAAEECQACAA4ApAADAAEECQADAEwAsgADAAEECQAEADgA/gADAAEECQAFAHgBNgADAAEECQAGADYBrgADAAEECQAIABYB5AADAAEECQAJABYB+gADAAEECQALACQCEAADAAEECQAMACQCNAADAAEECQATACQCWAADAAEECQDIABYCfAADAAEECQDJADACknd3dy5nbHlwaGljb25zLmNvbQBDAG8AcAB5AHIAaQBnAGgAdAAgAKkAIAAyADAAMQAzACAAYgB5ACAASgBhAG4AIABLAG8AdgBhAHIAaQBrAC4AIABBAGwAbAAgAHIAaQBnAGgAdABzACAAcgBlAHMAZQByAHYAZQBkAC4ARwBMAFkAUABIAEkAQwBPAE4AUwAgAEgAYQBsAGYAbABpAG4AZwBzAFIAZQBnAHUAbABhAHIAMQAuADAAMAAxADsAVQBLAFcATgA7AEcATABZAFAASABJAEMATwBOAFMASABhAGwAZgBsAGkAbgBnAHMALQBSAGUAZwB1AGwAYQByAEcATABZAFAASABJAEMATwBOAFMAIABIAGEAbABmAGwAaQBuAGcAcwAgAFIAZQBnAHUAbABhAHIAVgBlAHIAcwBpAG8AbgAgADEALgAwADAAMQA7AFAAUwAgADAAMAAxAC4AMAAwADEAOwBoAG8AdABjAG8AbgB2ACAAMQAuADAALgA3ADAAOwBtAGEAawBlAG8AdABmAC4AbABpAGIAMgAuADUALgA1ADgAMwAyADkARwBMAFkAUABIAEkAQwBPAE4AUwBIAGEAbABmAGwAaQBuAGcAcwAtAFIAZQBnAHUAbABhAHIASgBhAG4AIABLAG8AdgBhAHIAaQBrAEoAYQBuACAASwBvAHYAYQByAGkAawB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQBXAGUAYgBmAG8AbgB0ACAAMQAuADAATQBvAG4AIABKAGEAbgAgADIANwAgADAAOAA6ADAAMQA6ADMANAAgADIAMAAxADQAAgAAAAAAAP+1ADIAAAAAAAAAAAAAAAAAAAAAAAAAAADbAAABAgEDAAMADQAOAQQBBQEGAQcBCAEJAQoBCwEMAQ0BDgEPARABEQESAO8BEwEUARUBFgEXARgBGQEaARsBHAEdAR4BHwEgASEBIgEjASQBJQEmAScBKAEpASoBKwEsAS0BLgEvATABMQEyATMBNAE1ATYBNwE4ATkBOgE7ATwBPQE+AT8BQAFBAUIBQwFEAUUBRgFHAUgBSQFKAUsBTAFNAU4BTwFQAVEBUgFTAVQBVQFWAVcBWAFZAVoBWwFcAV0BXgFfAWABYQFiAWMBZAFlAWYBZwFoAWkBagFrAWwBbQFuAW8BcAFxAXIBcwF0AXUBdgF3AXgBeQF6AXsBfAF9AX4BfwGAAYEBggGDAYQBhQGGAYcBiAGJAYoBiwGMAY0BjgGPAZABkQGSAZMBlAGVAZYBlwGYAZkBmgGbAZwBnQGeAZ8BoAGhAaIBowGkAaUBpgGnAagBqQGqAasBrAGtAa4BrwGwAbEBsgGzAbQBtQG2AbcBuAG5AboBuwG8Ab0BvgG/AcABwQHCAcMBxAHFAcYBxwHIAckBygHLAcwBzQHOAc8B0AHRAdIB0wHUAdUB1gHXBmdseXBoMQd1bmkwMDBEB3VuaTAwQTAHdW5pMjAwMAd1bmkyMDAxB3VuaTIwMDIHdW5pMjAwMwd1bmkyMDA0B3VuaTIwMDUHdW5pMjAwNgd1bmkyMDA3B3VuaTIwMDgHdW5pMjAwOQd1bmkyMDBBB3VuaTIwMkYHdW5pMjA1RgRFdXJvB3VuaTI1RkMHdW5pMjYwMQd1bmkyNzA5B3VuaTI3MEYHdW5pRTAwMQd1bmlFMDAyB3VuaUUwMDMHdW5pRTAwNQd1bmlFMDA2B3VuaUUwMDcHdW5pRTAwOAd1bmlFMDA5B3VuaUUwMTAHdW5pRTAxMQd1bmlFMDEyB3VuaUUwMTMHdW5pRTAxNAd1bmlFMDE1B3VuaUUwMTYHdW5pRTAxNwd1bmlFMDE4B3VuaUUwMTkHdW5pRTAyMAd1bmlFMDIxB3VuaUUwMjIHdW5pRTAyMwd1bmlFMDI0B3VuaUUwMjUHdW5pRTAyNgd1bmlFMDI3B3VuaUUwMjgHdW5pRTAyOQd1bmlFMDMwB3VuaUUwMzEHdW5pRTAzMgd1bmlFMDMzB3VuaUUwMzQHdW5pRTAzNQd1bmlFMDM2B3VuaUUwMzcHdW5pRTAzOAd1bmlFMDM5B3VuaUUwNDAHdW5pRTA0MQd1bmlFMDQyB3VuaUUwNDMHdW5pRTA0NAd1bmlFMDQ1B3VuaUUwNDYHdW5pRTA0Nwd1bmlFMDQ4B3VuaUUwNDkHdW5pRTA1MAd1bmlFMDUxB3VuaUUwNTIHdW5pRTA1Mwd1bmlFMDU0B3VuaUUwNTUHdW5pRTA1Ngd1bmlFMDU3B3VuaUUwNTgHdW5pRTA1OQd1bmlFMDYwB3VuaUUwNjIHdW5pRTA2Mwd1bmlFMDY0B3VuaUUwNjUHdW5pRTA2Ngd1bmlFMDY3B3VuaUUwNjgHdW5pRTA2OQd1bmlFMDcwB3VuaUUwNzEHdW5pRTA3Mgd1bmlFMDczB3VuaUUwNzQHdW5pRTA3NQd1bmlFMDc2B3VuaUUwNzcHdW5pRTA3OAd1bmlFMDc5B3VuaUUwODAHdW5pRTA4MQd1bmlFMDgyB3VuaUUwODMHdW5pRTA4NAd1bmlFMDg1B3VuaUUwODYHdW5pRTA4Nwd1bmlFMDg4B3VuaUUwODkHdW5pRTA5MAd1bmlFMDkxB3VuaUUwOTIHdW5pRTA5Mwd1bmlFMDk0B3VuaUUwOTUHdW5pRTA5Ngd1bmlFMDk3B3VuaUUxMDEHdW5pRTEwMgd1bmlFMTAzB3VuaUUxMDQHdW5pRTEwNQd1bmlFMTA2B3VuaUUxMDcHdW5pRTEwOAd1bmlFMTA5B3VuaUUxMTAHdW5pRTExMQd1bmlFMTEyB3VuaUUxMTMHdW5pRTExNAd1bmlFMTE1B3VuaUUxMTYHdW5pRTExNwd1bmlFMTE4B3VuaUUxMTkHdW5pRTEyMAd1bmlFMTIxB3VuaUUxMjIHdW5pRTEyMwd1bmlFMTI0B3VuaUUxMjUHdW5pRTEyNgd1bmlFMTI3B3VuaUUxMjgHdW5pRTEyOQd1bmlFMTMwB3VuaUUxMzEHdW5pRTEzMgd1bmlFMTMzB3VuaUUxMzQHdW5pRTEzNQd1bmlFMTM2B3VuaUUxMzcHdW5pRTEzOAd1bmlFMTM5B3VuaUUxNDAHdW5pRTE0MQd1bmlFMTQyB3VuaUUxNDMHdW5pRTE0NAd1bmlFMTQ1B3VuaUUxNDYHdW5pRTE0OAd1bmlFMTQ5B3VuaUUxNTAHdW5pRTE1MQd1bmlFMTUyB3VuaUUxNTMHdW5pRTE1NAd1bmlFMTU1B3VuaUUxNTYHdW5pRTE1Nwd1bmlFMTU4B3VuaUUxNTkHdW5pRTE2MAd1bmlFMTYxB3VuaUUxNjIHdW5pRTE2Mwd1bmlFMTY0B3VuaUUxNjUHdW5pRTE2Ngd1bmlFMTY3B3VuaUUxNjgHdW5pRTE2OQd1bmlFMTcwB3VuaUUxNzEHdW5pRTE3Mgd1bmlFMTczB3VuaUUxNzQHdW5pRTE3NQd1bmlFMTc2B3VuaUUxNzcHdW5pRTE3OAd1bmlFMTc5B3VuaUUxODAHdW5pRTE4MQd1bmlFMTgyB3VuaUUxODMHdW5pRTE4NAd1bmlFMTg1B3VuaUUxODYHdW5pRTE4Nwd1bmlFMTg4B3VuaUUxODkHdW5pRTE5MAd1bmlFMTkxB3VuaUUxOTIHdW5pRTE5Mwd1bmlFMTk0B3VuaUUxOTUHdW5pRTE5Nwd1bmlFMTk4B3VuaUUxOTkHdW5pRTIwMLgB/4WwAY0AS7AIUFixAQGOWbFGBitYIbAQWUuwFFJYIbCAWR2wBitcWFmwFCsAAAABUuZYrgAA') format('truetype')}.glyphicon{position:relative;top:1px;display:inline-block;font-family:'Glyphicons Halflings';font-style:normal;font-weight:400;line-height:1;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased}.glyphicon-asterisk:before{content:'\\2a'}.glyphicon-plus:before{content:'\\2b'}.glyphicon-euro:before{content:'\\20ac'}.glyphicon-minus:before{content:'\\2212'}.glyphicon-cloud:before{content:'\\2601'}.glyphicon-envelope:before{content:'\\2709'}.glyphicon-pencil:before{content:'\\270f'}.glyphicon-glass:before{content:'\\e001'}.glyphicon-music:before{content:'\\e002'}.glyphicon-search:before{content:'\\e003'}.glyphicon-heart:before{content:'\\e005'}.glyphicon-star:before{content:'\\e006'}.glyphicon-star-empty:before{content:'\\e007'}.glyphicon-user:before{content:'\\e008'}.glyphicon-film:before{content:'\\e009'}.glyphicon-th-large:before{content:'\\e010'}.glyphicon-th:before{content:'\\e011'}.glyphicon-th-list:before{content:'\\e012'}.glyphicon-ok:before{content:'\\e013'}.glyphicon-remove:before{content:'\\e014'}.glyphicon-zoom-in:before{content:'\\e015'}.glyphicon-zoom-out:before{content:'\\e016'}.glyphicon-off:before{content:'\\e017'}.glyphicon-signal:before{content:'\\e018'}.glyphicon-cog:before{content:'\\e019'}.glyphicon-trash:before{content:'\\e020'}.glyphicon-home:before{content:'\\e021'}.glyphicon-file:before{content:'\\e022'}.glyphicon-time:before{content:'\\e023'}.glyphicon-road:before{content:'\\e024'}.glyphicon-download-alt:before{content:'\\e025'}.glyphicon-download:before{content:'\\e026'}.glyphicon-upload:before{content:'\\e027'}.glyphicon-inbox:before{content:'\\e028'}.glyphicon-play-circle:before{content:'\\e029'}.glyphicon-repeat:before{content:'\\e030'}.glyphicon-refresh:before{content:'\\e031'}.glyphicon-list-alt:before{content:'\\e032'}.glyphicon-lock:before{content:'\\e033'}.glyphicon-flag:before{content:'\\e034'}.glyphicon-headphones:before{content:'\\e035'}.glyphicon-volume-off:before{content:'\\e036'}.glyphicon-volume-down:before{content:'\\e037'}.glyphicon-volume-up:before{content:'\\e038'}.glyphicon-qrcode:before{content:'\\e039'}.glyphicon-barcode:before{content:'\\e040'}.glyphicon-tag:before{content:'\\e041'}.glyphicon-tags:before{content:'\\e042'}.glyphicon-book:before{content:'\\e043'}.glyphicon-bookmark:before{content:'\\e044'}.glyphicon-print:before{content:'\\e045'}.glyphicon-camera:before{content:'\\e046'}.glyphicon-font:before{content:'\\e047'}.glyphicon-bold:before{content:'\\e048'}.glyphicon-italic:before{content:'\\e049'}.glyphicon-text-height:before{content:'\\e050'}.glyphicon-text-width:before{content:'\\e051'}.glyphicon-align-left:before{content:'\\e052'}.glyphicon-align-center:before{content:'\\e053'}.glyphicon-align-right:before{content:'\\e054'}.glyphicon-align-justify:before{content:'\\e055'}.glyphicon-list:before{content:'\\e056'}.glyphicon-indent-left:before{content:'\\e057'}.glyphicon-indent-right:before{content:'\\e058'}.glyphicon-facetime-video:before{content:'\\e059'}.glyphicon-picture:before{content:'\\e060'}.glyphicon-map-marker:before{content:'\\e062'}.glyphicon-adjust:before{content:'\\e063'}.glyphicon-tint:before{content:'\\e064'}.glyphicon-edit:before{content:'\\e065'}.glyphicon-share:before{content:'\\e066'}.glyphicon-check:before{content:'\\e067'}.glyphicon-move:before{content:'\\e068'}.glyphicon-step-backward:before{content:'\\e069'}.glyphicon-fast-backward:before{content:'\\e070'}.glyphicon-backward:before{content:'\\e071'}.glyphicon-play:before{content:'\\e072'}.glyphicon-pause:before{content:'\\e073'}.glyphicon-stop:before{content:'\\e074'}.glyphicon-forward:before{content:'\\e075'}.glyphicon-fast-forward:before{content:'\\e076'}.glyphicon-step-forward:before{content:'\\e077'}.glyphicon-eject:before{content:'\\e078'}.glyphicon-chevron-left:before{content:'\\e079'}.glyphicon-chevron-right:before{content:'\\e080'}.glyphicon-plus-sign:before{content:'\\e081'}.glyphicon-minus-sign:before{content:'\\e082'}.glyphicon-remove-sign:before{content:'\\e083'}.glyphicon-ok-sign:before{content:'\\e084'}.glyphicon-question-sign:before{content:'\\e085'}.glyphicon-info-sign:before{content:'\\e086'}.glyphicon-screenshot:before{content:'\\e087'}.glyphicon-remove-circle:before{content:'\\e088'}.glyphicon-ok-circle:before{content:'\\e089'}.glyphicon-ban-circle:before{content:'\\e090'}.glyphicon-arrow-left:before{content:'\\e091'}.glyphicon-arrow-right:before{content:'\\e092'}.glyphicon-arrow-up:before{content:'\\e093'}.glyphicon-arrow-down:before{content:'\\e094'}.glyphicon-share-alt:before{content:'\\e095'}.glyphicon-resize-full:before{content:'\\e096'}.glyphicon-resize-small:before{content:'\\e097'}.glyphicon-exclamation-sign:before{content:'\\e101'}.glyphicon-gift:before{content:'\\e102'}.glyphicon-leaf:before{content:'\\e103'}.glyphicon-fire:before{content:'\\e104'}.glyphicon-eye-open:before{content:'\\e105'}.glyphicon-eye-close:before{content:'\\e106'}.glyphicon-warning-sign:before{content:'\\e107'}.glyphicon-plane:before{content:'\\e108'}.glyphicon-calendar:before{content:'\\e109'}.glyphicon-random:before{content:'\\e110'}.glyphicon-comment:before{content:'\\e111'}.glyphicon-magnet:before{content:'\\e112'}.glyphicon-chevron-up:before{content:'\\e113'}.glyphicon-chevron-down:before{content:'\\e114'}.glyphicon-retweet:before{content:'\\e115'}.glyphicon-shopping-cart:before{content:'\\e116'}.glyphicon-folder-close:before{content:'\\e117'}.glyphicon-folder-open:before{content:'\\e118'}.glyphicon-resize-vertical:before{content:'\\e119'}.glyphicon-resize-horizontal:before{content:'\\e120'}.glyphicon-hdd:before{content:'\\e121'}.glyphicon-bullhorn:before{content:'\\e122'}.glyphicon-bell:before{content:'\\e123'}.glyphicon-certificate:before{content:'\\e124'}.glyphicon-thumbs-up:before{content:'\\e125'}.glyphicon-thumbs-down:before{content:'\\e126'}.glyphicon-hand-right:before{content:'\\e127'}.glyphicon-hand-left:before{content:'\\e128'}.glyphicon-hand-up:before{content:'\\e129'}.glyphicon-hand-down:before{content:'\\e130'}.glyphicon-circle-arrow-right:before{content:'\\e131'}.glyphicon-circle-arrow-left:before{content:'\\e132'}.glyphicon-circle-arrow-up:before{content:'\\e133'}.glyphicon-circle-arrow-down:before{content:'\\e134'}.glyphicon-globe:before{content:'\\e135'}.glyphicon-wrench:before{content:'\\e136'}.glyphicon-tasks:before{content:'\\e137'}.glyphicon-filter:before{content:'\\e138'}.glyphicon-briefcase:before{content:'\\e139'}.glyphicon-fullscreen:before{content:'\\e140'}.glyphicon-dashboard:before{content:'\\e141'}.glyphicon-paperclip:before{content:'\\e142'}.glyphicon-heart-empty:before{content:'\\e143'}.glyphicon-link:before{content:'\\e144'}.glyphicon-phone:before{content:'\\e145'}.glyphicon-pushpin:before{content:'\\e146'}.glyphicon-usd:before{content:'\\e148'}.glyphicon-gbp:before{content:'\\e149'}.glyphicon-sort:before{content:'\\e150'}.glyphicon-sort-by-alphabet:before{content:'\\e151'}.glyphicon-sort-by-alphabet-alt:before{content:'\\e152'}.glyphicon-sort-by-order:before{content:'\\e153'}.glyphicon-sort-by-order-alt:before{content:'\\e154'}.glyphicon-sort-by-attributes:before{content:'\\e155'}.glyphicon-sort-by-attributes-alt:before{content:'\\e156'}.glyphicon-unchecked:before{content:'\\e157'}.glyphicon-expand:before{content:'\\e158'}.glyphicon-collapse-down:before{content:'\\e159'}.glyphicon-collapse-up:before{content:'\\e160'}.glyphicon-log-in:before{content:'\\e161'}.glyphicon-flash:before{content:'\\e162'}.glyphicon-log-out:before{content:'\\e163'}.glyphicon-new-window:before{content:'\\e164'}.glyphicon-record:before{content:'\\e165'}.glyphicon-save:before{content:'\\e166'}.glyphicon-open:before{content:'\\e167'}.glyphicon-saved:before{content:'\\e168'}.glyphicon-import:before{content:'\\e169'}.glyphicon-export:before{content:'\\e170'}.glyphicon-send:before{content:'\\e171'}.glyphicon-floppy-disk:before{content:'\\e172'}.glyphicon-floppy-saved:before{content:'\\e173'}.glyphicon-floppy-remove:before{content:'\\e174'}.glyphicon-floppy-save:before{content:'\\e175'}.glyphicon-floppy-open:before{content:'\\e176'}.glyphicon-credit-card:before{content:'\\e177'}.glyphicon-transfer:before{content:'\\e178'}.glyphicon-cutlery:before{content:'\\e179'}.glyphicon-header:before{content:'\\e180'}.glyphicon-compressed:before{content:'\\e181'}.glyphicon-earphone:before{content:'\\e182'}.glyphicon-phone-alt:before{content:'\\e183'}.glyphicon-tower:before{content:'\\e184'}.glyphicon-stats:before{content:'\\e185'}.glyphicon-sd-video:before{content:'\\e186'}.glyphicon-hd-video:before{content:'\\e187'}.glyphicon-subtitles:before{content:'\\e188'}.glyphicon-sound-stereo:before{content:'\\e189'}.glyphicon-sound-dolby:before{content:'\\e190'}.glyphicon-sound-5-1:before{content:'\\e191'}.glyphicon-sound-6-1:before{content:'\\e192'}.glyphicon-sound-7-1:before{content:'\\e193'}.glyphicon-copyright-mark:before{content:'\\e194'}.glyphicon-registration-mark:before{content:'\\e195'}.glyphicon-cloud-download:before{content:'\\e197'}.glyphicon-cloud-upload:before{content:'\\e198'}.glyphicon-tree-conifer:before{content:'\\e199'}.glyphicon-tree-deciduous:before{content:'\\e200'}";
  17.  
  18. if (typeof GM_addStyle != "undefined") {
  19. GM_addStyle(css);
  20. } else if (typeof PRO_addStyle != "undefined") {
  21. PRO_addStyle(css);
  22. } else if (typeof addStyle != "undefined") {
  23. addStyle(css);
  24. } else {
  25. var node = document.createElement("style");
  26. node.type = "text/css";
  27. node.appendChild(document.createTextNode(css));
  28. var heads = document.getElementsByTagName("head");
  29. if (heads.length > 0) {
  30. heads[0].appendChild(node);
  31. } else {
  32. document.documentElement.appendChild(node);
  33. }
  34. }
  35.  
  36. var $ = window.$;
  37. var jQuery = window.$;
  38. var pgf = window.pgf;
  39.  
  40. window.ext = (function (_ext) {
  41. "use strict";
  42.  
  43. //if (window.API_CLIENT > 'the_tale-v0.3.15.8') return;
  44.  
  45. setTimeout(function() {
  46.  
  47. $(document).on('ajaxSuccess.ext', function(event, XMLHttpRequest, setting, result) {
  48.  
  49. "use strict";
  50.  
  51. if (setting.url.indexOf('/game/api/info?api_client=') === 0) {
  52.  
  53. var game_data = result.data;
  54.  
  55.  
  56. try {
  57.  
  58. _ext.heroName = game_data.account.hero.base.name;
  59.  
  60. _ext.map_version = game_data.map_version;
  61.  
  62. } catch(e) {}
  63.  
  64.  
  65.  
  66. if (!_ext.preloadDone) {
  67.  
  68. // $(document).trigger(pgf.game.events.DATA_REFRESHED, game_data);
  69.  
  70. _ext.publish('preload', game_data);
  71.  
  72. }
  73.  
  74. _ext.preloadDone = 1;
  75.  
  76.  
  77. if (!game_data.account.is_old) {
  78.  
  79. _ext.publish('load', game_data);
  80.  
  81. $('.ext-wait').hide();
  82.  
  83. $(document).off('ajaxSuccess.ext');
  84.  
  85. delete _ext.preloadDone;
  86.  
  87. }
  88.  
  89. }
  90.  
  91. });
  92.  
  93. }, 15000);
  94.  
  95.  
  96. _ext.const = {
  97.  
  98. MAX_LOG_LENGTH: 1000,
  99.  
  100. MAX_ARCHIVE_LENGTH: 2000,
  101.  
  102. FIGHT_START: ['fight'],
  103.  
  104. FIGHT: ['hit', 'might', 'fire', 'poisoncloud', 'vamp', 'stunHit', 'crit', 'flame', 'poison', 'slow', 'mush', 'speed', 'ue', 'eva', 'stun', 'heal'],
  105.  
  106. FIGHT_VALUES: ['hit', 'might', 'fire', 'vamp', 'stunHit', 'crit', 'flame', 'poison', 'heal'],
  107.  
  108. FIGHT_COUNTS: [ 'poisoncloud', 'slow', 'mush', 'speed', 'ue', 'eva', 'stun' ],
  109.  
  110. DMG: ['hit', 'might', 'fire', 'poisoncloud', 'vamp', 'stunHit', 'crit' ],
  111.  
  112. DOT: [ 'flame', 'poison' ],
  113.  
  114. DEBUFF: [ 'slow' ],
  115.  
  116. BUFF: [ 'mush', 'speed', 'ue' ],
  117.  
  118. ACTIVE: ['hit', 'might', 'fire', 'poisoncloud', 'vamp', 'stunHit', 'mush' , 'ue' ],
  119.  
  120. PASSIVE: [ 'crit', 'flame', 'poison', 'slow', 'speed', 'eva', 'stun', 'heal'],
  121.  
  122.  
  123. SUM_TO_MAIN: {flame: 'fire', poison: 'poisoncloud', crit: 'hit'},
  124.  
  125. LOOT: ['pickup', 'empty', 'drop'],
  126.  
  127. REST: ['rest'],
  128.  
  129. SHORT: ['hit', 'might', 'fire', 'flame', 'poison', 'vamp', 'slow', 'poisoncloud', 'mush', 'ue', 'eva', 'heal', 'godheal', 'rest', 'coins', 'speed', 'stunHit', 'stun', 'pvpeff', 'pvpice', 'pvpflame', 'pvpfail'],
  130.  
  131. ACTION_TYPE_NAMES: {
  132.  
  133. 0: 'idle', //безделие
  134.  
  135. 1: 'quest', //задание
  136.  
  137. 2: 'walk', //путешествие между городами
  138.  
  139. 3: 'fight', //сражение 1x1 с монстром
  140.  
  141. 4: 'dead', //воскрешение
  142.  
  143. 5: 'city', //действия в городе
  144.  
  145. 6: 'rest', //отдых
  146.  
  147. 7: 'equip', //экипировка
  148.  
  149. 8: 'trade', //торговля
  150.  
  151. 9: 'nearcity', //путешествие около города
  152.  
  153. 10: 'energy', //восстановление энергии
  154.  
  155. 11: 'noeffect', //действие без эффекта на игру
  156.  
  157. 12: 'proxy', //прокси-действия для взаимодействия героев
  158.  
  159. 13: 'pvp', //PvP 1x1
  160.  
  161. 14: 'test', //проверочное действие
  162.  
  163. 15: 'companionHelp' //уход за спутником
  164.  
  165. },
  166.  
  167. ACTION_TYPE_TEXTS: {
  168.  
  169. idle: 'безделие',
  170.  
  171. quest: 'задание',
  172.  
  173. walk: 'путешествие между городами',
  174.  
  175. fight: 'сражение 1x1 с монстром',
  176.  
  177. dead: 'воскрешение',
  178.  
  179. city: 'действия в городе',
  180.  
  181. rest: 'отдых',
  182.  
  183. equip: 'экипировка',
  184.  
  185. trade: 'торговля',
  186.  
  187. nearcity: 'путешествие около города',
  188.  
  189. energy: 'восстановление энергии',
  190.  
  191. noeffect: 'действие без эффекта на игру',
  192.  
  193. proxy: 'прокси-действия для взаимодействия героев',
  194.  
  195. pvp: 'PvP 1x1',
  196.  
  197. undefined: 'неизвестное действие',
  198.  
  199. broken: 'неразобранное действие'
  200.  
  201. },
  202.  
  203. ACTION_TYPE_ICONS: {
  204.  
  205. idle: '<span class="glyphicon glyphicon-home"></span>',
  206.  
  207. quest: '<span class="glyphicon glyphicon-globe"></span>',
  208.  
  209. walk: '<span class="glyphicon glyphicon-road"></span>',
  210.  
  211. fight: '<span class="glyphicon glyphicon-leaf"></span>',
  212.  
  213. dead: '<span class="glyphicon glyphicon-remove"></span>',
  214.  
  215. city: '<span class="glyphicon glyphicon-home"></span>',
  216.  
  217. rest: '<span class="glyphicon glyphicon-heart-empty"></span>',
  218.  
  219. equip: '<span class="glyphicon glyphicon-collapse-down"></span>',
  220.  
  221. trade: '<span class="glyphicon glyphicon-copyright-mark"></span>',
  222.  
  223. nearcity: '<span class="glyphicon glyphicon-globe"></span>',
  224.  
  225. energy: '<span class="glyphicon glyphicon-flash"></span>',
  226.  
  227. noeffect: '<span class="glyphicon glyphicon-music"></span>',
  228.  
  229. broken: '<span class="glyphicon glyphicon-ban-circle"></span>'
  230.  
  231. },
  232.  
  233. ERROR_CODES: {
  234.  
  235. 1: 'Неполное действие',
  236.  
  237. 2: 'Не найдено начало боя',
  238.  
  239. 3: 'Не найден конец боя',
  240.  
  241. 4: 'Дыра в середине действия',
  242.  
  243. 5: 'Запись обрывается'
  244.  
  245. },
  246.  
  247. ICONS: {
  248.  
  249. hit: '<span class="glyphicon glyphicon-leaf"></span>',
  250.  
  251. crit: '<span class="glyphicon glyphicon-star"></span>',
  252.  
  253. might: '<span class="glyphicon glyphicon-tower"></span>',
  254.  
  255. fire: '<span class="glyphicon glyphicon-fire"></span>',
  256.  
  257. flame: '<span class="glyphicon glyphicon-fire small"></span>',
  258.  
  259. poisoncloud:'<span class="glyphicon glyphicon-tint"></span>',
  260.  
  261. poison: '<span class="glyphicon glyphicon-tint small"></span>',
  262.  
  263. vamp: '<span class="glyphicon glyphicon-cutlery"></span>',
  264.  
  265.  
  266. ue: '<span class="glyphicon glyphicon-refresh"></span> ',
  267.  
  268. eva: '<span class="glyphicon glyphicon-transfer"></span> ',
  269.  
  270. slow: '<span class="glyphicon glyphicon-link"></span>',
  271.  
  272. mush: '<span class="glyphicon glyphicon-arrow-up"></span>',
  273.  
  274. speed: '<span class="glyphicon glyphicon-forward"></span>',
  275.  
  276.  
  277. stunHit: '<span class="glyphicon glyphicon-step-forward"></span>',
  278.  
  279. stun: '<span class="glyphicon glyphicon-bell"></span>',
  280.  
  281.  
  282. rest: '<span class="glyphicon glyphicon-heart-empty"></span>',
  283.  
  284. heal: '<span class="glyphicon glyphicon-heart"></span>',
  285.  
  286.  
  287. dmgSum: '<span class="glyphicon glyphicon-asterisk"></span>',
  288.  
  289.  
  290. godheal: '<span class="glyphicon glyphicon-heart-empty"></span>',
  291.  
  292. coins: '<span class="glyphicon glyphicon-copyright-mark"></span>',
  293.  
  294.  
  295. pickup: '<span class="glyphicon glyphicon-ok-circle"></span>',
  296.  
  297. empty: '<span class="glyphicon glyphicon-remove-circle"></span>',
  298.  
  299. drop: '<span class="glyphicon glyphicon-ban-circle"></span>',
  300.  
  301.  
  302. pvpeff: '<span class="glyphicon glyphicon-flash"></span>',
  303.  
  304. pvpice: '<span class="glyphicon glyphicon-cloud-upload"></span>',
  305.  
  306. pvpflame: '<span class="glyphicon glyphicon-cloud-download"></span>',
  307.  
  308. pvpfail: '<span class="glyphicon glyphicon-cloud"></span>'
  309.  
  310. },
  311.  
  312. ACTION_TRANSLATE: {
  313.  
  314. hit: 'Удар',
  315.  
  316. crit: 'Критический удар',
  317.  
  318. stunHit: 'Разбег-толчок',
  319.  
  320. stun: 'Шок',
  321.  
  322. might: 'Тяжёлый удар',
  323.  
  324. slow: 'Заморозка',
  325.  
  326. speed: 'Ускорение',
  327.  
  328. fire: 'Шар огня',
  329.  
  330. flame: 'Пламя',
  331.  
  332. mush: 'Волшебный гриб',
  333.  
  334. ue: 'Шаг в сторону',
  335.  
  336. eva: 'Увернулся',
  337.  
  338. poisoncloud: 'Ядовитое облако',
  339.  
  340. poison: 'Отравление',
  341.  
  342. vamp: 'Удар вампира',
  343.  
  344. heal: 'Регенерация',
  345.  
  346. dmgSum: 'Всего'
  347.  
  348. },
  349.  
  350. MOBS: [null,"олень","одичавшая одежда","осы","хищный кустик","кабан-секач","бродячий дуб","волк","дух леса","оскорблённая эльфийка","блюститель природы","медведь","аристократ","единорог","призрачный волк","астральный охотник","головастик","домашний поползень","одичавшие светлячки","ползун","пиявка","гигантская росянка","сбежавший эксперимент","слизь","гидра","щупальце","химический мусорник","упырь","кикимора","бесхозный голем","охотник за реагентами","мутировавший гоблин","василиск","ожившее пугало","хищный хомяк","борец за справедливость","саранча","маньяк-извращенец","карточный шулер","богомол","благородный разбойник","одержимый колдун","механический дракончик","дварф-изгнанник","неистовый сектант","механический голем","убийца","полоумный маг","шакал","скорпион","налётчик","огненная лиса","орк-изгнанник","скарабей","гремучая змея","саламандра","пустынный дракончик","джинн","берсерк","смерч","заблудший шаман","шайтан","крыса","бандит","призрак","неупокоенный","гремлин","подыльник","странствующий рыцарь","нага","охотник за головами","суккуб","вирика","див","додо","ангиак","бхут","гвиллион","вилах","вендиго","галд","ачери","земляная даппи","огненная даппи","анчутка","жихарь","йиена","мара","боец Серого Ордена","спятивший лорд","волколак","медвелак","аколит огня","ламия","баргест","вий","гьюнальский паук","дэра","гриндель","грим","кагир","друджи","даса","стая ичетики","кера","куд","ларва","егви","каменный великан","каннибал","ламашту","лахама","наркоман","бывший палач","мародёр","страхолев","злоцвет","варвар","ночная хныка","омерзень","смрадень","глыдень","песчаный холерник","мать анаконд","белая гиена","мраморный бык","язвомор","лишайница","жирница","бульг","костогрыз","каменник","вирница","тигр","барбегаз","бродячий огонек","скальник","пупырник","стогница","куропатка гуанила","пёстробородый","капитан Серых Плащей","аколит льда","семиглаз","гюрза","медвежук","снежная росомаха","койот","горный манул","мастер огня","мастер льда","кобольд","пожиратель плоти","главарь пестробородых","ученик Силы","гомункул","некромант","тагар","стрига","приземник","лесной тролль","вурдалак","болотный тролль","бескуд","атач"]
  351.  
  352. };
  353.  
  354.  
  355.  
  356.  
  357. var _subscribeList = [];
  358.  
  359. function publish(e) {
  360.  
  361. var sp = e.split('.');
  362.  
  363. var event = sp[0];
  364.  
  365. var namespace = sp[1];
  366.  
  367. for (var i=0; i<_subscribeList.length; i++) {
  368.  
  369. var s = _subscribeList[i];
  370.  
  371. if ((!namespace || namespace === s.namespace) && (!event || event === s.event)) {
  372.  
  373. s.fn.apply(_ext, Array.prototype.slice.call(arguments, 1));
  374.  
  375. }
  376.  
  377. }
  378.  
  379. }
  380.  
  381. function subscribe(e, fn) {
  382.  
  383. var events = e.split(',');
  384.  
  385. if (events.length > 1) {
  386.  
  387. events.forEach(function(ev) {subscribe(ev, fn);});
  388.  
  389. return;
  390.  
  391. }
  392.  
  393. var sp = e.split('.');
  394.  
  395. var event = sp[0];
  396.  
  397. var namespace = sp[1];
  398.  
  399. _subscribeList.push({
  400.  
  401. event: event,
  402.  
  403. namespace: namespace,
  404.  
  405. fn: fn
  406.  
  407. });
  408.  
  409. }
  410.  
  411. var cacheStore = {};
  412.  
  413. var cacheTimeId = {};
  414.  
  415. function cache(name, value, time) {
  416.  
  417. if (arguments.length > 1) {
  418.  
  419. cacheStore[name] = value;
  420.  
  421. if (cacheTimeId[name]) window.clearTimeout(cacheTimeId[name]);
  422.  
  423. if (time) {
  424.  
  425. cacheTimeId[name] = window.setTimeout(function() {
  426.  
  427. delete cacheStore[name];
  428.  
  429. }, time)
  430.  
  431. }
  432.  
  433. } else {
  434.  
  435. return cacheStore[name];
  436.  
  437. }
  438.  
  439. }
  440.  
  441. $.extend(_ext, {
  442.  
  443. subscribeList: _subscribeList,
  444.  
  445. cacheStore: cacheStore,
  446.  
  447. cache: cache,
  448.  
  449. publish: publish,
  450.  
  451. subscribe: subscribe
  452.  
  453. });
  454.  
  455.  
  456. var _store = (function() {
  457.  
  458. var prefix = 'ext-';
  459.  
  460. function setStore(key, value) {
  461.  
  462. localStorage.setItem(prefix + key, JSON.stringify(value));
  463.  
  464. }
  465.  
  466. function getStore(key) {
  467.  
  468. return JSON.parse(localStorage.getItem(prefix + key) || localStorage.getItem(key));
  469.  
  470. }
  471.  
  472. if (window.localStorage && window.JSON) {
  473.  
  474. return {
  475.  
  476. get: getStore,
  477.  
  478. set: setStore
  479.  
  480. }
  481.  
  482. } else {
  483.  
  484. return {
  485.  
  486. get: function() {},
  487.  
  488. set: function() {}
  489.  
  490. }
  491.  
  492. }
  493.  
  494. })({});
  495.  
  496.  
  497. /* log */
  498.  
  499. var _log = (function() {
  500.  
  501. function setLog(name, messages) {
  502.  
  503. try {
  504.  
  505. if (name === 'messagesLog') {
  506.  
  507. var max = _settings.settingsValues.maxLogLength || _ext.const.MAX_LOG_LENGTH;
  508.  
  509. pgf.base.settings.set(name, JSON.stringify(messages.slice(messages.length - max)));
  510.  
  511. } else {
  512.  
  513. pgf.base.settings.set(name, JSON.stringify(messages));
  514.  
  515. }
  516.  
  517. } catch (e) {
  518.  
  519. console.warn('setLog', name, e);
  520.  
  521. }
  522.  
  523. }
  524.  
  525. function getLog(name) {
  526.  
  527. var g = pgf.base.settings.get(name);
  528.  
  529. return g ? JSON.parse(g) : '';
  530.  
  531. }
  532.  
  533. function logToConsole(name) {
  534.  
  535. var strLog = pgf.base.settings.get(name);
  536.  
  537. var s = strLog
  538.  
  539. .replace(/\],\[/g, '],\n\t[')
  540.  
  541. .replace(/\},{/g, '},\n\t{')
  542.  
  543. .replace(/"(action|base|energy|habits|secondary|turn)"/g, '\n\t\t"$1"');
  544.  
  545. console.log(s);
  546.  
  547. }
  548.  
  549. function toStr(messages) {
  550.  
  551. var strLog = JSON.stringify(messages);
  552.  
  553. var s = strLog
  554.  
  555. .replace(/\],\[/g, '],\n\t[')
  556.  
  557. .replace(/"(action|base|energy|habits|secondary|turn)"/g, '\n\t\t"$1"');
  558.  
  559. return s;
  560.  
  561. }
  562.  
  563. function size() {
  564.  
  565. var t = 0;
  566.  
  567. for(var x in localStorage){
  568.  
  569. t += localStorage[x].length * 2;
  570.  
  571. }
  572.  
  573. // console.log(t/1024+ " KB");
  574.  
  575. return t;
  576.  
  577. }
  578.  
  579. if (window.localStorage && window.JSON) {
  580.  
  581. return {
  582.  
  583. toConsole: logToConsole,
  584.  
  585. toStr: toStr,
  586.  
  587. get: getLog,
  588.  
  589. size: size,
  590.  
  591. set: setLog
  592.  
  593. };
  594.  
  595. } else {
  596.  
  597. return {
  598.  
  599. toConsole: function() {},
  600.  
  601. toStr: function() {},
  602.  
  603. get: function() {},
  604.  
  605. size: function() {},
  606.  
  607. set: function() {}
  608.  
  609. };
  610.  
  611. }
  612.  
  613. })();
  614.  
  615. /* eo log */
  616.  
  617.  
  618.  
  619.  
  620. /* settings */
  621.  
  622. var _settings = (function(_settings) {
  623.  
  624. function getSettingInput(key) {
  625.  
  626. return $('input[data-name="' + key + '"]');
  627.  
  628. }
  629.  
  630.  
  631. var sets = [
  632.  
  633. {
  634.  
  635. title: 'Персонаж',
  636.  
  637. fields: [{
  638.  
  639. label: 'Начало имени героя:',
  640.  
  641. note: 'Общая часть имени для всех падежей',
  642.  
  643. inputs: [{
  644.  
  645. name: 'heroNameStart',
  646.  
  647. type: 'text'
  648.  
  649. }]
  650.  
  651.  
  652. }]
  653.  
  654. },
  655.  
  656. {
  657.  
  658. title: 'Уведомления',
  659.  
  660. fields: [{
  661.  
  662. label: 'Отправлять уведомления',
  663.  
  664. name: 'notify',
  665.  
  666. isToggle: 1,
  667.  
  668. value: true,
  669.  
  670. subs: [{
  671.  
  672. label: 'Выбор в задании',
  673.  
  674. name: 'notifyQuestChoose',
  675.  
  676. isToggle: 1,
  677.  
  678. value: true
  679.  
  680. }, {
  681.  
  682. label: 'Герой бездействует',
  683.  
  684. name: 'notifyHeroIdle',
  685.  
  686. isToggle: 1,
  687.  
  688. value: true
  689.  
  690. /*}, {
  691.  
  692. label: 'Герой воскресает',
  693.  
  694. name: 'notifyHeroDead',
  695.  
  696. isToggle: 1,
  697.  
  698. value: true*/
  699.  
  700. }, {
  701.  
  702. label: 'Здоровье ниже ',
  703.  
  704. name: 'notifyHeroHp',
  705.  
  706. isToggle: 1,
  707.  
  708. value: true,
  709.  
  710. inputs: [{
  711.  
  712. type: 'num',
  713.  
  714. name: 'notifyHeroHpLowerValue',
  715.  
  716. value: 200
  717.  
  718. }]
  719.  
  720. }, {
  721.  
  722. label: 'Энергия выше ',
  723.  
  724. name: 'notifyHeroEnergy',
  725.  
  726. isToggle: 1,
  727.  
  728. value: true,
  729.  
  730. inputs: [{
  731.  
  732. type: 'num',
  733.  
  734. name: 'notifyHeroEnergyGreaterValue',
  735.  
  736. value: 9
  737.  
  738. }]
  739.  
  740. }]
  741.  
  742. }]
  743.  
  744. }, {
  745.  
  746. title: 'Статистика',
  747.  
  748. fields: [{
  749.  
  750. label: 'Уровень героя: последний, или',
  751.  
  752. name: 'statsByLevel',
  753.  
  754. isToggle: 1,
  755.  
  756. value: true,
  757.  
  758. inputs: [{
  759.  
  760. type: 'num',
  761.  
  762. name: 'statsByLevelValue'
  763.  
  764. }]
  765.  
  766. }, {
  767.  
  768. label: 'Монстр: последний, или',
  769.  
  770. name: 'statsByMob',
  771.  
  772. isToggle: 1,
  773.  
  774. value: true,
  775.  
  776. inputs: [{
  777.  
  778. type: 'num',
  779.  
  780. name: 'statsByMobId',
  781.  
  782. addOn: 'ID'
  783.  
  784. }],
  785.  
  786. subs: [{
  787.  
  788. label: 'Статистика героя против монстра',
  789.  
  790. name: 'myStatsByMob',
  791.  
  792. isToggle: 1,
  793.  
  794. value: false
  795.  
  796. }]
  797.  
  798. }, {
  799.  
  800. label: 'Количество действий в статистике:',
  801.  
  802. inputs: [{
  803.  
  804. name: 'statsActionsCount',
  805.  
  806. type: 'num'
  807.  
  808. }]
  809.  
  810. }]
  811.  
  812. }, {
  813.  
  814. title: 'Отображение',
  815.  
  816. fields: [{
  817.  
  818. label: 'Выводить подробности действий',
  819.  
  820. name: 'groupOpenOnDefault',
  821.  
  822. isToggle: 1,
  823.  
  824. value: true
  825.  
  826. }, {
  827.  
  828. label: 'Показывать архив',
  829.  
  830. name: 'showArchive',
  831.  
  832. isToggle: 1,
  833.  
  834. value: false
  835.  
  836. }]
  837.  
  838. }, {
  839.  
  840. title: 'Хранилище <span id="storage-size"></span>',
  841.  
  842. fields: [{
  843.  
  844. label: 'Сообщений в кратком журнале:',
  845.  
  846. inputs: [{
  847.  
  848. name: 'maxLogLength',
  849.  
  850. type: 'num',
  851.  
  852. value: _ext.const.MAX_LOG_LENGTH
  853.  
  854. }]
  855.  
  856. }, {
  857.  
  858. label: 'Действий в архиве:',
  859.  
  860. inputs: [{
  861.  
  862. name: 'maxArchiveLength',
  863.  
  864. type: 'num',
  865.  
  866. value: _ext.const.MAX_ARCHIVE_LENGTH
  867.  
  868. }]
  869.  
  870. }, {
  871.  
  872. label: '<span class="link-ajax" id="reset-stats">Очистить хранилище</span>'
  873.  
  874. }]
  875.  
  876. }
  877.  
  878. ];
  879.  
  880. _ext.subscribe('settingsChange', checkDependences);
  881.  
  882. function checkDependences(key, value, isDisabled) {
  883.  
  884. if (deps[key]) {
  885.  
  886. deps[key].forEach(function(keyName) {
  887.  
  888. if (keyName) getSettingInput(keyName).closest('.input-wrap').toggleClass('disabled', isDisabled || !value);//.prop('disabled', !value);
  889.  
  890. });
  891.  
  892. }
  893.  
  894. }
  895.  
  896.  
  897. var deps = {};
  898.  
  899. function addSets(sets) {
  900.  
  901. for (var i=0; i<sets.length;i++) {
  902.  
  903. settingsDefaults(sets[i].fields);
  904.  
  905. }
  906.  
  907.  
  908. function settingsDefaults(fields) {
  909.  
  910. var childs = [];
  911.  
  912. for (var i=0; i<fields.length;i++) {
  913.  
  914. var st = fields[i];
  915.  
  916. childs.push(st.name);
  917.  
  918. // console.log('defaults', st.name);
  919.  
  920. if (typeof settingsValues[st.name] === 'undefined') settingsValues[st.name] = st.value;
  921.  
  922. var subsChilds = [];
  923.  
  924. if (st.inputs) {
  925.  
  926. subsChilds = settingsDefaults(st.inputs).concat(subsChilds);
  927.  
  928. }
  929.  
  930. if (st.subs) {
  931.  
  932. subsChilds = settingsDefaults(st.subs).concat(subsChilds);
  933.  
  934. }
  935.  
  936. if (st.fields) subsChilds = settingsDefaults(st.fields).concat(subsChilds);
  937.  
  938.  
  939. if (subsChilds.length && st.name) {
  940.  
  941. deps[st.name] = subsChilds;
  942.  
  943. childs = subsChilds.concat(childs);
  944.  
  945. }
  946.  
  947. }
  948.  
  949. return childs;
  950.  
  951. }
  952.  
  953. }
  954.  
  955.  
  956. var settingsValues = _log.get('settings') || {};
  957.  
  958. addSets(sets);
  959.  
  960. function drawSets(sets) {
  961.  
  962. var $sets = _elements.getTabInner('sets');
  963.  
  964. var html = '';
  965.  
  966. for (var i=0; i<sets.length;i++) {
  967.  
  968. var st = sets[i];
  969.  
  970. html +=
  971.  
  972. '<div class="">' +
  973.  
  974. '<div class="sets-header">' + (st.title||'') + '</div>' +
  975.  
  976. drawSetsGroup(st.fields) +
  977.  
  978. '</div>';
  979.  
  980. }
  981.  
  982. html = '<div class="settings-form form-horizontal">' + html + '</div>';
  983.  
  984. $sets.append(html);
  985.  
  986. $sets.find('input').each(function() {
  987.  
  988. var $input = $(this);
  989.  
  990. var name = $input.data('name');
  991.  
  992. var value = $input.is('[type="checkbox"]') ? $input.prop('checked') : $input.val();
  993.  
  994. var isDisabled = $input.closest('.input-wrap').hasClass('disabled')//.prop('disabled');
  995.  
  996. checkDependences(name, !isDisabled && value);
  997.  
  998. });
  999.  
  1000. }
  1001.  
  1002.  
  1003. function init() {
  1004.  
  1005. var $sets = _elements.getTabInner('sets');
  1006.  
  1007.  
  1008. drawSets(sets);
  1009.  
  1010.  
  1011. $sets
  1012.  
  1013. .on('change', 'input', function() {
  1014.  
  1015. var $input = $(this);
  1016.  
  1017. var name = $input.data('name');
  1018.  
  1019. var value = $input.is('[type="checkbox"]') ? $input.prop('checked') : $input.val();
  1020.  
  1021. if ($input.data('type') == 'num') { value =+value; }
  1022.  
  1023. settingsValues[name] = value;
  1024.  
  1025. var isDisabled = $input.closest('.input-wrap').hasClass('disabled')//.prop('disabled');
  1026.  
  1027. _ext.publish('settingsChange', name, value, isDisabled);
  1028.  
  1029. _log.set('settings', settingsValues);
  1030.  
  1031. })
  1032.  
  1033. .on('click', '#reset-stats', function() {
  1034.  
  1035. if (confirm('Будет сброшена вся история\nПродолжить?')) {
  1036.  
  1037. _ext.log.set('messagesLog', '');
  1038.  
  1039. _ext.trace.messagesLog.splice(0, _ext.trace.messagesLog.length);
  1040.  
  1041. _ext.log.set('archiveGroups', '');
  1042.  
  1043. _ext.archive.archiveGroups.splice(0, _ext.archive.archiveGroups.length);
  1044.  
  1045. _ext.stats.drawStatsSide();
  1046.  
  1047. _ext.elements.getTabInner('group').html('')
  1048.  
  1049. }
  1050.  
  1051. });
  1052.  
  1053. }
  1054.  
  1055.  
  1056. function drawSetsGroup(fields) {
  1057.  
  1058. // console.log('drawSetsGroup', sets);
  1059.  
  1060. var html = '';
  1061.  
  1062. for (var i=0;i<fields.length;i++) {
  1063.  
  1064. var st = fields[i];
  1065.  
  1066.  
  1067. var inputsHtml = '';
  1068.  
  1069. if (st.inputs) {
  1070.  
  1071. inputsHtml = ' ' + st.inputs.map(drawInput).join('');
  1072.  
  1073. }
  1074.  
  1075. var label =
  1076.  
  1077. '<div class="input-wrap ' + (st.isToggle ? 'checkbox' : '') + '">' +
  1078.  
  1079. '<label>' +
  1080.  
  1081. drawInput(st) +
  1082.  
  1083. st.label +
  1084.  
  1085. '</label>' +
  1086.  
  1087. inputsHtml +
  1088.  
  1089. '</div>';
  1090.  
  1091.  
  1092.  
  1093. html +=
  1094.  
  1095. '<div class="sets' + (st.isInline ? ' sets-inline' : '') + '">' +
  1096.  
  1097. '<div class="sets-title">' +
  1098.  
  1099. label +
  1100.  
  1101. (st.note ? '<div class="note-block">' + st.note + '</div>' : '') +
  1102.  
  1103. '</div>' +
  1104.  
  1105. (st.subs ? drawSetsGroup(st.subs) : '') +
  1106.  
  1107. '</div>';
  1108.  
  1109. }
  1110.  
  1111. _log.set('settings', settingsValues);
  1112.  
  1113.  
  1114. return html;
  1115.  
  1116.  
  1117. function drawInput(st) {
  1118.  
  1119. if (typeof settingsValues[st.name] === 'undefined') settingsValues[st.name] = st.value;
  1120.  
  1121. var html = '';
  1122.  
  1123. if (st.isToggle || st.type === 'checkbox') {
  1124.  
  1125. html = '<input type="checkbox" data-name="' + st.name + '""' + (settingsValues[st.name] ? ' checked' : '') + '> ';
  1126.  
  1127. } else if (st.type === 'text' || st.type === 'num') {
  1128.  
  1129. html = '<input type="text" class="input-tiny input-tiny-' + st.type + '"' +
  1130.  
  1131. ' data-name="' + st.name + '"' +
  1132.  
  1133. (' data-type="' + st.type + '"') +
  1134.  
  1135. (settingsValues[st.name] ? ' value="' + settingsValues[st.name] + '"' : '') + '>';
  1136.  
  1137. if (st.addOn) {
  1138.  
  1139. html = '<div class="input-wrap input-append">' + html + '<span class="add-on">' + st.addOn + '</span></div>';
  1140.  
  1141. } else {
  1142.  
  1143. html = '<div class="input-wrap input-wrap-inline">' + html + '</div>';
  1144.  
  1145. }
  1146.  
  1147. }
  1148.  
  1149. return html;
  1150.  
  1151. }
  1152.  
  1153. }
  1154.  
  1155.  
  1156. _ext.isMyName = function(str) {
  1157.  
  1158. return str.indexOf(_settings.settingsValues.heroNameStart) >= 0;
  1159.  
  1160. };
  1161.  
  1162.  
  1163.  
  1164. $.extend(_settings, {
  1165.  
  1166. init: init,
  1167.  
  1168. addSets: addSets,
  1169.  
  1170. drawSets: drawSets,
  1171.  
  1172. getSettingInput: getSettingInput,
  1173.  
  1174. settingsValues: settingsValues
  1175.  
  1176. });
  1177.  
  1178. return _settings;
  1179.  
  1180. })({});
  1181.  
  1182. _ext.subscribe('init', function() {
  1183.  
  1184. _settings.init();
  1185.  
  1186. });
  1187.  
  1188.  
  1189. _ext.subscribe('preload', function() {
  1190.  
  1191. if (!_settings.settingsValues.heroNameStart) {
  1192.  
  1193. var heroName = _ext.heroName;
  1194.  
  1195. _settings.settingsValues.heroNameStart = heroName.substring(0, Math.max(3, heroName.length-2));
  1196.  
  1197. _settings.getSettingInput('heroNameStart').val(_settings.settingsValues.heroNameStart).trigger('change');
  1198.  
  1199. }
  1200.  
  1201. });
  1202.  
  1203.  
  1204. _ext.subscribe('settingsChange', function(key, value) {
  1205.  
  1206. if (key === 'heroNameStart') {
  1207.  
  1208. _trace.traceInit();
  1209.  
  1210. _ext.groupMessages.drawMessages( _ext.groupMessages.list );
  1211.  
  1212. }
  1213.  
  1214. });
  1215.  
  1216.  
  1217. /* eo settings */
  1218.  
  1219.  
  1220.  
  1221. var lastNonifyMessagesText = '';
  1222.  
  1223. _ext.subscribe('newMessages', function(messagesNew, gameData) {
  1224.  
  1225. var hero = gameData.account.hero;
  1226.  
  1227. var _settingsValues = _settings.settingsValues;
  1228.  
  1229. if (!_settingsValues.notify) return;
  1230.  
  1231. var notifyMessages = [];
  1232.  
  1233. if (_settingsValues.notifyHeroHp) {
  1234.  
  1235. var health = hero.base.health;
  1236.  
  1237. // var healthMax = hero.base.max_health;
  1238.  
  1239. // var healthPercent = health / healthMax * 100;
  1240.  
  1241. var minHp = _settingsValues.notifyHeroHpLowerValue;
  1242.  
  1243. if (health < minHp) {
  1244.  
  1245. notifyMessages.push('Низкое здоровье: ' + health + ' HP');
  1246.  
  1247. }
  1248.  
  1249. }
  1250.  
  1251. if (_settingsValues.notifyHeroEnergy) {
  1252.  
  1253. var energy = hero.energy.value;
  1254.  
  1255. // var energyMax = hero.energy.max;
  1256.  
  1257. // var energyPercent = energy / energyMax * 100;
  1258.  
  1259. var maxEnergy = _settingsValues.notifyHeroEnergyGreaterValue;
  1260.  
  1261. if (energy > maxEnergy) {
  1262.  
  1263. notifyMessages.push('Энергия накопилась: ' + energy);
  1264.  
  1265. }
  1266.  
  1267. }
  1268.  
  1269. if (_settingsValues.notifyHeroIdle) {
  1270.  
  1271. var actionType = hero.action.type;
  1272.  
  1273. if (actionType === 0) {
  1274.  
  1275. notifyMessages.push('Герой бездействует');
  1276.  
  1277. }
  1278.  
  1279. }
  1280.  
  1281.  
  1282. if (notifyMessages.length) {
  1283.  
  1284. var notifyMessagesText = notifyMessages.join('\n');
  1285.  
  1286. if (notifyMessagesText != lastNonifyMessagesText) {
  1287.  
  1288. _notification.sendNotify('The Tale Extended - ' + _ext.heroName, {
  1289.  
  1290. tag: 'send',
  1291.  
  1292. body: notifyMessagesText
  1293.  
  1294. });
  1295.  
  1296. }
  1297.  
  1298. lastNonifyMessagesText = notifyMessagesText;
  1299.  
  1300. }
  1301.  
  1302. });
  1303.  
  1304.  
  1305.  
  1306.  
  1307.  
  1308.  
  1309. /* trace */
  1310.  
  1311. var _trace = (function(_trace) {
  1312.  
  1313.  
  1314. var messagesLog = _log.get('messagesLog') || [];
  1315.  
  1316. function traceInit() {
  1317.  
  1318. for (var i=0; i<messagesLog.length; i++) {
  1319.  
  1320. var messageNew = messagesLog[i];
  1321.  
  1322. messageNew[4] = _ext.parse.short(messageNew[2]) || false;
  1323.  
  1324. }
  1325.  
  1326. }
  1327.  
  1328.  
  1329. function traceData(game_data) {
  1330.  
  1331.  
  1332. var hero = game_data.account.hero;
  1333.  
  1334. if (!hero) return;
  1335.  
  1336. var heroData = {
  1337.  
  1338. action: {
  1339.  
  1340. description: hero.action.description,
  1341.  
  1342. type: hero.action.type,
  1343.  
  1344. info_link: hero.action.info_link,
  1345.  
  1346. is_boss: hero.action.is_boss,
  1347.  
  1348. percents: hero.action.percents
  1349.  
  1350. },
  1351.  
  1352. base: {
  1353.  
  1354. health: hero.base.health,
  1355.  
  1356. max_health: hero.base.max_health,
  1357.  
  1358. level: hero.base.level,
  1359.  
  1360. alive: hero.base.alive,
  1361.  
  1362. money: hero.base.money
  1363.  
  1364. },
  1365.  
  1366. energy: {
  1367.  
  1368. bonus: hero.energy.bonus,
  1369.  
  1370. max: hero.energy.max,
  1371.  
  1372. value: hero.energy.value
  1373.  
  1374. },
  1375.  
  1376. secondary: {
  1377.  
  1378. // initiative: hero.secondary.initiative,
  1379.  
  1380. loot_items_count: hero.secondary.loot_items_count,
  1381.  
  1382. // max_bag_size: hero.secondary.max_bag_size,
  1383.  
  1384. // move_speed: hero.secondary.move_speed,
  1385.  
  1386. power: hero.secondary.power
  1387.  
  1388. },
  1389.  
  1390. turn: game_data.turn.number
  1391.  
  1392. };
  1393.  
  1394.  
  1395. var lastLog = messagesLog[messagesLog.length-1] || [];
  1396.  
  1397. var lastTimestamp = lastLog[0];
  1398.  
  1399. var messagesPack = hero.messages;
  1400.  
  1401. messagesPack[messagesPack.length-1][3] = heroData;
  1402.  
  1403. var messagesPackTimestamp = messagesPack[messagesPack.length-1][0];
  1404.  
  1405. var messagesNew = [];
  1406.  
  1407. for (var i=0; i<messagesPack.length; i++) {
  1408.  
  1409. if (!lastTimestamp || messagesPack[i][0] > lastTimestamp) {
  1410.  
  1411. var messageNew = messagesPack[i];
  1412.  
  1413. // if (messagesPackTimestamp === messageNew[0]) messageNew[3] = heroData;
  1414.  
  1415. messageNew[3] = messageNew[3] || false;
  1416.  
  1417. messageNew[4] = _ext.parse.short(messageNew[2]) || false;
  1418.  
  1419.  
  1420. messagesLog.push(messageNew);
  1421.  
  1422. messagesNew.push(messageNew);
  1423.  
  1424. }
  1425.  
  1426. }
  1427.  
  1428. _log.set('messagesLog', messagesLog.map(function(item) { return [item[0], item[1], item[2], item[3]]; }));
  1429.  
  1430. _trace.heroData = heroData;
  1431.  
  1432.  
  1433. _ext.publish('newTurn', messagesNew, game_data, messagesPackTimestamp);
  1434.  
  1435. if (messagesNew.length) {
  1436.  
  1437. _ext.publish('newMessages', messagesNew, game_data, messagesPackTimestamp);
  1438.  
  1439. }
  1440.  
  1441. }
  1442.  
  1443. function traceStart() {
  1444.  
  1445. $(document).bind(pgf.game.events.DATA_REFRESHED + '.ext-trace', function(e, game_data){
  1446.  
  1447. traceData(game_data);
  1448.  
  1449. });
  1450.  
  1451. }
  1452.  
  1453. function traceStop() {
  1454.  
  1455. $(document).unbind(pgf.game.events.DATA_REFRESHED + '.ext-trace');
  1456.  
  1457. }
  1458.  
  1459.  
  1460.  
  1461. $.extend(_trace, {
  1462.  
  1463. messagesLog: messagesLog, //.slice(messagesLog.length - 500)
  1464.  
  1465. traceData: traceData,
  1466.  
  1467. traceInit: traceInit,
  1468.  
  1469. traceStart: traceStart,
  1470.  
  1471. traceStop: traceStop
  1472.  
  1473. });
  1474.  
  1475.  
  1476. return _trace;
  1477.  
  1478. })({});
  1479.  
  1480.  
  1481. _ext.subscribe('init', function(game_data) {
  1482.  
  1483. _trace.traceInit();
  1484.  
  1485. _trace.traceStart();
  1486.  
  1487. });
  1488.  
  1489. /* eo trace */
  1490.  
  1491.  
  1492. _ext.subscribe('newMessages', function(messagesNew, gameData, timestamp) {
  1493.  
  1494.  
  1495.  
  1496. _log.set('game_data', gameData);
  1497.  
  1498. var hero = gameData.account.hero;
  1499.  
  1500. var levelsLog = _log.get('levelsLog') || [];
  1501.  
  1502. var lastLevel = (levelsLog[levelsLog.length-1] || [])[1];
  1503.  
  1504. if (hero.base.level !== lastLevel) {
  1505.  
  1506. console.info('level up!', hero.base.level);
  1507.  
  1508. levelsLog.push([timestamp, hero.base.level]);
  1509.  
  1510. _log.set('levelsLog', levelsLog);
  1511.  
  1512. }
  1513.  
  1514. var powersLog = _log.get('powersLog') || [];
  1515.  
  1516. powersLog = powersLog.filter(function(t) { return typeof t[1] === 'number'; } );
  1517.  
  1518. var lastPower = (powersLog[powersLog.length-1] || [])[1];
  1519.  
  1520. var powerSum = hero.secondary.power[0] + hero.secondary.power[1];
  1521.  
  1522. if (powerSum !== lastPower) {
  1523.  
  1524. console.info('power up!', powerSum);
  1525.  
  1526. powersLog.push([timestamp, powerSum]);
  1527.  
  1528. _log.set('powersLog', powersLog);
  1529.  
  1530. }
  1531.  
  1532. _ext.trace.lastGameData = gameData;
  1533.  
  1534. });
  1535.  
  1536.  
  1537.  
  1538. _ext.subscribe('init', function() {
  1539.  
  1540. $('#pgf-journal-container').on('click', '[data-ts]', function() {
  1541.  
  1542. var timestamp = $(this).data('ts');
  1543.  
  1544. _trace.messagesLog.forEach(function(message) {
  1545.  
  1546. if (message[0] === timestamp) {
  1547.  
  1548. console.log(message);
  1549.  
  1550. }
  1551.  
  1552. });
  1553.  
  1554. });
  1555.  
  1556. $('#pgf-journal-container').on('click', '[data-click]', function() {
  1557.  
  1558. console.log($(this).data('click'));
  1559.  
  1560. });
  1561.  
  1562. });
  1563.  
  1564.  
  1565. $('body').on('click', '.group-toggle', function() {
  1566.  
  1567. $(this).closest('.group').toggleClass('open');
  1568.  
  1569. });
  1570.  
  1571.  
  1572.  
  1573.  
  1574. /* elements */
  1575.  
  1576. var _elements = (function(_elements) {
  1577.  
  1578. var tabs = {};
  1579.  
  1580.  
  1581. var $diaryRoot = $('#pgf-diary-container').parent().parent();
  1582.  
  1583. var $diaryTabs = $diaryRoot.children('.nav-tabs');
  1584.  
  1585. var $diaryContainer = $diaryRoot.children('.tab-content');
  1586.  
  1587.  
  1588. var $equipRoot = $('#pgf-equipment-container').parent().parent();
  1589.  
  1590. var $equipTabs = $equipRoot.children('.nav-tabs');
  1591.  
  1592. var $equipContainer = $equipRoot.children('.tab-content');
  1593.  
  1594.  
  1595. var zones = {
  1596.  
  1597. main: {
  1598.  
  1599. $tabs: $diaryTabs,
  1600.  
  1601. $container: $diaryContainer
  1602.  
  1603. },
  1604.  
  1605. equip: {
  1606.  
  1607. $tabs: $equipTabs,
  1608.  
  1609. $container: $equipContainer
  1610.  
  1611. }
  1612.  
  1613. };
  1614.  
  1615.  
  1616.  
  1617.  
  1618. function addTab(name, opts) {
  1619.  
  1620. var zone = opts.zone || 'main';
  1621.  
  1622. var $tabs = zones[zone].$tabs;
  1623.  
  1624. var $container = zones[zone].$container;
  1625.  
  1626.  
  1627. var $tab = $('<li class="pull-right"><a href="#pgf-'+name+'-container" class="pgf-'+name+'-tab-button" data-toggle="tab">'+opts.title+'</a></li>');
  1628.  
  1629. if (zone == 'main') {
  1630.  
  1631. var $content = $('<div class="tab-pane log-block" id="pgf-'+name+'-container"></div>');
  1632.  
  1633. } else {
  1634.  
  1635. var $content = $('<div class="tab-pane" id="pgf-'+name+'-container"></div>');
  1636.  
  1637. }
  1638.  
  1639.  
  1640. $tabs.append($tab);
  1641.  
  1642. $container.append($content);
  1643.  
  1644.  
  1645. var $inner = $content;
  1646.  
  1647. if (opts.content) {
  1648.  
  1649. $inner = $(opts.content).appendTo($content);
  1650.  
  1651. }
  1652.  
  1653.  
  1654. tabs[name] = {
  1655.  
  1656. zone: zone,
  1657.  
  1658. $tab: $tab,
  1659.  
  1660. $content: $content,
  1661.  
  1662. $inner: $inner
  1663.  
  1664. };
  1665.  
  1666. return $inner;
  1667.  
  1668. }
  1669.  
  1670. function getTabInner(name) {
  1671.  
  1672. return tabs[name].$inner;
  1673.  
  1674. }
  1675.  
  1676. function getTab(name) {
  1677.  
  1678. return tabs[name].$tab;
  1679.  
  1680. }
  1681.  
  1682.  
  1683. function activeTab(name) {
  1684.  
  1685. // tabs[name].$tab.addClass('active').siblings().removeClass('active');
  1686.  
  1687. // if (tabs[name].zone === 'mainMiddle') {
  1688.  
  1689. // $('#pgf-journal-container').addClass('active').siblings().removeClass('active');
  1690.  
  1691. // tabs[name].$content.show().siblings().hide();
  1692.  
  1693. // } else {
  1694.  
  1695. // tabs[name].$content.addClass('active').siblings().removeClass('active');
  1696.  
  1697. // }
  1698.  
  1699. }
  1700.  
  1701.  
  1702. var controls = {};
  1703.  
  1704. var $controls = $('<div class="ext-controls pull-right"></div>').insertAfter('#current-action-block, #pvp-info-block');
  1705.  
  1706. function addControl(name, opts) {
  1707.  
  1708. var html = '<span class="ext-control link-ajax" id="ext-' + name + '" title="' + opts.title + '">' + (opts.content || '') + '</span>';
  1709.  
  1710.  
  1711. var $el = $(html).appendTo($controls);
  1712.  
  1713. controls[name] = {
  1714.  
  1715. $el: $el
  1716.  
  1717. };
  1718.  
  1719. return $el;
  1720.  
  1721. }
  1722.  
  1723. function getControl(name) {
  1724.  
  1725. return controls[name].$el;
  1726.  
  1727. }
  1728.  
  1729.  
  1730. $.extend(_elements, {
  1731.  
  1732. addTab: addTab,
  1733.  
  1734. activeTab: activeTab,
  1735.  
  1736. getTab: getTab,
  1737.  
  1738. getTabInner: getTabInner,
  1739.  
  1740.  
  1741. addControl: addControl,
  1742.  
  1743. getControl: getControl
  1744.  
  1745. });
  1746.  
  1747. return _elements;
  1748.  
  1749. })({});
  1750.  
  1751.  
  1752. /* eo elements */
  1753.  
  1754.  
  1755. _elements.addTab('sets', {zone: 'main', title: '<span class="glyphicon glyphicon-cog" title="Настройки &laquo;The Tale Extended&raquo;"></span>'});
  1756.  
  1757. _elements.addTab('towns', {zone: 'main', title: 'города'});
  1758.  
  1759. _elements.addTab('archive', {zone: 'main', title: 'архив'});
  1760.  
  1761. _elements.addTab('group', {zone: 'main', title: 'кратко'});
  1762.  
  1763. _ext.subscribe('init', function() {
  1764.  
  1765. _elements.activeTab('group');
  1766.  
  1767. });
  1768.  
  1769.  
  1770.  
  1771. _elements.addControl('journal-log', {title: 'Журнал', content: '<span class="value"></span> <span class="glyphicon glyphicon-th-list"></span></span>'})
  1772.  
  1773. .on('click', function() { _log.toConsole('messagesLog'); });
  1774.  
  1775.  
  1776. _elements.addControl('archive-log', {title: 'Архив', content: '<span class="value"></span> <span class="glyphicon glyphicon-th"></span></span>'})
  1777.  
  1778. .on('click', function() { _log.toConsole('archiveGroups'); });
  1779.  
  1780.  
  1781.  
  1782. _ext.subscribe('newTurn', function(messagesNew) {
  1783.  
  1784. window.setTimeout(function() {
  1785.  
  1786. _ext.elements.getControl('journal-log')
  1787.  
  1788. .find('.value').text(_trace.messagesLog.length);
  1789.  
  1790.  
  1791. _ext.elements.getControl('archive-log')
  1792.  
  1793. .find('.value').text(_ext.archive.archiveGroups.length);
  1794.  
  1795.  
  1796. $('#storage-size')
  1797.  
  1798. .text('(занято ' + Math.round( _ext.log.size() / 1024 / 1024 * 100 )/100 + 'Мб)');
  1799.  
  1800. }, 10);
  1801.  
  1802. });
  1803.  
  1804.  
  1805.  
  1806. /* notifications */
  1807.  
  1808. var _notification = (function(_notification) {
  1809.  
  1810. var rndStr = Math.random() + '';
  1811.  
  1812.  
  1813. function request() {
  1814.  
  1815. if (Notification.permission.toLowerCase() !== 'granted') {
  1816.  
  1817. Notification.requestPermission( newMessage );
  1818.  
  1819. }
  1820.  
  1821. function newMessage(permission) {
  1822.  
  1823. if( permission !== "granted" ) return false;
  1824.  
  1825. var notify = new Notification("Thanks for letting notify you");
  1826.  
  1827. return true;
  1828.  
  1829. }
  1830.  
  1831. }
  1832.  
  1833. $('body').one('click', request);
  1834.  
  1835.  
  1836.  
  1837.  
  1838. function sendNotify(name, options) {
  1839.  
  1840. // if (!_settings.settingsValues.notify) return;
  1841.  
  1842.  
  1843. var d = new Date();
  1844.  
  1845. var h = d.getHours();
  1846.  
  1847. var m = d.getMinutes();
  1848.  
  1849. var time = h + ':' + (m<10?'0'+m:m);
  1850.  
  1851. var nt = new Notification(name, {
  1852.  
  1853. tag: options.tag + rndStr,
  1854.  
  1855. body: options.body + (options.addTime ? '\n' + time : ''),
  1856.  
  1857. icon: options.icon || (window.extPass + 'img/128.png')
  1858.  
  1859. });
  1860.  
  1861. nt.onclick = nt.onclose = function() {
  1862.  
  1863. rndStr = Math.random() + '';
  1864.  
  1865. };
  1866.  
  1867. }
  1868.  
  1869.  
  1870.  
  1871. $.extend(_notification, {
  1872.  
  1873. sendNotify: sendNotify
  1874.  
  1875. });
  1876.  
  1877. return _notification;
  1878.  
  1879.  
  1880. })({});
  1881.  
  1882.  
  1883.  
  1884. /* quests */
  1885.  
  1886. var _quests = (function(_quests) {
  1887.  
  1888. var lastQuests;
  1889.  
  1890. function checkQuests(quests) {
  1891.  
  1892. var line = quests[1].line;
  1893.  
  1894. var lineOld = lastQuests && lastQuests[1] && lastQuests[1].line || [];
  1895.  
  1896. var newLines = [];
  1897.  
  1898. for (var i=0; i<line.length; i++) {
  1899.  
  1900. var q = line[i];
  1901.  
  1902. var qOld = lineOld[i];
  1903.  
  1904. if (!qOld || !isSameQuest(q, qOld)) {
  1905.  
  1906.  
  1907. if (_settings.settingsValues.notifyQuestChoose && q.choice_alternatives && q.choice_alternatives.length) {
  1908.  
  1909. // console.info('quest!', q.type, _ext.heroName + ' ' + q.action + '!', q);
  1910.  
  1911. if (_settings.settingsValues.notify) {
  1912.  
  1913. _notification.sendNotify(q.name, {
  1914.  
  1915. tag: 'quest',
  1916.  
  1917. body: _ext.heroName + ' ' + q.action + '!',
  1918.  
  1919. icon: window.extPass + 'img/quest/caravan.png', //window.extPass + 'img/quest/' + q.type + '.png',
  1920.  
  1921. addTime: 1
  1922.  
  1923. });
  1924.  
  1925. }
  1926.  
  1927. } else {
  1928.  
  1929. // console.info('quest.', q.type, q.experience, q.power, _ext.heroName + ' ' + q.action + '!', q);
  1930.  
  1931. }
  1932.  
  1933. newLines.push(q);
  1934.  
  1935.  
  1936. }
  1937.  
  1938. }
  1939.  
  1940. if (newLines.length) {
  1941.  
  1942. _ext.publish('questUpdate', line);
  1943.  
  1944. }
  1945.  
  1946. lastQuests = quests;
  1947.  
  1948. }
  1949.  
  1950.  
  1951. function isSameQuest(q1,q2) {
  1952.  
  1953. var tests = ['action', 'choice', 'name', 'type', 'uid'];
  1954.  
  1955. for (var s in tests) if (tests.hasOwnProperty(s)) {
  1956.  
  1957. var key = tests[s];
  1958.  
  1959. if (q1[key] !== q2[key]) return false;
  1960.  
  1961. }
  1962.  
  1963. return true;
  1964.  
  1965. }
  1966.  
  1967.  
  1968. $.extend(_quests, {
  1969.  
  1970. checkQuests: checkQuests
  1971.  
  1972. });
  1973.  
  1974. return _quests;
  1975.  
  1976. })({});
  1977.  
  1978.  
  1979. _ext.subscribe('preload', function() {
  1980.  
  1981. _ext.subscribe('newMessages', function(messagesNew, game_data) {
  1982.  
  1983. var hero = game_data.account.hero;
  1984.  
  1985. var quests = hero.quests.quests;
  1986.  
  1987. _quests.checkQuests(quests);
  1988.  
  1989. });
  1990.  
  1991. });
  1992.  
  1993. /* questss */
  1994.  
  1995.  
  1996.  
  1997.  
  1998. /* utils functions */
  1999.  
  2000.  
  2001. var _utils = {};
  2002.  
  2003. _utils.capitalize = function(text) {
  2004.  
  2005. return text.substring(0,1).toUpperCase() + text.substring(1);
  2006.  
  2007. };
  2008.  
  2009. _utils.declenstionByNumber = function(number, titles, addCount) {
  2010.  
  2011. return (addCount ? number + ' ' : '' ) + titles[ (number%100>4 && number%100<20)? 2 : [2, 0, 1, 1, 1, 2][(number%10<5)?number%10:5] ];
  2012.  
  2013. };
  2014.  
  2015. _utils.timeSpan = function(timeSpan) {
  2016.  
  2017. timeSpan = timeSpan|0;
  2018.  
  2019. var h = (timeSpan / 60 / 60)|0;
  2020.  
  2021. var m = ((timeSpan / 60)|0) % 60;
  2022.  
  2023. var s = timeSpan % 60;
  2024.  
  2025. return (h ? h + ':' : '') + (m<10?'0':'') + m + ':' + (s<10?'0':'') + s ;
  2026.  
  2027. };
  2028.  
  2029.  
  2030. /* eo utils functions */
  2031.  
  2032.  
  2033.  
  2034. _ext.store = _store;
  2035.  
  2036. _ext.settings = _settings;
  2037.  
  2038. _ext.elements = _elements;
  2039.  
  2040. _ext.quests = _quests;
  2041.  
  2042. _ext.notification = _notification;
  2043.  
  2044. _ext.utils = _utils;
  2045.  
  2046. _ext.elements = _elements;
  2047.  
  2048. _ext.log = _log;
  2049.  
  2050. _ext.trace = _trace;
  2051.  
  2052.  
  2053.  
  2054. var _parse = (function(_parse) {
  2055.  
  2056.  
  2057. var parseShortRaw = [
  2058.  
  2059. /* --pvp-- */
  2060.  
  2061. '[pvpeff]Мощный поток энергии вздыбил землю вокруг {actor}. Эффективность {} увеличена на {eff} единиц(а|у|ы).',
  2062.  
  2063. '[pvpeff]Кровь на мгновение застилает глаза {actor}. Эффективность увеличена на {eff} единиц(а|у|ы).',
  2064.  
  2065. '[pvpeff]{actor} почувствовал(а|о|и) как (его|ее|их) мышцы крепчают, а сознание очищается. Эффективность боя увеличилась на {eff} единиц(а|у|ы).',
  2066.  
  2067. '[pvpeff]Хранитель усилил связь с {actor} и увеличил (его|ее|их) эффективность на {eff} единиц(а|у|ы).',
  2068.  
  2069.  
  2070. '[pvpice]Хранитель {actor} сконцентрировался и увеличил скорость прироста энергии.',
  2071.  
  2072. '[pvpice]Холодная голова и кристально чистые мысли помогли хранителю {actor} увеличить скорость прироста энергии.',
  2073.  
  2074.  
  2075. '[pvpflame]Хранитель {actor} нарушил(а|о|и) концентрацию противник(а|ов) и уменьшил скорость прироста (его|ее|их) энергии.',
  2076.  
  2077.  
  2078. '[pvpfail]Тучи за спиной {actor} сошлись… И снова разошлись, ничего не произошло.',
  2079.  
  2080. '[pvpfail]Хранитель {actor} не смог сконцентрироваться и потратил энергию впустую.',
  2081.  
  2082. '[pvpfail]Хранителю не хватило концентрации для установления связи с {actor}. Энергия ушла в никуда.',
  2083.  
  2084.  
  2085.  
  2086. /* Удар */
  2087.  
  2088. '[hit]Изучив тактику {victim}, {actor} провел(а|о|и) атаку и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2089.  
  2090. '[hit]Сосредоточившись, {actor} провел(а|о|и) атаку и лишил(а|о|и) {victim} {value} HP.',
  2091.  
  2092. '[hit]Обманув {victim}, {actor} смог(ла|ло|ли) провести неожиданную атаку, лишив противник(а|ов) {value} HP.',
  2093.  
  2094. '[hit]Усыпив бдительность {victim}, {actor} набросил(ся|ась|ось|ись) на противник(а|ов) и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2095.  
  2096. '[hit]{actor} нанес(ла|ло|ли) точный удар, уменьшая здоровье {victim} на {value} HP.',
  2097.  
  2098. '[hit]Апперкот, и вокруг {victim} взметнул(ась|ись) {value} звездоч(ка|ки|ек).',
  2099.  
  2100. '[hit]Быстрая атака {actor} лишила {victim} {value} HP.',
  2101.  
  2102. '[hit]{actor} виртуозно провел(а|о|и) атаку и нанес(ла|ло|ли) {victim} {value} единиц(у|а|ы) урона.',
  2103.  
  2104. '[hit]Рассекающий удар {actor} нан(ёс|есла|есло|если) {value} единиц(у|а|ы) урона {victim}.',
  2105.  
  2106. '[hit]Стоило только {victim} замахнуться, как {actor} резко ударил(а|о|и) (его|ее|их) по больному месту и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2107.  
  2108. '[hit]Умелая подсечка, и придорожный камень нанёс голов(е|ам) {victim} {value} урона.',
  2109.  
  2110. '[hit]Ловко парировав удар, {actor} провод(ит|ят) успешный контрудар, лишая {victim} {value} HP.',
  2111.  
  2112. '[hit]{actor} поцарапал(а|о|и) {victim} на {value} HP.',
  2113.  
  2114. '[hit]{actor} быстро прицелил(ся|ась|ось|ись), изготовил(ся|ась|ось|ись) и в стремительном рывке обрушил(а|о|и) разящий удар в {value} единиц(у|а|ы) урона на {victim}.',
  2115.  
  2116. '[hit]{victim} пропустил(а|о|и) удар и теперь страда(ет|ют) отсутствием {value} HP.',
  2117.  
  2118. '[hit]{victim} пропуска(ет|ют) лёгкий тычок в {value} единиц(у|ы) урона.',
  2119.  
  2120. '[hit]Быстрота реакции {actor} помогла (ему|ей|им) нанести {victim} {value} единиц(у|а|ы) урона.',
  2121.  
  2122. '[hit]{actor} отош(ел|ла|ло|ли) в сторону, и {victim} с разбега врезал(ся|ась|ось|ись) в стену, потеряв {value} HP.',
  2123.  
  2124. '[hit]{actor} поднатужил(ся|ась|ось|ись) и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2125.  
  2126. '[hit]С кошачьей грацией {actor} извернул(ся|ась|ось|ись) и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2127.  
  2128. '[hit]{actor} точным ударом поразил(а|о|и) {victim} на {value} единиц(у|а|ы) урона.',
  2129.  
  2130. '[hit]{victim} потерял(а|о|и) {value} HP, пропустив атаку {actor}.',
  2131.  
  2132. '[hit]«Скрытая атака!» — заорал(а|о|и) {actor} и выбил(а|о|и) из {victim} {value} HP.',
  2133.  
  2134. '[hit]Замахиваясь, {victim} пропустил(а|о|и) резкий и быстрый выпад {actor}, забравший {value} HP.',
  2135.  
  2136. '[hit]Молниеносная атака {actor} нанесла {value} единиц(у|а|ы) урона.',
  2137.  
  2138. '[hit]После обмена серией стремительных ударов с {victim}, {actor} наконец наш(ел|ла|ло|ли) брешь в защите и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2139.  
  2140. '[hit]Кувырком уходя из-под удара и контратакуя, {actor} сумел(а|о|и) нанести {victim} {value} единиц(у|а|ы) урона.',
  2141.  
  2142. '[hit]{victim} приготовил(ся|ась|ось|ись) парировать удар, но не ожидал(а|о|и) от {actor} такой стремительности и теря(ет|ют) {value} HP.',
  2143.  
  2144. '[hit]{victim} не ожидал(а|о|и) такого выпада от {actor} и получа(ет|ют) {value} единиц(у|а|ы) урона.',
  2145.  
  2146. '[hit]{actor} провод(ит|ят) сногсшибательный приём, {victim} пада(ет|ют) и теря(ет|ют) {value} HP.',
  2147.  
  2148. '[hit]Вложив всю злость в удар, {actor} нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2149.  
  2150. '[hit]{actor} поднатужил(ся|ась|ось|ись) и нанес(а|о|и) {value} единиц(у|а|ы) урона.',
  2151.  
  2152. '[hit]Издавая зловещие звуки, {actor} сжима(ет|ют) {} что есть силы в своих объятиях! {victim} кряхт(ит|ят) и получа(ет|ют) {value} единиц(у|а|ы) урона.',
  2153.  
  2154. '[hit]{victim} не ожидал(а|о|и) такого выпада от {actor}, и получа(е|ю)т {value} единиц(у|а|ы) урона.',
  2155.  
  2156. '[hit]{victim} не ожидал(а|о|и) такого выпада от {actor}, и получил(а|о|и) {value} единиц(у|ы) урона.',
  2157.  
  2158. '[hit]Вложив всю злость в удар, {actor} нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2159.  
  2160. '[hit]{actor} точным ударом поразил(а|о|и) {victim} на {value} единиц(у|ы) урона.',
  2161.  
  2162. '[hit]{actor} провел(а|о|и) сногсшибательный прием, {victim} пада(ет|ют) и теря(ет|ют) {value} HP.',
  2163.  
  2164. '[hit]Усыпив бдительность {victim}, {actor} набросил(ся|ась|ось|ись) на противника и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2165.  
  2166. '[hit]После обмена серией стремительных ударов с {victim}, {actor} наконец наш(ел|ёл|ла|ло|ли) брешь в защите и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2167.  
  2168. '[hit]Рассекающий удар {actor} нанес {value} единиц(у|ы) урона {victim}.',
  2169.  
  2170. '[hit]Кувырком уходя из-под удара и контратакуя, {actor} сумел(а|о|и) нанести {victim} {value} единиц(у|ы) урона.',
  2171.  
  2172. '[hit]«Скрытая атака!» — заорал(а|о|и) {actor} и выбил(а|о|и) из {victim} {value} HP.',
  2173.  
  2174. '[hit]Ловко парировав удар, {actor} провел(а|о|и) успешный контрудар, лишая {victim} {value} HP.',
  2175.  
  2176. '[hit]Издавая зловещие звуки, {actor} сжима(ет|ют) {victim} что есть силы в своих объятиях! {victim} кряхт(ит|ят) и получа(ет|ют) {value} единиц(у|ы) урона.',
  2177.  
  2178. '[hit]Замахиваясь, {victim} пропустил(а|о|и) резкий и быстрый выпад {actor}, забравший {value} HP.',
  2179.  
  2180. '[hit]{victim} приготовил(ся|ась|ось|ись) парировать удар, но не ожидал(а|о|и) от {actor} такой стремительности и теря(ет|ют) {value} HP.',
  2181.  
  2182. '[hit]Стоило только {victim} замахнуться, как {actor} резко ударил(а|о|и) {victim} по больному месту и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2183.  
  2184. '[hit]{victim} пропуска(ет|ют) легкий тычок в {value} единиц(у|ы) урона.',
  2185.  
  2186. '[hit]Сосредоточившись, {actor} провел(а|о|и) атаку и лишил(а|о|и) {victim} {value} HP.',
  2187.  
  2188. '[hit]Быстрая атака {actor} лишила {victim} {value} HP.',
  2189.  
  2190. '[hit]Обманув {victim}, {actor} смог(ла|ло|ли) провести неожиданную атаку, лишив {victim} {value} HP.',
  2191.  
  2192. '[hit]{victim} потерял(а|о|и) {value} HP, пропустив атаку {actor}.',
  2193.  
  2194. '[hit]{actor} нанес(ла|ло|ли) точный удар, уменьшая здоровье {victim} на {value} HP.',
  2195.  
  2196. '[hit]{actor} виртуозно провел(а|о|и) атаку и нанес(ла|ло|ли) {victim} {value} единиц(у|ы) урона.',
  2197.  
  2198. '[hit]Молниеносная атака {actor} нанесла {value} единиц(у|ы) урона.',
  2199.  
  2200. '[hit]Быстрота реакции {actor} помогла (ему|ей|ему|им) нанести {victim} {value} единиц(у|ы) урона.',
  2201.  
  2202. '[hit]Изучив тактику {victim}, {actor} провел(а|о|и) атаку и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2203.  
  2204. '[hit]С кошачьей грацией {actor} извернул(ся|ась|ось|ись) и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2205.  
  2206. '[hit]{actor} быстро прицелил(ся|ась|ось|ись), изготовил(ся|ась|ось|ись) и в стремительном рывке обрушил(а|о|и) разящий удар в {value} единиц(у|ы) урона на {victim}.',
  2207.  
  2208. '[hit]{victim} пропустил(а|о|и) удар и теперь страда(ет|ют) отсутствием {value} HP.',
  2209.  
  2210. '[hit]{actor} отош(ел|ёл|ла|ло|ли) в сторону, и {victim} с разбега врезал(ся|ась|ось|ись) в стену, потеряв {value} HP.',
  2211.  
  2212. '[hit]{actor} поцарапал(а|о|и) {victim} на {value} HP.',
  2213.  
  2214. '[hit]Апперкот, и вокруг {victim} взметнул(ась|ось|ись) {value} звездоч(ка|ки|ек).',
  2215.  
  2216. '[hit]Умелая подсечка, и придорожный камень нанес голов(е|ам) {victim} {value} урона.',
  2217.  
  2218. '[hit]{actor} поднатужил(ся|ась|ось|ись) и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2219.  
  2220. '[hit]{actor} ударил(а|о|и) так, что {value} НР {victim} как и не бывало.',
  2221.  
  2222.  
  2223.  
  2224. /* Тяжёлый удар */
  2225.  
  2226. '[might]Как следует размахнувшись, {actor} одним сильным ударом вынес(ла|ло|ли) из {victim} {value} HP.',
  2227.  
  2228. '[might]Вложив всю свою силу в удар, {actor} нанес(ла|ло|ли) {victim} {value} единиц(у|а|ы) урона.',
  2229.  
  2230. '[might]После короткого и очень сильного удара {actor} по телу {victim} прокатилась волна боли на {value} единиц(у|а|ы) урона.',
  2231.  
  2232. '[might]Сильнейший удар обрушил(а|о|и) {actor}, выколачивая из {victim} {value} HP.',
  2233.  
  2234. '[might]{victim} потерял(а|о|и) {value} HP от чрезвычайно сильного удара {actor}.',
  2235.  
  2236. '[might]Собрав всю злобу и силу в один удар, {actor} уменьшил(а|о|и) здоровье {victim} на {value} HP.',
  2237.  
  2238. '[might]{actor} размахнул(ся|ась|ось|ись) сильнее обычного и нанес(ла|ло|ли) {value} единиц(у|а|ы) урона.',
  2239.  
  2240. '[might]Взяв размах от плеча, {actor} сокрушительным ударом выбил(а|о|и) из {victim} {value} HP.',
  2241.  
  2242. '[might]Затаив злобу лютую, с особой жестокостью ударил {actor} {victim} на {value} единиц(у|а|ы) урона.',
  2243.  
  2244.  
  2245. /* Отравление */
  2246.  
  2247. '[poison]Яд тянет из {victim} силы и {value} HP.',
  2248.  
  2249. '[poison]{victim} чувству(ет|ют) себя всё хуже. Отравление съедает {value} HP.',
  2250.  
  2251. '[poison]Отрава вытягивает из {victim} {value} HP.',
  2252.  
  2253. '[poison]Яд неумолимо отнимает здоровье {victim}. Еще {value} HP потрачен(о|ы) на борьбу с отравой.',
  2254.  
  2255. '[poison]Чувствуя действие яда всё сильнее, {victim} теря(е|ю)т {value} HP.',
  2256.  
  2257. '[poison]{victim} теря(е|ю)т {value} HP от отравления.',
  2258.  
  2259. '[poison]Головокружение и тошнота — не самое страшное, куда серьёзней {victim} показалась потеря {value} HP.',
  2260.  
  2261. '[poison]Отрава вытягивает из {victim} {value} HP.',
  2262.  
  2263. '[poison]{victim} отравлен(а|о|ы) и теря(е|ю)т {value} единиц(у|ы) здоровья.',
  2264.  
  2265. '[poison]{victim} страда(е|ю)т от отравления, получая {value} единиц(у|ы) урона.',
  2266.  
  2267. '[poison]Медленно убивая {victim} изнутри, отравление наносит {value} единиц(у|ы) урона.',
  2268.  
  2269. '[poison]Сильное отравление лишает {victim} {value} HP.',
  2270.  
  2271. '[poison]Нанося {victim} {value} единиц(у|ы) урона, отравление грозит (ему|ей|им) летальным исходом.',
  2272.  
  2273. '[poison]Охваченн(ый|ая|ое|ые) ядовитыми парами, {victim} теряет {value} HP.',
  2274.  
  2275. '[poison]Смерть подступает всё ближе, и {victim} теря(е|ю)т от отравления {value} HP.',
  2276.  
  2277. '[poison]Яд растекается по венам {victim}. Он(а|о|и) теря(е|ю)т {value} HP.',
  2278.  
  2279. '[poison]{victim} задыха(е|ю)тся и кашля(е|ю)т в созданном {actor} ядовитом облаке.',
  2280.  
  2281.  
  2282. /* Ядовитое облако */
  2283.  
  2284. '[poisoncloud]{victim} оказал(ся|ась|ось|ись) в созданном {actor} облаке яда.',
  2285.  
  2286. '[poisoncloud]Быстро создав ядовитое облако, {actor} окутал(а|о|и) им {victim}.',
  2287.  
  2288. '[poisoncloud]{actor} создал(а|о|и) вокруг {victim} ядовитое облако.',
  2289.  
  2290. '[poisoncloud]{victim} задыха(е|ю)тся и кашляет в созданном {actor} ядовитом облаке.',
  2291.  
  2292. '[poisoncloud]Ядовитое облако, сотворённое {actor}, окутало {victim}.',
  2293.  
  2294. '[poisoncloud]Ядовитое облако, созданное {actor}, обволокло {victim}.',
  2295.  
  2296. '[poisoncloud]{actor} создал(а|о|и) странный пар вокруг {victim} и (тот|та|то|те) начал(а|о|и) задыхаться.',
  2297.  
  2298. '[poisoncloud]{actor} создал(а|о|и) ядовитое облако вокруг {victim}.',
  2299.  
  2300.  
  2301.  
  2302. /* Удар вампира */
  2303.  
  2304. '[vamp]Жаждя крови {victim}, {actor} вонзил(а|о|и) зубы в (его|ее|их) тело. От такой атаки {victim} потерял(а|о|и) {value} HP, а {} приобрел(а|о|и) {vamp} HP.',
  2305.  
  2306. '[vamp]Вконец остервенев от злости, {actor} вцепил(ся|ась|ось|ись) зубами в {victim} и нанес(ла|ло|ли) (ему|ей|им) {value} единиц(у|ы) урона, при этом восстановив {vamp} HP.',
  2307.  
  2308. '[vamp]{actor} вспомнил(а|о|и), что из оружия у (него|нее|ней|них) есть ещё и зубы. Вцепившись ими в {victim}, он(а|о|и) восстановил(а|о|и) {vamp} HP и нанес(ла|ло|ли) противник(у|ам) {value} единиц(у|ы) урона.',
  2309.  
  2310. '[vamp]Хорошо прицелившись, {actor} прыгнул(а|о|и) и впил(ся|ась|ось|ись) зубами в {victim}, нанеся (ему|ей|им) {value} единиц(у|ы) урона и восстановив себе {vamp} HP.',
  2311.  
  2312. '[vamp]Выждав подходящий момент для сложной атаки, {actor} с силой грызанул(а|о|и) {victim} и, почувствовав (его|ее|их) кровь, восстановил(а|о|и) {vamp} HP. От укуса {} потерял(а|о|и) {value} HP.',
  2313.  
  2314. '[vamp]{actor} зубами впил(ся|ась|ось|ись) в {victim}, нанес(ла|ло|ли) {value} единиц(у|а|ы) урона и восстановил(а|о|и) {vamp} HP.',
  2315.  
  2316. '[vamp]Даже самая толстая кожа не спасла бы от укуса {actor}. {victim} проща(е|ю)тся с {value} HP, из которых {vamp} HP переход(и|я)т к {}.',
  2317.  
  2318. '[vamp]{actor} вытянул(а|о|и) из {} {value} HP и излечил(а|о|и) свои раны на {vamp} HP.',
  2319.  
  2320.  
  2321.  
  2322. /* Шаг в сторону */
  2323.  
  2324. '[ue]Серией быстрых движений {actor} дезориентиру(ет|ют) {victim}.',
  2325.  
  2326. '[ue]{actor} ловко уш(ел|ла|ло|ли) в сторону от {victim}.',
  2327.  
  2328. '[ue]{actor} мастерски уш(ел|ла|ло|ли) от атаки {victim}.',
  2329.  
  2330.  
  2331. /* Увернулся */
  2332.  
  2333. '[eva]{actor} носит(ся|ась|ось|ись) кругами вокруг {victim}, вынуждая (его|ее|их) промахнуться.',
  2334.  
  2335. '[eva]Отточенным движением {actor} уш(ел|ла|ло|ли) из-под удара.',
  2336.  
  2337. '[eva]{actor} увернул(ся|ась|ось|ись) от {victim} и (его|ее|их) чудовищной силы удара.',
  2338.  
  2339. '[eva]{actor} увернул(ся|ась|ось|ись) от удара.',
  2340.  
  2341. '[eva]{actor} резко качнул(ся|ась|ось|ись) в сторону и {victim} промахнул(ся|ась|ось|ись).',
  2342.  
  2343. '[eva]{actor} провел(а|о|и) обманное движение, и {victim} промазал(а|о|и).',
  2344.  
  2345.  
  2346. '[eva]Солнце ослепило {victim}, и (тот|та|то|те) промахнул(ся|ась|ось|ись) по {actor}.',
  2347.  
  2348. '[eva]{victim} слишком долго собирал(ся|ась|ось|ись) с мыслями, и удар пришёлся в никуда.',
  2349.  
  2350. '[eva]{victim} «крякнул(а|о|и)» и промахнул(ся|ась|ось|ись).',
  2351.  
  2352. '[eva]Перемудрив с обманными движениями, {victim} промахнул(ся|ась|ось|ись).',
  2353.  
  2354. '[eva]{victim} поднатужил(ся|ась|ось|ись)… и промахнул(ся|ась|ось|ись)…',
  2355.  
  2356. '[eva]{victim} промахнул(ся|ась|ось|ись)! {victim} промахнул(ся|ась|ось|ись)!',
  2357.  
  2358. '[eva]{actor} вовремя подал(ся|ась|ось|ись) назад, и {victim} промахнул(ся|ась|ось|ись).',
  2359.  
  2360. '[eva]{actor} успел(а|о|и) отбежать подальше, пока {victim} замахивал(ся|ась|ось|ись).',
  2361.  
  2362. '[eva]Слишком сильно размахнувшись, {victim} сбил(ся|ась|ось|ись) с темпа и не попал(а|о|и) по {actor}.',
  2363.  
  2364. '[eva]Сильнейший удар {victim} со свистом рассёк воздух, даже не задев {actor}.',
  2365.  
  2366. '[eva]Попавшая в глаз {victim} соринка сорвала атаку, и он(а|о|и) промахнул(ся|ась|ось|ись).',
  2367.  
  2368. '[eva]{victim}, готовясь к сильному удару, совсем забыл(а|о|и) о точности и промахнул(ся|ась|ось|ись).',
  2369.  
  2370. '[eva]Явно считая, что сила есть, а точности не надо, {victim} промахнул(ся|ась|ось|ись) по {actor}.',
  2371.  
  2372. '[eva]{victim} крякнул(а|о|и) и промахнул(ся|ась|ось|ись).',
  2373.  
  2374. '[eva]Прыжок, кувырок, перекат, снова перекат... Отскок! {actor} так верт(и|я)тся, что {victim} никак не мо(жет|гут) попасть по (нему|ней|ним).',
  2375.  
  2376.  
  2377. '[eva,fire]Слишком долго готовясь к атаке, {victim} упустил(а|о|и) подходящий момент, и (его|ее|их) огненный шар не попал в цель.',
  2378.  
  2379. '[eva,fire]{victim} запустил(а|о|и) шар огня в {}, но не тут-то было! {actor} виртуозно уклонил(ся|ась|ось|ись) от пламени.',
  2380.  
  2381. '[eva,fire]{actor} завороженно проводил(а|о|и) взглядом пролетевший мимо огненный шар.',
  2382.  
  2383. '[eva,stunHit]{victim} разбежал(ся|ась|ось|ись) и пронес(ся|лась|лось|лись) мимо {actor}.',
  2384.  
  2385. '[eva,stunHit]{victim} бросил(ся|ась|ось|ись) в сторону {actor}, но увы, проскочил(а|о|и) мимо.',
  2386.  
  2387. '[eva,vamp]{victim} клацнул(а|о|и) зубами рядом с {actor}, не сумев вцепиться в (него|нее|них).',
  2388.  
  2389. '[eva,vamp]{actor} ускользнул(а|о|и) от зубов {victim}.',
  2390.  
  2391. '[eva,vamp]{victim} укусил(а|о|и) себя вместо {actor}.',
  2392.  
  2393. '[eva,vamp]Слишком сильно открыв рот для укуса, {victim} ощутил(а|о|и) боль в челюсти и не смог(ла|ло|ли) укусить {actor}.',
  2394.  
  2395. '[eva,vamp]{victim} попытал(ся|ась|ось|ись) укусить {actor}, но промахнул(ся|ась|ось|ись).',
  2396.  
  2397. '[eva,vamp]{victim} раззявил(а|о|и) рот, но подходящий для укуса момент уже был упущен.',
  2398.  
  2399. '[eva,vamp]{victim} промахнул(ся|ась|ось|ись), бесполезно клацнув зубами.',
  2400.  
  2401.  
  2402. /* Заморозка */
  2403.  
  2404. '[slow]Легко, почти непринуждённо, {actor} охлажда(ет|ют) воздух вокруг {victim}, сковывая (его|ее|их) движения.',
  2405.  
  2406. '[slow]{actor}, сосредоточившись, ударил(а|о|и) {victim} потоком липкого холодного воздуха, сковав (его|ее|их) члены и замедлив движения.',
  2407.  
  2408. '[slow]Собравшись, {actor} направля(ет|ют) инеевые искры в сторону врага. Воздух вокруг {victim} замерзает и густеет, замедляя (его|ее|их) движения.',
  2409.  
  2410. '[slow]{actor} взывает к зимним ветрам и {victim} окутывает ужасный холод, (его|ее|их) движения замедляются.',
  2411.  
  2412. '[slow]Призывая в помощь зимние ледяные ветра, {actor} бьет холодом, и {victim}, покрываясь инеем, замедля(ется|ются).',
  2413.  
  2414. '[slow]{actor} встряхнул(а|о|и) кисти рук и на кончиках (его|ее|их) пальцев образовались ледяные иглы. Сконцентрировавшись, (он|она|оно|они) нанес(ла|ло|ли) удар холодом. Движения {victim} замедлились.',
  2415.  
  2416. '[slow]Собравшись, {actor} направляет инеевые искры в сторону враг(а|ов). Воздух вокруг {victim} замерзает и густеет, замедляя (его|ее|их) движения.',
  2417.  
  2418. '[slow]{actor} сделал(а|о|и) глубокий вдох и резко выдохнул(а|о|и). {victim} ощутил(а|о|и) как идёт мороз по коже и замедляются (его|ее|их) движения.',
  2419.  
  2420. '[slow]{actor} вызыва(е|ю)т лютую стужу и направля(е|ю)т холод в сторону {victim}, замедляя (его|ее|их) движения.',
  2421.  
  2422.  
  2423. /* Ускорение */
  2424.  
  2425. '[speed]Скорость движений {actor} сильно возросла после того как он(а|о|и) сделал(а|о|и) магический пасс.',
  2426.  
  2427. '[speed]{actor} использу(е|ю)т скрытые резервы организма и заметно ускоря(е|ю)тся.',
  2428.  
  2429. '[speed]{actor} перенаправля(е|ю)т свою ци в нижнее русло и начина(е|ю)т двигаться быстрее.',
  2430.  
  2431. '[speed]{actor} замыка(е|ю)т на себе потоки магии и начина(е|ю)т двигаться значительно быстрее.',
  2432.  
  2433. '[speed]{actor} концентрируется и скорость (его|ее|их) движений возрастает.',
  2434.  
  2435. '[speed]Обуздав стихию ветра, {actor} ускоря(е|ю)т движения.',
  2436.  
  2437. '[speed]Воспользовавшись магией, {actor} начина(е|ю)т двигаться невероятно быстро.',
  2438.  
  2439. '[speed]{actor} сотворя(е|ю)т сложный знак и значительно ускоря(е|ю)тся.',
  2440.  
  2441. '[speed]Сделав сложный пасс, {actor} начина(е|ю)т двигаться быстрее.',
  2442.  
  2443.  
  2444. /* Волшебный гриб */
  2445.  
  2446. '[mush]«Волшебный гриб!» — воскликнул(а|о|и) {actor}, запихивая в рот мухомор.',
  2447.  
  2448. '[mush]Откуда-то {actor} достал(а|о|и) и отправил(а|о|и) в рот волшебный синий гриб.',
  2449.  
  2450. '[mush]{actor} случайно съел(а|о|и) обычный шампиньон, однако, благодаря самовнушению, все-равно получил(а|о|и) приток сил.',
  2451.  
  2452.  
  2453. /* Шар огня */
  2454.  
  2455. '[fire]{actor} броса(ет|ют) огненный шар в {victim} и сжига(ет|ют) (ему|ей|им) {value} HP.',
  2456.  
  2457. '[fire]{actor} кида(ет|ют) огненный шар, опалив {victim} и нанеся {value} единиц(у|ы) урона (его|ее|их) здоровью.',
  2458.  
  2459. '[fire]{actor} концентриру(ет|ют) магическую энергию и выпуска(ет|ют) пылающий огненный шар. {victim} получа(ет|ют) урон в {value} HP и сильные ожоги.',
  2460.  
  2461. '[fire]{actor} выпустил(а|о|и) два огненных шара, но {victim} ловко увернул(ся|ась|ось|ись) и уже готовил(ся|ась|ось|ись) к удару, как оба шара, закрутившись по спирали, развернулись и врезались (ему|ей|им) в спину. Коварное самонаводящееся заклинание нанесло {value} единиц(у|ы) урона и подожгло {}.',
  2462.  
  2463. '[fire]{actor} создал(а|о|и) огненный шар и направил(а|о|и) его прямо во врага… Есть контакт! Повреждение — {value} единиц(а|у|ы) урона.',
  2464.  
  2465. '[fire]Выпущенный {actor} шар огня ожёг {victim} на {value} HP.',
  2466.  
  2467. '[fire]Взывая к пламени, {actor} выпуска(ет|ют) огненный шар в ничего не подозревающ(его|ую|ее|их|ий) {victim}, уменьшая (его|ее|их) здоровье на {value} HP.',
  2468.  
  2469. '[fire]Взывая к пламени, {actor} выпуска(ет|ют) огненный шар в ничего не подозревавший {victim}, уменьшая (его|ее|их) здоровье на {value} HP.',//err
  2470.  
  2471. '[fire]{victim} не успел(а|о|и) увернуться от возникшего прямо из воздуха всполоха пламени. От ожогов (он|она|оно|они) потерял(а|о|и) {value} HP.',
  2472.  
  2473.  
  2474. '[fire]{actor} выпустил(а|о|и) шар огня. Волшебное пламя нанесло {value} единиц(у|ы) урона ошарашенн(ому|ой|ым) {victim}.',
  2475.  
  2476. '[fire]Почувствовав тепло, {actor} сформировал(а|о|и) его в огненный шар и ударил(а|о|и) им по {victim}, нанеся {value} единиц(у|ы) урона.',
  2477.  
  2478. '[fire]{actor} взмахива(е|ю)т руками и магическое пламя стремительно покрывает {victim}, забирая {value} HP.',
  2479.  
  2480.  
  2481. /* Пламя (после Шара огня) */
  2482.  
  2483. '[flame]Теряя {value} HP, {victim} пыта(ется|ются) сбить с себя пламя.',
  2484.  
  2485. '[flame]Пламя окутывает {victim}, нанося {value} единиц(у|ы) урона.',
  2486.  
  2487. '[flame]Вместе с запахом паленого ветер уносит в сторону от {victim} {value} HP.',
  2488.  
  2489. '[flame]Хрустящей корочки еще нет, но {value} HP у {victim} уже сгорело.',
  2490.  
  2491. '[flame]Жаркое пламя наносит {victim} {value} единиц(у|ы) урона.',
  2492.  
  2493. '[flame]Подожженн(ый|ая|ое|ые) {victim} теря(ет|ют) {value} HP.',
  2494.  
  2495. '[flame]{victim} стара(ется|ются) сбить с себя пламя, но оно всё равно сжигает {value} HP.',
  2496.  
  2497. '[flame]Весёлые языки пламени пляшут по {victim}, нанося {value} единиц(у|ы) урона.',
  2498.  
  2499. '[flame]Пламя слизывает с {victim} {value} HP, как корова языком.',
  2500.  
  2501. '[flame]Безжалостный огонь наносит {victim} {value} единиц(у|ы) урона.',
  2502.  
  2503. '[flame]Пламя в недоступном месте вызвало у {victim} ужасную боль и нанесло {value} единиц(у|ы) урона.',
  2504.  
  2505. '[flame]{victim} горит и получа(ет|ют) {value} единиц(у|ы) урона.',
  2506.  
  2507. '[flame]Языки пламени гуляют по {victim}, и (он|она|оно|они) теря(ет|ют) {value} HP.',
  2508.  
  2509. '[flame]Охваченн(ый|ая|ое|ые) огнём, {victim} лишил(ся|ась|ось|ись) {value} HP.',
  2510.  
  2511. '[flame]{victim} меч(е|у)тся, охваченн(ый|ая|ое|ые) пламенем, и получа(е|ю)т {value} единиц(у|ы) урона.',
  2512.  
  2513. '[flame]{victim} пыла(е|ю)т словно факел(ы) и получа(е|ю)т {value} единиц(у|ы) урона.',
  2514.  
  2515. '[flame]От ожогов {victim} получа(е|ю)т {value} единиц(у|ы) урона.',
  2516.  
  2517.  
  2518. /* Разбег-толчок */
  2519.  
  2520. '[stunHit]{actor} разбежал(ся|ась|ось|ись) и в прыжке, толкнув {victim}, нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2521.  
  2522. '[stunHit]{actor} с разбега пнул(а|о|и) {victim} и нанес(ла|ло|ли) {value} единиц(у|ы) урона.',
  2523.  
  2524. '[stunHit]Разбежавшись, {actor} толкнул(а|о|и) {victim}, выколачивая из (него|нее|них) {value} HP.',
  2525.  
  2526. '[stunHit]Набрав скорость, {actor} врезал(ся|ась|ось|ись) всей массой в противника. {victim} получил(а|о|и) ушибы и потерял(а|о|и) {value} HP.',
  2527.  
  2528. '[stunHit]Немного отступив, {actor} разогнал(ся|ась|ось|ись) и толчком нанес(ла|ло|ли) {victim} {value} единиц(у|ы) урона.',
  2529.  
  2530. '[stunHit]Дождавшись, когда {victim} окаж(е|у)тся на достаточном расстоянии, {actor} разбежал(ся|ась|ось|ись) и ударил(а|о|и), нанеся {value} единиц(у|ы) урона.',
  2531.  
  2532. '[stunHit]Примерившись, {actor} толчком выколотил(а|о|и) {value} единиц(у|ы) урона из {victim}.',
  2533.  
  2534. '[stunHit]Подобно дикому кабану {actor} нес(е|у)тся вперед и таран(и|я)т {victim} так, что он(а|о|и) отлета(е|ю)т на несколько метров. От удара {} потерял(а|о|и) {value} HP.',
  2535.  
  2536. '[stunHit]{actor} разбежал(ся|ась|ось|ись) и впечатал(а|о|и) {victim} в дерево, несчастн(ый|ая|ое|ые) не досчитал(ся|ась|ось|ись) {value} HP.',
  2537.  
  2538.  
  2539. /* Шок (от Разбег-толчок)*/
  2540.  
  2541. '[stun]{victim} в шоке и не мо(жет|гут) атаковать.',
  2542.  
  2543. '[stun]{victim} пересчитыва(е|ю)т звёздочки, кружащиеся у (него|нее|них) над голов(ой|ами).',
  2544.  
  2545. '[stun]{victim} пыта(е|ю)тся восстановить равновесие.',
  2546.  
  2547. '[stun]{victim} было подумал(а|о|и) ударить, но оказал(ся|ась|ось|ись) не в состоянии додумать мысль.',
  2548.  
  2549. '[stun]Оглушенн(ый|ая|ое|ые) {victim} вяло крут(и|я)т голов(ой|ами), не решаясь атаковать.',
  2550.  
  2551. '[stun]{victim} пыта(е|ю)тся прийти в себя.',
  2552.  
  2553. '[stun]Пропустив сильный удар, {victim} пыта(е|ю)тся устоять на ногах.',
  2554.  
  2555. '[stun]{victim} с увлечением лов(и|я)т летающие вокруг голов(ы) звёздочки.',
  2556.  
  2557. '[stun]У {victim} троится в глазах.',
  2558.  
  2559.  
  2560. /* Регенерация */
  2561.  
  2562. '[heal]Дав себе установку выздороветь, {actor} восстановил(а|о|и) {value} HP.',
  2563.  
  2564. '[heal]Сконцентрировавшись, {actor} восстановил(а|о|и) {value} HP.',
  2565.  
  2566. '[heal]{actor} восстановил(а|о|и) {value} HP.',
  2567.  
  2568. '[heal]«Исцеление — это то, что сейчас важнее всего», — подумал(а|о|и) {actor} и восстановил(а|о|и) {value} HP.',
  2569.  
  2570. '[heal]{actor} восстановил(а|о|и) {value} HP и смахнул(а|о|и) пот.',
  2571.  
  2572. '[heal]{actor} отскакива(е|ю)т в сторону и, воспользовавшись короткой паузой, восстанавлива(е|ю)т {value} HP.',
  2573.  
  2574.  
  2575. /* Помощь игрока */
  2576.  
  2577. '[godheal]Вокруг {} засветился воздух и (его|ее|их) раны затянулись, излечив {value} HP.',
  2578.  
  2579. '[godheal]Банка с йодом? Что мне с ней сделать, выпить?',
  2580.  
  2581. '[godheal]Почувствовав жжение в теле, {} замечает, что (его|ее|их) раны затягиваются.',
  2582.  
  2583. '[godheal]{} исцелен на {value} HP.',
  2584.  
  2585. '[godheal]Зелёный луч ударил с небес, исцеляя {} на {value} HP.', //crit
  2586.  
  2587. '[godheal]Снег посыпался на {}, восстанавливая {value} HP.',
  2588.  
  2589. '[godheal]Банка с йодом? Что мне с ней сделать, выпить?',
  2590.  
  2591. '[godheal]Ха-ха, божественная регенерация!',
  2592.  
  2593.  
  2594. '[godhit]Неведомая сила подняла {} в воздух и швырнула под ноги {}.',
  2595.  
  2596. '[godhit]Молния существенно поджарила {}.',
  2597.  
  2598. '[godhit]Вот значит как работают вакуумные бомбы…',
  2599.  
  2600. '[godhit]Бабочка неудачно приземлилась на {}, выведя (того|ту|то|тех) из равновесия, результат — серьёзная шишка у {}.',
  2601.  
  2602. '[godhit]На горизонте показалась яркая точка, стремительно приближающаяся к {}. Мгновение и шар огня выжег аккуратный круг там, где находился противник.',
  2603.  
  2604. '[godhit]«Неужели… Это… Случилось?!» — ошарашенно шепчет {}, наблюдая за тем, что подошедший отряд стражников делает с {}.',
  2605.  
  2606. '[godhit]Неожиданно огромный перст опустился с небес прямо на {}. {} издал(а|о|и) сдавленный визг.',
  2607.  
  2608. '[godhit]Лёгкий ветерок внезапно снёс {} с места и впечатал в скалу. {} злобно смеётся над оглушённым монстром.',
  2609.  
  2610. '[godhit]Почувствовав приятный аромат жареного, {} замечает, что {} охватило яркое пламя. С душераздирающим визгом он(а|о|и) медленно обуглива(е|ю)тся.',
  2611.  
  2612. '[godhit]{} оказал(ся|ась|ось|ись) не готов(а|о|ы) к падению метеорита. В образовавшемся кратере будет сложно искать добычу.',
  2613.  
  2614. '[godhit]«Падающие наковальни — это, конечно, хорошо, но вот куда они потом пропадают?» — задался {} вопросом, созерцая то, во что превратил(ся|ась|ось|ись) {}.',
  2615.  
  2616. '[godhit]{} еле успел(а|о|и) отскочить от взрыва шаровой молнии. {} подлетел(а|о|и) в воздух и больно упал(а|о|и) вниз.',
  2617.  
  2618.  
  2619. /* Отдых */
  2620.  
  2621. '[rest]Раны начинают затягиваться, и {actor} восстанавлива(ет|ют) {value} HP.',
  2622.  
  2623. '[rest]Немного отдохнув и набравшись сил, {actor} восстановил(а|о|и) {value} HP.',
  2624.  
  2625. '[rest]Увидев неподалёку небольшой кожаный мешочек, {actor} решил(а|о|и) изучить его и обнаружил(а|о|и) несколько пробирок с красной жидкостью, по запаху напоминавшей чай из целебных трав. Выпив последний сосуд, {actor} стал(а|о|и) чувствовать себя бодрее на {value} HP, а вот мешочек развеялся в прах. Увы, его с собой не забрать.',
  2626.  
  2627. '[rest]{actor} стиснул(а|о|и) зубы и со второй попытки всё-таки вправил(а|о|и) недавно полученный вывих. Восстановлено {value} HP.',
  2628.  
  2629. '[rest]{actor} уделил(а|о|и) внимание своему исстрадавшемуся телу: сделал(а|о|и) перевязку, промыл(а|о|и) раны, ссадины, — и облегчение не заставило себя долго ждать. Исцелено {value} HP.',
  2630.  
  2631. '[rest]{actor} накладыва(ет|ют) повязку, как учили. Видать, не зря учили — восстанавливается {value} HP.',
  2632.  
  2633. '[rest]{actor} прикладыва(ет|ют) прохладные примочки к ушибленному плечу, тем самым снимая отёк и успокаивая боль. Вылечено {value} HP.',
  2634.  
  2635. '[rest]Как приятно на время отбросить грозное оружие, вездесущий рюкзак и прилечь на землицу родную. В блаженном покое {actor} восстанавлива(ет|ют) {value} HP.',
  2636.  
  2637. '[rest]Использовав совет одного целителя, {actor} собрал(а|о|и) немного паутины и аккуратно уложил(а|о|и) её в промытую рану. Буквально на глазах кровь свернулась, образовав защитный слой. Вылечен(о) {value} HP.',
  2638.  
  2639. '[rest]«Эльфийских капель не найти полезней! Эльфийские капли от всех болезней!» — напева(ет|ют) {actor}, восстанавливая {value} HP.',
  2640.  
  2641. '[rest]Обнаружив неподалёку небольшой источник, {actor} решил(а|о|и) сделать несколько глотков и промыть раны. Появилось ощущение свежести, спокойствия и уверенности. Вылечено {value} HP.',
  2642.  
  2643. '[rest]«Эх, а водичка-то из родничка просто прелесть», — {actor} поправля(ет|ют) {value} HP.',
  2644.  
  2645. '[rest]{actor} слегка подкрепил(ся|ась|ось|ись), вылечено {value} HP.',
  2646.  
  2647. '[rest]Проходящий мимо монах ткнул дорожной тростью в {actor}, и на (нем|ней|нем|них) сразу начали заживляться раны. Вылечено {value} HP.',
  2648.  
  2649. '[rest]«Как хорошо просто полежать, дав своему телу немного отдохнуть, восстановить силы и здоровье». Исцелено {value} HP.',
  2650.  
  2651. '[rest]Проходящий мимо шаман кастанул чайник и исцелил на {value} HP',
  2652.  
  2653. '[rest]«Волшебный пластырь — действительно эффективная штука! За мгновение восстановил(а|о|и) {value} HP!»',
  2654.  
  2655. '[rest]Растущие рядом лопухи помогли восстановить {value} HP.',
  2656.  
  2657. '[rest]Целебные орочьи горчичники вылечили {value} HP, но оставили пару ожогов.',
  2658.  
  2659. '[rest]Пара съеденных яблок улучшила настроение и восстановила {value} HP.',
  2660.  
  2661. '[rest]«Наверное, стоит приложить что-нибудь холодное к ушибленной ноге и дать ей немного отдыха». Вылечено {value} HP.',
  2662.  
  2663. '[rest]Ух ты, а подорожник помогает: вылечил(а|о|и) {value} HP',
  2664.  
  2665. '[rest]Пожевал(а|о|и) корешок, он оказался целебным и восстановил {value} HP. «Хоть какая-то от этих диет польза».',
  2666.  
  2667. '[rest]Крепкий отвар помог восстановить {value} HP.',
  2668.  
  2669. '[rest]«Рана загнила. Делать нечего, чтобы зараза не пошла дальше, придётся прижигать». Вылечено {value} HP.',
  2670.  
  2671. '[rest]Магическое зелье Й.О.Д. начинает действовать и восстанавливает {value} HP.',
  2672.  
  2673. '[rest]«Ужас, как же раскалывается голова… Может есть смысл просто полежать ни о чём не думая?» Восстановлено {value} HP.',
  2674.  
  2675. '[rest]Вино помогло вернуть {value} HP всего за несколько глотков.',
  2676.  
  2677. '[rest]Пролетавшая мимо фея помогла восстановить {value} HP.',
  2678.  
  2679. '[rest]Применил(а|о|и) магию вуду и исцелил(ся|ась|ось|ись) на {value} HP',
  2680.  
  2681. '[rest]Удачно наложенная повязка восстановила {value} HP.',
  2682.  
  2683. '[rest]Отвар из чистотела помог {actor} избавиться от раздражений на коже, чем сильно поднял настроение. Вылечено {value} HP.',
  2684.  
  2685. '[rest]Втирая пиявок в кожу, {actor} восстановил(а|о|и) {value} HP.',
  2686.  
  2687. '[rest]«Так вот от чего у меня плечо зудит!» — облегченно восклица(ет|ют) {actor}, выдирая кусок зуба из своей раны. Вылечено {value} HP.',
  2688.  
  2689. '[rest]Отвар целебных трав возымел силу и вернул {value} HP.',
  2690.  
  2691. '[rest]Сделав настой календулы, {actor} промыва(ет|ют) свои воспалённые и слезящиеся от пыли глаза. Вылечено {value} HP.',
  2692.  
  2693. '[rest]Втерев в раны наскоро приготовленную мазь, {actor} сразу же почувствовал(а|о|и) себя лучше. Вылечено {value} HP.',
  2694.  
  2695. '[rest]«Ну и гадость!» — {actor} выпивает очередную порцию запасенного когда-то лечебного отвара.',
  2696.  
  2697. '[rest]Стиснув зубы, зашивает глубокий порез.',
  2698.  
  2699. '[rest]Пьёт целебный отвар.',
  2700.  
  2701. '[rest]{actor} чувствует, как лёгкие наполняются воздухом, а рассудок — ясностью.',
  2702.  
  2703. '[rest]Война войной, а обед — по расписанию. Скушав маленько припасов, {} подлечил(ся|ась|ось|ись) на {value} HP.',
  2704.  
  2705. '[rest]«Дела делами, а отдыхать и кушать надо — все-таки не из железа и камня я сделан(а|о).»',
  2706.  
  2707. '[rest]«Вдали виднеется многовековой дуб. Прилягу под ним, думу подумаю…»',
  2708.  
  2709. '[rest]Пьет целебный отвар.',
  2710.  
  2711. '[rest]Пожевал(а|о|и) корешок, он оказался целебным и восстановил {value} HP. «Хоть какая-то от этих диет польза.»',
  2712.  
  2713. '[rest]Использовав совет одного целителя, {} собрал(а|о|и) немного паутины и аккуратно уложил(а|о|и) ее в промытую рану. Буквально на глазах кровь свернулась, образовав защитный слой. Вылечено {value} HP.',
  2714.  
  2715.  
  2716.  
  2717. /* Торговля */
  2718.  
  2719. '[coins]Нет, таких торгашей надо вешать. {value} монет(а|у|ы) за {item}, он с ума сошёл!',
  2720.  
  2721. '[coins]«Получил(а|о|и) {value} монет(а|у|ы) в награду за {item}.»',
  2722.  
  2723. '[coins]{value} монет(а|у|ы) за {item}. Честная сделка.',
  2724.  
  2725. '[coins]«За {item} {value} монет(а|у|ы). Вот же ей Бо… бей лавочников, спасай Пандору!»',
  2726.  
  2727. '[coins]«Эх, не зря я тащил(а|о|и) {item}», — решил(а|о|и) {}, пряча {value} монет(а|у|ы).',
  2728.  
  2729. '[coins]«Мне чужого не надо, вот и продаю {item}. Тем более, {value} монет(а|у|ы) не лишн(яя|ие|их)».',
  2730.  
  2731. '[coins]Применив пытливость и изворотливость ума, {} продал(а|о|и) {item} аж за {value} монет(а|у|ы). Перед продажей оставил(а|о|и) подпись «{}» на самом заметном месте — хоть какой-то заметный след в истории.',
  2732.  
  2733. '[coins]«Цел(ая|ые|ых) {value} монет(а|у|ы) за {item}? В чём подвох, торговец?»',
  2734.  
  2735. '[coins]«{value} монет(а|у|ы) за {item}? Ладно, забирай, я сегодня добр(ый|ая|ое).»',
  2736.  
  2737. '[coins]«Что?! {value} монет(а|у|ы) за мои подвиги?! А {item} на обочине валял(ся|ась|ось|ись), по-твоему?! Ладно, не ори. По рукам. Проклятый торгаш.»',
  2738.  
  2739. '[coins]«Глупый торговец, отдал {value} монет(а|у|ы) за дешев(ый|ую|ое|ые) {item}.»',
  2740.  
  2741. '[coins]{} скептически смотр(ит|ят) на {value} монет(а|у|ы), вырученн(ую|ые|ых) за {item}, и, пожимая плечами, убира(ет|ют) деньги в карман.',
  2742.  
  2743. '[coins]{}, остервенело торгуясь, получа(ет|ют) таки {value} монет(а|у|ы) за {item}.',
  2744.  
  2745. '[coins]«Отличная цена за {item}. Положил(а|о|и) {value} монет(а|у|ы) в кошелёк.»',
  2746.  
  2747. '[coins]{} грозно посмотрел(а|о|и) на спекулянта: «Ладно, давай сюда сво(ю|и|их|й) {value} монет(а|у|ы) и забирай {item}, пока я не передумал(а|о|и).»', //err
  2748.  
  2749. '[coins]«Не торговцы, а жлобы! Я ему {item}, а он мне жалк(ую|ие|их) {value} монет(а|у|ы)! Небось втридорога продаст!»',
  2750.  
  2751. '[coins]«А {value} монет(а|у|ы) за {item} гораздо лучше чем ничего…»',
  2752.  
  2753. '[coins]«Слышь, ты, хапуга?! {value} монет(а|у|ы) за {item}?! Окстись! Давай деньги. Молчи, с тобой торг закончен…»',
  2754.  
  2755. '[coins]«{value} монет(а|у|ы) за {item}? Да это же последний писк моды!»',
  2756.  
  2757. '[coins]«{value} монет(а|у|ы) за {item} — это надо обмыть!»',
  2758.  
  2759. '[coins]«{value} монет(а|у|ы) за {item}… Выйдешь ты на тракт… Нет, нет, ничего! Удачной торговли вам! И детям вашим! Крохобор…»',
  2760.  
  2761. '[coins]С обречённым видом {} расста(ё|ю)тся с {item} всего за {value} монет(а|у|ы).',
  2762.  
  2763. '[coins]День явно был не самым удачным. Всего {value} монет(а|у|ы) за первоклассн(ый|ую|ое|ые) {item}. Но это был единственный покупатель, и {} со смиренным лицом спрятал(а|о|и) деньги в карман.',
  2764.  
  2765. '[coins]Торговка была хороша, даже слишком. И {} пришлось отдать {item} всего за {value} монет(а|у|ы).',
  2766.  
  2767. '[coins]«{value} монет(а|у|ы) против плохоньк(ой|ого|их) {item}? Соглас(ен|на|но).»',
  2768.  
  2769. '[coins]«Обманывать людей нехорошо… но так выгодно…» — думал(а|о|и) {}, пересчитывая {value} монет(а|у|ы), полученн(ую|ые|ых) за {item}.',
  2770.  
  2771. '[coins]«{value} монет(а|у|ы) за {item} и неприятный запах — ну что за место?!»',
  2772.  
  2773. '[coins]«{item} стоит не меньше {value} монет(а|у|ы)! Нет, так я найду другого покупателя! То-то же!»',
  2774.  
  2775. '[coins]«Отличная сделка! {value} монет(а|у|ы) лучше, чем {item}.»',
  2776.  
  2777. '[coins]{item} остал(ся|ась|ось|ись) в лавке, а {value} монет(а|у|ы) забрал(а|о|и) {}.',
  2778.  
  2779. '[coins]«Всего {value} монет(а|у|ы) за {item}?! Да они издеваются!»',
  2780.  
  2781. '[coins]После долгих споров и препираний торговец все же согласился отдать {value} монет(а|у|ы) за {item}.',
  2782.  
  2783. '[coins]«За прекрасн(ый|ую|ое|ые) {item} жалк(ую|их|ие) {value} монет(а|у|ы). Сволочи!»',
  2784.  
  2785. '[coins]«Вынудил(а|о|и) торговца купить {item} за {value} монет(а|у|ы).»',
  2786.  
  2787. '[coins]«Вшив(ая|ые|ых) {value} монет(а|у|ы) за {item}. Ворьё!»',
  2788.  
  2789. '[coins]{value} монет(а|у|ы) — достойная плата за {item}.',
  2790.  
  2791. '[coins]«Мошенники! Всего {value} монет(а|у|ы) за прекрасн(ый|ую|ое|ые) {item}!»',
  2792.  
  2793. '[coins]«Безобразие! Жалк(ая|ие|их) {value} монет(а|у|ы) за редчайш(ий|ую|ее|ие) {item}?! Грабёж! Ох и жульё на здешнем рынке…»',
  2794.  
  2795. // '[coins]{} споткнул(ся|ась|ось|ись) о камень, а это оказался кошель с {value} монетами! Вот так повезло!',
  2796.  
  2797. '[coins]«Городской травник, гоблин преклонных лет, долго изучал состав принесённых сухих трав, попутно рассказывая о их свойствах. Травы были оценены в {value} монет(а|у|ы)».',
  2798.  
  2799. '[coins]Заметив в моем инвентаре {item}, торговец в ужасе отпрянул и заплатил мне {value} монет(а|у|ы) за то, чтобы я выкинул(а|о) эту мерзость подальше.',
  2800.  
  2801. '[coins]Творчески использовав классику торговли — обсчет, обвес и пересорт, продал(а|о) торговцу {item} за {value} монет(а|у|ы).',
  2802.  
  2803.  
  2804.  
  2805. /* Поднял */
  2806.  
  2807. '[pickup]{item} лег(ла|ло|ли) в рюкзак как родн(ая|ой|ое|ые).',
  2808.  
  2809. '[pickup]«Вроде бы всего лишь {item}, а приятно.»',
  2810.  
  2811. '[pickup]Обыскав место сражения, {} находит {item} и забирает добычу себе.',
  2812.  
  2813. '[pickup]«{} всё равно больше не нужно, а мне {item} пригод(ится|ятся).»',
  2814.  
  2815. '[pickup]«{item} самое место у меня в рюкзаке.»',
  2816.  
  2817. '[pickup]{}, довольн(ый|ая|ое|ые) собой, обыскав {}, убира(ет|ют) найденн(ый|ую|ое|ые) {item} к себе в сумку.',
  2818.  
  2819. '[pickup]«Эх, с паршивой овцы хоть шерсти клок!» — думал(а|о|и) {}, пряча {item} в рюкзак.',
  2820.  
  2821. '[pickup]«Что с {} взять, кроме {item}?»',
  2822.  
  2823. '[pickup]Попрыгав и похлопав в ладоши от восторга, {} упаковыва(ет|ют) обнаруженн(ый|ую|ое|ые) {item} в рюкзак.',
  2824.  
  2825. '[pickup]«Превосходно, вот и трофеями судьба не обделила», — думает {}, убирая {item} в рюкзак.',
  2826.  
  2827. '[pickup]Битва закончена лишь тогда, когда собраны трофеи. {}, положив {item} к себе в рюкзак, продолжа(ет|ют) своё странствие.',
  2828.  
  2829. '[pickup]Вытерев {item} о траву, {} упаковал(а|о|и) добычу в рюкзак.',
  2830.  
  2831. '[pickup]«{}, воровато озираясь, прикарманил(а|о|и) {item}.»',
  2832.  
  2833. '[pickup]{} мертв(а|о|ы). {item} в рюкзаке. {} счастлив(а|о|ы).',
  2834.  
  2835. '[pickup]«Нет худа без добра! Вот отличн(ый|ая|ое|ые) {item}, надо прибрать себе в рюкзак.»',
  2836.  
  2837. '[pickup]«Что тут у нас? {item}! Неплохо…»',
  2838.  
  2839. '[pickup]«Сегодня явно мой день, столь восхитительн(ый|ую|ое|ые) {item} не каждый раз найдёшь. Надо бы (его|ее|их) к себе в рюкзак положить.»',
  2840.  
  2841. '[pickup]«Ну вот так всегда, никогда на них живого места нет: тут как-то неестественно вывернуто, там кость торчит, глаз вытек, рожа изуродована, а про шкуру так и вообще говорить нечего… Постойте-ка! А вот {item}, кажется, не очень помят(а|о|ы), да и кровью не пропитал(ся|ась|ось|ись). Решено. Беру!»',
  2842.  
  2843. '[pickup]«{item}. А что ещё взять с {}?»',
  2844.  
  2845. '[pickup]{} перебрал(а|о|и) добычу и закинул(а|о|и) {item} к себе в сумку.',
  2846.  
  2847. '[pickup]Присвистнув от радости, {} пряч(ет|ат) найденн(ый|ую|ое|ые) {item} в сумку.',
  2848.  
  2849. '[pickup]Какая удача! Обшарив останки {}, {} наш(ел|ла|ло|ли) {item}!',
  2850.  
  2851. '[pickup]«Как(ой|ая|ое|ие) замечательн(ый|ая|ое|ые) {item}, в хозяйстве пригод(ится|ятся)», — {} убира(ет|ат) добычу в рюкзак.',
  2852.  
  2853. '[pickup]«Невесть что, но горсть монет я за это уж точно получу» — пробормотал(а|о|и) {}, убирая {item} в рюкзак.',
  2854.  
  2855. '[pickup]Чистая победа — грязн(ый|ая|ое|ые) {item}.',
  2856.  
  2857. '[pickup]«Древняя мудрость гласит: {item} в рюкзак — авантюристу приятней… Неее… баба с возу — авантюристу легче?…»',
  2858.  
  2859. '[pickup]«О, {item}! Пригод(и|я)тся.»',
  2860.  
  2861. '[pickup]{item} воня(ет|ют) даже сильнее, чем грязн(ый|ая|ое|ые) {}, но {} не из брезгливых.',
  2862.  
  2863. '[pickup]«Отлично, {item}! Доволочь до города — там можно будет продать.»',
  2864.  
  2865. '[pickup]«Вот это я понимаю! {item}! То что надо!»',
  2866.  
  2867. '[pickup]«Ух ты, как(ой|ая|ое|ие) прекрасн(ый|ая|ое|ые) {item}!» — {} с блаженным лицом укладывает трофей к себе в сумку.',
  2868.  
  2869. '[pickup]«Так, что тут у нас? {item}? Долж(ен|на|но|ны) пригодиться.»',
  2870.  
  2871. '[pickup]{} морщит нос, но вытира(ет|ют) дурно пахнущ(ий|ую|ее|ое|ие|их) {item} и запихива(е|ю)т в рюкзак.',//err
  2872.  
  2873. '[pickup]«Не густо, всего лишь {item}, но хоть что-то.»',
  2874.  
  2875. '[pickup]«Что это тут», — думал(а|о|и) {}, выуживая {item} из того, что осталось от {}.',
  2876.  
  2877. '[pickup]«Хлам, конечно, но может и удастся продать…» — подумал(а|о|и) {}, убирая {item} в сумку.',
  2878.  
  2879. '[pickup]Воровато оглядываясь по сторонам, {} запихивает в рюкзак {item}.',
  2880.  
  2881. '[pickup]«Вот в такие моменты я понимаю, что всё не зря», — думает {}, умиляясь найденн(ому|ой|ым) {item} и пряча (его|ее|их) к себе в рюкзак.',
  2882.  
  2883. '[pickup]«Хм… Не мешок бриллиантов, но и {item} с {} тоже ничего.»',
  2884.  
  2885. '[pickup]Изумительн(ый|ая|ое|ые) {item} — достойная награда победителя.',
  2886.  
  2887. '[pickup]«{item}? В рюкзак! В хозяйстве пригодится.»',
  2888.  
  2889. '[pickup]«Не дело трофеями пренебрегать.» — пробормотал(а|о|и) {}, засовывая {item} к себе в рюкзак.',
  2890.  
  2891. '[pickup]«Моя Прелесть», — подумал(а|о|и) {} и, оглядываясь по сторонам, бережно спрятал(а|о|и) {item} в рюкзак.',
  2892.  
  2893. '[pickup]«Сражение превращается в бесполезный риск, если нет трофеев», — замечает про себя {}, пряча {item} в рюкзак.',
  2894.  
  2895. '[pickup]«Приш(ел|ла|ло|ли), увидел(а|о|и), победил(а|о|и) — забрал(а|о|и) {item}!»',
  2896.  
  2897. '[pickup]{} — смерть, победителю — {item} — вот естественный ход вещей.',
  2898.  
  2899. '[pickup]«{item}! Всё в дом, всё в дом!»',
  2900.  
  2901. '[pickup]Один из лучших экземпляров {item}, и победитель забирает (его|ее|их) по праву.',
  2902.  
  2903. '[pickup]«Хороший враг — мёртвый враг, а хорош(ий|ая|ее|ие) {item} — {item} в рюкзаке.»',
  2904.  
  2905. '[pickup]«Так себе {item}, но не бросать же.»',
  2906.  
  2907. '[pickup]«Дело за малым: продать {item} и купить что-нибудь нужное.»',
  2908.  
  2909. '[pickup]«{item}… Неплохой трофей…»',
  2910.  
  2911. '[pickup]«Мертв(ый|ая|ое|ые) {} на первое, сладкое чувство победы на второе, и {item} на десерт.»',
  2912.  
  2913. '[pickup]«Неплохая вещица, (этот|эта|это|эти) {item} — надо оставить себе.»',
  2914.  
  2915. '[pickup]«После славной битвы и трофей должен быть достойным победителя! Что тут у нас… {item}, значит. Отлично!»',
  2916.  
  2917. '[pickup]«С паршив(ого|ой|ых) {} хоть {item} клок… А, там про овцу и шерсть…»',
  2918.  
  2919. '[pickup]«Раны болят не так сильно, если после боя тебя ждёт достойная награда… Вот прекрасн(ый|ая|ое|ые) {item}, положим (его|ее|их) в рюкзак… Хм-м, определённо, самочувствие стало лучше.»',
  2920.  
  2921. '[pickup]«Нет лучшего места для {item}, чем мой рюкзак.»',
  2922.  
  2923. '[pickup]{item} и слава — вот награда {} за победу над {} коварн(ым|ой|ыми)!',
  2924.  
  2925. '[pickup]Грёзы о трофеях были не напрасны. {} восхищается {item} и убирает (его|ее|их) в рюкзак.',
  2926.  
  2927. '[pickup]Взглядом опытного мародера {} осмотрел(а|о|и) останки {} и спрятал(а|о|и) {item} в рюкзак.',
  2928.  
  2929. '[pickup]{}, воровато озираясь, прикарманил(а|о|и) {item}.',
  2930.  
  2931. '[pickup]«Так, {item}, положу-ка я тебя вот сюда».',
  2932.  
  2933. '[pickup]{} подбросил(а|о) {item} над головой. Т(от|о|а|е) перелетел(а|о|и) за спину и упал(а|о|и) прямо в рюкзак к остальной добыче. «В яблочко!» – довольно улыбнул(ся|ась|ось) {}.',
  2934.  
  2935. '[pickup]«Что-то упало. Я точно видел(а|о), как что-то упало с {}. Да вот только непонятно, куда? О, наш(ел|ла|ло)! {item}!»',
  2936.  
  2937. '[pickup]Немного постояв над найденн(ым|ой|ыми) {item}, {} аккуратно (его|ее|их) поднял(а|о) и спрятал(а|о) к себе в рюкзак.',
  2938.  
  2939. '[pickup]«{item}? Возьму. Продам кому-нибудь на рынке».',
  2940.  
  2941.  
  2942.  
  2943. /* Пусто */
  2944.  
  2945. '[empty]«Нет, мешок денег было бы чересчур, но хоть что-то могло достаться в награду?»',
  2946.  
  2947. '[empty]{} склоняется над поверженн(ым|ой|ыми) {}… и ничего не находит.',
  2948.  
  2949. '[empty]Порывшись в останках {}, {} ничего не наш(ел|ла|ло), вытер(ла|ло|ли) руки и уш(ел|ла|ло).',
  2950.  
  2951. '[empty]«Ненавижу, когда так получается: бьёшься, жизнью рискуешь, а награды нет!»',
  2952.  
  2953. '[empty]{} посмотрел(а|о), что осталось от {}, и с досадой отметил(а|о): «Добычи нет.»',
  2954.  
  2955. '[empty]{} не наш(ел|ла|ло) ничего полезного на тушк(е|ах) монстр(а|ов).',
  2956.  
  2957. '[empty]«Тут брать нечего. Если так дело и дальше будет, я по миру пойду.»',
  2958.  
  2959. '[empty]«И за что было сражаться? Добычи явно нет.»',
  2960.  
  2961. '[empty]«Какая уж тут добыча? В живых остал(ся|ась|ось|ись), и то славно.»',
  2962.  
  2963. '[empty]«Нечего взять с {}… Ладно, найду ещё кого-нибудь.»',
  2964.  
  2965. '[empty]{} отдубасил(а|о) бандита так, что ничего ценного уже не осталось.',
  2966.  
  2967. '[empty]{} с унынием смотрит на то, что осталось от {}. Совершенно очевидно, что здесь поживиться нечем.',
  2968.  
  2969. '[empty]«И какой смысл столько времени тратить на сражение, если после этого даже взять нечего?»',
  2970.  
  2971. '[empty]Сегодня {} без награды, но главное, что (он|она|оно) жив(а|о).',
  2972.  
  2973. '[empty]«Что тут? Пусто…» — Пнув {}, {} продолжает свой путь.',
  2974.  
  2975. '[empty]{} склоняется над поверженн(ым|ой|ыми) {} и… плюясь и ворча, не находит добычи.',
  2976.  
  2977. '[empty]{} со злостью плюёт на останки {}, всё равно трофеев никаких.',
  2978.  
  2979. '[empty]{} настолько увлек(ся|лась|лось|лись), что от {} ничего не осталось.',
  2980.  
  2981. '[empty]{} понял(а|о|и), что поживиться тут нечем, но не унывает: «Легче сумка — шире шаг.»',
  2982.  
  2983. '[empty]«Можно было бы строго судить авантюриста, если опять нет добычи, и этот авантюрист пошёл бы грабить караваны?…»',
  2984.  
  2985. '[empty]В этот раз {} остал(ся|ась|ось|ись) без добычи.',
  2986.  
  2987. '[empty]Небрежно осмотрев останки {}, и не найдя ничего полезного, {} отправил(ся|ась|ось|ись) дальше.',
  2988.  
  2989. '[empty]«Тьфу ты ж! И взять с {} нечего!»',
  2990.  
  2991. '[empty]«Трофеев нет, но будет ещё и на нашей улице праздник!»',
  2992.  
  2993. '[empty]«Ангел хранитель, эй! Победили — это спасибо, конечно, но что я буду есть, если снова нет трофеев?»',
  2994.  
  2995. '[empty]«Хранитель, эй! Победили — это спасибо, конечно, но что я буду есть, если снова нет трофеев?»',
  2996.  
  2997. '[empty]«Победа есть, а добычи нет…»',
  2998.  
  2999. '[empty]{} отдубасил(а|о) {} так, что ничего ценного уже не осталось.',
  3000.  
  3001. '[empty]«Вот же тварь какая, и взять-то нечего.»',
  3002.  
  3003. '[empty]Раздосадованн(ый|ая|ое|ые) отсутствием трофеев, {} глумится над останками {}.',
  3004.  
  3005. '[empty]«Ну что за день сегодня такой?… Даже с {} взять нечего.»',
  3006.  
  3007. '[empty]Измотанн(ый|ая|ое|ые) битвой {} с брезгливостью смотрит на место сражения, понимая, что трофеев здесь явно нет.',
  3008.  
  3009. '[empty]Плохо. Очень плохо. {} остал(ся|ась|ось|ись) без добычи.',
  3010.  
  3011. '[empty]Очередн(ой|ые) монстр(ы) без добычи, куда катится этот мир?',
  3012.  
  3013. '[empty]Останки поверженн(ого|ой|ых) {} испарились прямо на глазах. Стоп! А где трофей?',
  3014.  
  3015. '[empty]«И снова нечего взять.»',
  3016.  
  3017. '[empty]Останки {} выглядят слишком противно, чтобы пытаться найти что-то ценное.',
  3018.  
  3019. '[empty]Покойся с миром, несчастн(ый|ая|ое|ые) {}. Не принесла победа ни добычи, ни славы.',
  3020.  
  3021. '[empty]«Да уж… Добычи нет, и стоило ли рисковать?»',
  3022.  
  3023. '[empty]Ничего ценного. В следующий раз повезёт.',
  3024.  
  3025. '[empty]«Совсем нет добычи? Будем искать!»',
  3026.  
  3027. '[empty]И кукиш с маслом, как награда герою.',
  3028.  
  3029. '[empty]«Вот же твари какие, и взять-то нечего.»',
  3030.  
  3031. '[empty]«Рискуешь жизнью, терпишь лишения, сбиваешь ноги в кровь, и что в итоге? Ни-че-го!»',
  3032.  
  3033. '[empty]После ожесточённой схватки {} оказал(ся|ась|ось|ись) настолько искромсан(а|о|ы), что о добыче можно и не думать.',
  3034.  
  3035. '[empty]«Как-то и зимы холоднее стали, и рюкзаки делают маленькие. Тяжёлые времена ждут этот мир.»',
  3036.  
  3037. '[empty]Столько сил потрачено и все напрасно.',
  3038.  
  3039.  
  3040.  
  3041. /* Бросил */
  3042.  
  3043. '[drop]«Вот несправедливость какая: отличн(ый|ая|ое|ые) {drop}, а забрать не получится — рюкзак уже полный.»',
  3044.  
  3045. '[drop]«Надо что ли рюкзак побольше купить, а то такими вещами приходится разбрасываться… Неплох(ой|ая|ое|ие) был(и) {drop}.»',
  3046.  
  3047. // '[drop]Короткая вспышка — и {} с удивлением смотр(ит|ят) на горстку пепла, оставшуюся от {drop}.', //manual drop
  3048.  
  3049. '[drop]Найденн(ый|ая|ое|ые) {drop} не помеща(ется|ются) в рюкзак. «Кто ты, тот счастливчик, что получит (его|ее|их) задарма? И когда уже я найду брошенный кем-то трофей?»',
  3050.  
  3051. '[drop]Сердце кровью обливается, когда приходится бросать {drop}. Но {} смотрит в набитый доверху рюкзак, и становится легче.',
  3052.  
  3053. '[drop]Повертев в руках {drop}, доставш(ийся|уюся|ееся|иеся|егося) от {}, {} с сожалением выкинул(а|о|и) (его|ее|их) — в рюкзаке не осталось места.', //err
  3054.  
  3055. '[drop]{} пытается запихнуть трофей в рюкзак, но, потерпев сокрушительное фиаско, решает, что {drop} ему не так уж сильно и нужен.',
  3056.  
  3057. '[drop]{}, повертев в руках {item}, и не найдя куда (его|ее|их) засунуть, решил(а|о|и) в следующий раз купить штаны с дополнительными карманами.',
  3058.  
  3059. '[drop]{item} остал(ся|ась|ось|ись) валяться на обочине дороги.',
  3060.  
  3061. '[drop]В рюкзаке нет свободного места и {item} пришлось оставить.',
  3062.  
  3063. '[drop]«Это не дело: рюкзак — битком — трофеи складывать некуда.»',
  3064.  
  3065. '[drop]«Нужен рюкзак побольше, а то в этот уже не влезает.»',
  3066.  
  3067. '[drop]«Хм… {item}. А запихнуть-то некуда — рюкзак набит — придется выкинуть.»',
  3068.  
  3069. '[drop]{} пыжится, всё вертит и пытается впихнуть {item} в рюкзак, но там слишком мало места. Вздохнув, он выбрасывает {item}.',
  3070.  
  3071. '[drop]Рюкзак полон и {} не может взять добыт(ый|ую|ое|ые) {item}.',
  3072.  
  3073. '[drop]Хмуро глядя в набитый доверху рюкзак, {} оставляет добычу на земле.',
  3074.  
  3075. '[drop]«Опять забыл(а|о|и) от хлама избавиться!» — ворчит {}, выбрасывая подобранн(ый|ую|ое|ые) {item}.',
  3076.  
  3077. '[drop]«Надо срочно продать чего-нибудь, освободить место в рюкзаке. Ну жалко ж бросать {item}!»',
  3078.  
  3079. '[drop]«Был бы рюкзак побольше… А пока {item} придется выбросить.»',
  3080.  
  3081. '[drop]Найденн(ый|ая|ое|ые) {item} не помеща(ется|ются) в рюкзак. «Кто ты, тот счастливчик, что получит (его|ее|их) задарма? И когда уже я найду брошенный кем-то трофей?»',
  3082.  
  3083. '[drop]Оставлю я {item} здесь, может кому сгодится.',
  3084.  
  3085. '[drop]{item} никак не влаз(ит|ят) в рюкзак, придётся выкинуть.',
  3086.  
  3087. '[drop]«Некуда уже пихать, ну и демон с {item}.»',
  3088.  
  3089. '[drop]{} попытал(ся|ась|ось|ись) впихнуть {item} к себе в рюкзак, но, услышав подозрительный треск швов, оставил(а|о|и) эту затею и выкинул(а|о|и) добычу.',
  3090.  
  3091. '[drop]«Нет, всё же грех жаловаться. Вон уже и трофеи складывать некуда!»',
  3092.  
  3093. '[drop]«Надо что ли рюкзак побольше купить, а то такими вещами приходится разбрасываться… Неплох(ой|ая|ое|ие) был(а|о|и) {item}.»',
  3094.  
  3095. '[drop]«Святой лук-порей! Опять куча хлама в инвентаре!» — со слезами на глазах, {} уходит прочь, а {item} остается одиноко лежать на земле.',
  3096.  
  3097. '[drop]«Вот и {item} уже не влезает. Решено — пора продавать трофеи.»',
  3098.  
  3099. '[drop]В придорожные кусты лет(и|я)т {item}. В рюкзак (его|ее|их) уже не впихнуть.',
  3100.  
  3101. '[drop]Израненн(ый|ая|ое) {} смотрит в переполненный рюкзак и понимает, что {item} туда уже не влезет, и (его|ее|их) придётся бросить. Знакомая каждому, извечная и обидная ситуация.',
  3102.  
  3103. '[drop]«В рюкзаке совсем нет места. Зачем мне столько барахла?» — думает {}, выбрасывая добытый {item}.',
  3104.  
  3105. '[drop]Рюкзак полон настолько, что чуть ли не трещит по швам. {} с грустью оставляет добыт(ый|ую|ое|ые) {item} и идёт дальше.',
  3106.  
  3107. '[drop]Красиво жить не запретишь! Рюкзак уже битком, и неплох(ой|ая|ое|ие) {item} летит в канаву.',
  3108.  
  3109. '[drop]Рюкзак и так слишком тяжёлый. Вздохнув, {} оставляет {item} на поле боя.',
  3110.  
  3111. '[drop]«То совсем пусто, то густо, так что уже и в рюкзак не влезает — придется выкидывать хорош(ий|ую|ое|ие) {item}.»',
  3112.  
  3113. '[drop]{} безуспешно пытается засунуть {item} в трещащий по швам рюкзак: «Да здесь еще три артефакта влезет, глупая авоська! Где заклинание утрамбовывания?!»',
  3114.  
  3115. '[drop]Обливаясь слезами, {} последний раз смотр(и|я)т на найденн(ый|ую|ое|ые) {item} и оставляет (его|ее|их) на обочине — в рюкзаке уже нет места.',
  3116.  
  3117. '[drop]Неплох(ой|ая|ое|ие) {item}, но в рюкзак уже не влез(е|ю)т.',
  3118.  
  3119. '[drop]«Так бывает», — отмеча(е|ю)т {}, глядя в доверху набитый рюкзак и выкидыва(е|ю)т {item}.',
  3120.  
  3121. '[drop]«Если я возьму с собой ещё и {item}, то спина мне этого не простит.»',
  3122.  
  3123. '[drop]«Так не достанешься же ты никому, {item}!» — иступлённо крикнул(а|о|и) {} и выбросил(а|о|и) трофей, не поместившийся в рюкзак, в канаву.',
  3124.  
  3125. '[drop]{} пытается запихнуть трофей в рюкзак, но, потерпев сокрушительное фиаско, решает, что {item} (ему|ей|им) не так уж сильно и нуж(ен|на|но|ны).',
  3126.  
  3127.  
  3128.  
  3129. /* Начало боя */
  3130.  
  3131. '[fight]{} и {} сошлись на узкой дорожке.',
  3132.  
  3133. '[fight]За спиной {} хрустнула ветка. Он(а|о|и) ме-едленно обернул(ся|ась|ось|ись), доставая оружие, и увидел(а|о|и) {}, стояв(шего|шую|шее|ших) с видом полной невинности.',
  3134.  
  3135. '[fight]«А это ещё что за бести(я|и)? {}?»',
  3136.  
  3137. '[fight]«Ни на минуту нельзя расслабиться!» — {} атаку(е|ю)т {}.',
  3138.  
  3139. '[fight]Впереди {}, ищущ(ий|ая|ое|ие) неприятностей, — «Эй, я здесь!»',
  3140.  
  3141. '[fight]{} преградил(а|о|и) дорогу. {} вынужден(а|о|и) принять сражение.',
  3142.  
  3143. '[fight]«{}? Что ж, потанцуем!»',
  3144.  
  3145. '[fight]{} на горизонте, в бой!',
  3146.  
  3147. '[fight]Битвы не избежать — впереди {}.',
  3148.  
  3149. '[fight]Как {} ни пытал(ся|ась|ось|ись) притвориться мертв(ым|ой), {} всё равно пош(ел|ла|ло|ли) в наступление.',
  3150.  
  3151. '[fight]«Что там впереди? {}? Ближе… Сейчас я буду добывать трофеи.»',
  3152.  
  3153. '[fight]«Что-то тут не так. Не в засаду ли меня несё… Проклятье! {}!»',
  3154.  
  3155. '[fight]«Судьба сводит меня с {}? Что ж…»',
  3156.  
  3157. '[fight]Ужасн(ый|ая|ое|ые) {} выскочил(а|о|и) из кустов.',
  3158.  
  3159. '[fight]«Это {}! К бою!»',
  3160.  
  3161. '[fight]«{}! Оружие наголо!»',
  3162.  
  3163. '[fight]«Ни минуты покоя!» — на пути {}!',
  3164.  
  3165. '[fight]«Ого, да это {}. Исчезающий вид!»',
  3166.  
  3167. '[fight]«И так жить на свете страшно, а тут еще и {}!»',
  3168.  
  3169. '[fight]«Проклятье! Не золотуха, так {}!»',
  3170.  
  3171. '[fight]«Это была самая большая ошибка в твоей жизни… и последняя…» — грозно произнес(ла|ло|ли) {} и приготовил(ся|ась|ось|ись) к бою против {}.',
  3172.  
  3173. '[fight]Обернул(ся|ась|ось|ись) на странный шум и увидел(а|о|и) {}.',
  3174.  
  3175. '[fight]«{}, как мне вас не хватало!» — проворчал(а|о|и) {}, разминая кисти рук.',
  3176.  
  3177. '[fight]«Неужто передо мной {}? Отлично! Будет чем пополнить мою коллекцию трофеев!»',
  3178.  
  3179. '[fight]«Говорят, места тут спокойные. Тогда что тут дела(е|ю)т {}?!»',
  3180.  
  3181. '[fight]«Ха-ха! Вот мы и сошлись, {}!»',
  3182.  
  3183. '[fight]Вовремя заметив приближающ(егося|уюся|ееся|ихся|ийся) {}, {} приготовил(ся|ась|ось|ись) к бою.',
  3184.  
  3185. '[fight]«На ловца и зверь бежит! Сегодня это {}!»',
  3186.  
  3187. '[fight]Неожиданно перед {} появля(е|ю)тся {}, заставляя (его|ее|их) подпрыгнуть от испуга! «Никому не дозволено безнаказанно меня пугать!» — вопит {} и отчаянно бросается в бой.',
  3188.  
  3189. '[fight]{} на пути!',
  3190.  
  3191. '[fight]{} видит, как дорогу прегражда(е|ю)т {}.',
  3192.  
  3193. '[fight]За очередным поворотом дороги {} сталкива(е|ю)тся с {}.',
  3194.  
  3195. '[fight]«Подозрительно тихо, не нравится мне это… Так и есть, {}!»',
  3196.  
  3197. '[fight]Из придорожных зарослей выскочил(а|о|и) {}!',
  3198.  
  3199. '[fight]{}, чувствуя опасность, насторожил(ся|ась|ось|ись) и вовремя заметил(а|о|и) {}.',
  3200.  
  3201. '[fight]{} заметил(а|о|и) впереди {} и готовится к схватке.',
  3202.  
  3203. '[fight]«Впереди {}! Здоров(ый|ая|ое|ые), зараз(а|ы)!»',
  3204.  
  3205. '[fight]«Вот сколько раз уже сражал(ся|ась|ось|ись) с {}, а популяция их всё не убывает… О! Вот опять!»',
  3206.  
  3207. '[fight]Из-за кустов показыва(ется|ются) {}. «Не случайно нынешней ночью тревожные сны снились.»',
  3208.  
  3209. '[fight]{} в опасности — на (его|ее|их) пути {}.',
  3210.  
  3211. '[fight]«Я, отважн(ый|ая|ое) {}, смело принимаю бой! Никто не выстоит против меня! Даже {}!»',
  3212.  
  3213. '[fight]{} преградил(а|о|и) дорогу. {} вынужден(а|о|и) принять сражение.',
  3214.  
  3215. '[fight]«Хм, {}. Давно пора очистить от них Пандору!»',
  3216.  
  3217. '[fight]«В книгах пишут, что нападение это лучшая защита. Как бы проверить?… Ха, {}! Как по заказу».',
  3218.  
  3219. '[fight]{} бросил(а|о) камень в кусты, чем разозлил(а|о) {}, стоявш(его|ую|ее|их) за ними.',
  3220.  
  3221. '[fight]«Кажется, я нарушил(а|о) границы территории {}. Сейчас меня будут прогонять».',
  3222.  
  3223. '[fight]«Ну, кто на меня? {}!»',
  3224.  
  3225. '[fight]«Ни проехать, ни пройти пока {} сто(ит|ят) на пути!»',
  3226.  
  3227.  
  3228. '[fight]Вовремя заметив {victim}, {actor} подкрал(ся|ась|ось|ись) сзади и сразил(а|о|и) противник(а|ов) одним ударом.',
  3229.  
  3230. '[fight]{actor} зловеще ухмыльнул(ся|ась|ось), когда {victim} попал(ся|ась|ось|ись) в предусмотрительно установленную на дороге ловушку.'
  3231.  
  3232. // '[walk]Проезжавший мимо крестьянин узнал {actor} и немного подвёз на своей телеге.'
  3233.  
  3234.  
  3235. /* */
  3236.  
  3237. ];
  3238.  
  3239.  
  3240. var parseHighlightRaw = [
  3241.  
  3242. '[value]\\d+ звездоч(?:ка|ки|ек)?',
  3243.  
  3244. '[value]\\d+ HP',
  3245.  
  3246. '[value]\\d+ единиц(:?у|а|ы)? урона',
  3247.  
  3248. '[value]\\d+ урона?',
  3249.  
  3250. '[energy]\\d+ единиц(:?у|а|ы)? энергии',
  3251.  
  3252. '[coins]\\d+ монет(?:ами|ой|а|у|ы)?',
  3253.  
  3254. '[exp]\\d+ опыт(?:а)?',
  3255.  
  3256. '[level]\\d+ уровень'
  3257.  
  3258. ];
  3259.  
  3260.  
  3261. function processHighlightRaw(parseHighlightRaw) {
  3262.  
  3263. var typeReg = /^\[[a-zA-Z]+\]/g;
  3264.  
  3265. var result = [];
  3266.  
  3267. for (var i = 0; i<parseHighlightRaw.length; i++) {
  3268.  
  3269. var cfgString = parseHighlightRaw[i];
  3270.  
  3271. var parsedCfg = {};
  3272.  
  3273.  
  3274. var p = typeReg.exec(cfgString);
  3275.  
  3276. if (p && p[0]) parsedCfg.type = p[0].substring(1, p[0].length-1);
  3277.  
  3278. cfgString = cfgString.replace(typeReg, '');
  3279.  
  3280.  
  3281. parsedCfg.regex = new RegExp(cfgString);
  3282.  
  3283. result.push(parsedCfg)
  3284.  
  3285. }
  3286.  
  3287. return result;
  3288.  
  3289. }
  3290.  
  3291. function processShortRaw(cfgRaw) {
  3292.  
  3293. var ESCAPE = /[?.]/g;
  3294.  
  3295. var anyRegExp = new RegExp('({})', 'g');
  3296.  
  3297. var numbersRegExp = new RegExp('({value})', 'g');
  3298.  
  3299. var paramsRegExp = new RegExp('({[a-zA-Z]+})', 'g');
  3300.  
  3301. var namesRegExp = '([а-яА-ЯёЁa-zA-Z\\d _\\-\']+)';
  3302.  
  3303.  
  3304. var paramReg = /{([a-zA-Z]+)}/g;
  3305.  
  3306. var result = [];
  3307.  
  3308.  
  3309. var typeReg = /^\[([a-zA-Z]+)(?:,([a-zA-Z]+))?\]/;
  3310.  
  3311.  
  3312. for(var i = 0; i<cfgRaw.length; i++) {
  3313.  
  3314. var cfgString = cfgRaw[i].replace(/ё/g, 'е');
  3315.  
  3316. var parsedCfg = {};
  3317.  
  3318.  
  3319. var p = typeReg.exec(cfgString);
  3320.  
  3321. if (p) {
  3322.  
  3323. if (p[1]) parsedCfg.type = p[1];
  3324.  
  3325. if (p[2]) parsedCfg.sec = p[2];
  3326.  
  3327. // else parsedCfg.sec = 'vamp';
  3328.  
  3329. cfgString = cfgString.replace(p[0], '');
  3330.  
  3331. }
  3332.  
  3333. if (cfgString.charAt(0) != '~') {
  3334.  
  3335. cfgString = '^' + cfgString;
  3336.  
  3337. cfgString = cfgString + '$';
  3338.  
  3339. }
  3340.  
  3341.  
  3342. var paramNames = [];
  3343.  
  3344. do {
  3345.  
  3346. var p = paramReg.exec(cfgString);
  3347.  
  3348. if (p) paramNames.push(p[1]);
  3349.  
  3350. } while (p);
  3351.  
  3352. parsedCfg.params = paramNames;
  3353.  
  3354.  
  3355. cfgString = cfgString
  3356.  
  3357. .replace(ESCAPE, '\\$&')
  3358.  
  3359. .replace(/\(/g, '(?:')
  3360.  
  3361. .replace(/\)/g, ')?')
  3362.  
  3363. .replace(anyRegExp, '.+')
  3364.  
  3365. .replace(numbersRegExp, '(\\d+)')
  3366.  
  3367. .replace(paramsRegExp, namesRegExp);
  3368.  
  3369. parsedCfg.regex = new RegExp(cfgString);
  3370.  
  3371. result[i] = parsedCfg;
  3372.  
  3373. }
  3374.  
  3375. return result;
  3376.  
  3377. }
  3378.  
  3379.  
  3380. var cfgShort = processShortRaw(parseShortRaw);
  3381.  
  3382. var cfgHighlight = processHighlightRaw(parseHighlightRaw);
  3383.  
  3384.  
  3385. function parseShort(msg) {
  3386.  
  3387. msg = msg.replace(/ё/g, 'е');
  3388.  
  3389. for(var i = 0; i<cfgShort.length; i++) {
  3390.  
  3391. var cfgParsedLine = cfgShort[i];
  3392.  
  3393. var paramNames = cfgParsedLine.params;
  3394.  
  3395. var values = cfgParsedLine.regex.exec(msg);
  3396.  
  3397.  
  3398. if (values) {
  3399.  
  3400. values = values.slice(1);
  3401.  
  3402.  
  3403. var act = {};
  3404.  
  3405. act.type = cfgParsedLine.type;
  3406.  
  3407. if (cfgParsedLine.sec) act.sec = cfgParsedLine.sec;
  3408.  
  3409. for(var paramIndex = 0; paramIndex < paramNames.length; paramIndex++) {
  3410.  
  3411. var param = paramNames[paramIndex];
  3412.  
  3413. act[param] = isNaN(values[paramIndex]) ? values[paramIndex] : +values[paramIndex];
  3414.  
  3415. }
  3416.  
  3417.  
  3418. act.isMe = (act.actor ? _ext.isMyName(act.actor) : (act.victim ? !_ext.isMyName(act.victim) : true ))|0;
  3419.  
  3420. if (!_ext.settings.settingsValues.heroNameStart) act.isMe = 0;
  3421.  
  3422.  
  3423. }
  3424.  
  3425. }
  3426.  
  3427.  
  3428. return act;
  3429.  
  3430. }
  3431.  
  3432. function parseHighlight(msg, act) {
  3433.  
  3434. for (var i = 0; i<cfgHighlight.length; i++) {
  3435.  
  3436. var regExp = cfgHighlight[i].regex;
  3437.  
  3438. var type = cfgHighlight[i].type || 'value';
  3439.  
  3440. msg = msg.replace(regExp, '<span class="' + type + '">$&</span>')
  3441.  
  3442. }
  3443.  
  3444. for (var cls in act) if (act.hasOwnProperty(cls)) {
  3445.  
  3446. var value = act[cls];
  3447.  
  3448. if (cls != 'value' && cls != 'type') {
  3449.  
  3450. msg = msg.replace(value, '<span class="'+cls+'">'+value+'</span>')
  3451.  
  3452. }
  3453.  
  3454. }
  3455.  
  3456. return msg;
  3457.  
  3458. }
  3459.  
  3460.  
  3461.  
  3462. $.extend(_parse, {
  3463.  
  3464. short: parseShort,
  3465.  
  3466. highlight: parseHighlight
  3467.  
  3468. });
  3469.  
  3470. return _parse;
  3471.  
  3472. })({});
  3473.  
  3474.  
  3475. _ext.parse = _parse;
  3476.  
  3477.  
  3478.  
  3479.  
  3480. var _const = _ext.const;
  3481.  
  3482. var _trace = _ext.trace;
  3483.  
  3484. var _elements = _ext.elements;
  3485.  
  3486. var _icons = _const.ICONS;
  3487.  
  3488.  
  3489. function isActType(types, actType) {
  3490.  
  3491. return _const[types].indexOf(actType) >= 0;
  3492.  
  3493. }
  3494.  
  3495.  
  3496.  
  3497. /* towns */
  3498.  
  3499. var _towns = (function(_towns) {
  3500.  
  3501. var $townsContent = _elements.getTabInner('towns');
  3502.  
  3503. $('body')
  3504.  
  3505. .on('click.town', '.town', function() {
  3506.  
  3507. if (!mapData) return;
  3508.  
  3509. var id = $(this).data('place-id');
  3510.  
  3511. showMapDialogById(id);
  3512.  
  3513. })
  3514.  
  3515. .on('click.town', '.reload', function() {
  3516.  
  3517. if (_ext.map_version) {
  3518.  
  3519. _towns.mapDataUpdate(_ext.map_version)
  3520.  
  3521. .done(function(mapData) {
  3522.  
  3523. townParams(mapData);
  3524.  
  3525. });
  3526.  
  3527. }
  3528.  
  3529. });
  3530.  
  3531.  
  3532.  
  3533. $townsContent.html('<span class="link-ajax pull-right reload glyphicon glyphicon-repeat"></span>');
  3534.  
  3535.  
  3536.  
  3537.  
  3538. var mapData;
  3539.  
  3540. function mapDataUpdate(map_version) {
  3541.  
  3542. return $.ajax({
  3543.  
  3544. url: '/dcont/map/region-' + map_version + '.js',
  3545.  
  3546. dataType: 'json',
  3547.  
  3548. type: 'get'
  3549.  
  3550. })
  3551.  
  3552. .done(function(map_data) {
  3553.  
  3554. _towns.mapData = mapData = map_data;
  3555.  
  3556. _ext.publish('townsInit', map_data);
  3557.  
  3558. });
  3559.  
  3560. }
  3561.  
  3562.  
  3563. function init() {
  3564.  
  3565. var places = mapData.places;
  3566.  
  3567. var html = '';
  3568.  
  3569. for (var i in places) if (places.hasOwnProperty(i)) {
  3570.  
  3571. var place = places[i];
  3572.  
  3573. html +=
  3574.  
  3575. '<tr class="place-row" data-place-id="' + place.id + '">' +
  3576.  
  3577. '<td class="size"><span class="badge">' + place.size + '</span></td>' +
  3578.  
  3579. '<td>' +
  3580.  
  3581. '<span class="link-ajax town" data-place-id="' + place.id + '">' + place.name + '</span>' +
  3582.  
  3583. ' <span class="quest"></span>' +
  3584.  
  3585. '</td>' +
  3586.  
  3587. // '<td data-city-param="1" class="production"></td>' +
  3588.  
  3589. '<td data-city-param="размер экономики"></td>' +
  3590.  
  3591. '<td data-city-param="безопасность"></td>' +
  3592.  
  3593. '<td data-city-param="транспорт"></td>' +
  3594.  
  3595. '<td data-city-param="свобода"></td>' +
  3596.  
  3597. '</tr>';
  3598.  
  3599. }
  3600.  
  3601. html =
  3602.  
  3603. '<table class="table table-towns table-noborder table-hover-dark table-condensed">' +
  3604.  
  3605. '<thead>' +
  3606.  
  3607. '<th class="size" title="Размер города">Р</th>' +
  3608.  
  3609. '<th>Название</th>' +
  3610.  
  3611. // '<th>Произв.</th>' +
  3612.  
  3613. '<th title="Размер экономики">Э</th>' +
  3614.  
  3615. '<th title="Безопасность">Безоп.</th>' +
  3616.  
  3617. '<th title="Транспорт">Транс.</th>' +
  3618.  
  3619. '<th title="Свобода">Своб.</th>' +
  3620.  
  3621. '</thead>' +
  3622.  
  3623. '<tbody>' +
  3624.  
  3625. html +
  3626.  
  3627. '</tbody>' +
  3628.  
  3629. '</table>';
  3630.  
  3631. var $table = $(html).appendTo($townsContent);
  3632.  
  3633. window.tables.makeSortable($table);
  3634.  
  3635.  
  3636.  
  3637.  
  3638. _towns.townQuestUpdate = function(quests) {
  3639.  
  3640. $('.place-row .quest').html('');
  3641.  
  3642. for (var questsIndex=0;questsIndex<quests.length;questsIndex++) {
  3643.  
  3644. var quest = quests[questsIndex];
  3645.  
  3646. var actors = quest.actors;
  3647.  
  3648. for (var i=0; i<actors.length; i++) {
  3649.  
  3650. var actor = actors[i];
  3651.  
  3652. var isFrom = i === 0 && actors.length > 1;
  3653.  
  3654. var actorType = actor[0];
  3655.  
  3656. var actorTypeId = actor[1];
  3657.  
  3658. var placeId = actorTypeId == 1 ? actor[2].id : actor[2].place;
  3659.  
  3660. var $placeRow = $('.place-row[data-place-id="' + placeId + '"]');
  3661.  
  3662. var $townQuest = $placeRow.find('.quest');
  3663.  
  3664. var questHtml =
  3665.  
  3666. (isFrom ? '<span class="glyphicon glyphicon-arrow-right"></span>' : '<span class="glyphicon glyphicon-arrow-left"></span>') +
  3667.  
  3668. '<span class="quest-icon-mini pgf-quest-icon ' + quest.type + '" title="' + actorType + '"></span> ';
  3669.  
  3670. $townQuest.append(questHtml);
  3671.  
  3672. }
  3673.  
  3674. }
  3675.  
  3676. };
  3677.  
  3678.  
  3679. for (var i=0; i<_prevParams.length; i++) {
  3680.  
  3681. _towns.townQuestUpdate.apply(_towns, _prevParams[i]);
  3682.  
  3683. }
  3684.  
  3685.  
  3686. }
  3687.  
  3688. var _prevParams = [];
  3689.  
  3690. _towns.townQuestUpdate = function() {
  3691.  
  3692. _prevParams.push(arguments);
  3693.  
  3694. };
  3695.  
  3696.  
  3697. function townParams(mapData) {
  3698.  
  3699. var places = mapData.places;
  3700.  
  3701. for (var i in places) {
  3702.  
  3703. (function(placeIndex) {
  3704.  
  3705. var place = places[placeIndex];
  3706.  
  3707. requestPlace(place.pos.x, place.pos.y)
  3708.  
  3709. .done(function(html) {
  3710.  
  3711. var parsed = parsePlaceHtml(html);
  3712.  
  3713. var $placeRow = $('.place-row[data-place-id="' + placeIndex + '"]');
  3714.  
  3715. parsed.cityParams.forEach(function(item) {
  3716.  
  3717. var val = item.value;
  3718.  
  3719. if (val < 100) val = '&nbsp;' + val;
  3720.  
  3721. $placeRow.children('[data-city-param="' + item.name + '"]').html(val);
  3722.  
  3723. });
  3724.  
  3725. });
  3726.  
  3727. })(i);
  3728.  
  3729. }
  3730.  
  3731. }
  3732.  
  3733.  
  3734.  
  3735. function parsePlaceHtml(html) {
  3736.  
  3737. var $info = $(html);
  3738.  
  3739.  
  3740. var cityParams = [];
  3741.  
  3742. var $cityParamsRows = $info.find('#pgf-cell-place-parameters').find('tr').slice(1);
  3743.  
  3744. $cityParamsRows.each(function() {
  3745.  
  3746. var $row = $(this);
  3747.  
  3748. var paramName = $.trim($row.children('th').first().text());
  3749.  
  3750. var valueText = $row.children('td').first().text();
  3751.  
  3752. var value = parseFloat(valueText);
  3753.  
  3754. cityParams.push({
  3755.  
  3756. name: paramName,
  3757.  
  3758. value: value
  3759.  
  3760. });
  3761.  
  3762. });
  3763.  
  3764. return {
  3765.  
  3766. cityParams: cityParams
  3767.  
  3768. };
  3769.  
  3770. }
  3771.  
  3772.  
  3773. function requestPlace(x,y) {
  3774.  
  3775. return $.ajax({
  3776.  
  3777. url: '/game/map/cell-info?x=' + x + '&y=' + y + '&_=' + (+new Date()),
  3778.  
  3779. method: 'get',
  3780.  
  3781. dataType: 'html'
  3782.  
  3783. });
  3784.  
  3785. }
  3786.  
  3787.  
  3788.  
  3789. function showMapDialogById(id) {
  3790.  
  3791. var place = mapData.places[id];
  3792.  
  3793. showMapDialog(place.pos.x, place.pos.y);
  3794.  
  3795. }
  3796.  
  3797. function showMapDialog(x, y) {
  3798.  
  3799. pgf.ui.dialog.Create({ fromUrl: pgf.urls['game:map:cell_info'](x, y),
  3800.  
  3801. OnOpened: function(dialog) {
  3802.  
  3803. pgf.base.InitializeTabs('game-map-cell-info', 'map',
  3804.  
  3805. [[jQuery('.pgf-cell-description-button', dialog), 'description'],
  3806.  
  3807. [jQuery('.pgf-cell-persons-button', dialog), 'persons'],
  3808.  
  3809. [jQuery('.pgf-cell-place-parameters-button', dialog),'place-parameters'],
  3810.  
  3811. [jQuery('.pgf-cell-place-demographics-button', dialog),'place-demographics'],
  3812.  
  3813. [jQuery('.pgf-cell-place-bills-button', dialog),'place-bills'],
  3814.  
  3815. [jQuery('.pgf-cell-place-modifiers-button', dialog), 'place-modifiers'],
  3816.  
  3817. [jQuery('.pgf-cell-place-chronicle-button', dialog), 'place-chronicle'],
  3818.  
  3819. [jQuery('.pgf-cell-building-button', dialog), 'building'],
  3820.  
  3821. [jQuery('.pgf-cell-map-button', dialog), 'map'],
  3822.  
  3823. [jQuery('.pgf-cell-debug-button', dialog), 'debug']]);
  3824.  
  3825. jQuery('[rel="tooltip"]', dialog).tooltip(pgf.base.tooltipsArgs);
  3826.  
  3827.  
  3828. if (widgets.abilities) {
  3829.  
  3830. widgets.abilities.UpdateButtons();
  3831.  
  3832. widgets.abilities.RenderAbility(pgf.game.constants.abilities.building_repair);
  3833.  
  3834. jQuery('.angel-ability', dialog).toggleClass('pgf-hidden', false);
  3835.  
  3836. }
  3837.  
  3838.  
  3839. },
  3840.  
  3841. OnClosed: function(dialog) {
  3842.  
  3843. pgf.base.HideTooltips(dialog);
  3844.  
  3845. }
  3846.  
  3847. });
  3848.  
  3849. }
  3850.  
  3851.  
  3852. $.extend(_towns, {
  3853.  
  3854. init : init,
  3855.  
  3856. mapDataUpdate : mapDataUpdate,
  3857.  
  3858. showMapDialogById : showMapDialogById
  3859.  
  3860. });
  3861.  
  3862. return _towns;
  3863.  
  3864. })({});
  3865.  
  3866.  
  3867. _ext.subscribe('preload', function() {
  3868.  
  3869. if (_ext.map_version) {
  3870.  
  3871. _towns.mapDataUpdate(_ext.map_version)
  3872.  
  3873. .done(function() {
  3874.  
  3875. _towns.init();
  3876.  
  3877. });
  3878.  
  3879. }
  3880.  
  3881. });
  3882.  
  3883. _ext.subscribe('questUpdate', function(quest) {
  3884.  
  3885. _towns.townQuestUpdate(quest);
  3886.  
  3887. });
  3888.  
  3889. /* eo towns */
  3890.  
  3891.  
  3892. /* shortMessages */
  3893.  
  3894. var _shortMessages = (function(_shortMessages) {
  3895.  
  3896. // var $shortContainer = _elements.getTabInner('short');
  3897.  
  3898. function htmlMessages(messages) {
  3899.  
  3900. var html = '';
  3901.  
  3902. for (var i=0; i<messages.length; i++) {
  3903.  
  3904. var message = messages[i];
  3905.  
  3906. // var m = JSON.stringify(message); data-m=\''+ m +'\'
  3907.  
  3908. var htmlShortMsg = htmlMessage(message);
  3909.  
  3910. var timestamp = message[0];
  3911.  
  3912. var time = message[1];
  3913.  
  3914. var htmlMsg;
  3915.  
  3916. if (htmlShortMsg) {
  3917.  
  3918. while (messages[i+1] && messages[i+1][1] === time) {
  3919.  
  3920. var htmlShortMsg2 = htmlMessage(messages[i+1]);
  3921.  
  3922. if (!htmlShortMsg2) break;
  3923.  
  3924. htmlShortMsg += htmlShortMsg2;
  3925.  
  3926. i++;
  3927.  
  3928. }
  3929.  
  3930. htmlMsg = '<li data-ts="'+timestamp+'" class="log-record-short">' + htmlShortMsg + '</li>';
  3931.  
  3932. } else {
  3933.  
  3934. var htmlLongMsg = htmlLongMessage(message);
  3935.  
  3936. htmlMsg = '<li data-ts="'+timestamp+'" class="log-record">' + htmlLongMsg + '</li>';
  3937.  
  3938. }
  3939.  
  3940. html = htmlMsg + html;
  3941.  
  3942. }
  3943.  
  3944. return html;
  3945.  
  3946. }
  3947.  
  3948.  
  3949. function htmlMessage(message) {
  3950.  
  3951. var time = message[1];
  3952.  
  3953. var msg = message[2];
  3954.  
  3955. var act = message[4]; //_ext.parse.message(message, 'message');
  3956.  
  3957.  
  3958. var htmlMsg;
  3959.  
  3960. if (!act || !act.type) return '';
  3961.  
  3962. var isMe = act.isMe;
  3963.  
  3964. var type = act.type;
  3965.  
  3966. var sec = act.sec;
  3967.  
  3968. var t = '';
  3969.  
  3970. var val = act.value || '';
  3971.  
  3972. var icon = _icons[type];
  3973.  
  3974. // if (sec) console.log(sec, message[3], time)
  3975.  
  3976. if (sec) icon += '<span class="sub-icon">' + _icons[sec] + '</span>';
  3977.  
  3978. if (type === 'hit') t = val;
  3979.  
  3980. else if (type === 'vamp') t = val + icon + '<span class="vamp">' + act.vamp + '</span>';
  3981.  
  3982. else if (isActType('SHORT', type)) t = val + icon;
  3983.  
  3984.  
  3985. if (t) htmlMsg = '<span class="submessage act act-' + act.type + (isMe ? ' me' : ' enemy') + '" title="' + time + '> ' + msg + '">' + t + '</span>';
  3986.  
  3987. return htmlMsg || '';
  3988.  
  3989. }
  3990.  
  3991.  
  3992. function htmlLongMessage(message) {
  3993.  
  3994. var time = message[1];
  3995.  
  3996. var msg = message[2];
  3997.  
  3998. var act = message[4];
  3999.  
  4000.  
  4001. var actType = '';
  4002.  
  4003. if (act && act.type) {
  4004.  
  4005. var isMe = act.isMe ;
  4006.  
  4007. actType = ' msg msg-' + act.type + (isMe ? ' me' : ' enemy');
  4008.  
  4009. }
  4010.  
  4011. var messageHighlight = _ext.parse.highlight(msg, act);
  4012.  
  4013.  
  4014.  
  4015. var htmlLongMsg =
  4016.  
  4017. '<div class="pgf-time time">' + time + '</div>' +
  4018.  
  4019. '<div class="pgf-message message">' +
  4020.  
  4021. '<div class="submessage' + actType + '">' + messageHighlight + '</div>' +
  4022.  
  4023. '</div>';
  4024.  
  4025.  
  4026. return htmlLongMsg;
  4027.  
  4028. }
  4029.  
  4030.  
  4031. $.extend(_shortMessages, {
  4032.  
  4033. htmlMessage: htmlMessage,
  4034.  
  4035. htmlLongMessage: htmlLongMessage,
  4036.  
  4037. htmlMessages: htmlMessages
  4038.  
  4039. });
  4040.  
  4041. return _shortMessages;
  4042.  
  4043. })({});
  4044.  
  4045.  
  4046. /* eo shortMessages */
  4047.  
  4048.  
  4049.  
  4050. /* group */
  4051.  
  4052. var _groupMessages = (function(_groupMessages) {
  4053.  
  4054.  
  4055. var $groupsContent = _elements.getTabInner('group');
  4056.  
  4057. var $lastGroup;
  4058.  
  4059.  
  4060. var messagesGrouped = [];
  4061.  
  4062.  
  4063. function addMessages(messagesList) {
  4064.  
  4065. for (var i=0; i<messagesList.length; i++) {
  4066.  
  4067. addMessage(messagesList[i]);
  4068.  
  4069. }
  4070.  
  4071. return messagesGrouped;
  4072.  
  4073. }
  4074.  
  4075.  
  4076. /*
  4077.  
  4078. data: {
  4079.  
  4080. actionName: "пытается одолеть волка"
  4081.  
  4082. info_link: "/guide/mobs/7/info"
  4083.  
  4084. type: "fight"
  4085.  
  4086. typeId: 3
  4087.  
  4088. }*/
  4089.  
  4090. function addMessage(message) {
  4091.  
  4092. var hero = message[3];
  4093.  
  4094. var act = message[4];
  4095.  
  4096. var action = hero.action || false;
  4097.  
  4098.  
  4099. var currGr = messagesGrouped[messagesGrouped.length-1];
  4100.  
  4101. var isFirstGrouop = !currGr;
  4102.  
  4103.  
  4104. if (isFirstGrouop) {
  4105.  
  4106. /* самая первая группа */
  4107.  
  4108. currGr = { /* заглушка */
  4109.  
  4110. data: {}
  4111.  
  4112. }
  4113.  
  4114. } else {
  4115.  
  4116. var currActionName = currGr.data.actionName;
  4117.  
  4118. var currType = currGr.data.type;
  4119.  
  4120. var currInfoLink = currGr.data.info_link;
  4121.  
  4122. var currTypeId = currGr.data.typeId;
  4123.  
  4124. }
  4125.  
  4126.  
  4127.  
  4128. if (action) {
  4129.  
  4130. var actionName = action.description;
  4131.  
  4132. var actionTypeId = action.type;
  4133.  
  4134. var actionType = _const.ACTION_TYPE_NAMES[actionTypeId];
  4135.  
  4136. var actionInfoLink = action.info_link;
  4137.  
  4138.  
  4139. if (!currActionName) currGr.data.actionName = currActionName = actionName;
  4140.  
  4141. if (!currType) currGr.data.type = currType = actionType;
  4142.  
  4143. if (!currInfoLink) currGr.data.info_link = currInfoLink = actionInfoLink;
  4144.  
  4145. if (!currTypeId) currGr.data.typeId = currTypeId = actionTypeId;
  4146.  
  4147. var grData = {
  4148.  
  4149. actionName: actionName,
  4150.  
  4151. type: actionType,
  4152.  
  4153. info_link: actionInfoLink,
  4154.  
  4155. typeId: actionTypeId
  4156.  
  4157. };
  4158.  
  4159. if (isFirstGrouop) grData.isBroken = 1; // first group
  4160.  
  4161. }
  4162.  
  4163.  
  4164.  
  4165. if (act) {
  4166.  
  4167. var actType = act.type;
  4168.  
  4169. var isFightStart = isActType('FIGHT_START', actType);
  4170.  
  4171. var isFightEnd = isActType('LOOT', actType);
  4172.  
  4173. if (actType === 'godheal' || actType === 'godhit') {
  4174.  
  4175. grData = $.extend(grData, {god: 1});
  4176.  
  4177. }
  4178.  
  4179.  
  4180. if (isFightStart) {
  4181.  
  4182. if (!isFirstGrouop) finishGroup();
  4183.  
  4184. newGroup(message, {fightStarted: 1}); //fight started, not finished
  4185.  
  4186. return;
  4187.  
  4188. }
  4189.  
  4190. if (isFightEnd) {
  4191.  
  4192. if (isFirstGrouop) {
  4193.  
  4194. newGroup(message, grData)
  4195.  
  4196. } else {
  4197.  
  4198. var lastG = addToLastGroup(message, {
  4199.  
  4200. actionName: 'в бою',
  4201.  
  4202. type: 'fight',
  4203.  
  4204. info_link: '',
  4205.  
  4206. typeId: 3
  4207.  
  4208. });
  4209.  
  4210. if (lastG.data.fightStarted) {
  4211.  
  4212. delete lastG.data.fightStarted;
  4213.  
  4214. if (lastG.data.isBroken !== 4 && lastG.data.isBroken !== 5) {
  4215.  
  4216. delete lastG.data.isBroken;
  4217.  
  4218. }
  4219.  
  4220. } else {
  4221.  
  4222. lastG.data.isBroken = 2; //fight ended, but not started
  4223.  
  4224. }
  4225.  
  4226. finishGroup();
  4227.  
  4228. }
  4229.  
  4230.  
  4231. if (action && actionName !== currActionName) {
  4232.  
  4233. newGroup([message[0], message[1], actionName, false, false], grData);
  4234.  
  4235. }
  4236.  
  4237. return;
  4238.  
  4239. }
  4240.  
  4241. if (currGr.messages && currGr.messages.length && !currGr.data.fightStarted && isActType('FIGHT', act.type)) {
  4242.  
  4243. grData = $.extend(grData, {isBroken: 5});
  4244.  
  4245. }
  4246.  
  4247.  
  4248. }
  4249.  
  4250.  
  4251.  
  4252. if (action && actionName && currActionName && actionName !== currActionName) {
  4253.  
  4254. finishGroup();
  4255.  
  4256. newGroup(message, grData);
  4257.  
  4258. return;
  4259.  
  4260. }
  4261.  
  4262. if (isFirstGrouop) {
  4263.  
  4264. newGroup(message, grData)
  4265.  
  4266. } else {
  4267.  
  4268. addToLastGroup(message, grData);
  4269.  
  4270. }
  4271.  
  4272.  
  4273. function addToLastGroup(message, data) {
  4274.  
  4275. var lastG = messagesGrouped[messagesGrouped.length-1];
  4276.  
  4277. if (data) lastG.data = $.extend(data, lastG.data);
  4278.  
  4279. if (lastG.messages.length) {
  4280.  
  4281. var prevMsg = lastG.messages[lastG.messages.length-1];
  4282.  
  4283. if (message[0] - prevMsg[0] > 1200) lastG.data.isBroken = 4; // hole in messages 20 min
  4284.  
  4285. }
  4286.  
  4287. lastG.messages.push(message);
  4288.  
  4289. return lastG;
  4290.  
  4291. }
  4292.  
  4293.  
  4294. function finishGroup(data) {
  4295.  
  4296. var lastG = messagesGrouped[messagesGrouped.length-1];
  4297.  
  4298. delete lastG.data.unfinished;
  4299.  
  4300. if (lastG.data.fightStarted) {
  4301.  
  4302. lastG.data.isBroken = 3; // fight started, not ended
  4303.  
  4304. }
  4305.  
  4306. if (data) lastG.data = $.extend(data, lastG.data);
  4307.  
  4308. _ext.publish('groupFinished', lastG, messagesGrouped.length-1);
  4309.  
  4310. }
  4311.  
  4312. function newGroup(message, data) {
  4313.  
  4314. var newG = {
  4315.  
  4316. data: {unfinished: 1},
  4317.  
  4318. messages: []
  4319.  
  4320. };
  4321.  
  4322. if (data) newG.data = $.extend(data, newG.data);
  4323.  
  4324. if (message) newG.messages.push(message);
  4325.  
  4326. messagesGrouped[messagesGrouped.length] = newG;
  4327.  
  4328. _ext.publish('groupStarted', newG, messagesGrouped.length-1);
  4329.  
  4330. }
  4331.  
  4332. }
  4333.  
  4334.  
  4335. function redrawGroup(index, isOpen) {
  4336.  
  4337. var $group = $groupsContent.children('.group[data-index="'+index+'"]');
  4338.  
  4339. if ($group.length) {
  4340.  
  4341. if (typeof isOpen !== 'undefined') {
  4342.  
  4343. $group.toggleClass('open', isOpen);
  4344.  
  4345. }
  4346.  
  4347. $group.html(drawGroupInner(messagesGrouped[index], messagesGrouped[index+1]));
  4348.  
  4349. } else {
  4350.  
  4351. $groupsContent.prepend(drawGroup(messagesGrouped[index], index, isOpen));
  4352.  
  4353. }
  4354.  
  4355. }
  4356.  
  4357.  
  4358.  
  4359. function drawGroup(group, index, isOpen) {
  4360.  
  4361. isOpen = isOpen || _ext.settings.settingsValues.groupOpenOnDefault;
  4362.  
  4363. var html =
  4364.  
  4365. '<div class="group' + (isOpen ? ' open' : '') + '" data-index="' + index + '">' +
  4366.  
  4367. drawGroupInner(group, messagesGrouped[index+1]) +
  4368.  
  4369. '</div>';
  4370.  
  4371. return html;
  4372.  
  4373. }
  4374.  
  4375. function drawGroupInner(group, groupNext) {
  4376.  
  4377. if (!group || !group.messages || !group.messages.length) { return ''; }
  4378.  
  4379. var messages = group.messages;
  4380.  
  4381. var groupData = group.data;
  4382.  
  4383.  
  4384. var messageFirst = messages[0];
  4385.  
  4386. var messageLast = messages[messages.length-1];
  4387.  
  4388.  
  4389. var groupType = groupData.type;
  4390.  
  4391. var groupLink = (groupData.info_link || '').replace('/info', '');
  4392.  
  4393. var htmlGroupIcon = _const.ACTION_TYPE_ICONS[groupType] || '';
  4394.  
  4395. var iconAttr = 'class="action-icon ' + groupType + '" title="' + _const.ACTION_TYPE_TEXTS[groupType] + '"';
  4396.  
  4397.  
  4398. if (groupData.isBroken && !groupData.unfinished) {
  4399.  
  4400. htmlGroupIcon = _const.ACTION_TYPE_ICONS['broken'];
  4401.  
  4402. var title = _const.ERROR_CODES[groupData.isBroken || 1];
  4403.  
  4404. iconAttr = 'class="action-icon broken" title="' + title + '"';
  4405.  
  4406. }
  4407.  
  4408. if (groupLink) {
  4409.  
  4410. htmlGroupIcon = '<a ' + iconAttr + ' href="'+groupLink+'" target="_blank">' + htmlGroupIcon + '</a>';
  4411.  
  4412. } else {
  4413.  
  4414. htmlGroupIcon = '<span ' + iconAttr + '">' + htmlGroupIcon + '</span>';
  4415.  
  4416. }
  4417.  
  4418.  
  4419.  
  4420. var htmlTitle = '<span class="action-name">' + (groupData.actionName || 'неизвестное действие') + '</span>';
  4421.  
  4422.  
  4423. var timeStart = messageFirst[0];
  4424.  
  4425. var timeEnd = messageLast[0];
  4426.  
  4427. if (groupType !== 'fight' && groupNext && groupNext.messages[0]) {
  4428.  
  4429. var timeStartNext = groupNext.messages[0] && groupNext.messages[0][0];
  4430.  
  4431. if (timeStartNext - timeEnd < 120) timeEnd = timeStartNext; // проверка на случай сломанной группы
  4432.  
  4433. }
  4434.  
  4435. var timeSpan = timeEnd - timeStart;
  4436.  
  4437. var htmlTime = /*'<span class="glyphicon glyphicon-time"></span> ' +*/
  4438.  
  4439. '<span class="group-time ' + (timeSpan > 600 ? 'bad' : timeSpan > 300 ? 'average' : '') + '">' +
  4440.  
  4441. _ext.utils.timeSpan(timeSpan) +
  4442.  
  4443. '</span> ';
  4444.  
  4445.  
  4446.  
  4447.  
  4448. var htmlGroupList = _shortMessages.htmlMessages(messages);
  4449.  
  4450.  
  4451. var html =
  4452.  
  4453. '<div class="group-title on-close' + (groupData.god ? ' god' : '') + '">' + htmlGroupIcon + htmlTime + htmlTitle +'</div>' +
  4454.  
  4455. // '<div class="group-stats on-close">' + htmlStats + '</div>' +
  4456.  
  4457. '<div class="group-controls">' +
  4458.  
  4459. '<span class="group-toggle on-close text-muted glyphicon glyphicon-chevron-up"></span>' +
  4460.  
  4461. '<span class="group-toggle on-open text-muted glyphicon glyphicon-chevron-down"></span>' +
  4462.  
  4463. '</div>' +
  4464.  
  4465. '<ul class="unstyled pgf-log-list on-open">' + htmlGroupList + '</ul>';
  4466.  
  4467.  
  4468. return html;
  4469.  
  4470. }
  4471.  
  4472. function drawMessages(messagesGrouped) {
  4473.  
  4474. var html = '';
  4475.  
  4476. for (var i=0; i<messagesGrouped.length; i++) {
  4477.  
  4478. html = drawGroup(messagesGrouped[i], i) + html;
  4479.  
  4480. }
  4481.  
  4482. $groupsContent.html(html);
  4483.  
  4484. $lastGroup = $groupsContent.children('.group').first();
  4485.  
  4486. }
  4487.  
  4488. function drawFakeMessage(message) {
  4489.  
  4490. var html = _shortMessages.htmlLongMessage(message);
  4491.  
  4492. $lastGroup.prepend(html);
  4493.  
  4494. }
  4495.  
  4496.  
  4497.  
  4498. $.extend(_groupMessages, {
  4499.  
  4500. list: messagesGrouped,
  4501.  
  4502. addMessages: addMessages,
  4503.  
  4504. drawFakeMessage: drawFakeMessage,
  4505.  
  4506. // addMessage: addMessage,
  4507.  
  4508. // group: group,
  4509.  
  4510. drawMessages: drawMessages,
  4511.  
  4512. redrawGroup: redrawGroup
  4513.  
  4514. });
  4515.  
  4516.  
  4517. return _groupMessages;
  4518.  
  4519. })({});
  4520.  
  4521.  
  4522. _ext.subscribe('init', function() {
  4523.  
  4524. _groupMessages.addMessages(_trace.messagesLog);
  4525.  
  4526. _groupMessages.drawMessages( _groupMessages.list );
  4527.  
  4528. // _ext.subscribe('groupFinished', function(group, index) {
  4529.  
  4530. // _groupMessages.redrawGroup(index);
  4531.  
  4532. // });
  4533.  
  4534. _ext.subscribe('groupStarted', function(group, index) {
  4535.  
  4536. _groupMessages.redrawGroup(index-1);
  4537.  
  4538. });
  4539.  
  4540. _ext.subscribe('newTurn', function(messagesNew) {
  4541.  
  4542. _groupMessages.addMessages(messagesNew);
  4543.  
  4544. _groupMessages.redrawGroup(_groupMessages.list.length-1);
  4545.  
  4546. });
  4547.  
  4548.  
  4549. });
  4550.  
  4551.  
  4552. _elements.getTabInner('group').on('click', '.group-title', function() {
  4553.  
  4554. var $group = $(this).closest('.group');
  4555.  
  4556. var index = $group.data('index');
  4557.  
  4558. console.log('group>', _groupMessages.list[index]);
  4559.  
  4560. });
  4561.  
  4562. /* eo group */
  4563.  
  4564.  
  4565. _ext.elements.addControl('group-toggle', {title: 'Только действия / Подробности', content: '<span class="glyphicon glyphicon-chevron-' + (_ext.settings.settingsValues.groupOpenOnDefault ? 'down' : 'up') + '"></span>'})
  4566.  
  4567. .on('click', function() {
  4568.  
  4569. // _ext.elements.activeTab('group');
  4570.  
  4571. var $icon = $(this).children('.glyphicon');
  4572.  
  4573. var isOpen = $icon.hasClass('glyphicon-chevron-up');
  4574.  
  4575. _elements.getTabInner('group')
  4576.  
  4577. .children('.group').toggleClass('open', isOpen);
  4578.  
  4579. _elements.getTabInner('archive')
  4580.  
  4581. .find('.group').toggleClass('open', isOpen);
  4582.  
  4583. $icon.toggleClass('glyphicon-chevron-down glyphicon-chevron-up');
  4584.  
  4585. });
  4586.  
  4587.  
  4588.  
  4589.  
  4590. /* archive */
  4591.  
  4592. var _archive = (function(_archive) {
  4593.  
  4594. var showArchive = _ext.settings.settingsValues.showArchive;
  4595.  
  4596. var $archiveTab = _elements.getTab('archive').toggle(showArchive);
  4597.  
  4598.  
  4599. _ext.subscribe('settingsChange', function(key, value) {
  4600.  
  4601. if (key === 'showArchive') {
  4602.  
  4603. showArchive = value;
  4604.  
  4605. $archiveTab.toggle(showArchive);
  4606.  
  4607. if (!showArchive) $archiveContent.html('');
  4608.  
  4609. }
  4610.  
  4611. });
  4612.  
  4613.  
  4614. var $archiveTabContent = _elements.getTabInner('archive');
  4615.  
  4616.  
  4617. $('<span class="link-ajax archive-renew">обновить</span>').appendTo($archiveTabContent)
  4618.  
  4619. .on('click', function() {
  4620.  
  4621. drawArchiveGroups(_archive.archiveGroups);
  4622.  
  4623. });
  4624.  
  4625. var $archiveContent = $('<div class="archive-content"></div>').appendTo($archiveTabContent);
  4626.  
  4627.  
  4628. $archiveContent.on('click', '.group-toggle', function() {
  4629.  
  4630. $(this).closest('.group').toggleClass('open');
  4631.  
  4632. });
  4633.  
  4634.  
  4635. function drawArchiveGroups(archiveGroups) {
  4636.  
  4637. var levelIndex = 0;
  4638.  
  4639. var levelsLog = _ext.log.get('levelsLog') || [];
  4640.  
  4641. for (var i=0; i<archiveGroups.length; i++) {
  4642.  
  4643. var group = archiveGroups[i];
  4644.  
  4645. var ts = group.ts[1];
  4646.  
  4647. var lv = levelsLog[levelIndex] || [];
  4648.  
  4649. while (ts > lv[0]) {
  4650.  
  4651. levelIndex++;
  4652.  
  4653. var lvlHtml = '<div class="level">' + lv[1] + ' уровень!</div>';
  4654.  
  4655. $archiveContent.prepend(lvlHtml);
  4656.  
  4657. var lv = levelsLog[levelIndex] || [];
  4658.  
  4659. }
  4660.  
  4661. drawArchiveGroup(archiveGroups[i], i, archiveGroups);
  4662.  
  4663. }
  4664.  
  4665. }
  4666.  
  4667.  
  4668. function drawArchiveGroup(archiveGroup, index, archiveGroups) {
  4669.  
  4670. var isOpen = _ext.settings.settingsValues.groupOpenOnDefault;
  4671.  
  4672. var groupType = archiveGroup.broken ? 'broken' : archiveGroup.type;
  4673.  
  4674. if (archiveGroup.mobId) {
  4675.  
  4676. var groupLink = '/guide/mobs/' + archiveGroup.mobId;
  4677.  
  4678. }
  4679.  
  4680. var htmlGroupIcon = _const.ACTION_TYPE_ICONS[groupType] || '';
  4681.  
  4682. if (groupLink) {
  4683.  
  4684. htmlGroupIcon = '<a class="action-icon ' + groupType + '" href="'+groupLink+'" target="_blank">' + htmlGroupIcon + '</a>';
  4685.  
  4686. } else {
  4687.  
  4688. htmlGroupIcon = '<span class="action-icon ' + groupType + '">' + htmlGroupIcon + '</span>';
  4689.  
  4690. }
  4691.  
  4692. var title = (archiveGroup.text || 'неизвестное действие');
  4693.  
  4694. var htmlTitle = '<span class="action-name">' + title + '</span>';
  4695.  
  4696.  
  4697.  
  4698. var timeStart = archiveGroup.ts[0];
  4699.  
  4700. var timeEnd = archiveGroup.ts[1];
  4701.  
  4702. var archiveGroupNext = archiveGroups[index+1];
  4703.  
  4704. if (groupType !== 'fight' && archiveGroupNext && archiveGroupNext.ts) {
  4705.  
  4706. var timeStartNext = archiveGroupNext.ts[0];
  4707.  
  4708. if (timeStartNext - timeEnd < 120) timeEnd = timeStartNext; // проверка на случай сломанной группы (3)
  4709.  
  4710. }
  4711.  
  4712. var timeSpan = timeEnd - timeStart;
  4713.  
  4714.  
  4715. var htmlTime = /*'<span class="glyphicon glyphicon-time"></span> ' +*/
  4716.  
  4717. '<span class="group-time ' + (timeSpan > 600 ? 'bad' : timeSpan > 300 ? 'average' : '') + '">' +
  4718.  
  4719. _ext.utils.timeSpan(timeSpan) +
  4720.  
  4721. '</span> ';
  4722.  
  4723.  
  4724.  
  4725. var htmlGroupList = '';
  4726.  
  4727. if (archiveGroup.total && (archiveGroup.type === 'fight' || archiveGroup.type === 'fight-god')) {
  4728.  
  4729. htmlGroupList =
  4730.  
  4731. '<span class="stats-archive stats-archive-me">' +
  4732.  
  4733. drawArchiveActStat(archiveGroup.total.me) +
  4734.  
  4735. '</span>' +
  4736.  
  4737. '<span class="stats-archive stats-archive-enemy">' +
  4738.  
  4739. drawArchiveActStat(archiveGroup.total.enemy) +
  4740.  
  4741. '</span>';
  4742.  
  4743. } else {
  4744.  
  4745. var actName = _const.ACTION_TYPE_TEXTS[archiveGroup.type || 'undefined'];
  4746.  
  4747. if (archiveGroup.broken) {
  4748.  
  4749. actName = _const.ERROR_CODES[archiveGroup.broken || 1]
  4750.  
  4751. }
  4752.  
  4753. htmlGroupList = '<span class="stats-archive">' + actName + '</span>';
  4754.  
  4755. }
  4756.  
  4757. var html =
  4758.  
  4759. '<div class="group-title">' + htmlGroupIcon + htmlTime + htmlTitle +'</div>' +
  4760.  
  4761. '<div class="group-controls">' +
  4762.  
  4763. '<span class="group-toggle on-close text-muted glyphicon glyphicon-chevron-up"></span>' +
  4764.  
  4765. '<span class="group-toggle on-open text-muted glyphicon glyphicon-chevron-down"></span>' +
  4766.  
  4767. '</div>' +
  4768.  
  4769. '<div class="archive-log-list on-open">' + htmlGroupList + '</div>';
  4770.  
  4771.  
  4772. html =
  4773.  
  4774. '<div class="group' + (isOpen ? ' open' : '') + '" data-index="' + index + '">' +
  4775.  
  4776. html +
  4777.  
  4778. '</div>';
  4779.  
  4780.  
  4781. $archiveContent.prepend(html);
  4782.  
  4783.  
  4784. }
  4785.  
  4786.  
  4787. function drawArchiveActStat(stats) {
  4788.  
  4789.  
  4790. var types = ['dmgSum'].concat(_const.FIGHT);
  4791.  
  4792. var html = '';
  4793.  
  4794. // html += JSON.stringify(stats)
  4795.  
  4796. for (var i=0; i<types.length; i++) {
  4797.  
  4798. var type = types[i];
  4799.  
  4800. if (stats[type]) {
  4801.  
  4802. var stat = stats[type];
  4803.  
  4804. var htmlStat = '';
  4805.  
  4806. var htmlStatName = _icons[type];
  4807.  
  4808. var count = stat.count;
  4809.  
  4810. var sum = stat.sum;
  4811.  
  4812. if (isActType('FIGHT_COUNTS', type)) {
  4813.  
  4814. htmlStat += count + 'x' + htmlStatName;
  4815.  
  4816. } else {
  4817.  
  4818. if (type === 'dmgSum') {
  4819.  
  4820. htmlStat += htmlStatName + '=<b>' + sum + '</b> ';
  4821.  
  4822. } else {
  4823.  
  4824. htmlStat += count + 'x' + htmlStatName;
  4825.  
  4826. htmlStat += '=' + sum + ' ';
  4827.  
  4828. }
  4829.  
  4830. }
  4831.  
  4832.  
  4833. html += '<span class="stats-archive-act stats-archive-' + type + '">' + htmlStat + '</span>';
  4834.  
  4835. }
  4836.  
  4837. }
  4838.  
  4839. return html;
  4840.  
  4841.  
  4842. }
  4843.  
  4844.  
  4845.  
  4846. /* пересчет группы в архив */
  4847.  
  4848. function countArchiveFromGroup(group) {
  4849.  
  4850. var groupData = group.data;
  4851.  
  4852. var messages = group.messages;
  4853.  
  4854. if (!messages.length) return false;
  4855.  
  4856. var isBroken = groupData.isBroken;
  4857.  
  4858. var dataType = groupData.type;
  4859.  
  4860. var first = messages[0];
  4861.  
  4862. var last = messages[messages.length-1];
  4863.  
  4864. var archiveGroup = $.extend({}, {
  4865.  
  4866. ts: [first[0], last[0]],
  4867.  
  4868. type: dataType,
  4869.  
  4870. text: groupData.actionName,
  4871.  
  4872. broken: isBroken
  4873.  
  4874. });
  4875.  
  4876.  
  4877. if (dataType === 'fight' && !isBroken) {
  4878.  
  4879. archiveGroup.me = {};
  4880.  
  4881. archiveGroup.enemy = {};
  4882.  
  4883. if (groupData.info_link) {
  4884.  
  4885. var mobId = groupData.info_link.replace('/guide/mobs/', '').replace('/info', '');
  4886.  
  4887. if (+mobId) archiveGroup.mobId = +mobId;
  4888.  
  4889. }
  4890.  
  4891.  
  4892. /* пересчет последней группы */
  4893.  
  4894. for (var i=0; i<messages.length; i++) {
  4895.  
  4896. var message = messages[i];
  4897.  
  4898. var act = message[4];
  4899.  
  4900. if (!act) continue;
  4901.  
  4902. var type = act.type; /* тип фразы */
  4903.  
  4904. if (isActType('FIGHT', type)) {
  4905.  
  4906. var isMe = act.isMe;
  4907.  
  4908. var addTo = archiveGroup[isMe ? 'me' : 'enemy'];
  4909.  
  4910. if (isActType('FIGHT_VALUES', type)) {
  4911.  
  4912. addTo[type] = addTo[type] || [];
  4913.  
  4914. addTo[type].push(act.value);
  4915.  
  4916. } else {
  4917.  
  4918. addTo[type] = addTo[type] || 0;
  4919.  
  4920. addTo[type]++;
  4921.  
  4922. }
  4923.  
  4924. } else if (isActType('LOOT', type)) {
  4925.  
  4926. archiveGroup.loot = type;
  4927.  
  4928. }
  4929.  
  4930. }
  4931.  
  4932. }
  4933.  
  4934.  
  4935. upgradeArchiveGroup(archiveGroup);
  4936.  
  4937. return archiveGroup;
  4938.  
  4939. }
  4940.  
  4941.  
  4942.  
  4943.  
  4944. /* добавляет к архивам поле total */
  4945.  
  4946. function upgradeArchiveGroup(archiveGroup, index) {
  4947.  
  4948. if (!archiveGroup || !archiveGroup.ts) return archiveGroup;
  4949.  
  4950. if (archiveGroup.me) {
  4951.  
  4952. $.extend(archiveGroup, {
  4953.  
  4954. total:{
  4955.  
  4956. me: countTotalFromArchive(archiveGroup.me, 'me'),
  4957.  
  4958. enemy: countTotalFromArchive(archiveGroup.enemy, 'enemy', archiveGroup.mobId == 66)
  4959.  
  4960. }
  4961.  
  4962. });
  4963.  
  4964. }
  4965.  
  4966. return archiveGroup;
  4967.  
  4968.  
  4969. /* подсчет всех сумм из архива */
  4970.  
  4971. function countTotalFromArchive(archiveGroupFrom) {
  4972.  
  4973. var addTo = {dmgSum: {count:0, sum:0}};
  4974.  
  4975. for (var type in archiveGroupFrom) if (archiveGroupFrom.hasOwnProperty(type)) {
  4976.  
  4977. var vals = archiveGroupFrom[type];
  4978.  
  4979. var isPassive = isActType('PASSIVE', type);
  4980.  
  4981. var isHeal = type === 'heal';
  4982.  
  4983.  
  4984. var count;
  4985.  
  4986. var sum;
  4987.  
  4988. if (typeof vals === 'number') {
  4989.  
  4990. count = vals;
  4991.  
  4992. sum = 0;
  4993.  
  4994. } else {
  4995.  
  4996. count = vals.length;
  4997.  
  4998. sum = vals.reduce(function(pv, cv) { return pv + cv; }, 0) || 0;
  4999.  
  5000. }
  5001.  
  5002.  
  5003. var av;
  5004.  
  5005. var critMin;
  5006.  
  5007. var critVals;
  5008.  
  5009. var critCount = 0;
  5010.  
  5011. addTo[type] = addTo[type] || {sum: 0, count: 0};
  5012.  
  5013. if (type === 'hit' && sum) {
  5014.  
  5015. av = sum/count;
  5016.  
  5017. critMin = av * 1.35;
  5018.  
  5019. critVals = vals.filter(function(item) { return item > critMin;});
  5020.  
  5021. critCount = critVals.length;
  5022.  
  5023. }
  5024.  
  5025. if (critCount) {
  5026.  
  5027. addTo.crit = addTo.crit || {sum: 0, count: 0};
  5028.  
  5029. var critSum = critVals.reduce(function(pv, cv) { return pv + cv; }, 0)||0;
  5030.  
  5031. addTo.crit.count += critCount;
  5032.  
  5033. addTo.crit.sum += critSum;
  5034.  
  5035. }
  5036.  
  5037. addTo[type].count += count;
  5038.  
  5039. addTo[type].sum += sum;
  5040.  
  5041.  
  5042.  
  5043. var typeSumTo = _const.SUM_TO_MAIN[type];
  5044.  
  5045. if (typeSumTo) {
  5046.  
  5047. addTo[typeSumTo] = addTo[typeSumTo] || {sum: 0, count: 0};
  5048.  
  5049. addTo[typeSumTo].sum += sum;
  5050.  
  5051. }
  5052.  
  5053. if (!isHeal) {
  5054.  
  5055. if (!isPassive) {
  5056.  
  5057. addTo.dmgSum.count += count;
  5058.  
  5059. }
  5060.  
  5061. addTo.dmgSum.sum += sum;
  5062.  
  5063. }
  5064.  
  5065. }
  5066.  
  5067. return addTo;
  5068.  
  5069. }
  5070.  
  5071. }
  5072.  
  5073. function downgradeArchiveGroup(fullStats) {
  5074.  
  5075. var archiveGroup = $.extend({}, fullStats);
  5076.  
  5077. delete archiveGroup.total;
  5078.  
  5079. delete archiveGroup.level;
  5080.  
  5081. return archiveGroup;
  5082.  
  5083. }
  5084.  
  5085.  
  5086. function addArchiveGroup(group) {
  5087.  
  5088. var _archiveGroups = _archive.archiveGroups;
  5089.  
  5090. if (!group) return;
  5091.  
  5092. var archiveGroup = countArchiveFromGroup(group);
  5093.  
  5094.  
  5095. if (!archiveGroup) return;
  5096.  
  5097.  
  5098. var lastStatGroup = _archiveGroups[_archiveGroups.length-1];
  5099.  
  5100. if (!lastStatGroup || lastStatGroup.ts[0] === archiveGroup.ts[0]) {
  5101.  
  5102. /* последняя группа обновляется каждый ход*/
  5103.  
  5104. _archiveGroups[_archiveGroups.length-1] = archiveGroup;
  5105.  
  5106. } else if (lastStatGroup.ts[0] < archiveGroup.ts[0]) {
  5107.  
  5108. _archiveGroups.push(archiveGroup);
  5109.  
  5110. } else {
  5111.  
  5112. var isInArr = false;
  5113.  
  5114. for (var i=0; i<_archiveGroups.length; i++) {
  5115.  
  5116. var sg = _archiveGroups[i];
  5117.  
  5118. if (sg.ts[0] === archiveGroup.ts[0]) {
  5119.  
  5120. _archiveGroups[i] = archiveGroup;
  5121.  
  5122. isInArr = true;
  5123.  
  5124. break;
  5125.  
  5126. }
  5127.  
  5128. }
  5129.  
  5130. if (!isInArr) {
  5131.  
  5132. _archiveGroups.push(archiveGroup);
  5133.  
  5134. _archiveGroups.sort(function(a,b) {
  5135.  
  5136. return a.ts[0] - b.ts[0];
  5137.  
  5138. });
  5139.  
  5140. // console.log('add to millde', archiveGroup);
  5141.  
  5142. }
  5143.  
  5144. }
  5145.  
  5146. }
  5147.  
  5148. function loadArchiveGroups() {
  5149.  
  5150. var _archiveGroups = _ext.log.get('archiveGroups') || [];
  5151.  
  5152. _archiveGroups.sort(function(a,b) {
  5153.  
  5154. return a.ts[0] - b.ts[0];
  5155.  
  5156. });
  5157.  
  5158. for (var i=0; i<_archiveGroups.length-1; i++) { /* remove doubles */
  5159.  
  5160. if (_archiveGroups[i].ts[0] === _archiveGroups[i+1].ts[0]) {
  5161.  
  5162. _archiveGroups.splice(i, 1);
  5163.  
  5164. i--;
  5165.  
  5166. }
  5167.  
  5168. }
  5169.  
  5170. _archiveGroups.map(upgradeArchiveGroup);
  5171.  
  5172. _archive.archiveGroups = _archiveGroups;
  5173.  
  5174. }
  5175.  
  5176. function saveArchiveGroups() {
  5177.  
  5178. var _archiveGroups = _archive.archiveGroups;
  5179.  
  5180. var max = _ext.settings.settingsValues.maxArchiveLength || _const.MAX_ARCHIVE_LENGTH;
  5181.  
  5182.  
  5183. _archiveGroups = _archiveGroups.slice(_archiveGroups.length-max);
  5184.  
  5185. var toSave = _archiveGroups.map(downgradeArchiveGroup);
  5186.  
  5187. _ext.log.set('archiveGroups', toSave);
  5188.  
  5189. }
  5190.  
  5191.  
  5192. $.extend(_archive, {
  5193.  
  5194. drawArchiveGroups: drawArchiveGroups,
  5195.  
  5196.  
  5197. loadArchiveGroups: loadArchiveGroups,
  5198.  
  5199. saveArchiveGroups: saveArchiveGroups,
  5200.  
  5201. addArchiveGroup: addArchiveGroup
  5202.  
  5203. });
  5204.  
  5205. return _archive;
  5206.  
  5207. })({});
  5208.  
  5209. _ext.subscribe('init', function() {
  5210.  
  5211. _archive.loadArchiveGroups();
  5212.  
  5213. for (var i=1; i<_groupMessages.list.length; i++) {
  5214.  
  5215. var gr = _groupMessages.list[i];
  5216.  
  5217. _archive.addArchiveGroup(gr);
  5218.  
  5219. }
  5220.  
  5221. _archive.saveArchiveGroups();
  5222.  
  5223.  
  5224.  
  5225. _ext.subscribe('newMessages', function() {
  5226.  
  5227. var group = _groupMessages.list[_groupMessages.list.length - 1];
  5228.  
  5229.  
  5230. _archive.addArchiveGroup(group);
  5231.  
  5232. });
  5233.  
  5234. _ext.subscribe('groupFinished', function(group, index) {
  5235.  
  5236. _archive.addArchiveGroup(group);
  5237.  
  5238. _archive.saveArchiveGroups();
  5239.  
  5240. });
  5241.  
  5242. });
  5243.  
  5244. _elements.getTabInner('archive').on('click', '.group-title', function() {
  5245.  
  5246. var $group = $(this).closest('.group');
  5247.  
  5248. var index = $group.data('index');
  5249.  
  5250. var ts = _archive.archiveGroups[index].ts || [];
  5251.  
  5252. console.log('archive>', _archive.archiveGroups[index], index, new Date(ts[0] * 1000));
  5253.  
  5254. });
  5255.  
  5256. /* eo archive */
  5257.  
  5258.  
  5259.  
  5260.  
  5261. /* stats */
  5262.  
  5263. /* статистика собирается из архива */
  5264.  
  5265. var _stats = (function(_stats) {
  5266.  
  5267. _ext.elements.addTab('stats-side', {zone: 'equip', title: 'стат', content: '<div class="stats" />'});
  5268.  
  5269. _ext.elements.activeTab('stats-side');
  5270.  
  5271. var $stats = _ext.elements.getTabInner('stats-side');
  5272.  
  5273.  
  5274.  
  5275. function addToStats(addTo, addFrom) {
  5276.  
  5277. for (var type in addFrom) if (addFrom.hasOwnProperty(type)) {
  5278.  
  5279. var st = addFrom[type];
  5280.  
  5281. addTo[type] = addTo[type] || {sum: 0, count: 0};
  5282.  
  5283. if (typeof st === 'number') {
  5284.  
  5285. addTo[type].count += st;
  5286.  
  5287. } else {
  5288.  
  5289. addTo[type].count += st.count;
  5290.  
  5291. addTo[type].sum += st.sum;
  5292.  
  5293. }
  5294.  
  5295. }
  5296.  
  5297.  
  5298. }
  5299.  
  5300.  
  5301. function countStatsTotal(archiveGroups, count) {
  5302.  
  5303. archiveGroups = archiveGroups || _archive.archiveGroups;
  5304.  
  5305. count = count || archiveGroups.length;
  5306.  
  5307. var me = {};
  5308.  
  5309. var enemy = {};
  5310.  
  5311. var fights = 0;
  5312.  
  5313. var loot = { pickup: 0, empty: 0, drop: 0};
  5314.  
  5315. var meByMob = {};
  5316.  
  5317. var enemyByMob = {};
  5318.  
  5319. var actionsTimes = {};
  5320.  
  5321. var actionsCounts = {};
  5322.  
  5323. var actionsSum = 0;
  5324.  
  5325. var actionsTime = 0;
  5326.  
  5327. var fightRestTime = 0;
  5328.  
  5329. var otherTime = 0;
  5330.  
  5331. var mobId;
  5332.  
  5333.  
  5334. for (var i=Math.max(archiveGroups.length-count, 0); i<archiveGroups.length; i++) {
  5335.  
  5336. var fullStats = archiveGroups[i];
  5337.  
  5338. var type = fullStats.broken ? 'broken' : fullStats.type;
  5339.  
  5340. if (!fullStats.ts || type==='broken') continue;
  5341.  
  5342. if (type === 'fight') {
  5343.  
  5344. mobId = fullStats.mobId;
  5345.  
  5346. addToStats(me, fullStats.total.me );
  5347.  
  5348. addToStats(enemy, fullStats.total.enemy);
  5349.  
  5350. meByMob[mobId] = meByMob[mobId] || {};
  5351.  
  5352. addToStats(meByMob[mobId], fullStats.total.me);
  5353.  
  5354. enemyByMob[mobId] = enemyByMob[mobId] || {};
  5355.  
  5356. addToStats(enemyByMob[mobId], fullStats.total.enemy);
  5357.  
  5358.  
  5359. fights++;
  5360.  
  5361. var lt = fullStats.loot;
  5362.  
  5363. if (lt) {
  5364.  
  5365. loot[lt]++;
  5366.  
  5367. }
  5368.  
  5369. }
  5370.  
  5371. var nextFullStats = archiveGroups[i+1];
  5372.  
  5373. var timeStart = fullStats.ts[0];
  5374.  
  5375. var timeEnd = fullStats.ts[1];
  5376.  
  5377. if (type !== 'fight' && nextFullStats && nextFullStats.ts) {
  5378.  
  5379. var timeStartNext = nextFullStats.ts[0];
  5380.  
  5381. if (timeStartNext - timeEnd < 120) timeEnd = timeStartNext; // проверка на случай сломанной группы (2)
  5382.  
  5383. }
  5384.  
  5385. var time = timeEnd - timeStart;
  5386.  
  5387. // lastTimeEnd = fullStats.ts[1];
  5388.  
  5389. if (type === 'fight' || type === 'rest') {
  5390.  
  5391. fightRestTime += time;
  5392.  
  5393. } else {
  5394.  
  5395. otherTime += time;
  5396.  
  5397. }
  5398.  
  5399. type = type || 'undefined';
  5400.  
  5401. actionsTimes[type] = (actionsTimes[type] || 0) + time;
  5402.  
  5403. actionsCounts[type] = (actionsCounts[type] | 0) + 1;
  5404.  
  5405. actionsTime += time;
  5406.  
  5407. actionsSum++;
  5408.  
  5409. }
  5410.  
  5411.  
  5412.  
  5413.  
  5414. var statsTotal = {
  5415.  
  5416. fights: fights,
  5417.  
  5418. fightRestTime: fightRestTime,
  5419.  
  5420. otherTime: otherTime,
  5421.  
  5422. actionsTime: actionsTime,
  5423.  
  5424. actionsTimes: actionsTimes,
  5425.  
  5426. actionsCounts: actionsCounts,
  5427.  
  5428. actionsSum: actionsSum,
  5429.  
  5430. loot: loot,
  5431.  
  5432. me: me,
  5433.  
  5434. enemy: enemy,
  5435.  
  5436. meByMob: meByMob,
  5437.  
  5438. enemyByMob: enemyByMob
  5439.  
  5440. };
  5441.  
  5442. if (mobId) statsTotal.lastMobId = mobId;
  5443.  
  5444. return statsTotal;
  5445.  
  5446. }
  5447.  
  5448.  
  5449. _ext.subscribe('settingsChange', function(key, value) {
  5450.  
  5451. if (key==='statsByMob' || key==='statsByMobId' || key==='myStatsByMob' || key==='statsActionsCount' || key==='statsByLevel' || key==='statsByLevelValue') {
  5452.  
  5453. drawStatsSide();
  5454.  
  5455. }
  5456.  
  5457. });
  5458.  
  5459. // var statsActionsCount = _ext.settings.settingsValues.statsActionsCount;
  5460.  
  5461. // var statsByMob = _ext.settings.settingsValues.statsByMob;
  5462.  
  5463. function groupsByLevel(archiveGroups, level) {
  5464.  
  5465. var levelsLog = _ext.log.get('levelsLog') || [];
  5466.  
  5467. if (level) {
  5468.  
  5469. var lv1 = levelsLog.filter(function(item) { return item[1] === level;})[0] || [];
  5470.  
  5471. var lv2 = levelsLog.filter(function(item) { return item[1] === level+1;})[0] || [];
  5472.  
  5473. } else {
  5474.  
  5475. lv1 = levelsLog[levelsLog.length-1];
  5476.  
  5477. lv2 = [];
  5478.  
  5479. }
  5480.  
  5481. var time1 = lv1[0];
  5482.  
  5483. var time2 = lv2[0];
  5484.  
  5485. var i1 = null;
  5486.  
  5487. var i2 = null;
  5488.  
  5489. for (var i=0; i<archiveGroups.length; i++) {
  5490.  
  5491. var ts1 = archiveGroups[i].ts[0];
  5492.  
  5493. var ts2 = archiveGroups[archiveGroups.length - 1 - i].ts[0];
  5494.  
  5495. if (i1===null && ts1>time1) i1 = i;
  5496.  
  5497. if (i2===null && ts2<time2) i2 = archiveGroups.length - 1 - i;
  5498.  
  5499. }
  5500.  
  5501. return (i1!==null && !time2) ? archiveGroups.slice(i1) : archiveGroups.slice(i1|0, i2|0);
  5502.  
  5503. }
  5504.  
  5505. function drawStatsSide(archiveGroups) {
  5506.  
  5507. archiveGroups = archiveGroups || _archive.archiveGroups;
  5508.  
  5509. var groups = _ext.settings.settingsValues.statsByLevel ? groupsByLevel(archiveGroups, _ext.settings.settingsValues.statsByLevelValue) : archiveGroups;
  5510.  
  5511.  
  5512. var statsTotal = countStatsTotal(groups, _ext.settings.settingsValues.statsActionsCount);
  5513.  
  5514. var mobId = _ext.settings.settingsValues.statsByMob && (_ext.settings.settingsValues.statsByMobId || statsTotal.lastMobId);
  5515.  
  5516.  
  5517. var html = '';
  5518.  
  5519. var htmlMe;
  5520.  
  5521. if(_ext.settings.settingsValues.myStatsByMob && mobId) {
  5522.  
  5523. htmlMe = drawStatsSideByActor(statsTotal.meByMob[mobId]);
  5524.  
  5525. } else {
  5526.  
  5527. htmlMe = drawStatsSideByActor(statsTotal.me);
  5528.  
  5529. }
  5530.  
  5531. var htmlEnemy;
  5532.  
  5533. if (mobId) {
  5534.  
  5535. htmlEnemy =
  5536.  
  5537. '<tr class="unhover">' +
  5538.  
  5539. '<td class="stats-against" colspan="5"><a href="/guide/mobs/' + mobId + '" target="_blank">' + _const.MOBS[mobId] + '</a></td>' +
  5540.  
  5541. '</tr>' +
  5542.  
  5543. drawStatsSideByActor(statsTotal.enemyByMob[mobId]);
  5544.  
  5545. } else {
  5546.  
  5547. htmlEnemy =
  5548.  
  5549. '<tr class="unhover">' +
  5550.  
  5551. '<td class="stats-against" colspan="5">бестии</td>' +
  5552.  
  5553. '</tr>' +
  5554.  
  5555. drawStatsSideByActor(statsTotal.enemy);
  5556.  
  5557. }
  5558.  
  5559. var statsTable =
  5560.  
  5561. '<table class="table table-condensed table-noborder table-hover-dark table-stats">' +
  5562.  
  5563. '<tr class="unhover">' +
  5564.  
  5565. '<th class="stats-name"></th>' +
  5566.  
  5567. '<th class="stats-average" title="среднее значение">средн</th>' +
  5568.  
  5569. '<th class="stats-count" title="количество срабатываний из 100 ударов">шанс</th>' +
  5570.  
  5571. '<th class="stats-sum" title="доля от общего урона">урон</th>' +
  5572.  
  5573. '<th class="stats-bonus" title="прибавка к урону от умения">эфф</th>' +
  5574.  
  5575. '</tr>' +
  5576.  
  5577. '<tbody class="stats-me">' +
  5578.  
  5579. htmlMe +
  5580.  
  5581. '</tbody>' +
  5582.  
  5583. '<tbody class="stats-enemy">' +
  5584.  
  5585. htmlEnemy +
  5586.  
  5587. '</tbody>' +
  5588.  
  5589. '</table>';
  5590.  
  5591.  
  5592.  
  5593. var loot = statsTotal.loot;
  5594.  
  5595. var htmlLoot = '<span class="stats-name" title="Поднял/Пусто/Выбросил">' + _icons.pickup + '</span> ' +
  5596.  
  5597. '<span title="Поднял">' + loot.pickup + '</span> / <span title="Пусто">' + loot.empty + '</span> / <span title="Выбросил">' + loot.drop + '</span>';
  5598.  
  5599.  
  5600.  
  5601. var htmlTime =
  5602.  
  5603. '<b>' + statsTotal.actionsSum + '</b> ' + _ext.utils.declenstionByNumber(statsTotal.actionsSum, ['действие', 'действия', 'действий']) + ' за ' + _ext.utils.timeSpan(statsTotal.actionsTime) + '<br />';
  5604.  
  5605. var interestAverageActions = [{
  5606.  
  5607. type: 'fight,rest',
  5608.  
  5609. text: 'бой/отдых',
  5610.  
  5611. icon: '<span class="glyphicon glyphicon-flag"></span>'
  5612.  
  5613. }, {
  5614.  
  5615. type: 'walk',
  5616.  
  5617. text: 'в пути'
  5618.  
  5619. }, {
  5620.  
  5621. type: 'quest,city,nearcity,noeffect',
  5622.  
  5623. text: 'задания'
  5624.  
  5625. }, {
  5626.  
  5627. type: 'idle',
  5628.  
  5629. text: 'безделье'
  5630.  
  5631. }, {
  5632.  
  5633. type: 'dead',
  5634.  
  5635. text: 'воскресание'
  5636.  
  5637. }, {
  5638.  
  5639. type: 'energy',
  5640.  
  5641. text: 'восстановление'
  5642.  
  5643. }, {
  5644.  
  5645. type: 'trade,equip',
  5646.  
  5647. text: 'торговля и экипировка'
  5648.  
  5649. }, {
  5650.  
  5651. type: 'broken,proxy,pvp,undefined',
  5652.  
  5653. text: 'остальное'
  5654.  
  5655. }, {
  5656.  
  5657. title: '<br/><b>В среднем:</b>'
  5658.  
  5659. }, {
  5660.  
  5661. type: 'fight,rest',
  5662.  
  5663. countType: 'fight',
  5664.  
  5665. countAverage: 1,
  5666.  
  5667. icon: '<span class="glyphicon glyphicon-flag"></span>',
  5668.  
  5669. text: 'на бой с учетом отдыха'
  5670.  
  5671. }, {
  5672.  
  5673. type: 'fight',
  5674.  
  5675. countAverage: 1,
  5676.  
  5677. text: 'на бой'
  5678.  
  5679. }, {
  5680.  
  5681. type: 'rest',
  5682.  
  5683. countAverage: 1,
  5684.  
  5685. text: 'на отдых'
  5686.  
  5687. }];
  5688.  
  5689. for (var i=0; i<interestAverageActions.length; i++) {
  5690.  
  5691. var act = interestAverageActions[i];
  5692.  
  5693. if (act.title) {
  5694.  
  5695. htmlTime += act.title + '<br />';
  5696.  
  5697. continue;
  5698.  
  5699. }
  5700.  
  5701.  
  5702. var types = act.type.split(',');
  5703.  
  5704. var count = 0;
  5705.  
  5706. var countTotal = 0;
  5707.  
  5708. var time = 0;
  5709.  
  5710. for (var j=0; j<types.length; j++) {
  5711.  
  5712. var type = types[j];
  5713.  
  5714. time += statsTotal.actionsTimes[type]||0;
  5715.  
  5716. countTotal += statsTotal.actionsCounts[type]||0;
  5717.  
  5718. if (!act.countType) count += statsTotal.actionsCounts[type];
  5719.  
  5720. }
  5721.  
  5722. type = types[0];
  5723.  
  5724. if (act.countType) count = statsTotal.actionsCounts[act.countType];
  5725.  
  5726. var timePercent = Math.round(time / statsTotal.actionsTime * 1000) / 10;
  5727.  
  5728. if (act.countAverage) time = time/count;
  5729.  
  5730.  
  5731.  
  5732. if (time) {
  5733.  
  5734. htmlTime += '<span class="action-icon ' + (act.countType || type) + '" title="' + types.map(function(item) { return _const.ACTION_TYPE_TEXTS[item]; }).join(', ') + '">' + (act.icon || _const.ACTION_TYPE_ICONS[type]) + '</span>';
  5735.  
  5736. if (!act.countAverage) {
  5737.  
  5738. htmlTime += '<span title="' + _ext.utils.timeSpan(time) + '">' + timePercent.toFixed(1) + '%</span> - ';
  5739.  
  5740. } else {
  5741.  
  5742. htmlTime += _ext.utils.timeSpan(time);
  5743.  
  5744. }
  5745.  
  5746. htmlTime += ' ' + act.text + ' (' + countTotal + ')<br />';
  5747.  
  5748. }
  5749.  
  5750. }
  5751.  
  5752.  
  5753.  
  5754. html += statsTable;
  5755.  
  5756. html += '<div class="stats-side stats-loot">' + htmlLoot + '</div>';
  5757.  
  5758. html += '<div class="stats-side stats-time">' + htmlTime + '</div>';
  5759.  
  5760.  
  5761. $stats.html(html);
  5762.  
  5763. }
  5764.  
  5765. function drawStatsSideByActor(stats) {
  5766.  
  5767. var html = '';
  5768.  
  5769. if (!stats) return html;
  5770.  
  5771. var types = [].concat(_const.ACTIVE, ['dmgSum'], _const.PASSIVE);
  5772.  
  5773. for (var i=0; i<types.length; i++) {
  5774.  
  5775. var type = types[i];
  5776.  
  5777. var isDot = isActType('DOT', type);
  5778.  
  5779. var isPassive = isActType('PASSIVE', type);
  5780.  
  5781. var sumTo = _const.SUM_TO_MAIN[type];
  5782.  
  5783. var dmgSum = stats.dmgSum || {};
  5784.  
  5785. var hit = stats.hit || {};
  5786.  
  5787.  
  5788. if (stats[type]) {
  5789.  
  5790. var stat = stats[type];
  5791.  
  5792. var title = _const.ACTION_TRANSLATE[type] +
  5793.  
  5794. (sumTo ? ', включено в ' + _const.ACTION_TRANSLATE[sumTo] :
  5795.  
  5796. (isPassive ? ', не учитывается в сумме' : '' )
  5797.  
  5798. );
  5799.  
  5800. var htmlStat = '<td class="stats-name" title="' + title + '">' + _icons[type] + '</td> ';
  5801.  
  5802.  
  5803. var count = stat.count;
  5804.  
  5805. var sum = stat.sum;
  5806.  
  5807. var average = (Math.round(sum / count * 100) / 100)||0;
  5808.  
  5809. var hitCount = hit.count;
  5810.  
  5811. var hitSum = hit.sum;
  5812.  
  5813. var totalSum = dmgSum.sum;
  5814.  
  5815.  
  5816. var chance = type === 'dmgSum' ? 100 : count/(hitCount + count) * 100;
  5817.  
  5818. var chanceText = type === 'hit' ? '-' : chance>=100 ? Math.round(chance * 10)/10 : chance.toFixed(2);
  5819.  
  5820.  
  5821. var countText = 'сработал ' + _ext.utils.declenstionByNumber(count, ['раз', 'раза', 'раз'], 1);
  5822.  
  5823.  
  5824. if (!sum) {
  5825.  
  5826. htmlStat += '<td class="stats-average"></td>';
  5827.  
  5828. htmlStat += '<td class="stats-count" title="' + countText + '">' + chanceText + '</td>';
  5829.  
  5830. htmlStat += '<td class="stats-count"></td>';
  5831.  
  5832. htmlStat += '<td class="stats-sum"></td>';
  5833.  
  5834. } else {
  5835.  
  5836. var averagePercents = sum/count * hitCount/hitSum * 100;
  5837.  
  5838. var averagePercentsText = Math.round(averagePercents * 100)/100 + '';
  5839.  
  5840.  
  5841. var dmgPercents = sum/totalSum * 100;
  5842.  
  5843. var dmgPercentsText = '' + (dmgPercents<100 ? dmgPercents.toFixed(1) : Math.round(dmgPercents)) + '%';
  5844.  
  5845. var sumText = 'всего ' + _ext.utils.declenstionByNumber(sum, ['урон', 'урона', 'урона'], 1);
  5846.  
  5847.  
  5848. var bonusPercent = Math.round((averagePercents - 100) * chance) / 100;
  5849.  
  5850. var bonusPercentText = bonusPercent && !isDot ?
  5851.  
  5852. (bonusPercent >= 0 ? '+' : '&ndash;') + Math.abs(bonusPercent) + '%' :
  5853.  
  5854. '';
  5855.  
  5856. var bpTranslateText = (averagePercentsText >= 100 ? '+' : '&ndash;') + Math.abs(Math.round(averagePercentsText*100 - 10000)/100) + '% x ' + chanceText;
  5857.  
  5858.  
  5859. htmlStat += '<td class="stats-average" title="' + averagePercentsText + '%">' + average.toFixed(2) + '</td>';
  5860.  
  5861. htmlStat += '<td class="stats-count" title="' + countText + '">' + chanceText + '</td>';
  5862.  
  5863. htmlStat += '<td class="stats-sum" title="' + sumText + '">' + dmgPercentsText + '</td>';
  5864.  
  5865. htmlStat += '<td class="stats-bonus" title="' + bpTranslateText + '">' + bonusPercentText + '</td>';
  5866.  
  5867. }
  5868.  
  5869.  
  5870. html += '<tr class="stats-row stats-row-' + type + '">' + htmlStat + '</tr>';
  5871.  
  5872. }
  5873.  
  5874. }
  5875.  
  5876. return html;
  5877.  
  5878. }
  5879.  
  5880.  
  5881.  
  5882. $.extend(_stats, {
  5883.  
  5884. drawStatsSide: drawStatsSide,
  5885.  
  5886. countStatsTotal: countStatsTotal
  5887.  
  5888. });
  5889.  
  5890. return _stats;
  5891.  
  5892. })({});
  5893.  
  5894.  
  5895. _ext.subscribe('preload', function() {
  5896.  
  5897. _stats.drawStatsSide();
  5898.  
  5899.  
  5900. _ext.subscribe('newMessages', function() {
  5901.  
  5902. _stats.drawStatsSide();
  5903.  
  5904. });
  5905.  
  5906. });
  5907.  
  5908.  
  5909. /* eo stats */
  5910.  
  5911.  
  5912.  
  5913.  
  5914.  
  5915. _ext.archive = _archive;
  5916.  
  5917. _ext.groupMessages = _groupMessages;
  5918.  
  5919. _ext.shortMessages = _shortMessages;
  5920.  
  5921. _ext.towns = _towns;
  5922.  
  5923. _ext.stats = _stats;
  5924.  
  5925.  
  5926.  
  5927.  
  5928.  
  5929.  
  5930. var _auto = (function(_auto) {
  5931.  
  5932. var sets = [{
  5933.  
  5934. title: 'Помощь герою (<span class="link-ajax" data-auto="rest">миролюбие</span>/<span class="link-ajax" data-auto="fight">агрессия</span>)',
  5935.  
  5936. fields: [{
  5937.  
  5938. label: 'Автоматическая помощь',
  5939.  
  5940. note: 'Внимание! Настройки применяются в момент нажатия',
  5941.  
  5942. name: 'autohelp',
  5943.  
  5944. isToggle: 1,
  5945.  
  5946. value: false,
  5947.  
  5948. subs: [{
  5949.  
  5950. label: 'Уведомлять о помощи',
  5951.  
  5952. name: 'autohelpNotify',
  5953.  
  5954. isToggle: 1,
  5955.  
  5956. value: false
  5957.  
  5958. }, {
  5959.  
  5960. label: 'Использовать бонусную энергию, оставить',
  5961.  
  5962. note: 'Указанный запас энергии тратиться не будет',
  5963.  
  5964. name: 'autohelpEnergyBonus',
  5965.  
  5966. isToggle: 1,
  5967.  
  5968. value: false,
  5969.  
  5970. inputs: [{
  5971.  
  5972. name: 'autohelpEnergyBonusMax',
  5973.  
  5974. type: 'num'
  5975.  
  5976. }]
  5977.  
  5978. }, {
  5979.  
  5980. label: 'Герой бездействует',
  5981.  
  5982. name: 'autohelpIdle',
  5983.  
  5984. isToggle: 1,
  5985.  
  5986. value: true
  5987.  
  5988. }, {
  5989.  
  5990. label: 'Герой воскресает',
  5991.  
  5992. name: 'autohelpDead',
  5993.  
  5994. isToggle: 1,
  5995.  
  5996. value: true
  5997.  
  5998. }, {
  5999.  
  6000. label: 'Здоровье в бою ниже ',
  6001.  
  6002. name: 'autohelpHp',
  6003.  
  6004. isToggle: 1,
  6005.  
  6006. value: true,
  6007.  
  6008. inputs: [{
  6009.  
  6010. type: 'num',
  6011.  
  6012. name: 'autohelpHpLowerValue',
  6013.  
  6014. value: 200
  6015.  
  6016. }],
  6017.  
  6018. subs: [{
  6019.  
  6020. label: 'только против босса',
  6021.  
  6022. name: 'autohelpHpBoss',
  6023.  
  6024. isToggle: 1,
  6025.  
  6026. isInline: 1,
  6027.  
  6028. value: true
  6029.  
  6030. }]
  6031.  
  6032. }, {
  6033.  
  6034. label: 'Лечение спутника, если его здоровье меньше',
  6035.  
  6036. name: 'autohelpСompanion',
  6037.  
  6038. isToggle: 1,
  6039.  
  6040. isInline: 1,
  6041.  
  6042. value: true,
  6043.  
  6044. inputs: [{
  6045.  
  6046. type: 'num',
  6047.  
  6048. name: 'autohelpСompanionHp',
  6049.  
  6050. isInline: 1,
  6051.  
  6052. value: 20
  6053.  
  6054. }]
  6055.  
  6056. }, {
  6057.  
  6058. label: 'Энергия выше ',
  6059.  
  6060. name: 'autohelpEnergy',
  6061.  
  6062. isToggle: 1,
  6063.  
  6064. value: true,
  6065.  
  6066. inputs: [{
  6067.  
  6068. type: 'num',
  6069.  
  6070. name: 'autohelpEnergyGreaterValue',
  6071.  
  6072. value: 10
  6073.  
  6074. }],
  6075.  
  6076. subs: [{
  6077.  
  6078. label: 'в бою',
  6079.  
  6080. name: 'autohelpEnergyFight',
  6081.  
  6082. isToggle: 1,
  6083.  
  6084. isInline: 1,
  6085.  
  6086. value: false
  6087.  
  6088. }, {
  6089.  
  6090. label: 'вне боя',
  6091.  
  6092. name: 'autohelpEnergyRest',
  6093.  
  6094. isToggle: 1,
  6095.  
  6096. isInline: 1,
  6097.  
  6098. value: false
  6099.  
  6100. }, {
  6101.  
  6102. label: 'в пути',
  6103.  
  6104. name: 'autohelpEnergyWalk',
  6105.  
  6106. isToggle: 1,
  6107.  
  6108. isInline: 1,
  6109.  
  6110. value: true
  6111.  
  6112. }, {
  6113.  
  6114. label: 'торг/медитация',
  6115.  
  6116. name: 'autohelpEnergyTradeMed',
  6117.  
  6118. isToggle: 1,
  6119.  
  6120. isInline: 1,
  6121.  
  6122. value: true
  6123.  
  6124. }/*, {
  6125.  
  6126. label: 'Починка',
  6127.  
  6128. name: 'autohelpEnergyRepairBuilding',
  6129.  
  6130. isToggle: 1,
  6131.  
  6132. value: false,
  6133.  
  6134. inputs: [{
  6135.  
  6136. type: 'num',
  6137.  
  6138. addOn: 'ID',
  6139.  
  6140. name: 'autohelpEnergyRepairBuildingID'
  6141.  
  6142. }, {
  6143.  
  6144. type: 'num',
  6145.  
  6146. addOn: 'X',
  6147.  
  6148. name: 'autohelpEnergyRepairBuildingX'
  6149.  
  6150. }, {
  6151.  
  6152. type: 'num',
  6153.  
  6154. addOn: 'Y',
  6155.  
  6156. name: 'autohelpEnergyRepairBuildingY'
  6157.  
  6158. }, {
  6159.  
  6160. type: 'num',
  6161.  
  6162. addOn: '%',
  6163.  
  6164. name: 'autohelpEnergyRepairBuildingPercent'
  6165.  
  6166. }]
  6167.  
  6168. }*/]
  6169.  
  6170. }]
  6171.  
  6172. }, {
  6173.  
  6174. label: 'Автоматический выбор в задании',
  6175.  
  6176. name: 'autoquest',
  6177.  
  6178. isToggle: 1,
  6179.  
  6180. value: false,
  6181.  
  6182. subs: [{
  6183.  
  6184. label: 'Уведомлять о выборе',
  6185.  
  6186. name: 'autoquestNotify',
  6187.  
  6188. isToggle: 1,
  6189.  
  6190. value: false
  6191.  
  6192. }, {
  6193.  
  6194. label: 'Задания, влияющие на честь',
  6195.  
  6196. // note: 'Доставка, сопроводить караван, пошпионить',
  6197.  
  6198. name: 'autoquestHonor',
  6199.  
  6200. isToggle: 1,
  6201.  
  6202. value: true,
  6203.  
  6204. subs: [{
  6205.  
  6206. label: 'честно',
  6207.  
  6208. name: 'autoquestHonorPlus',
  6209.  
  6210. isToggle: 1,
  6211.  
  6212. isInline: 1,
  6213.  
  6214. value: true
  6215.  
  6216. }, {
  6217.  
  6218. label: 'бесчестно',
  6219.  
  6220. name: 'autoquestHonorMinus',
  6221.  
  6222. isToggle: 1,
  6223.  
  6224. isInline: 1,
  6225.  
  6226. value: false
  6227.  
  6228. }]
  6229.  
  6230. }, {
  6231.  
  6232. label: 'Задания, влияющие на миролюбие',
  6233.  
  6234. // note: 'Выбить долг',
  6235.  
  6236. name: 'autoquestPeace',
  6237.  
  6238. isToggle: 1,
  6239.  
  6240. value: true,
  6241.  
  6242. subs: [{
  6243.  
  6244. label: 'миролюбиво',
  6245.  
  6246. name: 'autoquestPeacePlus',
  6247.  
  6248. isToggle: 1,
  6249.  
  6250. isInline: 1,
  6251.  
  6252. value: true
  6253.  
  6254. }, {
  6255.  
  6256. label: 'воинственно',
  6257.  
  6258. name: 'autoquestPeaceMinus',
  6259.  
  6260. isToggle: 1,
  6261.  
  6262. isInline: 1,
  6263.  
  6264. value: false
  6265.  
  6266. }]
  6267.  
  6268. }]
  6269.  
  6270. }, {
  6271.  
  6272. label: 'Брать карты',
  6273.  
  6274. name: 'autocard',
  6275.  
  6276. isToggle: 1,
  6277.  
  6278. value: false
  6279.  
  6280. }]
  6281.  
  6282. }];
  6283.  
  6284.  
  6285. _ext.subscribe('preload', function(gameData) {
  6286.  
  6287. if (gameData) {
  6288.  
  6289. var hero = gameData.account.hero;
  6290.  
  6291. var energyBonus = hero.energy.bonus;
  6292.  
  6293. var $inputMax = _ext.settings.getSettingInput('autohelpEnergyBonusMax');
  6294.  
  6295. if (!+$inputMax.val()) $inputMax.val(Math.max(0, energyBonus-10)).trigger('change');
  6296.  
  6297. }
  6298.  
  6299. });
  6300.  
  6301.  
  6302. function initSets() {
  6303.  
  6304. _ext.settings.addSets(sets);
  6305.  
  6306. _ext.settings.drawSets(sets);
  6307.  
  6308.  
  6309. var $sets = _ext.elements.getTabInner('sets');
  6310.  
  6311. $sets.on('click', '[data-auto]', function(e) {
  6312.  
  6313. e.preventDefault();
  6314.  
  6315. var type = $(this).data('auto');
  6316.  
  6317. var types = {
  6318.  
  6319. rest: {
  6320.  
  6321. autohelpIdle: 1,
  6322.  
  6323. autohelpDead: 1,
  6324.  
  6325. autohelpHp: 1,
  6326.  
  6327. autohelpHpBoss: 1,
  6328.  
  6329. autohelpEnergy: 1,
  6330.  
  6331. autohelpEnergyFight: 0,
  6332.  
  6333. autohelpEnergyRest: 0,
  6334.  
  6335. autohelpEnergyWalk: 1,
  6336.  
  6337. autohelpEnergyTradeMed: 1
  6338.  
  6339. },
  6340.  
  6341. fight: {
  6342.  
  6343. autohelpIdle: 0,
  6344.  
  6345. autohelpDead: 0,
  6346.  
  6347. autohelpHp: 1,
  6348.  
  6349. autohelpHpBoss: 0,
  6350.  
  6351. autohelpEnergy: 1,
  6352.  
  6353. autohelpEnergyFight: 1,
  6354.  
  6355. autohelpEnergyRest: 0,
  6356.  
  6357. autohelpEnergyWalk: 0,
  6358.  
  6359. autohelpEnergyTradeMed: 0
  6360.  
  6361. }
  6362.  
  6363. };
  6364.  
  6365. var conf = types[type];
  6366.  
  6367. for (var key in conf) if (conf.hasOwnProperty(key)) {
  6368.  
  6369. var value = !!conf[key];
  6370.  
  6371. var $input = _ext.settings.getSettingInput(key);
  6372.  
  6373. $input.prop('checked', value).trigger('change');
  6374.  
  6375. }
  6376.  
  6377. })
  6378.  
  6379. }
  6380.  
  6381.  
  6382. function checkHero(gameData) {
  6383.  
  6384. var _settingsValues = _ext.settings.settingsValues;
  6385.  
  6386.  
  6387. var hero = gameData.account.hero;
  6388.  
  6389. var actionType = hero.action.type;
  6390.  
  6391. var actionPercent = hero.action.percents;
  6392.  
  6393. var actionName = _ext.const.ACTION_TYPE_NAMES[actionType];
  6394.  
  6395.  
  6396. var energy = hero.energy.value;
  6397.  
  6398. var energyBonus = _settingsValues.autohelpEnergyBonus ? hero.energy.bonus - _settingsValues.autohelpEnergyBonusMax : 0 ;
  6399.  
  6400. if (energyBonus < 0) energyBonus = 0;
  6401.  
  6402.  
  6403. if (energy + energyBonus < 4) return;
  6404.  
  6405.  
  6406.  
  6407. var isFight = actionName === 'fight';
  6408.  
  6409. var isRest = actionType < 10 && actionName !== 'fight';
  6410.  
  6411.  
  6412. var isBoss = !!hero.action.is_boss;
  6413.  
  6414. if (_settingsValues.autohelpHp && hero.base.health < _settingsValues.autohelpHpLowerValue && isFight && (!_settingsValues.autohelpHpBoss || isBoss)) {
  6415.  
  6416. godHelp('Низкое здоровье: ' + hero.base.health);
  6417.  
  6418. return;
  6419.  
  6420. }
  6421.  
  6422.  
  6423. var isHeplingСompanion = actionName === 'companionHelp';
  6424.  
  6425. if (_settingsValues.autohelpСompanion && isHeplingСompanion && hero.companion.health < _settingsValues.autohelpСompanionHp) {
  6426.  
  6427. godHelp('Низкое здоровье спутника: ' + hero.companion.health);
  6428.  
  6429.  
  6430. return;
  6431.  
  6432. }
  6433.  
  6434.  
  6435. if (_settingsValues.autohelpEnergy && energy > _settingsValues.autohelpEnergyGreaterValue && (
  6436.  
  6437. (_settingsValues.autohelpEnergyFight && isFight) ||
  6438.  
  6439. (_settingsValues.autohelpEnergyRest && isRest) ||
  6440.  
  6441. (_settingsValues.autohelpEnergyWalk && actionName === 'walk') ||
  6442.  
  6443. (_settingsValues.autohelpEnergyTradeMed && (actionName === 'trade' || actionName === 'energy'))
  6444.  
  6445. )) {
  6446.  
  6447. //console.log('check passed', energy, _settingsValues.autohelpEnergyGreaterValue)
  6448.  
  6449. // if (_settingsValues.autohelpEnergyRepairBuilding && _settingsValues.autohelpEnergyRepairBuildingID) {
  6450.  
  6451. // var x = _settingsValues.autohelpEnergyRepairBuildingX;
  6452.  
  6453. // var y = _settingsValues.autohelpEnergyRepairBuildingY;
  6454.  
  6455. // getBuildingState(x,y)
  6456.  
  6457. // .done(function(integrity) {
  6458.  
  6459. // if (integrity && integrity < (_settingsValues.autohelpEnergyRepairBuildingPercent/100 || 0.982)) {
  6460.  
  6461. // godHelp('repair', 'building_repair', {building: _settingsValues.autohelpEnergyRepairBuildingID});
  6462.  
  6463. // }
  6464.  
  6465. // });
  6466.  
  6467. // } else {
  6468.  
  6469. godHelp('Накопилась энергия: ' + energy);
  6470.  
  6471.  
  6472. return;
  6473.  
  6474. }
  6475.  
  6476. if (_settingsValues.autohelpIdle && actionType === 0 && !isFight && actionPercent > 0 && actionPercent < 0.8) {
  6477.  
  6478. godHelp('Герой бездействует');
  6479.  
  6480. return;
  6481.  
  6482. }
  6483.  
  6484. if (_settingsValues.autohelpDead && actionType === 4 && !isFight && actionPercent > 0 && actionPercent < 0.8) {
  6485.  
  6486. godHelp('Герой умер');
  6487.  
  6488. return;
  6489.  
  6490. }
  6491.  
  6492.  
  6493. var hasCard = !$('.pgf-get-card-button').hasClass('pgf-hidden');
  6494.  
  6495. if (_settingsValues.autocard && hasCard) {
  6496.  
  6497. $('.pgf-get-card-button a').trigger('click');
  6498.  
  6499. return;
  6500.  
  6501. }
  6502.  
  6503.  
  6504. function godHelp(msg, ability, getParams) {
  6505.  
  6506. ability = ability || 'help';
  6507.  
  6508. console.log('god ' + ability + '!', getParams, actionName, msg, $.extend({}, hero));
  6509.  
  6510. if (_settingsValues.autohelpNotify) {
  6511.  
  6512. _ext.notification.sendNotify('The Tale Extended - ' + ext.heroName, {
  6513.  
  6514. tag: 'autohelp',
  6515.  
  6516. body: 'Сработала автоматическая помощь ' +
  6517.  
  6518. '\n' + msg + '' +
  6519.  
  6520. '\nТекущее действие: ' + _ext.const.ACTION_TYPE_TEXTS[actionName] + '',
  6521.  
  6522. addTime: 1
  6523.  
  6524. });
  6525.  
  6526. }
  6527.  
  6528. // console.log('godHelp! real');
  6529.  
  6530. if (!_settingsValues.autohelp) return;
  6531.  
  6532. var paramsStr = '';
  6533.  
  6534. for (var key in getParams) if (getParams.hasOwnProperty(key)) {
  6535.  
  6536. paramsStr += '&' + key + '=' + getParams[key];
  6537.  
  6538. }
  6539.  
  6540. var url = '/game/abilities/' + ability + '/api/use?api_version=1.0&api_client='+window.API_CLIENT + paramsStr;
  6541.  
  6542. // console.log('url: ', url)
  6543.  
  6544. // if (!_settingsValues.autohelp) return;
  6545.  
  6546. hero.energy.value -= 4;
  6547.  
  6548. $.ajax({
  6549.  
  6550. url: url,
  6551.  
  6552. dataType: 'json',
  6553.  
  6554. type: 'post',
  6555.  
  6556. data: {}
  6557.  
  6558. })
  6559.  
  6560. }
  6561.  
  6562. }
  6563.  
  6564.  
  6565.  
  6566. var CHOICES = {
  6567.  
  6568. /* peacefullnes */
  6569.  
  6570. peacePlus: [
  6571.  
  6572. 'прибегнуть к дипломатии' /* collect_debt */
  6573.  
  6574. ],
  6575.  
  6576. peaceMinus: [
  6577.  
  6578. 'задействовать грубую силу' /* collect_debt */
  6579.  
  6580. ],
  6581.  
  6582. honorPlus: [
  6583.  
  6584. 'довести дело до конца', /* spying */
  6585.  
  6586. 'защищать торговца', /* caravan */
  6587.  
  6588. 'честно выполнить свои обязательства' /* delivery */
  6589.  
  6590. ],
  6591.  
  6592. honorMinus: [
  6593.  
  6594. 'поддаться укорам совести и раскрыться', /* spying */
  6595.  
  6596. 'шантажировать самостоятельно', /* spying */
  6597.  
  6598. 'присвоить письмо и продать', /* delivery */
  6599.  
  6600. 'украсть-украсть-украсть', /* delivery */
  6601.  
  6602. 'подделать письмо' /* delivery */
  6603.  
  6604. ]
  6605.  
  6606. };
  6607.  
  6608.  
  6609. var lastquest = '';
  6610.  
  6611. function checkQuest(gameData) {
  6612.  
  6613. var _settingsValues = _ext.settings.settingsValues;
  6614.  
  6615. var selectChoices = {};
  6616.  
  6617. if (_settingsValues.autoquestPeacePlus) selectChoices.peacePlus = 1;
  6618.  
  6619. if (_settingsValues.autoquestPeaceMinus) selectChoices.peaceMinus = 1;
  6620.  
  6621. if (_settingsValues.autoquestHonorPlus) selectChoices.honorPlus = 1;
  6622.  
  6623. if (_settingsValues.autoquestHonorMinus) selectChoices.honorMinus = 1;
  6624.  
  6625.  
  6626. var hero = gameData.account.hero;
  6627.  
  6628. var quests = hero.quests.quests;
  6629.  
  6630. var line = quests[1].line;
  6631.  
  6632. for (var i=0; i<line.length; i++) {
  6633.  
  6634. var q = line[i];
  6635.  
  6636. for (var choiceIndex=0; choiceIndex < q.choice_alternatives.length; choiceIndex++) {
  6637.  
  6638. var choiceName = q.choice_alternatives[choiceIndex][1];
  6639.  
  6640. var option_uid = q.choice_alternatives[choiceIndex][0];
  6641.  
  6642. for (var reward in CHOICES) {
  6643.  
  6644. if (CHOICES[reward].indexOf(choiceName) >=0 ) {
  6645.  
  6646. if (selectChoices[reward]) {
  6647.  
  6648. chooseQuest(option_uid, choiceName);
  6649.  
  6650. }
  6651.  
  6652. }
  6653.  
  6654. }
  6655.  
  6656. }
  6657.  
  6658. }
  6659.  
  6660. function chooseQuest(uid, name) {
  6661.  
  6662. if (_settingsValues.autoquestNotify && lastquest != name) {
  6663.  
  6664. lastquest = name;
  6665.  
  6666. _ext.notification.sendNotify('The Tale Extended - ' + ext.heroName, {
  6667.  
  6668. tag: 'autoquest',
  6669.  
  6670. body: 'Сделан выбор! \n— ' + name + '',
  6671.  
  6672. addTime: 1,
  6673.  
  6674. icon: window.extPass + 'img/quest/caravan.png'
  6675.  
  6676. });
  6677.  
  6678. }
  6679.  
  6680.  
  6681. if (!_settingsValues.autoquest) return;
  6682.  
  6683. $.ajax({
  6684.  
  6685. url: '/game/quests/api/choose?api_version=1.0&api_client='+window.API_CLIENT+'&option_uid='+encodeURIComponent(uid),
  6686.  
  6687. dataType: 'json',
  6688.  
  6689. type: 'post',
  6690.  
  6691. data: {
  6692.  
  6693. }
  6694.  
  6695. })
  6696.  
  6697. }
  6698.  
  6699. }
  6700.  
  6701.  
  6702.  
  6703.  
  6704. function getBuildingState(x,y) {
  6705.  
  6706. var dfr = $.Deferred();
  6707.  
  6708. var name = 'integrity-' + x + '-' + y;
  6709.  
  6710. if (_ext.cache(name)) {
  6711.  
  6712. dfr.resolve(_ext.cache(name));
  6713.  
  6714. } else {
  6715.  
  6716. requestPlaceHtml(x,y)
  6717.  
  6718. .done(function(html) {
  6719.  
  6720. // var $info = $(html);
  6721.  
  6722. var integrityParse = /data-building-integrity="([\d.]+)"/.exec(html);
  6723.  
  6724. var integrity = integrityParse && integrityParse[1];
  6725.  
  6726. _ext.cache(name, integrity, 10 * 60 * 1000);
  6727.  
  6728. console.log('integrity request:', integrity)
  6729.  
  6730. dfr.resolve(integrity);
  6731.  
  6732. });
  6733.  
  6734. }
  6735.  
  6736. return dfr;
  6737.  
  6738. }
  6739.  
  6740.  
  6741. function requestPlaceHtml(x,y) {
  6742.  
  6743. return $.ajax({
  6744.  
  6745. url: '/game/map/cell-info?x=' + x + '&y=' + y + '&_=' + (+new Date),
  6746.  
  6747. method: 'get',
  6748.  
  6749. dataType: 'html'
  6750.  
  6751. })
  6752.  
  6753. }
  6754.  
  6755.  
  6756.  
  6757. $.extend(_auto, {
  6758.  
  6759. checkHero: checkHero,
  6760.  
  6761. checkQuest: checkQuest,
  6762.  
  6763. initSets: initSets
  6764.  
  6765. });
  6766.  
  6767. return _auto;
  6768.  
  6769. })({});
  6770.  
  6771. _ext.subscribe('newTurn', function(messagesNew, gameData) {
  6772.  
  6773. window.setTimeout(function() {
  6774.  
  6775. _auto.checkHero(gameData);
  6776.  
  6777. }, 1000);
  6778.  
  6779. });
  6780.  
  6781. _ext.subscribe('newMessages', function(messagesNew, gameData) {
  6782.  
  6783. window.setTimeout(function() {
  6784.  
  6785. _auto.checkQuest(gameData);
  6786.  
  6787. }, 1000);
  6788.  
  6789. });
  6790.  
  6791.  
  6792. _ext.subscribe('init', function() {
  6793.  
  6794. _auto.initSets();
  6795.  
  6796. });
  6797.  
  6798. _ext.auto = _auto;
  6799.  
  6800.  
  6801.  
  6802. _ext.publish('init');
  6803.  
  6804. return _ext;
  6805. })(window.ext || {});
  6806.  
  6807. //console.log('tables.js')
  6808.  
  6809. window.tables = (function(_tables) {
  6810.  
  6811. "use strict";
  6812.  
  6813.  
  6814. function makeSortable($table) {
  6815.  
  6816. var $head = $table.find('thead tr').first();
  6817.  
  6818. var $rows = $table.find('tbody tr');
  6819.  
  6820.  
  6821. $head.children('th')
  6822.  
  6823. .wrapInner('<span class="sort" />')
  6824.  
  6825. .each(function(){
  6826.  
  6827.  
  6828. var $th = $(this);
  6829.  
  6830. var thIndex = $th.index();
  6831.  
  6832.  
  6833. $th.children('.sort').click(function(){
  6834.  
  6835. $th.siblings('th').children('.sort').attr('class', 'sort');
  6836.  
  6837. var inverse = $(this).hasClass('sort-up');
  6838.  
  6839. $(this).attr('class', 'sort sort-' + (inverse ? 'down' : 'up'));
  6840.  
  6841. var arr = [];
  6842.  
  6843. $rows.each(function() {
  6844.  
  6845. var valueText = $.trim($(this).children('td').eq(thIndex).text());
  6846.  
  6847. var value;
  6848.  
  6849.  
  6850. value = (-valueText) || parseDate(valueText) || valueText;
  6851.  
  6852.  
  6853.  
  6854. arr.push({
  6855.  
  6856. $item: this,
  6857.  
  6858. value: value
  6859.  
  6860. })
  6861.  
  6862. });
  6863.  
  6864. arr.sort(function(a,b) {
  6865.  
  6866. if (a.value == b.value) return 0;
  6867.  
  6868. return a.value > b.value ?
  6869.  
  6870. inverse ? 1 : -1
  6871.  
  6872. : inverse ? -1 : 1;
  6873.  
  6874. })
  6875.  
  6876. arr.forEach(function(item) {
  6877.  
  6878. $table.append(item.$item)
  6879.  
  6880. })
  6881.  
  6882. });
  6883.  
  6884.  
  6885. });
  6886.  
  6887. }
  6888.  
  6889.  
  6890.  
  6891. $('.table').each(function() {
  6892.  
  6893. makeSortable($(this))
  6894.  
  6895. });
  6896.  
  6897.  
  6898. $.extend(_tables, {
  6899.  
  6900. makeSortable: makeSortable
  6901.  
  6902. });
  6903.  
  6904.  
  6905. function parseDate(str) { //02.04.2014 10:50
  6906.  
  6907. var p = /(\d{2})\.(\d{2})\.(\d{4})\s(\d{1,2})\:(\d{2})/.exec(str);
  6908.  
  6909. if (!p) return 0;
  6910.  
  6911. return +new Date(p[3],p[2]-1,p[1],p[4],p[5]);
  6912.  
  6913. }
  6914.  
  6915.  
  6916. return _tables;
  6917.  
  6918. })({});
  6919.  
  6920.  
  6921.  
  6922. injectDone();
  6923. })((typeof this.unsafeWindow !== 'undefined') ? this.unsafeWindow : window);