forked from Roll20/roll20-api-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDupCharToken.js
270 lines (252 loc) · 13.2 KB
/
DupCharToken.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
//
// DupCharToken
//
// This script will duplicate a character sheet and token giving the new characters and tokens an identifying number and
// linking each new token to it's own individual new character sheet.
// This script is useful for those who want to create multiple copies of monsters because they prefer multiple character sheets to linking tokens as Mooks.
// Using an argument of "clean" causes it to clean up (delete) numbered tokens and characters such as this script creates.
//
// Examples:
// If you select a token named "Skel" that is linked to a character named "Skeleton", and in the chat window enter
// !DupCharToken 5
// It will make 5 new characters, identical to the original, named "Skeleton 1" through "Skeleton 5",
// and make 5 new tokens, identical to the original and stacked in the exact same location, named "Skel 1" through "Skel 5".
// Each token will be linked to its corresponding character sheet in the exact same manor as the originals were linked.
//
// If later on you select any token named "Skel" or "Skel (number)" and enter
// !DupCharToken clean
// It will delete all tokens where the name is "Skel (any number)" and all characters named "Skeleton (any number)".
// The original token and character (without a number) will not be deleted.
//
//
// Known bugs/issues:
// 1 - If the avatar image of the character is not in a user library (i.e: it's a marketplace image or from the Monster Manual, or the like),
// due to limits placed upon the API, it can't set an avatar image. You will have to do that yourself, or (better yet)
// copy the image to your own library on roll20, and set the token to be copied to use the image in your library.
// Two ways to copy the image to your own library on roll20 are as follows:
// A) download an image and then upload it manually, either by downloading the set,
// or placing a copy on the VTT, pressing "z" and right clicking the preview.
// or
// B) Right click the image in your marketplace collection (in the art library) and choosing "Copy to Library" or "Copy to Folder"
// B is probably faster, unless you are working with a whole set.
// After downloading the image to your own Roll20 library, make sure the Character and Token to be copied uses that image.
// 2 - It takes a while to make copies, If making lots of copies of characters with lots of attributes, it might be several minutes between the
// message that the copy has started, and the message that it has ended.
// This is OK except that the character copy is not done until the ending msg comes.
// 3 - Version 01.00 had unknown problems with the Roll20 script library, where the script could not be added or imported from the script library.
// The script would work if it was gotten from GitHub and pasted into an API screen, but did not work from the script library.
// It is hoped that version 1.01 will fix this, but if not, get it from the script library.
//
//
// This script is based upon a script by "The Aaron".
// https://app.roll20.net/forum/post/5687127/duplicate-character-sheet-plus-linked-token-script/?pageforid=5687127#post-5687127
// It has been modified to it's current form by Chris Dickey (Roll20 user "Chris D").
// Version 01.00 Dec 2018 Chris D. Original release.
// Version 1.01 Nov 2019 Chris D. After all characters have been created, do loops to read attributes and abilities once and create for each new character.
// Version 1.02 Jan 2020 Chris D. Fixed bug where script would crash API if clean was run with no tokens selected.
// Version 1.03 Mar 2020 Chris D. Tokens dragged from the journal would not be correctly linked. Moved setDefaultTokenForCharacter to occur after attributes are copied.
// Update 1.031 May 2020 Chris D. Minor bugfix with copying bio and gmnotes.
// Update 1.04 Sep 2020 Chris D. Fixed the bug where it would timeout, killing api sandbox, if copying too many attributes.
//
//
// Commands:
// !DupCharToken (Number of tokens to create) (Starting number)
// !DupCharToken clean
// Number to create is the number of characters and tokens to copy/create. It defaults to 1.
// Starting number is the number to start numbering the tokens and characters from. Defaults to 1.
// So !dupCharacter 5 3 would start numbering at 3 ending at number 7.
//
// You can add the following text to a macro
// !dupCharToken ?{How many Duplicates|1} ?{Starting Number|1}
on('ready',()=>{
'use strict';
on('chat:message',(msg)=>{
'use strict';
if( 'api' === msg.type && /^!dupCharToken\b/i.test( msg.content ) && playerIsGM( msg.playerid )) {
try {
let start = new Date().getTime();
function sChat( txt ) {
sendChat('DupCharToken', '/w ' + msg.who.replace(" (GM)","") + ' <div style="color: #993333;font-weight:bold;">' + txt + '</div>', null, {noarchive:true});
}
if( /\bclean\b/i.test( msg.content ) || /\bdelete\b/i.test( msg.content )) { // We are not duplicating, we are cleaning up characters and tokens that we created previously.
let tnames = [],
cnames = [];
if( !msg.selected || msg.selected.length == 0 )
sChat( "Please select exactly one token representing the character to duplicate. Script usage is !DupCharToken (Number of tokens to create) (Starting number) or !DupCharToken clean");
else {
_.each( msg.selected, function( sel ) { // for each token selected, find base token name and base character name. add unique ones to list.
let tokenObj = getObj('graphic', sel._id);
if ( _.isUndefined( tokenObj ) )
sChat( "Token is invalid in some way." );
else {
let tn = tokenObj.get( "name" ).replace( /\s+\d+\s*$/, ""); // one or more whitespaces, one or more digits, zero or more whitespaces, then the end of line get trimmed off.
if( tnames.indexOf( tn ) == -1 )
tnames.push( tn );
let charObj = getObj('character', tokenObj.get('represents'));
if ( _.isUndefined( charObj ) )
sChat( "Token is not correctly linked to a good character." );
else {
let cn = charObj.get( "name" ).replace( /\s+\d+\s*$/, ""); // one or more whitespaces, one or more digits, zero or more whitespaces, then the end of line get trimmed off.
if( cnames.indexOf( cn ) == -1 )
cnames.push( cn );
}
}
}); // end each selected token
function getRegEx( arr ) { // skel, ork.
let re = "^(";
arr.forEach( function(nm ) { re += nm + "|"; });
return new RegExp ( re.slice( 0, -1) + ")\\s\\d+\\s*$" );
};
if( tnames.length ) { // search for token names. for each found token, delete.
let reg = getRegEx( tnames ),
d = [];
_.each(findObjs({type:"graphic", _subtype: "token" }),( tObj )=>{
if( reg.test( tObj.get( 'name' ) ) ) {
d.push( tObj.get( 'name' ) );
tObj.remove();
}
});
sChat( "Deleted tokens: " + d.join() + "." );
}
if( cnames.length ) { // search for character names, for each, delete.
let reg = getRegEx( cnames ),
d = [];
_.each(findObjs({type:"character" }),( cObj )=>{
if( reg.test( cObj.get( 'name' ) ) ) {
d.push( cObj.get( 'name' ) );
cObj.remove();
}
});
sChat( "Deleted characters: " + d.join() + "." );
}
}
// End clean
} else if( !msg.selected || msg.selected.length !== 1 )
sChat( "Please select exactly one token representing the character to duplicate. Script usage is !DupCharToken (Number of tokens to create) (Starting number) or !DupCharToken clean");
else { // Not cleaning up, presumably duplicating.
let num = 1,
startnum = 1;
let cl = msg.content.split( /\s+/ );
if( cl.length > 1 ) {
if( parseInt( cl[ 1 ] ) )
num = parseInt( cl[ 1 ] );
else {
sChat( "Script usage is !DupCharToken (Number of tokens to create) (Starting number) or !DupCharToken clean");
return;
}
if( cl.length > 2 && parseInt( cl[ 2 ] ) )
startnum = parseInt( cl[ 2 ] );
}
let tokenObj = getObj('graphic', msg.selected[ 0 ]._id);
if ( _.isUndefined( tokenObj ) )
sChat( "Token is invalid in some way." );
else { // Good command line.
let charArray = [],
tokenArray = [],
links = [];
function getLink( n, lnk ) {
let l = tokenObj.get( lnk );
if( l && l != "" ) {
let a = getObj("attribute", l);
if( a && a.get( "name" ) != "" )
links [ n ] = a.get( "name" );
}
}
getLink( 0, "bar1_link" );
getLink( 1, "bar2_link" );
getLink( 2, "bar3_link" );
let charObj = getObj('character', tokenObj.get('represents'));
if ( _.isUndefined( charObj ) )
sChat( "Token is not correctly linked to a good character." );
else { // Everything is good. We are duplicating.
let oldCid = charObj.id;
let tmpC = JSON.stringify( charObj ).replace ( /\,"bio"\:.*?\,/gi, ',"bio":"",').replace ( /\,"gmnotes"\:.*?\,/gi, ',"gmnotes":"",'),
tmpT = JSON.stringify( tokenObj );
for( let i = startnum; i < (startnum + num); ++i) { // We are going to want to duplicate both the character and token this many times.
let newC = JSON.parse( tmpC ); // Simple true copy of object.
delete newC._id;
newC.name= charObj.get( 'name' ) + " " + i;
let parts = newC.avatar.match(/(.*\/images\/.*)(thumb|med|original|max)([^?]*)(\?[^?]+)?$/);
if(parts)
newC.avatar = parts[1]+'thumb'+parts[3]+(parts[4]?parts[4]:`?${Math.round(Math.random()*9999999)}`);
else
newC.avatar = "";
let newT = JSON.parse( tmpT );
delete newT._id;
newT.name = tokenObj.get( 'name' ) + " " + i;
parts = newT.imgsrc.match(/(.*\/images\/.*)(thumb|med|original|max)([^?]*)(\?[^?]+)?$/);
if(parts)
newT.imgsrc = parts[1]+'thumb'+parts[3]+(parts[4]?parts[4]:`?${Math.round(Math.random()*9999999)}`);
else
newT.imgsrc = "";
let newCObj = createObj('character', newC);
let newTObj = createObj('graphic', newT);
newTObj.set('represents', newCObj.id);
charArray.push( newCObj );
tokenArray.push( newTObj );
} // End For each character to create.
let attQueue = findObjs({_type:'attribute', _characterid:oldCid}); // create the queue we'll be processing asynchronously
const attBurndown = () => { // create the function that will process the next element of the queue
if(attQueue.length){
let a = attQueue.shift();
let sa = JSON.parse(JSON.stringify(a));
delete sa._id;
delete sa._type;
for( let i = 0; i < charArray.length; ++i ) {
delete sa._characterid;
sa._characterid = charArray[ i ].id;
if( links.indexOf( sa.name ) > -1 ) {
let newA = createObj('attribute', sa);
tokenArray[ i ].set( "bar" + (links.indexOf( sa.name ) + 1).toString() + "_link", newA.id); // Link the new token to the new attribute.
} else
createObj('attribute', sa);
};
setTimeout(attBurndown,0);
} else // Have finished the last attribute.
sChat( "Done duplicating: " + charObj.get( "name" ) + " " + num + " times. It took "
+ Math.round(((new Date().getTime()) - start) / 1000) + " seconds.");
};
// start the execution by doing the first element
attBurndown();
let abQueue = findObjs({_type:'ability', _characterid:oldCid}); // create the queue we'll be processing asynchronously
const abBurndown = () => { // create the function that will process the next element of the queue
if(abQueue.length){
let a = abQueue.shift();
let sa = JSON.parse(JSON.stringify(a));
delete sa._id;
delete sa._type;
for( let i = 0; i < charArray.length; ++i ) {
delete sa._characterid;
sa._characterid = charArray[ i ].id;
createObj('ability', sa);
};
setTimeout(abBurndown,0);
}
};
abBurndown(); // start the execution by doing the first element
for( let i = 0; i < charArray.length; ++i ) {
setDefaultTokenForCharacter( charArray[ i ], tokenArray[ i ]);
toFront( tokenArray[ i ] );
}
charObj.get("bio", function(bio) {
_.each( charArray, c => {
if( !_.isNull( bio ) && !_.isUndefined( bio) && bio != "null" && bio != "undefined" && (typeof bio === 'string' && bio.trim() !== "") )
c.set( 'bio', bio );
}); });
charObj.get("gmnotes", function(gmnotes) {
_.each( charArray, c => {
if( !_.isNull( gmnotes ) && !_.isUndefined( gmnotes) && gmnotes != "null" && gmnotes != "undefined" && (typeof gmnotes === 'string' && gmnotes.trim() !== "") )
c.set( 'gmnotes', gmnotes );
}); });
sChat( "Working on duplicating: " + charObj.get( "name" ) + " " + num + " times." );
}
} // End tokenObj is OK.
} // End have one token selected.
} catch(err) {
log( msg);
log( "DupCharToken.js error caught: " + err );
}
} // End msg if for this script.
}); // End on Chat.
// End on Ready.
});