2021-09-16 14:05:40 -04:00
|
|
|
/*
|
|
|
|
* TTSDeck.js
|
|
|
|
*
|
|
|
|
* Creates a deck image and corresponding "Saved Object" JSON for use
|
|
|
|
* in Tabletop Simulator
|
|
|
|
*/
|
|
|
|
|
|
|
|
useLibrary('project');
|
|
|
|
useLibrary('imageutils');
|
2021-09-30 03:00:32 -04:00
|
|
|
useLibrary('threads');
|
2021-09-16 14:05:40 -04:00
|
|
|
useLibrary('uilayout');
|
2021-09-18 01:26:15 -04:00
|
|
|
importClass(arkham.project.CopiesList);
|
2021-09-16 14:05:40 -04:00
|
|
|
|
2021-09-29 18:05:58 -04:00
|
|
|
const TTSJson = require('./TTSJson.js');
|
|
|
|
|
2021-09-16 14:05:40 -04:00
|
|
|
// The resolution (in pixels per inch) of the exported images
|
|
|
|
const RESOLUTION = 200;
|
|
|
|
// The extension of the image file format to use, e.g., png, jpg
|
|
|
|
const FORMAT = ImageUtils.FORMAT_JPEG;
|
|
|
|
|
2021-09-30 02:35:13 -04:00
|
|
|
const TTS_CARDS_PER_IMAGE = 69;
|
|
|
|
const TTS_MAX_ROWS = 7;
|
|
|
|
|
2021-09-16 14:05:40 -04:00
|
|
|
|
2021-09-30 00:29:27 -04:00
|
|
|
const getName = () => 'TTSDeck';
|
|
|
|
const getDescription = () => 'Generates a TTS deck image and JSON file';
|
|
|
|
const getVersion = () => 1.0;
|
|
|
|
const getPluginType = () => arkham.plugins.Plugin.INJECTED;
|
2021-09-16 14:05:40 -04:00
|
|
|
|
|
|
|
function unload() {
|
|
|
|
unregisterAll();
|
|
|
|
}
|
|
|
|
|
2021-09-17 17:04:17 -04:00
|
|
|
// Creates a test button during development that calls unload() to clean up.
|
|
|
|
testProjectScript();
|
|
|
|
|
2021-09-18 01:26:15 -04:00
|
|
|
// Hack to override the default return value of 1
|
|
|
|
function copyCount(copies_list, name) {
|
2021-09-30 00:29:27 -04:00
|
|
|
const entries = copies_list.getListEntries().map(x => String(x));
|
2021-09-18 01:26:15 -04:00
|
|
|
if (entries.indexOf(String(name)) == -1) {
|
|
|
|
return 2;
|
|
|
|
} else {
|
|
|
|
return copies_list.getCopyCount(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-30 01:30:23 -04:00
|
|
|
// export front face, or retrive it from a cached file
|
|
|
|
// TODO: handle two-sided cards
|
|
|
|
function makeCardImage(card) {
|
|
|
|
const component = ResourceKit.getGameComponentFromFile(card.file);
|
|
|
|
|
|
|
|
const cache_dir = new File(card.parent.file, '.ttsdeck_cache');
|
|
|
|
const cached_file = new File(cache_dir, card.file.name + '.' + FORMAT);
|
|
|
|
|
|
|
|
if (cached_file.exists() && cached_file.lastModified() > card.file.lastModified()) {
|
|
|
|
println("Got cached image for card", card);
|
|
|
|
return ImageUtils.read(cached_file);
|
|
|
|
} else {
|
|
|
|
println("Generating image for card ", card);
|
|
|
|
const sheets = component.createDefaultSheets();
|
|
|
|
const card_image = sheets[0].paint(arkham.sheet.RenderTarget.EXPORT, RESOLUTION);
|
|
|
|
|
|
|
|
cache_dir.mkdir();
|
|
|
|
ImageUtils.write(card_image, cached_file, FORMAT, -1, false, RESOLUTION);
|
|
|
|
|
|
|
|
return card_image;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
function TTSDeckPage(busy_props, page_num, page_cards, copies_list) {
|
2021-09-30 02:35:13 -04:00
|
|
|
this.rows = Math.min(Math.ceil(Math.sqrt(page_cards.length)), TTS_MAX_ROWS);
|
|
|
|
this.columns = Math.ceil(page_cards.length / this.rows);
|
|
|
|
this.deck_image = null;
|
2021-09-29 18:12:09 -04:00
|
|
|
let deck_graphics;
|
|
|
|
|
2021-09-30 02:35:13 -04:00
|
|
|
this.card_jsons = [];
|
|
|
|
for (let row = 0; row < this.rows; row++) {
|
|
|
|
for (let col = 0; col < this.columns && row * this.columns + col < page_cards.length; col++) {
|
2021-09-30 03:00:32 -04:00
|
|
|
if (busy_props.cancelled) return;
|
2021-09-30 02:35:13 -04:00
|
|
|
let index = row * this.columns + col;
|
|
|
|
let card = page_cards[index];
|
2021-09-30 03:00:32 -04:00
|
|
|
busy_props.status = "Processing Card " + card;
|
|
|
|
busy_props.currentProgress = (page_num - 1) * TTS_CARDS_PER_IMAGE + index;
|
2021-09-29 18:12:09 -04:00
|
|
|
|
|
|
|
try {
|
|
|
|
let component = ResourceKit.getGameComponentFromFile(card.file);
|
|
|
|
let copies = copyCount(copies_list, card.baseName);
|
|
|
|
|
|
|
|
for (let ii = 0; ii < copies; ii++) {
|
2021-09-30 02:35:13 -04:00
|
|
|
this.card_jsons.push(TTSJson.makeCardJSON(page_num * 100 + index, component.getName()));
|
2021-09-29 18:12:09 -04:00
|
|
|
}
|
|
|
|
|
2021-09-30 01:30:23 -04:00
|
|
|
let card_image = makeCardImage(card);
|
2021-09-29 18:12:09 -04:00
|
|
|
|
2021-09-30 02:35:13 -04:00
|
|
|
if (!this.deck_image) {
|
|
|
|
this.deck_image = ImageUtils.create(
|
|
|
|
card_image.width * this.columns, card_image.height * this.rows, false);
|
|
|
|
deck_graphics = this.deck_image.createGraphics();
|
2021-09-29 18:12:09 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
deck_graphics.drawImage(card_image, col * card_image.width, row * card_image.height, null);
|
|
|
|
} catch (ex) {
|
|
|
|
alert('Error while processing ' + card + ': ' + ex, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
println("End of Row ", row);
|
|
|
|
}
|
|
|
|
|
2021-09-30 02:35:13 -04:00
|
|
|
// TODO
|
|
|
|
this.face_url = "TODO";
|
|
|
|
this.back_url = "TODO";
|
|
|
|
}
|
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
function makeTTSDeck(busy_props, cards, copies_list) {
|
2021-09-30 02:35:13 -04:00
|
|
|
const pages = [];
|
2021-09-29 18:12:09 -04:00
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
busy_props.title = "Processing Cards";
|
|
|
|
busy_props.maximumProgress = cards.length;
|
|
|
|
|
2021-09-30 02:35:13 -04:00
|
|
|
for (let page_num = 0; page_num * TTS_CARDS_PER_IMAGE < cards.length; page_num++) {
|
|
|
|
let page_cards = cards.slice(page_num * TTS_CARDS_PER_IMAGE, (page_num + 1) * TTS_CARDS_PER_IMAGE);
|
|
|
|
printf("Making page %d, with %d cards:\n", page_num + 1, page_cards.length);
|
2021-09-30 03:00:32 -04:00
|
|
|
pages.push(new TTSDeckPage(busy_props, page_num + 1, page_cards, copies_list));
|
|
|
|
if (busy_props.cancelled) return [,];
|
2021-09-30 02:35:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const deck_json = TTSJson.makeDeckJSON(pages);
|
|
|
|
|
|
|
|
return [deck_json, pages.map(page => page.deck_image)];
|
2021-09-29 18:12:09 -04:00
|
|
|
}
|
|
|
|
|
2021-09-16 14:05:40 -04:00
|
|
|
function run() {
|
|
|
|
const ttsDeckAction = JavaAdapter(TaskAction, {
|
2021-09-30 00:29:27 -04:00
|
|
|
getLabel: () => 'Generate TTS Deck',
|
|
|
|
getActionName: () => 'ttsdeck',
|
2021-09-16 14:05:40 -04:00
|
|
|
// Applies to Deck Tasks
|
|
|
|
appliesTo: function appliesTo(project, task, member) {
|
|
|
|
if (member != null || task == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const type = task.settings.get(Task.KEY_TYPE);
|
|
|
|
if (NewTaskType.DECK_TYPE.equals(type)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
perform: function perform(project, task, member) {
|
|
|
|
member = ProjectUtilities.simplify(project, task, member);
|
|
|
|
Eons.setWaitCursor(true);
|
|
|
|
try {
|
2021-09-30 03:00:32 -04:00
|
|
|
Thread.busyWindow(
|
|
|
|
(busy_props) => this.performImpl(busy_props, member),
|
|
|
|
'Setting up...',
|
|
|
|
true);
|
2021-09-16 14:05:40 -04:00
|
|
|
} catch (ex) {
|
|
|
|
Error.handleUncaught(ex);
|
|
|
|
} finally {
|
|
|
|
Eons.setWaitCursor(false);
|
|
|
|
}
|
|
|
|
},
|
2021-09-30 03:00:32 -04:00
|
|
|
performImpl: function performImpl(busy_props, member) {
|
2021-09-18 01:26:15 -04:00
|
|
|
let copies_list;
|
|
|
|
try {
|
|
|
|
copies_list = new CopiesList(member);
|
|
|
|
} catch (ex) {
|
|
|
|
copies_list = new CopiesList();
|
|
|
|
alert("unable to read copies list, using card count of 2 for all files", true);
|
|
|
|
}
|
|
|
|
|
2021-09-16 14:05:40 -04:00
|
|
|
const children = member.getChildren();
|
2021-09-30 02:35:13 -04:00
|
|
|
const page_cards = children.filter(child => {
|
2021-09-16 14:05:40 -04:00
|
|
|
if (ProjectUtilities.matchExtension(child, 'eon')) {
|
|
|
|
let component = ResourceKit.getGameComponentFromFile(child.file);
|
|
|
|
return component.isDeckLayoutSupported();
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
const [deck_json, deck_images] = makeTTSDeck(busy_props, page_cards, copies_list);
|
|
|
|
if (busy_props.cancelled) return;
|
2021-09-29 18:05:58 -04:00
|
|
|
const saved_object = TTSJson.makeSavedObjectJSON([deck_json], member.getName());
|
2021-09-16 16:46:15 -04:00
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
busy_props.status = "";
|
|
|
|
busy_props.maximumProgress = -1;
|
|
|
|
busy_props.title = "Writing JSON";
|
2021-09-16 16:46:15 -04:00
|
|
|
const json_file = new File(member.file, member.getName() + '.json');
|
|
|
|
ProjectUtilities.writeTextFile(json_file, JSON.stringify(saved_object, null, 4));
|
|
|
|
|
2021-09-30 03:00:32 -04:00
|
|
|
busy_props.title = "Writing Images";
|
|
|
|
busy_props.maximumProgress = deck_images.length;
|
2021-09-30 02:35:13 -04:00
|
|
|
deck_images.forEach((deck_image, index) => {
|
2021-09-30 03:00:32 -04:00
|
|
|
busy_props.currentProgress = index;
|
2021-09-30 02:35:13 -04:00
|
|
|
const image_file = new File(member.file, member.getName() + '_' + (index + 1) + '.' + FORMAT);
|
|
|
|
ImageUtils.write(deck_image, image_file, FORMAT, -1, false, RESOLUTION);
|
|
|
|
});
|
2021-09-16 14:05:40 -04:00
|
|
|
|
|
|
|
member.synchronize();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
ActionRegistry.register(ttsDeckAction, Actions.PRIORITY_IMPORT_EXPORT);
|
|
|
|
}
|