├── BloodSpatterAndStatusMarkers.js
├── CashMaster.js
├── readme.md
└── turnMarker.js
/BloodSpatterAndStatusMarkers.js:
--------------------------------------------------------------------------------
1 | var BloodSpatter = {
2 | version: '0.1.0',
3 | wiki: 'https://wiki.roll20.net/Script:Blood_And_Honor:_Automatic_blood_spatter,_pooling_and_trail_effects',
4 | hpBar: 3, //1, 2, or 3
5 | hpCountUp: false,
6 | baseSize: 0.75,
7 | // This will make it so only the GM can use the !clearblood command. Change to "true" if you want to check for authorization.
8 | onlyAllowGMtoRunCommands: true,
9 |
10 | // YOU MUST ADD YOUR OWN SPATTERS AND POOLS TO YOUR LIBRARY AND GET THE IMAGE LINK VIA YOUR WEB BROWSER. FOLLOW THE INSTRUCTIONS HERE: https://wiki.roll20.net/API:Objects#imgsrc_and_avatar_property_restrictions
11 | // You can add as many as you'd like to either category. Spatters are also used for blood trails.
12 | spatters: [
13 | {
14 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638270/8nmNImR20gW1hYYonWU9EA/thumb.png?1439938892',
15 | 'width': 200,
16 | 'height': 100
17 | },
18 | {
19 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638265/F2Mw_sx4YKSZEhQcoy2ncw/thumb.png?1439938884',
20 | 'width': 200,
21 | 'height': 83
22 | },
23 | {
24 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638261/-1Gv5e3O5f4h_JeFLH5A7g/thumb.png?1439938877',
25 | 'width': 200,
26 | 'height': 40
27 | },
28 | {
29 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638259/xZxDeGB2Tr_q3UEt082_lw/thumb.png?1439938869',
30 | 'width': 200,
31 | 'height': 134
32 | },
33 | {
34 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638254/v6LFCczCxnX3AYmkngA8ow/thumb.png?1439938857',
35 | 'width': 200,
36 | 'height': 46
37 | },
38 | {
39 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638252/H13406ht412U_WRgYVtCwg/thumb.png?1439938846',
40 | 'width': 200,
41 | 'height': 179
42 | },
43 | {
44 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638246/5Sp_Dk2MR-_EMBpJLLiXcA/thumb.png?1439938833',
45 | 'width': 200,
46 | 'height': 146
47 | },
48 | {
49 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638244/iIJPw_YBucZWYMknD028Gw/thumb.png?1439938822',
50 | 'width': 200,
51 | 'height': 139
52 | },
53 | {
54 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638242/A7eIviKjyG6x9Q5wxy8xkQ/thumb.png?1439938811',
55 | 'width': 200,
56 | 'height': 100
57 | },
58 | {
59 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638235/ViN4vWlxbL1PSxkjOFASLg/thumb.png?1439938798',
60 | 'width': 200,
61 | 'height': 134
62 | },
63 | {
64 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638231/kKVxusJuBQMxDUSzNk0QAg/thumb.png?1439938780',
65 | 'width': 200,
66 | 'height': 40
67 | },
68 | {
69 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638226/nN6P5U5vWRoaAd5-XMS5dA/thumb.png?1439938768',
70 | 'width': 200,
71 | 'height': 99
72 | },
73 | {
74 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638217/ZZaTOGr5pPNY01RZKapl6g/thumb.png?1439938744',
75 | 'width': 200,
76 | 'height': 72
77 | },
78 | {
79 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638212/_eaJ7fr_QPWWEW7PmvPzqg/thumb.png?1439938727',
80 | 'width': 200,
81 | 'height': 35
82 | },
83 | {
84 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638206/RE2PGgNz11RZtV1rIzHfpA/thumb.png?1439938712',
85 | 'width': 200,
86 | 'height': 51
87 | },
88 | {
89 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638204/U90Jjp_cIYQRJsT_4XM8eA/thumb.png?1439938698',
90 | 'width': 200,
91 | 'height': 31
92 | },
93 | {
94 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638202/vRaCrr-SoQd1DORtK9IGfg/thumb.png?1439938687',
95 | 'width': 200,
96 | 'height': 130
97 | },
98 | {
99 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638194/JZgn6ALhb3DuCWeSEd2S8g/thumb.png?1439938672',
100 | 'width': 200,
101 | 'height': 182
102 | },
103 | {
104 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638185/BDOgQLzTlXPeWZHpt_AOYA/thumb.png?1439938657',
105 | 'width': 200,
106 | 'height': 100
107 | },
108 | {
109 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638182/nBWgsf-zsN6wHVTKAXMY1g/thumb.png?1439938643',
110 | 'width': 200,
111 | 'height': 146
112 | },
113 | {
114 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638174/qJvQwWJr20D7ShU6bBJ8Qw/thumb.png?1439938628',
115 | 'width': 200,
116 | 'height': 89
117 | },
118 | {
119 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638170/Qe-4mHgUs2zsYGDpjU_5TA/thumb.png?1439938618',
120 | 'width': 200,
121 | 'height': 193
122 | },
123 | {
124 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638163/rd-CJLyawUbLAb0YVxIAxg/thumb.png?1439938604',
125 | 'width': 200,
126 | 'height': 88
127 | }
128 | ],
129 | pools: [
130 | {
131 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638609/oowTOShdIP-PLoxtz-IkCg/thumb.png?1439939580',
132 | 'width': 181,
133 | 'height': 200
134 | },
135 | {
136 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638598/DIlJabyrE3ducucpa93v6A/thumb.png?1439939571',
137 | 'width': 200,
138 | 'height': 161
139 | },
140 | {
141 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638587/QPdMU5VryXp86pf5ZFsvDA/thumb.png?1439939564',
142 | 'width': 200,
143 | 'height': 128
144 | },
145 | {
146 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638577/havO5udqh2-bjX81r_V51g/thumb.png?1439939550',
147 | 'width': 200,
148 | 'height': 155
149 | },
150 | {
151 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638566/jxAyAP-FK5s52tcfV_oaUw/thumb.png?1439939537',
152 | 'width': 200,
153 | 'height': 77
154 | },
155 | {
156 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638558/xZgwCOqTFYTU4_hfKsDUAg/thumb.png?1439939528',
157 | 'width': 200,
158 | 'height': 196
159 | },
160 | {
161 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638550/VWjQJxKhtJ1qQrLb-Z4Cvg/thumb.png?1439939518',
162 | 'width': 180,
163 | 'height': 200
164 | },
165 | {
166 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638541/Xb1KTMqgty5tAEcr_-53Jg/thumb.png?1439939506',
167 | 'width': 200,
168 | 'height': 173
169 | },
170 | {
171 | 'src': 'https://s3.amazonaws.com/files.d20.io/images/11638533/a3xycjVLmVKHainc6a3htg/thumb.png?1439939496',
172 | 'width': 200,
173 | 'height': 200
174 | }
175 | ],
176 | chooseBlood: function (type) {
177 | if (type === 'spatter') {
178 | return BloodSpatter.spatters[randomInteger(BloodSpatter.spatters.length) - 1];
179 | }
180 | if (type === 'pool') {
181 | return BloodSpatter.pools[randomInteger(BloodSpatter.pools.length) - 1];
182 | }
183 | },
184 | getOffset: function () {
185 | return Math.random() < 0.5 ? 1 : -1;
186 | },
187 | bloodColor: function (gmnotes) {
188 | if (gmnotes.indexOf('bloodcolor_purple') !== -1) {
189 | return '#0000ff';
190 | } else if (gmnotes.indexOf('bloodcolor_blue') !== -1) {
191 | return '#00ffff';
192 | } else if (gmnotes.indexOf('bloodcolor_orange') !== -1) {
193 | return '#ffff00';
194 | } else {
195 | return 'transparent';
196 | }
197 | },
198 | createBlood: function (token, multiplier, bloodType) {
199 | var pages = _.union([Campaign().get('playerpageid')], _.values(Campaign().get('playerspecificpages')));
200 | var dropBlood = _.contains(pages, token.get('pageid'));
201 | var gmNotes = token.get('gmnotes');
202 |
203 | if (!dropBlood || token.get('layer') !== 'objects' || gmNotes.indexOf('noblood') !== -1) {
204 | return;
205 | }
206 |
207 | var size = Math.floor(BloodSpatter.getTokenSize(token) * multiplier);
208 | var bloodImage = BloodSpatter.chooseBlood(bloodType);
209 | var bloodTokenSource = bloodImage;
210 | var bloodTokenWidth = size * multiplier;
211 | var bloodTokenHeight = size * multiplier;
212 |
213 | if (bloodImage !== null && typeof(bloodImage) === 'object') {
214 | bloodTokenSource = bloodImage.src;
215 |
216 | var bloodImageAspectRatio = bloodImage.width / bloodImage.height;
217 | var widthRatioMultiplier = 1;
218 | var heightRatioMultiplier = 1;
219 |
220 | if (bloodImageAspectRatio < 1) {
221 | widthRatioMultiplier = bloodImageAspectRatio;
222 | } else if (bloodImageAspectRatio > 1) {
223 | heightRatioMultiplier = bloodImage.height / bloodImage.width;
224 | }
225 | bloodTokenWidth = bloodTokenWidth * widthRatioMultiplier;
226 | bloodTokenHeight = bloodTokenHeight * heightRatioMultiplier;
227 | }
228 |
229 | var widthIncrement = bloodTokenWidth * 0.1;
230 | var widthIncrementTotal = widthIncrement;
231 | var heightIncrement = bloodTokenHeight * 0.1;
232 | var heightIncrementTotal = heightIncrement;
233 | var spatterToken = BloodSpatter.fixedCreateObj('graphic', {
234 | pageid: token.get('_pageid'),
235 | imgsrc: bloodTokenSource,
236 | tint_color: BloodSpatter.bloodColor(gmNotes),
237 | gmnotes: 'blood',
238 | top: token.get('top') + (randomInteger(Math.floor(size / 2)) * BloodSpatter.getOffset()),
239 | left: token.get('left') + (randomInteger(Math.floor(size / 2)) * BloodSpatter.getOffset()),
240 | width: widthIncrement,
241 | height: heightIncrement,
242 | rotation: randomInteger(360) - 1,
243 | layer: 'map'
244 | });
245 |
246 | toFront(spatterToken);
247 |
248 | (function splatterEnlargeFunction(count) {
249 | if (count < 10) {
250 | setTimeout(function () {
251 | widthIncrementTotal += widthIncrement;
252 | heightIncrementTotal += heightIncrement;
253 | spatterToken.set({
254 | width: widthIncrementTotal,
255 | height: heightIncrementTotal
256 | });
257 | splatterEnlargeFunction(count + 1);
258 | }, 30);
259 | }
260 | })(0);
261 |
262 | },
263 | getTokenSize: function (token) {
264 | return (token.get('height') + token.get('width')) / 2; //average the height and the width
265 | },
266 | timeout: 0,
267 | increaseTimeout: function () {
268 | BloodSpatter.timeout += 4;
269 | BloodSpatter.watchTimeout();
270 | },
271 | interval: null,
272 | watchTimeout: function () {
273 | if (BloodSpatter.interval === null) {
274 | BloodSpatter.interval = setInterval(function () {
275 | if (BloodSpatter.timeout > 0) {
276 | BloodSpatter.timeout--;
277 | } else {
278 | clearInterval(BloodSpatter.interval);
279 | BloodSpatter.interval = null;
280 | }
281 | }, 2000);
282 | }
283 | },
284 | fixedCreateObj: function () {
285 | return function () {
286 | var obj = createObj.apply(this, arguments);
287 | if (obj && !obj.fbpath) {
288 | obj.fbpath = obj.changed._fbpath.replace(/([^\/]*\/){4}/, '/');
289 | }
290 | return obj;
291 | };
292 | }(),
293 | tokenHPChanged: function (token, maxHealth, currentHealth, damageTaken) {
294 | if (maxHealth === '') {
295 | return;
296 | }
297 | var percentOfHpLost = damageTaken / maxHealth;
298 | var damageMultiplier = BloodSpatter.baseSize + percentOfHpLost;
299 |
300 | if ((!BloodSpatter.hpCountUp && currentHealth <= maxHealth / 2) || (BloodSpatter.hpCountUp && currentHealth >= maxHealth / 2)) {
301 | token.set({
302 | status_red: true
303 | });
304 | // Create spatter near token if "bloodied". Chance of spatter depends on severity of damage
305 | if (damageTaken > 0 && currentHealth > 0 && ( (!BloodSpatter.hpCountUp && currentHealth < randomInteger(maxHealth)) || (BloodSpatter.hpCountUp && currentHealth > randomInteger(maxHealth)) )) {
306 | BloodSpatter.createBlood(token, damageMultiplier, 'spatter');
307 | }
308 | } else {
309 | token.set({
310 | status_red: false
311 | });
312 | }
313 | if ((!BloodSpatter.hpCountUp && currentHealth <= 0) || (BloodSpatter.hpCountUp && currentHealth >= maxHealth)) {
314 | token.set({
315 | status_dead: true
316 | });
317 | // Create pool near token if health drops below 1.
318 | if (damageTaken > 0) {
319 | BloodSpatter.createBlood(token, damageMultiplier, 'pool');
320 | }
321 | } else {
322 | token.set({
323 | status_dead: false
324 | });
325 | }
326 | }
327 | };
328 |
329 | on('ready', function () {
330 | on('change:graphic:bar' + BloodSpatter.hpBar + '_value', function (token, prev) {
331 | var maxHealth = token.get('bar' + BloodSpatter.hpBar + '_max');
332 | var currentHealth = token.get('bar' + BloodSpatter.hpBar + '_value');
333 | var previousHealth = prev['bar' + BloodSpatter.hpBar + '_value'];
334 | var damageTaken;
335 |
336 | if (!BloodSpatter.hpCountUp) {
337 | damageTaken = previousHealth - currentHealth;
338 | } else {
339 | damageTaken = currentHealth - previousHealth;
340 | }
341 |
342 | BloodSpatter.tokenHPChanged(token, maxHealth, currentHealth, damageTaken);
343 | });
344 |
345 | //Make blood trails, chance goes up depending on how injured a token is
346 | on('change:graphic:lastmove', function (token) {
347 | var maxHealth = token.get('bar' + BloodSpatter.hpBar + '_max');
348 |
349 | if (maxHealth === '' || BloodSpatter.timeout !== 0) {
350 | return;
351 | }
352 |
353 | var currentHealth = token.get('bar' + BloodSpatter.hpBar + '_value');
354 | var healthLost = maxHealth - currentHealth;
355 | var percentOfHpLost = healthLost / maxHealth;
356 | var damageMultiplier = (BloodSpatter.baseSize / 2) + percentOfHpLost;
357 |
358 | if ((!BloodSpatter.hpCountUp && currentHealth <= maxHealth / 2 && currentHealth < randomInteger(maxHealth)) || (BloodSpatter.hpCountUp && currentHealth >= maxHealth / 2 && currentHealth > randomInteger(maxHealth))) {
359 | BloodSpatter.createBlood(token, damageMultiplier, 'spatter');
360 | BloodSpatter.increaseTimeout();
361 | }
362 | });
363 |
364 | on('chat:message', function (msg) {
365 | if (msg.type === 'api' && (msg.content.indexOf('!clearblood') !== -1 || msg.content.indexOf('!clear blood') !== -1)) {
366 | if (BloodSpatter.onlyAllowGMtoRunCommands && !playerIsGM(msg.playerid)) {
367 | sendChat(msg.who, '/w ' + msg.who + ' You are not authorized to use that command!');
368 | return;
369 | } else {
370 | var objects = filterObjs(function (obj) {
371 | if (obj.get('type') === 'graphic' && obj.get('gmnotes') === 'blood') {
372 | return true;
373 | } else {
374 | return false;
375 | }
376 | });
377 | _.each(objects, function (obj) {
378 | obj.remove();
379 | });
380 | }
381 | }
382 | });
383 | });
--------------------------------------------------------------------------------
/CashMaster.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | /* global on log playerIsGM findObjs getObj getAttrByName sendChat */
4 |
5 | /*
6 | CASHMASTER
7 |
8 | A currency management script for the D&D 5e OGL sheets on roll20.net.
9 | Please use `!cmhelp` for inline help and examples.
10 |
11 | arthurbauer@me.com
12 | */
13 |
14 | var cashsplit = function cashsplit(c, m, x) {
15 | var ct = 0;
16 | var cr = 0;
17 | if (c !== null) {
18 | ct = Math.floor(c / m);
19 | cr = c % m;
20 | if (cr >= x || c < 0 && cr < 0 && -cr < x) {
21 | ct += 1;
22 | }
23 | }
24 | return ct;
25 | };
26 |
27 | var getattr = function getattr(cid, att) {
28 | var attr = findObjs({
29 | type: 'attribute',
30 | characterid: cid,
31 | name: att
32 | })[0];
33 | if (!attr) {
34 | return null;
35 | }
36 | return attr.get('current');
37 | };
38 |
39 | var setattr = function setattr(cid, att, val) {
40 | var attr = findObjs({
41 | type: 'attribute',
42 | characterid: cid,
43 | name: att
44 | })[0];
45 | if (!attr) {
46 | return;
47 | }
48 | attr.setWithWorker({
49 | current: parseInt(val, 10)
50 | });
51 | };
52 |
53 | on('ready', function () {
54 | var v = '0.4.1';
55 | var usd = 25;
56 | /*
57 | Change this if you want to have a rough estimation of a character's wealth in USD.
58 | Set it to something between 25 and 50 (25 USD per 1gp).
59 | Set it to 0 to disable it completely.
60 | */
61 | var scname = 'CashMaster';
62 |
63 | log(scname + ' v' + v + ' online. Select one or more party members, then use `!cmhelp`');
64 |
65 | on('chat:message', function (msg) {
66 | if (msg.type !== 'api' && !playerIsGM(msg.playerid)) {
67 | return;
68 | }
69 | if (!msg.content.startsWith('!cm')) {
70 | return;
71 | }
72 | var partytotal = 0;
73 | var output = '/w gm &{template:desc} {{desc=Party’s cash overview
';
74 | var partycounter = 0;
75 | var partymember = Object.entries(msg.selected).length;
76 | msg.selected.forEach(function (obj) {
77 | var token = getObj('graphic', obj._id); // eslint-disable-line no-underscore-dangle
78 | var character = void 0;
79 | if (token) {
80 | character = getObj('character', token.get('represents'));
81 | }
82 | if (character) {
83 | partycounter += 1;
84 | var name = getAttrByName(character.id, 'character_name');
85 | var pp = getattr(character.id, 'pp') * 1;
86 | var gp = getattr(character.id, 'gp') * 1;
87 | var ep = getattr(character.id, 'ep') * 1;
88 | var sp = getattr(character.id, 'sp') * 1;
89 | var cp = getattr(character.id, 'cp') * 1;
90 | var total = Math.round((pp * 10 + ep * 0.5 + gp + sp / 10 + cp / 100) * 10000) / 10000;
91 | partytotal = total + partytotal;
92 | output += '' + name + '
has ';
93 | if (pp !== 0) output += pp + ' platinum, ';
94 | if (gp !== 0) output += gp + ' gold, ';
95 | if (ep !== 0) output += ep + ' electrum, ';
96 | if (sp !== 0) output += sp + ' silver, ';
97 | if (cp !== 0) output += cp + ' copper.';
98 |
99 | output += '
Converted, this character has ';
100 | if (usd > 0) output += '';
101 | output += total + ' gp';
102 | if (usd > 0) output += '';
103 | output += ' in total.
';
104 | }
105 | });
106 |
107 | partytotal = Math.round(partytotal * 100, 0) / 100;
108 |
109 | output += 'Party total: ' + partytotal + ' gp}}';
110 | sendChat(scname, output);
111 |
112 | if (msg.content === '!cmhelp') {
113 | sendChat(scname, '/w gm Usage
First, select one or several party members.
Then use
!cm
to get an overview over the party’s cash,!cmshare
to share the money equally between party members, converting the amount into the best combination of gold, silver and copper (this should be used in smaller stores),!cmconvert
to convert and share the money equally between party members, converting the amount into the best combination of platinum, gold, electrum, silver and copper (this should only be used in larger stores that have a fair amount of cash),!cmadd [amount][currency]
to add/subtract an equal amount of money from each selected party member,!cmhoard [amount][currency]
to share a certain amount of coins between the party members. Note that in this case, no conversion between the different coin types is made - if a party of 5 shares 4 pp, then 4 party members receive one pp each, and the last member won’t get anything.
Examples
!cm
will show a cash overview.!cmshare
will collect all the money and share it evenly on the members, using gp, sp and cp only (pp and ep will be converted). Can also be used for one character to “exchange” money.!cmconvert
- same as !cmshare
, but will also use platinum and electrum.!cmadd 50gp
will add 50 gp to every selected character.!cmhoard 50gp
will (more or less evenly) distribute 50 gp among the party members.
Note: If you substract more coins than a character has, the coin value will become negative. Use !cmshare
on that one character to balance the coins (see examples below).
Advanced uses
- Changing multiple values at once:
!cmadd -1gp 10sp
will substract 1 gp and add 10 sp at the same time. - Paying services:
!cmadd -6cp
will subtract 6cp from each selected party member. Use !cmshare
or !cmconvert
afterwards to balance the amount of coins (e.g. it will substract 1 sp and add 4 cp if the character didn’t have copper pieces before).
');
114 | }
115 |
116 | if (msg.content === '!cmshare' || msg.content === '!cmconvert') {
117 | output = '';
118 | var cashshare = partytotal / partycounter;
119 | var newcounter = 0;
120 | var pps = Math.floor(cashshare / 10);
121 | if (msg.content === '!cmshare') pps = 0;
122 | var rest = cashshare - pps * 10;
123 | var gps = Math.floor(rest);
124 | rest = (rest - gps) * 2;
125 | var eps = Math.floor(rest);
126 | if (msg.content === '!cmshare') eps = 0;
127 | rest = (rest - eps) * 5;
128 | var sps = Math.floor(rest);
129 | rest = (rest - sps) * 10;
130 | var cps = Math.round(rest);
131 | rest = (rest - cps) * partycounter;
132 |
133 | sendChat(scname, '/w gm &{template:desc} {{desc=Let\'s share this!
Everyone receives the equivalent of ' + cashshare + ' gp: ' + pps + ' platinum, ' + gps + ' gold, ' + eps + ' electrum, ' + sps + ' silver, and ' + cps + ' copper.}}');
134 |
135 | msg.selected.forEach(function (obj) {
136 | var token = getObj('graphic', obj._id); // eslint-disable-line no-underscore-dangle
137 | var character = void 0;
138 | newcounter += 1;
139 |
140 | if (token) {
141 | character = getObj('character', token.get('represents'));
142 | }
143 | if (character) {
144 | setattr(character.id, 'pp', pps);
145 | setattr(character.id, 'gp', gps);
146 | setattr(character.id, 'ep', eps);
147 | setattr(character.id, 'sp', sps);
148 | // enough copper coins? If not, the last one in the group has to take the diff
149 | if (rest > 0.999 && newcounter === partycounter) {
150 | cps += Math.round(rest);
151 | }
152 | if (rest < -0.999 && newcounter === partycounter) {
153 | cps += Math.round(rest);
154 | }
155 | setattr(character.id, 'cp', cps);
156 | }
157 | });
158 | }
159 |
160 | if (msg.content.startsWith('!cmadd')) {
161 | var ppg = /([0-9 -]+)pp/;
162 | var ppa = ppg.exec(msg.content);
163 | var gpg = /([0-9 -]+)gp/;
164 | var gpa = gpg.exec(msg.content);
165 | var epg = /([0-9 -]+)ep/;
166 | var epa = epg.exec(msg.content);
167 | var spg = /([0-9 -]+)sp/;
168 | var spa = spg.exec(msg.content);
169 | var cpg = /([0-9 -]+)cp/;
170 | var cpa = cpg.exec(msg.content);
171 | output = '';
172 |
173 | msg.selected.forEach(function (obj) {
174 | var token = getObj('graphic', obj._id); // eslint-disable-line no-underscore-dangle
175 | var character = void 0;
176 | if (token) {
177 | character = getObj('character', token.get('represents'));
178 | }
179 | if (character) {
180 | partycounter += 1;
181 | var name = getAttrByName(character.id, 'character_name');
182 | var pp = getattr(character.id, 'pp') * 1;
183 | var gp = getattr(character.id, 'gp') * 1;
184 | var ep = getattr(character.id, 'ep') * 1;
185 | var sp = getattr(character.id, 'sp') * 1;
186 | var cp = getattr(character.id, 'cp') * 1;
187 | var total = Math.round((pp * 10 + ep * 0.5 + gp + sp / 10 + cp / 100) * 10000) / 10000;
188 | partytotal = total + partytotal;
189 | output += '
' + name + '';
190 | if (ppa) {
191 | setattr(character.id, 'pp', parseInt(pp, 10) + parseInt(ppa[1], 10));
192 | output += '
' + ppa[0];
193 | }
194 | if (gpa) {
195 | setattr(character.id, 'gp', parseInt(gp, 10) + parseInt(gpa[1], 10));
196 | output += '
' + gpa[0];
197 | }
198 | if (epa) {
199 | setattr(character.id, 'ep', parseInt(ep, 10) + parseInt(epa[1], 10));
200 | output += '
' + epa[0];
201 | }
202 | if (spa) {
203 | setattr(character.id, 'sp', parseInt(sp, 10) + parseInt(spa[1], 10));
204 | output += '
' + spa[0];
205 | }
206 | if (cpa) {
207 | setattr(character.id, 'cp', parseInt(cp, 10) + parseInt(cpa[1], 10));
208 | output += '
' + cpa[0];
209 | }
210 | }
211 | });
212 | sendChat(scname, '/w gm &{template:desc} {{desc=Cashing out - it\'s payday!
' + output + '}}');
213 | }
214 |
215 | if (msg.content.startsWith('!cmhoard')) {
216 | var _ppg = /([0-9 -]+)pp/;
217 | var _ppa = _ppg.exec(msg.content);
218 | var _gpg = /([0-9 -]+)gp/;
219 | var _gpa = _gpg.exec(msg.content);
220 | var _epg = /([0-9 -]+)ep/;
221 | var _epa = _epg.exec(msg.content);
222 | var _spg = /([0-9 -]+)sp/;
223 | var _spa = _spg.exec(msg.content);
224 | var _cpg = /([0-9 -]+)cp/;
225 | var _cpa = _cpg.exec(msg.content);
226 |
227 | output = '';
228 | partycounter = 0;
229 |
230 | msg.selected.forEach(function (obj) {
231 | var token = getObj('graphic', obj._id); // eslint-disable-line no-underscore-dangle
232 | var character = void 0;
233 | if (token) {
234 | character = getObj('character', token.get('represents'));
235 | }
236 | if (character) {
237 | partycounter += 1;
238 | var name = getAttrByName(character.id, 'character_name');
239 | var pp = getattr(character.id, 'pp') * 1;
240 | var gp = getattr(character.id, 'gp') * 1;
241 | var ep = getattr(character.id, 'ep') * 1;
242 | var sp = getattr(character.id, 'sp') * 1;
243 | var cp = getattr(character.id, 'cp') * 1;
244 | var ppt = void 0;
245 | var gpt = void 0;
246 | var ept = void 0;
247 | var spt = void 0;
248 | var cpt = void 0;
249 |
250 | if (_ppa !== null) {
251 | ppt = cashsplit(_ppa[1], partymember, partycounter);
252 | }
253 | if (_gpa !== null) {
254 | gpt = cashsplit(_gpa[1], partymember, partycounter);
255 | }
256 | if (_epa !== null) {
257 | ept = cashsplit(_epa[1], partymember, partycounter);
258 | }
259 | if (_spa !== null) {
260 | spt = cashsplit(_spa[1], partymember, partycounter);
261 | }
262 | if (_cpa !== null) {
263 | cpt = cashsplit(_cpa[1], partymember, partycounter);
264 | }
265 |
266 | output += '
' + name + '';
267 | if (_ppa) {
268 | setattr(character.id, 'pp', parseInt(pp, 10) + parseInt(ppt, 10));
269 | output += '
' + ppt + 'pp';
270 | }
271 | if (_gpa) {
272 | setattr(character.id, 'gp', parseInt(gp, 10) + parseInt(gpt, 10));
273 | output += '
' + gpt + 'gp';
274 | }
275 | if (_epa) {
276 | setattr(character.id, 'ep', parseInt(ep, 10) + parseInt(ept, 10));
277 | output += '
' + ept + 'ep';
278 | }
279 | if (_spa) {
280 | setattr(character.id, 'sp', parseInt(sp, 10) + parseInt(spt, 10));
281 | output += '
' + spt + 'sp';
282 | }
283 | if (_cpa) {
284 | setattr(character.id, 'cp', parseInt(cp, 10) + parseInt(cpt, 10));
285 | output += '
' + cpt + 'cp';
286 | }
287 | }
288 | });
289 | sendChat(scname, '/w gm &{template:desc} {{desc=You are splitting up the coins among you
' + output + '}}');
290 | }
291 | });
292 | });
293 |
--------------------------------------------------------------------------------
/readme.md:
--------------------------------------------------------------------------------
1 | This project is now maintained at https://github.com/mlenser/roll20-api-scripts/tree/master/5eShapedScript
2 |
3 | Please do not ask for copyright material.
--------------------------------------------------------------------------------
/turnMarker.js:
--------------------------------------------------------------------------------
1 | // Github: https://github.com/shdwjk/Roll20API/blob/master/TurnMarker1/TurnMarker1.js
2 | // By: The Aaron, Arcane Scriptomancer
3 | // Contact: https://app.roll20.net/users/104025/the-aaron
4 |
5 | /* ############################################################### */
6 | /* TurnMarker */
7 | /* ############################################################### */
8 |
9 | var TurnMarker = TurnMarker || (function () {
10 | "use strict";
11 |
12 | var version = '1.3.9',
13 | lastUpdate = 1500408657,
14 | schemaVersion = 1.17,
15 | active = false,
16 | threadSync = 1,
17 | autoPullOptions = {
18 | 'none': 'None',
19 | 'npcs': 'NPCs',
20 | 'all': 'All'
21 | },
22 |
23 | fixedSendPing = (function () {
24 | var last = {};
25 | return function (left, top, pageid, playerid, pull) {
26 | if (last.left === left &&
27 | last.top === top &&
28 | last.pageid === pageid) {
29 | sendPing(-100, -100, pageid, null, false);
30 | }
31 | sendPing(left, top, pageid, playerid, pull);
32 | last.left = left;
33 | last.top = top;
34 | last.pageid = pageid;
35 | };
36 | }()),
37 |
38 | checkInstall = function () {
39 | log('-=> TurnMarker v' + version + ' <=- [' + (new Date(lastUpdate * 1000)) + ']');
40 |
41 | if (!state.hasOwnProperty('TurnMarker') || state.TurnMarker.version !== schemaVersion) {
42 | log(' > Updating Schema to v' + schemaVersion + ' <');
43 | switch (state.TurnMarker && state.TurnMarker.version) {
44 | case 1.16:
45 | state.TurnMarker.autoPull = 'none';
46 | /* falls through */
47 |
48 | case 'UpdateSchemaVersion':
49 | state.TurnMarker.version = schemaVersion;
50 | break;
51 |
52 | default:
53 | state.TurnMarker = {
54 | version: schemaVersion,
55 | announceRounds: true,
56 | announceTurnChange: true,
57 | announcePlayerInTurnAnnounce: true,
58 | announcePlayerInTurnAnnounceSize: '100%',
59 | autoPull: 'none',
60 | autoskipHidden: true,
61 | tokenName: 'Round',
62 | tokenURL: 'https://s3.amazonaws.com/files.d20.io/images/4095816/086YSl3v0Kz3SlDAu245Vg/thumb.png?1400535580',
63 | playAnimations: false,
64 | rotation: false,
65 | animationSpeed: 5,
66 | scale: 1.7,
67 | aura1: {
68 | pulse: false,
69 | size: 5,
70 | color: '#ff00ff'
71 | },
72 | aura2: {
73 | pulse: false,
74 | size: 5,
75 | color: '#00ff00'
76 | }
77 | };
78 | break;
79 | }
80 | }
81 | if (Campaign().get('turnorder') === '') {
82 | Campaign().set('turnorder', '[]');
83 | }
84 | if ('undefined' !== typeof GroupInitiative && GroupInitiative.ObserveTurnOrderChange) {
85 | GroupInitiative.ObserveTurnOrderChange(handleExternalTurnOrderChange);
86 | }
87 | },
88 |
89 | showHelp = function (who) {
90 | var marker = getMarker();
91 | var rounds = parseInt(marker.get('bar2_value'), 10);
92 | sendChat('',
93 | '/w "' + who + '" ' +
94 | '' +
95 | '
' +
96 | 'TurnMarker v' + version +
97 | '
' +
98 | '
Commands' +
99 | '
!tm' +
100 | '
' +
101 | 'The following arguments may be supplied in order to change the configuration. All changes are persisted between script restarts.' +
102 | '
' +
103 | '' + rounds + '
' +
104 | '- reset [#] -- Sets the round counter back to 0 or the supplied #.
' +
105 | '' + autoPullOptions[state.TurnMarker.autoPull] + '
' +
106 | '- autopull <mode> -- Sets auto pulling to the token whose turn it is. Modes: ' + _.keys(autoPullOptions) + '
' +
107 | '' + ( state.TurnMarker.announceRounds ? 'ON' : 'OFF' ) + '
' +
108 | '- toggle-announce -- When on, each round will be announced to chat.
' +
109 | '' + ( state.TurnMarker.announceTurnChange ? 'ON' : 'OFF' ) + '
' +
110 | '- toggle-announce-turn -- When on, the transition between visible turns will be announced.
' +
111 | '' + ( state.TurnMarker.announcePlayerInTurnAnnounce ? 'ON' : 'OFF' ) + '
' +
112 | '- toggle-announce-player -- When on, the player(s) controlling the current turn are included in the turn announcement.
' +
113 | '' + ( state.TurnMarker.autoskipHidden ? 'ON' : 'OFF' ) + '
' +
114 | '- toggle-skip-hidden -- When on, turn order will automatically be advanced past any hidden turns.
' +
115 | '' + ( state.TurnMarker.playAnimations ? 'ON' : 'OFF' ) + '
' +
116 | '- toggle-animations -- Turns on turn marker animations. [Experimental!]
' +
117 | '' + ( state.TurnMarker.rotation ? 'ON' : 'OFF' ) + '
' +
118 | '- toggle-rotate -- When on, the turn marker will rotate slowly clockwise. [Animation]
' +
119 | '' + ( state.TurnMarker.aura1.pulse ? 'ON' : 'OFF' ) + '
' +
120 | '- toggle-aura-1 -- When on, aura 2 will pulse in and out. [Animation]
' +
121 | '' + ( state.TurnMarker.aura2.pulse ? 'ON' : 'OFF' ) + '
' +
122 | '- toggle-aura-2 -- When on, aura 2 will pulse in and out. [Animation]
' +
123 | '
' +
124 | '
' +
125 | '
' +
126 | '
!eot' +
127 | '
' +
128 | 'Players may execute this command to advance the initiative to the next turn. This only succeeds if the current token is one that the caller controls or if it is executed by a GM.' +
129 | '
' +
130 | '
' +
131 | '
'
132 | );
133 | },
134 |
135 | handleInput = function (msg) {
136 | var who, tokenized, command;
137 |
138 | if (msg.type !== "api") {
139 | return;
140 | }
141 |
142 | who = (getObj('player', msg.playerid) || {get: () => 'API'}).get('_displayname');
143 | tokenized = msg.content.split(/\s+/);
144 | command = tokenized[0];
145 |
146 | switch (command) {
147 | case "!tm":
148 | case "!turnmarker": {
149 | if (!playerIsGM(msg.playerid)) {
150 | return;
151 | }
152 | let tokens = _.rest(tokenized), marker, value;
153 | switch (tokens[0]) {
154 | case 'reset':
155 | marker = getMarker();
156 | value = parseInt(tokens[1], 10) || 0;
157 | marker.set({
158 | name: state.TurnMarker.tokenName + ' ' + value,
159 | bar2_value: value
160 | });
161 | sendChat('', '/w "' + who + '" Round count is reset to ' + value + '.');
162 | break;
163 |
164 | case 'ping-target':
165 | var obj = getObj('graphic', tokens[1]);
166 | if (obj) {
167 | fixedSendPing(obj.get('left'), obj.get('top'), obj.get('pageid'), null, true);
168 | }
169 | break;
170 |
171 | case 'autopull':
172 | if (_.contains(_.keys(autoPullOptions), tokens[1])) {
173 | state.TurnMarker.autoPull = tokens[1];
174 | sendChat('', '/w "' + who + '" AutoPull is now ' + (autoPullOptions[state.TurnMarker.autoPull]) + '.');
175 | } else {
176 | sendChat('', '/w "' + who + '" "' + tokens[1] + '" is not a valid AutoPull options. Please specify one of: ' + _.keys(autoPullOptions).join(', ') + '.');
177 | }
178 | break;
179 |
180 | case 'toggle-announce':
181 | state.TurnMarker.announceRounds = !state.TurnMarker.announceRounds;
182 | sendChat('', '/w "' + who + '" Announce Rounds is now ' + (state.TurnMarker.announceRounds ? 'ON' : 'OFF' ) + '.');
183 | break;
184 |
185 | case 'toggle-announce-turn':
186 | state.TurnMarker.announceTurnChange = !state.TurnMarker.announceTurnChange;
187 | sendChat('', '/w "' + who + '" Announce Turn Changes is now ' + (state.TurnMarker.announceTurnChange ? 'ON' : 'OFF' ) + '.');
188 | break;
189 |
190 | case 'toggle-announce-player':
191 | state.TurnMarker.announcePlayerInTurnAnnounce = !state.TurnMarker.announcePlayerInTurnAnnounce;
192 | sendChat('', '/w "' + who + '" Player Name in Announce is now ' + (state.TurnMarker.announcePlayerInTurnAnnounce ? 'ON' : 'OFF' ) + '.');
193 | break;
194 |
195 | case 'toggle-skip-hidden':
196 | state.TurnMarker.autoskipHidden = !state.TurnMarker.autoskipHidden;
197 | sendChat('', '/w "' + who + '" Auto-skip Hidden is now ' + (state.TurnMarker.autoskipHidden ? 'ON' : 'OFF' ) + '.');
198 | break;
199 |
200 | case 'toggle-animations':
201 | state.TurnMarker.playAnimations = !state.TurnMarker.playAnimations;
202 | if (state.TurnMarker.playAnimations) {
203 | stepAnimation(threadSync);
204 | } else {
205 | marker = getMarker();
206 | marker.set({
207 | aura1_radius: '',
208 | aura2_radius: ''
209 | });
210 | }
211 |
212 | sendChat('', '/w "' + who + '" Animations are now ' + (state.TurnMarker.playAnimations ? 'ON' : 'OFF' ) + '.');
213 | break;
214 |
215 | case 'toggle-rotate':
216 | state.TurnMarker.rotation = !state.TurnMarker.rotation;
217 | sendChat('', '/w "' + who + '" Rotation is now ' + (state.TurnMarker.rotation ? 'ON' : 'OFF' ) + '.');
218 | break;
219 |
220 | case 'toggle-aura-1':
221 | state.TurnMarker.aura1.pulse = !state.TurnMarker.aura1.pulse;
222 | sendChat('', '/w "' + who + '" Aura 1 is now ' + (state.TurnMarker.aura1.pulse ? 'ON' : 'OFF' ) + '.');
223 | break;
224 |
225 | case 'toggle-aura-2':
226 | state.TurnMarker.aura2.pulse = !state.TurnMarker.aura2.pulse;
227 | sendChat('', '/w "' + who + '" Aura 2 is now ' + (state.TurnMarker.aura2.pulse ? 'ON' : 'OFF' ) + '.');
228 | break;
229 |
230 | default:
231 | case 'help':
232 | showHelp(who);
233 | break;
234 |
235 | }
236 | }
237 | break;
238 |
239 | case "!eot":
240 | requestTurnAdvancement(msg.playerid);
241 | break;
242 | }
243 | },
244 |
245 | getMarker = function () {
246 | var marker = findObjs({
247 | imgsrc: state.TurnMarker.tokenURL,
248 | pageid: Campaign().get("playerpageid")
249 | })[0];
250 |
251 | if (marker === undefined) {
252 | marker = createObj('graphic', {
253 | name: state.TurnMarker.tokenName + ' 0',
254 | pageid: Campaign().get("playerpageid"),
255 | layer: 'gmlayer',
256 | imgsrc: state.TurnMarker.tokenURL,
257 | left: 0,
258 | top: 0,
259 | height: 70,
260 | width: 70,
261 | bar2_value: 0,
262 | showplayers_name: true,
263 | showplayers_aura1: true,
264 | showplayers_aura2: true
265 | });
266 | }
267 | if (!TurnOrder.HasTurn(marker.id)) {
268 | TurnOrder.AddTurn({
269 | id: marker.id,
270 | pr: -1,
271 | custom: "",
272 | pageid: marker.get('pageid')
273 | });
274 | }
275 | return marker;
276 | },
277 |
278 | stepAnimation = function (sync) {
279 | if (!state.TurnMarker.playAnimations || sync !== threadSync) {
280 | return;
281 | }
282 | var marker = getMarker();
283 | if (active === true) {
284 | var rotation = (marker.get('bar1_value') + state.TurnMarker.animationSpeed) % 360;
285 | marker.set('bar1_value', rotation);
286 | if (state.TurnMarker.rotation) {
287 | marker.set('rotation', rotation);
288 | }
289 | if (state.TurnMarker.aura1.pulse) {
290 | marker.set('aura1_radius', Math.abs(Math.sin(rotation * (Math.PI / 180))) * state.TurnMarker.aura1.size);
291 | } else {
292 | marker.set('aura1_radius', '');
293 | }
294 | if (state.TurnMarker.aura2.pulse) {
295 | marker.set('aura2_radius', Math.abs(Math.cos(rotation * (Math.PI / 180))) * state.TurnMarker.aura2.size);
296 | } else {
297 | marker.set('aura2_radius', '');
298 | }
299 | setTimeout(_.bind(stepAnimation, this, sync), 100);
300 | }
301 | },
302 | checkForTokenMove = function (obj) {
303 | var turnOrder, current, marker;
304 | if (active) {
305 | turnOrder = TurnOrder.Get();
306 | current = _.first(turnOrder);
307 | if (obj && current && current.id === obj.id) {
308 | threadSync++;
309 |
310 | marker = getMarker();
311 | marker.set({
312 | "layer": obj.get("layer"),
313 | "top": obj.get("top"),
314 | "left": obj.get("left")
315 | });
316 |
317 | setTimeout(_.bind(stepAnimation, this, threadSync), 300);
318 | }
319 | }
320 | },
321 | requestTurnAdvancement = function (playerid) {
322 | if (active) {
323 | let turnOrder = TurnOrder.Get(),
324 | current = getObj('graphic', _.first(turnOrder).id),
325 | character = getObj('character', (current && current.get('represents')));
326 | if (playerIsGM(playerid) ||
327 | ( current &&
328 | ( _.contains(current.get('controlledby').split(','), playerid) ||
329 | _.contains(current.get('controlledby').split(','), 'all') )
330 | ) ||
331 | ( character &&
332 | ( _.contains(character.get('controlledby').split(','), playerid) ||
333 | _.contains(character.get('controlledby').split(','), 'all') )
334 | )
335 | ) {
336 | TurnOrder.Next();
337 | turnOrderChange(true);
338 | }
339 | }
340 | },
341 | announceRound = function (round) {
342 | if (state.TurnMarker.announceRounds) {
343 | sendChat(
344 | '',
345 | "/direct " +
346 | "" +
356 | "

" +
357 | "Round " + round +
358 | "

" +
359 | "
" +
360 | 'Reset &' + '#x21ba;'
361 | );
362 | }
363 | },
364 |
365 | turnOrderChange = function (FirstTurnChanged) {
366 | var marker = getMarker();
367 |
368 | if (!Campaign().get('initiativepage')) {
369 | return;
370 | }
371 |
372 | var turnOrder = TurnOrder.Get();
373 |
374 | if (!turnOrder.length) {
375 | return;
376 | }
377 |
378 | var current = _.first(turnOrder);
379 |
380 | if (state.TurnMarker.playAnimations) {
381 | threadSync++;
382 | setTimeout(_.bind(stepAnimation, this, threadSync), 300);
383 | }
384 |
385 | if (current.id === "-1") {
386 | return;
387 | }
388 |
389 | handleMarkerTurn();
390 |
391 | if (state.TurnMarker.autoskipHidden) {
392 | TurnOrder.NextVisible();
393 | handleMarkerTurn();
394 | }
395 |
396 | turnOrder = TurnOrder.Get();
397 |
398 | if (turnOrder[0].id === marker.id) {
399 | return;
400 | }
401 |
402 | current = _.first(TurnOrder.Get());
403 |
404 | var currentToken = getObj("graphic", turnOrder[0].id),
405 | currentChar = getObj('character', (currentToken || {get: _.noop}).get('represents'));
406 | if (currentToken) {
407 |
408 | if (FirstTurnChanged) {
409 | handleAnnounceTurnChange();
410 | }
411 |
412 | var size = Math.max(currentToken.get("height"), currentToken.get("width")) * state.TurnMarker.scale;
413 |
414 | if (marker.get("layer") === "gmlayer" && currentToken.get("layer") !== "gmlayer") {
415 | marker.set({
416 | "top": currentToken.get("top"),
417 | "left": currentToken.get("left"),
418 | "height": size,
419 | "width": size
420 | });
421 | setTimeout(function () {
422 | marker.set({
423 | "layer": currentToken.get("layer")
424 | });
425 | }, 500);
426 | } else {
427 | marker.set({
428 | "layer": currentToken.get("layer"),
429 | "top": currentToken.get("top"),
430 | "left": currentToken.get("left"),
431 | "height": size,
432 | "width": size
433 | });
434 | }
435 | toFront(currentToken);
436 |
437 | if ('all' === state.TurnMarker.autoPull ||
438 | ('npcs' === state.TurnMarker.autoPull && (
439 | '' === currentToken.get('controlledby') &&
440 | ( !currentChar || '' === currentChar.get('controlledby'))
441 | ))
442 | ) {
443 | fixedSendPing(currentToken.get('left'), currentToken.get('top'), currentToken.get('pageid'), null, true);
444 | }
445 | }
446 | },
447 |
448 | handleDestroyGraphic = function (obj) {
449 | if (TurnOrder.HasTurn(obj.id)) {
450 | let prev = JSON.parse(JSON.stringify(Campaign()));
451 | TurnOrder.RemoveTurn(obj.id);
452 | handleTurnOrderChange(Campaign(), prev);
453 | }
454 | },
455 |
456 | handleTurnOrderChange = function (obj, prev) {
457 | var prevOrder = JSON.parse(prev.turnorder);
458 | var objOrder = JSON.parse(obj.get('turnorder'));
459 |
460 | if (_.isArray(prevOrder) &&
461 | _.isArray(objOrder) &&
462 | prevOrder.length &&
463 | objOrder.length &&
464 | objOrder[0].id !== prevOrder[0].id
465 | ) {
466 | turnOrderChange(true);
467 | }
468 | },
469 |
470 | handleExternalTurnOrderChange = function () {
471 | var marker = getMarker(),
472 | turnorder = Campaign().get('turnorder'),
473 | markerTurn;
474 |
475 | turnorder = ('' === turnorder) ? [] : JSON.parse(turnorder);
476 | markerTurn = _.filter(turnorder, function (i) {
477 | return marker.id === i.id;
478 | })[0];
479 |
480 | if (markerTurn.pr !== -1) {
481 | markerTurn.pr = -1;
482 | turnorder = _.union([markerTurn], _.reject(turnorder, function (i) {
483 | return marker.id === i.id || (getObj('graphic', i.id) || {get: _.noop}).get('imgsrc') === state.TurnMarker.tokenURL;
484 | }));
485 | Campaign().set('turnorder', JSON.stringify(turnorder));
486 | }
487 | _.defer(dispatchInitiativePage);
488 | },
489 |
490 | handleMarkerTurn = function () {
491 | var marker = getMarker(),
492 | turnOrder = TurnOrder.Get(),
493 | round;
494 |
495 | if (turnOrder[0].id === marker.id) {
496 | round = parseInt(marker.get('bar2_value')) + 1;
497 | marker.set({
498 | name: state.TurnMarker.tokenName + ' ' + round,
499 | bar2_value: round
500 | });
501 | announceRound(round);
502 | TurnOrder.Next();
503 | }
504 | },
505 | handleAnnounceTurnChange = function () {
506 |
507 | if (state.TurnMarker.announceTurnChange) {
508 | var marker = getMarker();
509 | var turnOrder = TurnOrder.Get();
510 | var currentToken = getObj("graphic", turnOrder[0].id);
511 | if ('gmlayer' === currentToken.get('layer')) {
512 | return;
513 | }
514 | var previousTurn = _.last(_.filter(turnOrder, function (element) {
515 | var token = getObj("graphic", element.id);
516 | return token &&
517 | token.get('layer') !== 'gmlayer' &&
518 | element.id !== marker.id;
519 | }));
520 |
521 | /* find previous token. */
522 | var previousToken = getObj("graphic", previousTurn.id);
523 | var pImage = previousToken.get('imgsrc');
524 | var cImage = currentToken.get('imgsrc');
525 | var pRatio = previousToken.get('width') / previousToken.get('height');
526 | var cRatio = currentToken.get('width') / currentToken.get('height');
527 |
528 | var pNameString = "The Previous turn is done.";
529 | if (previousToken && previousToken.get('showplayers_name')) {
530 | pNameString = '' +
535 | previousToken.get('name') +
536 | '\'s turn is done.';
537 | }
538 |
539 | var cNameString = '';
540 | if (currentToken && currentToken.get('showplayers_name')) {
541 | cNameString = '' +
545 | currentToken.get('name') +
546 | '';
547 | }
548 |
549 |
550 | var PlayerAnnounceExtra = 'EOT &' + '#x21e8;';
551 | if (state.TurnMarker.announcePlayerInTurnAnnounce) {
552 | var Char = currentToken.get('represents');
553 | if (Char) {
554 | Char = getObj('character', Char);
555 | if (Char && _.isFunction(Char.get)) {
556 | var Controllers = Char.get('controlledby').split(',');
557 | _.each(Controllers, function (c) {
558 | switch (c) {
559 | case 'all':
560 | PlayerAnnounceExtra += '' +
571 | 'All' +
572 | '
';
573 | break;
574 |
575 | default:
576 | var player = getObj('player', c);
577 | if (player) {
578 | var PlayerColor = player.get('color');
579 | var PlayerName = player.get('displayname');
580 | PlayerAnnounceExtra += '' +
593 | PlayerName +
594 | '
';
595 | }
596 | break;
597 | }
598 | });
599 | }
600 | }
601 | }
602 |
603 | var tokenSize = 50;
604 | sendChat(
605 | '',
606 | "/direct " +
607 | "" +
608 | '
' +
617 | PlayerAnnounceExtra +
618 | '
' +
619 | "
"
620 | );
621 | }
622 | },
623 | resetMarker = function () {
624 | active = false;
625 | threadSync++;
626 |
627 | var marker = getMarker();
628 |
629 | marker.set({
630 | layer: "gmlayer",
631 | aura1_radius: '',
632 | aura2_radius: '',
633 | left: 35,
634 | top: 35,
635 | height: 70,
636 | width: 70,
637 | rotation: 0,
638 | bar1_value: 0
639 | });
640 | },
641 | startMarker = function () {
642 | var marker = getMarker();
643 |
644 | if (state.TurnMarker.playAnimations && state.TurnMarker.aura1.pulse) {
645 | marker.set({
646 | aura1_radius: state.TurnMarker.aura1.size,
647 | aura1_color: state.TurnMarker.aura1.color
648 | });
649 | }
650 | if (state.TurnMarker.playAnimations && state.TurnMarker.aura2.pulse) {
651 | marker.set({
652 | aura2_radius: state.TurnMarker.aura2.size,
653 | aura2_color: state.TurnMarker.aura2.color
654 | });
655 | }
656 | active = true;
657 | stepAnimation(threadSync);
658 | turnOrderChange(true);
659 | },
660 | dispatchInitiativePage = function () {
661 | if (!Campaign().get('initiativepage')) {
662 | resetMarker();
663 | } else {
664 | startMarker();
665 | }
666 | },
667 | registerEventHandlers = function () {
668 | on("change:campaign:initiativepage", dispatchInitiativePage);
669 | on("change:campaign:turnorder", handleTurnOrderChange);
670 | on("change:graphic", checkForTokenMove);
671 | on("destroy:graphic", handleDestroyGraphic);
672 | on("chat:message", handleInput);
673 |
674 | dispatchInitiativePage();
675 | }
676 | ;
677 |
678 | return {
679 | CheckInstall: checkInstall,
680 | RegisterEventHandlers: registerEventHandlers,
681 | TurnOrderChange: handleExternalTurnOrderChange
682 | };
683 |
684 | }());
685 |
686 | on("ready", function () {
687 | 'use strict';
688 |
689 | TurnMarker.CheckInstall();
690 | TurnMarker.RegisterEventHandlers();
691 | });
692 |
693 | var TurnOrder = TurnOrder || (function () {
694 | "use strict";
695 |
696 | return {
697 | Get: function () {
698 | var to = Campaign().get("turnorder");
699 | to = ('' === to ? '[]' : to);
700 | return JSON.parse(to);
701 | },
702 | Set: function (turnOrder) {
703 | Campaign().set({turnorder: JSON.stringify(turnOrder)});
704 | },
705 | Next: function () {
706 | this.Set(TurnOrder.Get().rotate(1));
707 | if ("undefined" !== typeof Mark && _.has(Mark, 'Reset') && _.isFunction(Mark.Reset)) {
708 | Mark.Reset();
709 | }
710 | },
711 | NextVisible: function () {
712 | var turns = this.Get();
713 | var context = {skip: 0};
714 | var found = _.find(turns, function (element) {
715 | var token = getObj("graphic", element.id);
716 | if (
717 | (undefined !== token) &&
718 | (token.get('layer') !== 'gmlayer')
719 | ) {
720 | return true;
721 | }
722 | else {
723 | this.skip++;
724 | }
725 | }, context);
726 | if (undefined !== found && context.skip > 0) {
727 | this.Set(turns.rotate(context.skip));
728 | }
729 | },
730 | HasTurn: function (id) {
731 | return (_.filter(this.Get(), function (turn) {
732 | return id === turn.id;
733 | }).length !== 0);
734 | },
735 | AddTurn: function (entry) {
736 | var turnorder = this.Get();
737 | turnorder.push(entry);
738 | this.Set(turnorder);
739 | },
740 | RemoveTurn: function (id) {
741 | this.Set(_.reject(this.Get(), (o) => o.id === id));
742 | }
743 | };
744 | }());
745 |
746 | Object.defineProperty(Array.prototype, 'rotate', {
747 | enumerable: false,
748 | writable: true
749 | });
750 |
751 | Array.prototype.rotate = (function () {
752 | "use strict";
753 | var unshift = Array.prototype.unshift,
754 | splice = Array.prototype.splice;
755 |
756 | return function (count) {
757 | var len = this.length >>> 0;
758 | count = count >> 0;
759 |
760 | unshift.apply(this, splice.call(this, count % len, len));
761 | return this;
762 | };
763 | }());
764 |
--------------------------------------------------------------------------------