diff --git a/examples/css-snippets/dnd5e-compendium.css b/examples/css-snippets/dnd5e-compendium.css index 48ca14c9e..affa37d42 100644 --- a/examples/css-snippets/dnd5e-compendium.css +++ b/examples/css-snippets/dnd5e-compendium.css @@ -279,204 +279,399 @@ body { } .json5e-background div[src$="#center"], +.json5e-background img[src$="#center"], .json5e-class div[src$="#center"], +.json5e-class img[src$="#center"], .json5e-deck div[src$="#center"], +.json5e-deck img[src$="#center"], .json5e-deity div[src$="#center"], +.json5e-deity img[src$="#center"], .json5e-feat div[src$="#center"], +.json5e-feat img[src$="#center"], .json5e-hazard div[src$="#center"], +.json5e-hazard img[src$="#center"], .json5e-item div[src$="#center"], +.json5e-item img[src$="#center"], .json5e-monster div[src$="#center"], +.json5e-monster img[src$="#center"], .json5e-note div[src$="#center"], +.json5e-note img[src$="#center"], .json5e-object div[src$="#center"], +.json5e-object img[src$="#center"], .json5e-psionic div[src$="#center"], +.json5e-psionic img[src$="#center"], .json5e-race div[src$="#center"], +.json5e-race img[src$="#center"], .json5e-reward div[src$="#center"], +.json5e-reward img[src$="#center"], .json5e-spell div[src$="#center"], -.json5e-vehicle div[src$="#center"] { +.json5e-spell img[src$="#center"], +.json5e-vehicle div[src$="#center"], +.json5e-vehicle img[src$="#center"] { text-align: center; } .json5e-background div[src$="#card"], .json5e-background div[src$="#symbol"], .json5e-background div[src$="#portrait"], .json5e-background div[src$="#token"], .json5e-background div[src$="#right"], +.json5e-background img[src$="#card"], +.json5e-background img[src$="#symbol"], +.json5e-background img[src$="#portrait"], +.json5e-background img[src$="#token"], +.json5e-background img[src$="#right"], .json5e-class div[src$="#card"], .json5e-class div[src$="#symbol"], .json5e-class div[src$="#portrait"], .json5e-class div[src$="#token"], .json5e-class div[src$="#right"], +.json5e-class img[src$="#card"], +.json5e-class img[src$="#symbol"], +.json5e-class img[src$="#portrait"], +.json5e-class img[src$="#token"], +.json5e-class img[src$="#right"], .json5e-deck div[src$="#card"], .json5e-deck div[src$="#symbol"], .json5e-deck div[src$="#portrait"], .json5e-deck div[src$="#token"], .json5e-deck div[src$="#right"], +.json5e-deck img[src$="#card"], +.json5e-deck img[src$="#symbol"], +.json5e-deck img[src$="#portrait"], +.json5e-deck img[src$="#token"], +.json5e-deck img[src$="#right"], .json5e-deity div[src$="#card"], .json5e-deity div[src$="#symbol"], .json5e-deity div[src$="#portrait"], .json5e-deity div[src$="#token"], .json5e-deity div[src$="#right"], +.json5e-deity img[src$="#card"], +.json5e-deity img[src$="#symbol"], +.json5e-deity img[src$="#portrait"], +.json5e-deity img[src$="#token"], +.json5e-deity img[src$="#right"], .json5e-feat div[src$="#card"], .json5e-feat div[src$="#symbol"], .json5e-feat div[src$="#portrait"], .json5e-feat div[src$="#token"], .json5e-feat div[src$="#right"], +.json5e-feat img[src$="#card"], +.json5e-feat img[src$="#symbol"], +.json5e-feat img[src$="#portrait"], +.json5e-feat img[src$="#token"], +.json5e-feat img[src$="#right"], .json5e-hazard div[src$="#card"], .json5e-hazard div[src$="#symbol"], .json5e-hazard div[src$="#portrait"], .json5e-hazard div[src$="#token"], .json5e-hazard div[src$="#right"], +.json5e-hazard img[src$="#card"], +.json5e-hazard img[src$="#symbol"], +.json5e-hazard img[src$="#portrait"], +.json5e-hazard img[src$="#token"], +.json5e-hazard img[src$="#right"], .json5e-item div[src$="#card"], .json5e-item div[src$="#symbol"], .json5e-item div[src$="#portrait"], .json5e-item div[src$="#token"], .json5e-item div[src$="#right"], +.json5e-item img[src$="#card"], +.json5e-item img[src$="#symbol"], +.json5e-item img[src$="#portrait"], +.json5e-item img[src$="#token"], +.json5e-item img[src$="#right"], .json5e-monster div[src$="#card"], .json5e-monster div[src$="#symbol"], .json5e-monster div[src$="#portrait"], .json5e-monster div[src$="#token"], .json5e-monster div[src$="#right"], +.json5e-monster img[src$="#card"], +.json5e-monster img[src$="#symbol"], +.json5e-monster img[src$="#portrait"], +.json5e-monster img[src$="#token"], +.json5e-monster img[src$="#right"], .json5e-note div[src$="#card"], .json5e-note div[src$="#symbol"], .json5e-note div[src$="#portrait"], .json5e-note div[src$="#token"], .json5e-note div[src$="#right"], +.json5e-note img[src$="#card"], +.json5e-note img[src$="#symbol"], +.json5e-note img[src$="#portrait"], +.json5e-note img[src$="#token"], +.json5e-note img[src$="#right"], .json5e-object div[src$="#card"], .json5e-object div[src$="#symbol"], .json5e-object div[src$="#portrait"], .json5e-object div[src$="#token"], .json5e-object div[src$="#right"], +.json5e-object img[src$="#card"], +.json5e-object img[src$="#symbol"], +.json5e-object img[src$="#portrait"], +.json5e-object img[src$="#token"], +.json5e-object img[src$="#right"], .json5e-psionic div[src$="#card"], .json5e-psionic div[src$="#symbol"], .json5e-psionic div[src$="#portrait"], .json5e-psionic div[src$="#token"], .json5e-psionic div[src$="#right"], +.json5e-psionic img[src$="#card"], +.json5e-psionic img[src$="#symbol"], +.json5e-psionic img[src$="#portrait"], +.json5e-psionic img[src$="#token"], +.json5e-psionic img[src$="#right"], .json5e-race div[src$="#card"], .json5e-race div[src$="#symbol"], .json5e-race div[src$="#portrait"], .json5e-race div[src$="#token"], .json5e-race div[src$="#right"], +.json5e-race img[src$="#card"], +.json5e-race img[src$="#symbol"], +.json5e-race img[src$="#portrait"], +.json5e-race img[src$="#token"], +.json5e-race img[src$="#right"], .json5e-reward div[src$="#card"], .json5e-reward div[src$="#symbol"], .json5e-reward div[src$="#portrait"], .json5e-reward div[src$="#token"], .json5e-reward div[src$="#right"], +.json5e-reward img[src$="#card"], +.json5e-reward img[src$="#symbol"], +.json5e-reward img[src$="#portrait"], +.json5e-reward img[src$="#token"], +.json5e-reward img[src$="#right"], .json5e-spell div[src$="#card"], .json5e-spell div[src$="#symbol"], .json5e-spell div[src$="#portrait"], .json5e-spell div[src$="#token"], .json5e-spell div[src$="#right"], +.json5e-spell img[src$="#card"], +.json5e-spell img[src$="#symbol"], +.json5e-spell img[src$="#portrait"], +.json5e-spell img[src$="#token"], +.json5e-spell img[src$="#right"], .json5e-vehicle div[src$="#card"], .json5e-vehicle div[src$="#symbol"], .json5e-vehicle div[src$="#portrait"], .json5e-vehicle div[src$="#token"], -.json5e-vehicle div[src$="#right"] { +.json5e-vehicle div[src$="#right"], +.json5e-vehicle img[src$="#card"], +.json5e-vehicle img[src$="#symbol"], +.json5e-vehicle img[src$="#portrait"], +.json5e-vehicle img[src$="#token"], +.json5e-vehicle img[src$="#right"] { float: right; padding-left: 5px; } .json5e-background div[src$="#center"] img, .json5e-background div[src$="#right"] img, +.json5e-background img[src$="#center"] img, +.json5e-background img[src$="#right"] img, .json5e-class div[src$="#center"] img, .json5e-class div[src$="#right"] img, +.json5e-class img[src$="#center"] img, +.json5e-class img[src$="#right"] img, .json5e-deck div[src$="#center"] img, .json5e-deck div[src$="#right"] img, +.json5e-deck img[src$="#center"] img, +.json5e-deck img[src$="#right"] img, .json5e-deity div[src$="#center"] img, .json5e-deity div[src$="#right"] img, +.json5e-deity img[src$="#center"] img, +.json5e-deity img[src$="#right"] img, .json5e-feat div[src$="#center"] img, .json5e-feat div[src$="#right"] img, +.json5e-feat img[src$="#center"] img, +.json5e-feat img[src$="#right"] img, .json5e-hazard div[src$="#center"] img, .json5e-hazard div[src$="#right"] img, +.json5e-hazard img[src$="#center"] img, +.json5e-hazard img[src$="#right"] img, .json5e-item div[src$="#center"] img, .json5e-item div[src$="#right"] img, +.json5e-item img[src$="#center"] img, +.json5e-item img[src$="#right"] img, .json5e-monster div[src$="#center"] img, .json5e-monster div[src$="#right"] img, +.json5e-monster img[src$="#center"] img, +.json5e-monster img[src$="#right"] img, .json5e-note div[src$="#center"] img, .json5e-note div[src$="#right"] img, +.json5e-note img[src$="#center"] img, +.json5e-note img[src$="#right"] img, .json5e-object div[src$="#center"] img, .json5e-object div[src$="#right"] img, +.json5e-object img[src$="#center"] img, +.json5e-object img[src$="#right"] img, .json5e-psionic div[src$="#center"] img, .json5e-psionic div[src$="#right"] img, +.json5e-psionic img[src$="#center"] img, +.json5e-psionic img[src$="#right"] img, .json5e-race div[src$="#center"] img, .json5e-race div[src$="#right"] img, +.json5e-race img[src$="#center"] img, +.json5e-race img[src$="#right"] img, .json5e-reward div[src$="#center"] img, .json5e-reward div[src$="#right"] img, +.json5e-reward img[src$="#center"] img, +.json5e-reward img[src$="#right"] img, .json5e-spell div[src$="#center"] img, .json5e-spell div[src$="#right"] img, +.json5e-spell img[src$="#center"] img, +.json5e-spell img[src$="#right"] img, .json5e-vehicle div[src$="#center"] img, -.json5e-vehicle div[src$="#right"] img { +.json5e-vehicle div[src$="#right"] img, +.json5e-vehicle img[src$="#center"] img, +.json5e-vehicle img[src$="#right"] img { max-height: 60vh; } .json5e-background div[src$="#right"], +.json5e-background img[src$="#right"], .json5e-class div[src$="#right"], +.json5e-class img[src$="#right"], .json5e-deck div[src$="#right"], +.json5e-deck img[src$="#right"], .json5e-deity div[src$="#right"], +.json5e-deity img[src$="#right"], .json5e-feat div[src$="#right"], +.json5e-feat img[src$="#right"], .json5e-hazard div[src$="#right"], +.json5e-hazard img[src$="#right"], .json5e-item div[src$="#right"], +.json5e-item img[src$="#right"], .json5e-monster div[src$="#right"], +.json5e-monster img[src$="#right"], .json5e-note div[src$="#right"], +.json5e-note img[src$="#right"], .json5e-object div[src$="#right"], +.json5e-object img[src$="#right"], .json5e-psionic div[src$="#right"], +.json5e-psionic img[src$="#right"], .json5e-race div[src$="#right"], +.json5e-race img[src$="#right"], .json5e-reward div[src$="#right"], +.json5e-reward img[src$="#right"], .json5e-spell div[src$="#right"], -.json5e-vehicle div[src$="#right"] { +.json5e-spell img[src$="#right"], +.json5e-vehicle div[src$="#right"], +.json5e-vehicle img[src$="#right"] { max-width: 50%; } .json5e-background div[src$="#card"], .json5e-background div[src$="#token"], +.json5e-background img[src$="#card"], +.json5e-background img[src$="#token"], .json5e-class div[src$="#card"], .json5e-class div[src$="#token"], +.json5e-class img[src$="#card"], +.json5e-class img[src$="#token"], .json5e-deck div[src$="#card"], .json5e-deck div[src$="#token"], +.json5e-deck img[src$="#card"], +.json5e-deck img[src$="#token"], .json5e-deity div[src$="#card"], .json5e-deity div[src$="#token"], +.json5e-deity img[src$="#card"], +.json5e-deity img[src$="#token"], .json5e-feat div[src$="#card"], .json5e-feat div[src$="#token"], +.json5e-feat img[src$="#card"], +.json5e-feat img[src$="#token"], .json5e-hazard div[src$="#card"], .json5e-hazard div[src$="#token"], +.json5e-hazard img[src$="#card"], +.json5e-hazard img[src$="#token"], .json5e-item div[src$="#card"], .json5e-item div[src$="#token"], +.json5e-item img[src$="#card"], +.json5e-item img[src$="#token"], .json5e-monster div[src$="#card"], .json5e-monster div[src$="#token"], +.json5e-monster img[src$="#card"], +.json5e-monster img[src$="#token"], .json5e-note div[src$="#card"], .json5e-note div[src$="#token"], +.json5e-note img[src$="#card"], +.json5e-note img[src$="#token"], .json5e-object div[src$="#card"], .json5e-object div[src$="#token"], +.json5e-object img[src$="#card"], +.json5e-object img[src$="#token"], .json5e-psionic div[src$="#card"], .json5e-psionic div[src$="#token"], +.json5e-psionic img[src$="#card"], +.json5e-psionic img[src$="#token"], .json5e-race div[src$="#card"], .json5e-race div[src$="#token"], +.json5e-race img[src$="#card"], +.json5e-race img[src$="#token"], .json5e-reward div[src$="#card"], .json5e-reward div[src$="#token"], +.json5e-reward img[src$="#card"], +.json5e-reward img[src$="#token"], .json5e-spell div[src$="#card"], .json5e-spell div[src$="#token"], +.json5e-spell img[src$="#card"], +.json5e-spell img[src$="#token"], .json5e-vehicle div[src$="#card"], -.json5e-vehicle div[src$="#token"] { +.json5e-vehicle div[src$="#token"], +.json5e-vehicle img[src$="#card"], +.json5e-vehicle img[src$="#token"] { width: 150px; } .json5e-background div[src$="#symbol"], .json5e-background div[src$="#portrait"], +.json5e-background img[src$="#symbol"], +.json5e-background img[src$="#portrait"], .json5e-class div[src$="#symbol"], .json5e-class div[src$="#portrait"], +.json5e-class img[src$="#symbol"], +.json5e-class img[src$="#portrait"], .json5e-deck div[src$="#symbol"], .json5e-deck div[src$="#portrait"], +.json5e-deck img[src$="#symbol"], +.json5e-deck img[src$="#portrait"], .json5e-deity div[src$="#symbol"], .json5e-deity div[src$="#portrait"], +.json5e-deity img[src$="#symbol"], +.json5e-deity img[src$="#portrait"], .json5e-feat div[src$="#symbol"], .json5e-feat div[src$="#portrait"], +.json5e-feat img[src$="#symbol"], +.json5e-feat img[src$="#portrait"], .json5e-hazard div[src$="#symbol"], .json5e-hazard div[src$="#portrait"], +.json5e-hazard img[src$="#symbol"], +.json5e-hazard img[src$="#portrait"], .json5e-item div[src$="#symbol"], .json5e-item div[src$="#portrait"], +.json5e-item img[src$="#symbol"], +.json5e-item img[src$="#portrait"], .json5e-monster div[src$="#symbol"], .json5e-monster div[src$="#portrait"], +.json5e-monster img[src$="#symbol"], +.json5e-monster img[src$="#portrait"], .json5e-note div[src$="#symbol"], .json5e-note div[src$="#portrait"], +.json5e-note img[src$="#symbol"], +.json5e-note img[src$="#portrait"], .json5e-object div[src$="#symbol"], .json5e-object div[src$="#portrait"], +.json5e-object img[src$="#symbol"], +.json5e-object img[src$="#portrait"], .json5e-psionic div[src$="#symbol"], .json5e-psionic div[src$="#portrait"], +.json5e-psionic img[src$="#symbol"], +.json5e-psionic img[src$="#portrait"], .json5e-race div[src$="#symbol"], .json5e-race div[src$="#portrait"], +.json5e-race img[src$="#symbol"], +.json5e-race img[src$="#portrait"], .json5e-reward div[src$="#symbol"], .json5e-reward div[src$="#portrait"], +.json5e-reward img[src$="#symbol"], +.json5e-reward img[src$="#portrait"], .json5e-spell div[src$="#symbol"], .json5e-spell div[src$="#portrait"], +.json5e-spell img[src$="#symbol"], +.json5e-spell img[src$="#portrait"], .json5e-vehicle div[src$="#symbol"], -.json5e-vehicle div[src$="#portrait"] { +.json5e-vehicle div[src$="#portrait"], +.json5e-vehicle img[src$="#symbol"], +.json5e-vehicle img[src$="#portrait"] { width: 200px; } diff --git a/src/main/java/dev/ebullient/convert/io/FontRef.java b/src/main/java/dev/ebullient/convert/io/FontRef.java new file mode 100644 index 000000000..3d026047d --- /dev/null +++ b/src/main/java/dev/ebullient/convert/io/FontRef.java @@ -0,0 +1,53 @@ +package dev.ebullient.convert.io; + +public class FontRef { + /** Font family */ + public final String fontFamily; + /** Path to font source (unresolved local or remote) */ + public final String sourcePath; + + boolean hasTextReference = false; + + private FontRef(String fontFamily, String sourcePath) { + this.fontFamily = fontFamily; + this.sourcePath = sourcePath; + } + + public void addTextReference() { + hasTextReference = true; + } + + public boolean hasTextReference() { + return hasTextReference; + } + + @Override + public String toString() { + return "FontRef [fontFamily=" + fontFamily + ", sourcePath=" + sourcePath + "]"; + } + + public static String fontFamily(String fontPath) { + fontPath = fontPath.trim(); + int pos1 = fontPath.lastIndexOf('/'); + int pos2 = fontPath.lastIndexOf('.'); + if (pos1 > 0 && pos2 > 0) { + fontPath = fontPath.substring(pos1 + 1, pos2); + } else if (pos1 > 0) { + fontPath = fontPath.substring(pos1 + 1); + } else if (pos2 > 0) { + fontPath = fontPath.substring(0, pos2); + } + return fontPath; + } + + public static FontRef of(String fontString) { + return of(fontFamily(fontString), fontString); + } + + public static FontRef of(String fontFamily, String fontString) { + if (fontString == null || fontString.isEmpty()) { + return null; + } + return new FontRef(fontFamily, fontString); + } +} diff --git a/src/main/java/dev/ebullient/convert/io/Templates.java b/src/main/java/dev/ebullient/convert/io/Templates.java index 06a5a3746..f0e8a01ed 100644 --- a/src/main/java/dev/ebullient/convert/io/Templates.java +++ b/src/main/java/dev/ebullient/convert/io/Templates.java @@ -1,8 +1,10 @@ package dev.ebullient.convert.io; import java.io.IOException; +import java.io.InputStream; import java.nio.file.Files; import java.nio.file.Path; +import java.util.Base64; import java.util.Collection; import jakarta.enterprise.context.ApplicationScoped; @@ -103,4 +105,23 @@ public String renderIndex(String name, Collection resources) { return "%% ERROR: " + message + " %%"; } } + + public String renderCss(FontRef fontRef, InputStream data) throws IOException { + Template tpl = customTemplateOrDefault("css-font.txt"); + try { + String encoded = Base64.getEncoder().encodeToString(data.readAllBytes()); + int extpos = fontRef.sourcePath.lastIndexOf("."); + String type = fontRef.sourcePath.substring(extpos + 1); + return tpl + .data("fontFamily", fontRef.fontFamily) + .data("type", type) + .data("encoded", encoded) + .render(); + } catch (TemplateException tex) { + Throwable cause = tex.getCause(); + String message = cause != null ? cause.toString() : tex.toString(); + tui.error(tex, message); + return "%% ERROR: " + message + " %%"; + } + } } diff --git a/src/main/java/dev/ebullient/convert/io/Tui.java b/src/main/java/dev/ebullient/convert/io/Tui.java index ac1013c1b..423e8e56f 100644 --- a/src/main/java/dev/ebullient/convert/io/Tui.java +++ b/src/main/java/dev/ebullient/convert/io/Tui.java @@ -1,9 +1,11 @@ package dev.ebullient.convert.io; +import java.io.BufferedInputStream; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; +import java.net.URI; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; @@ -55,6 +57,12 @@ @ApplicationScoped public class Tui { + static Tui instance; + + public static Tui instance() { + return instance; + } + public final static TypeReference> LIST_STRING = new TypeReference<>() { }; public final static TypeReference> LIST_INT = new TypeReference<>() { @@ -176,6 +184,8 @@ public Tui() { this.err = new PrintWriter(System.err); this.debug = false; this.verbose = true; + + Tui.instance = this; } public void init(CommandSpec spec, boolean debug, boolean verbose) { @@ -304,6 +314,33 @@ public Optional resolvePath(Path path) { .findFirst(); } + public void copyFonts(Collection fonts, Map fallbackPaths) { + for (FontRef fontRef : fonts) { + Path targetPath = output.resolve(Path.of("css-snippets", slugify(fontRef.fontFamily) + ".css")); + targetPath.getParent().toFile().mkdirs(); + + printlnf("⏱️ Generating CSS snippet for %s", fontRef.sourcePath); + if (fontRef.sourcePath.startsWith("http")) { + try (InputStream is = URI.create(fontRef.sourcePath.replace(" ", "%20")).toURL().openStream()) { + Files.writeString(targetPath, templates.renderCss(fontRef, is)); + } catch (IOException e) { + errorf(e, "Unable to copy font from %s to %s", fontRef.sourcePath, targetPath); + } + } else { + Optional resolvedSource = resolvePath(Path.of(fontRef.sourcePath)); + if (resolvedSource.isEmpty()) { + errorf("Unable to find font '%s'", fontRef.sourcePath); + continue; + } + try (BufferedInputStream is = new BufferedInputStream(Files.newInputStream(resolvedSource.get()))) { + Files.writeString(targetPath, templates.renderCss(fontRef, is)); + } catch (IOException e) { + errorf(e, "Unable to copy font from %s to %s", fontRef.sourcePath, targetPath); + } + } + } + } + public void copyImages(Collection images, Map fallbackPaths) { for (ImageRef image : images) { Path targetPath = output.resolve(image.targetFilePath()); diff --git a/src/main/java/dev/ebullient/convert/tools/JsonNodeReader.java b/src/main/java/dev/ebullient/convert/tools/JsonNodeReader.java index 0122d3173..44df59afe 100644 --- a/src/main/java/dev/ebullient/convert/tools/JsonNodeReader.java +++ b/src/main/java/dev/ebullient/convert/tools/JsonNodeReader.java @@ -85,6 +85,20 @@ default boolean existsIn(JsonNode source) { return source.has(this.nodeName()); } + default boolean isArrayIn(JsonNode source) { + if (source == null || !source.has(this.nodeName())) { + return false; + } + return source.get(this.nodeName()).isArray(); + } + + default boolean isObjectIn(JsonNode source) { + if (source == null || !source.has(this.nodeName())) { + return false; + } + return source.get(this.nodeName()).isObject(); + } + default T fieldFromTo(JsonNode source, Class classTarget, Tui tui) { return tui.readJsonValue(source.get(this.nodeName()), classTarget); } @@ -268,24 +282,24 @@ default boolean valueEquals(JsonNode previous, JsonNode next) { } default ArrayNode withArrayFrom(JsonNode source) { - if (source == null || !source.has(this.nodeName())) { - return Tui.MAPPER.createArrayNode(); + if (isArrayIn(source)) { + return source.withArray(this.nodeName()); } - return source.withArray(this.nodeName()); + return Tui.MAPPER.createArrayNode(); } default Iterable iterateArrayFrom(JsonNode source) { - if (source == null || !source.has(this.nodeName())) { - return List.of(); + if (isArrayIn(source)) { + return () -> source.withArray(this.nodeName()).elements(); } - return () -> source.withArray(this.nodeName()).elements(); + return List.of(); } default Iterable> iterateFieldsFrom(JsonNode source) { - if (source == null) { - return List.of(); + if (isObjectIn(source)) { + return () -> source.get(this.nodeName()).fields(); } - return source::fields; + return List.of(); } /** Destructive! */ diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/JsonTextReplacement.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/JsonTextReplacement.java index 31334275e..18cac4463 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/JsonTextReplacement.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/JsonTextReplacement.java @@ -31,6 +31,7 @@ public interface JsonTextReplacement extends JsonTextConverter Pattern dicePattern = Pattern.compile("\\{@(dice|damage) ([^{}]+)}"); Pattern chancePattern = Pattern.compile("\\{@chance ([^}]+)}"); + Pattern fontPattern = Pattern.compile("\\{@font ([^}]+)}"); Pattern homebrewPattern = Pattern.compile("\\{@homebrew ([^}]+)}"); Pattern quickRefPattern = Pattern.compile("\\{@quickref ([^}]+)}"); Pattern notePattern = Pattern.compile("\\{@note (\\*|Note:)?\\s?([^}]+)}"); @@ -186,6 +187,17 @@ default String _replaceTokenText(String input, boolean nested) { result = linkifyPattern.matcher(result) .replaceAll(this::linkify); + result = fontPattern.matcher(result) + .replaceAll((match) -> { + String[] parts = match.group(1).split("\\|"); + String fontFamily = Tools5eSources.getFontReference(parts[1]); + if (fontFamily != null) { + return String.format("%s", + fontFamily, parts[0]); + } + return parts[0]; + }); + try { result = result .replace("{@hitYourSpellAttack}", "the summoner's spell attack modifier") @@ -216,8 +228,8 @@ default String _replaceTokenText(String input, boolean nested) { .replaceAll("\\{@cult ([^|}]+)}", "$1") .replaceAll("\\{@language ([^|}]+)\\|?[^}]*}", "$1") .replaceAll("\\{@book ([^}|]+)\\|?[^}]*}", "\"$1\"") - .replaceAll("\\{@hit ([+-][^}<]+)}", "$1") - .replaceAll("\\{@hit ([^}<]+)}", "+$1") + .replaceAll("\\{@(hit|h) ([+-][^}<]+)}", "$2") + .replaceAll("\\{@(hit|h) ([^}<]+)}", "+$2") .replaceAll("\\{@h}", "*Hit:* ") .replaceAll("\\{@m}", "*Miss:* ") .replaceAll("\\{@atk a}", "*Area Attack:*") @@ -235,6 +247,7 @@ default String _replaceTokenText(String input, boolean nested) { .replaceAll("\\{@atk ms}", "*Melee Spell Attack:*") .replaceAll("\\{@atk rs}", "*Ranged Spell Attack:*") .replaceAll("\\{@atk ms,rs}", "*Melee or Ranged Spell Attack:*") + .replaceAll("\\{@spell\\s*}", "") // error in homebrew .replaceAll("\\{@color ([^|}]+)\\|?[^}]*}", "$1") .replaceAll("\\{@style ([^|}]+)\\|?[^}]*}", "$1") .replaceAll("\\{@b ([^}]+?)}", "**$1**") @@ -269,6 +282,7 @@ default String _replaceTokenText(String input, boolean nested) { String[] parts = match.group(1).split("\\|"); if (parts[0].contains("")) { // This already assumes what the footnote name will be + // TODO: Note content is lost on this path at the moment return String.format("%s", parts[0]); } if (parts.length > 2) { diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndex.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndex.java index 6499e5cbf..00a909bac 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndex.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndex.java @@ -98,6 +98,8 @@ private void indexTypes(String filename, JsonNode node) { Tools5eIndexType.spellFluff.withArrayFrom(node, this::addToIndex); Tools5eIndexType.vehicleFluff.withArrayFrom(node, this::addToIndex); + Tools5eIndexType.language.withArrayFrom(node, this::addToIndex); + Tools5eIndexType.itemEntry.withArrayFrom(node, this::addToIndex); Tools5eIndexType.itemTypeAdditionalEntries.withArrayFrom(node, this::addToIndex); Tools5eIndexType.magicvariant.withArrayFrom(node, this::addToIndex); @@ -246,6 +248,7 @@ private boolean addHomebrewSourcesIfPresent(String filename, JsonNode node) { metaTypes.setSkillType(skillName, skill); } } + Tools5eSources.addFonts(SourceField._meta.getFrom(node), HomebrewFields.fonts); return true; } @@ -307,7 +310,9 @@ void addToIndex(Tools5eIndexType type, JsonNode node) { SourceAndPage sp = new SourceAndPage(node); tableIndex.computeIfAbsent(sp, k -> new ArrayList<>()).add(node); } - + if (type == Tools5eIndexType.language && HomebrewFields.fonts.existsIn(node)) { + Tools5eSources.addFonts(node, HomebrewFields.fonts); + } if (node.has("srd")) { srdKeys.add(key); } @@ -1172,6 +1177,7 @@ public void setItemProperty(String key, CustomItemProperty value) { enum HomebrewFields implements JsonNodeReader { abbreviation, + fonts, full, json, optionalFeatureTypes, diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndexType.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndexType.java index ecfa44a3d..dd4c970dc 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndexType.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eIndexType.java @@ -43,6 +43,7 @@ public enum Tools5eIndexType implements IndexType, JsonNodeReader { itemType, itemTypeAdditionalEntries, itemProperty, + language, legendaryGroup, magicvariant, monster, diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eMarkdownConverter.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eMarkdownConverter.java index 3c7b3eb8d..8f5ba5e09 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eMarkdownConverter.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eMarkdownConverter.java @@ -44,6 +44,7 @@ public Tools5eMarkdownConverter writeNotesAndTables() { public Tools5eMarkdownConverter writeImages() { index.tui().copyImages(Tools5eSources.getImages(), fallbackPaths); + index.tui().copyFonts(Tools5eSources.getFonts(), fallbackPaths); return this; } diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eSources.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eSources.java index 004be4ea5..42d426530 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eSources.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/Tools5eSources.java @@ -6,16 +6,20 @@ import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.Map.Entry; import java.util.Optional; import java.util.Set; import com.fasterxml.jackson.databind.JsonNode; import dev.ebullient.convert.config.TtrpgConfig; +import dev.ebullient.convert.io.FontRef; +import dev.ebullient.convert.io.Tui; import dev.ebullient.convert.qute.ImageRef; import dev.ebullient.convert.qute.QuteBase; import dev.ebullient.convert.tools.CompendiumSources; import dev.ebullient.convert.tools.IndexType; +import dev.ebullient.convert.tools.JsonNodeReader; import dev.ebullient.convert.tools.JsonTextConverter.SourceField; import dev.ebullient.convert.tools.ToolsIndex.TtrpgValue; import dev.ebullient.convert.tools.dnd5e.JsonSource.JsonMediaHref; @@ -27,6 +31,7 @@ public class Tools5eSources extends CompendiumSources { private static final Map keyToSources = new HashMap<>(); private static final Map imageSourceToRef = new HashMap<>(); + private static final Map fontSourceToRef = new HashMap<>(); private static final Map> keyToInlineNotes = new HashMap<>(); public static Tools5eSources findSources(String key) { @@ -86,6 +91,51 @@ public void addInlineNote(QuteBase note) { keyToInlineNotes.computeIfAbsent(this.key, k -> new ArrayList<>()).add(note); } + public static Collection getFonts() { + return fontSourceToRef.values().stream() + .filter(FontRef::hasTextReference) + .toList(); + } + + public static void addFonts(JsonNode source, JsonNodeReader field) { + if (field.isArrayIn(source)) { + for (JsonNode font : field.iterateArrayFrom(source)) { + addFont(font.asText()); + } + } else if (field.isObjectIn(source)) { + for (Entry font : field.iterateFieldsFrom(source)) { + addFont(font.getKey(), font.getValue().asText()); + } + } + } + + static void addFont(String fontFamily, String fontString) { + FontRef ref = FontRef.of(fontFamily, fontString); + if (ref == null) { + Tui.instance().warnf("Font '%s' is invalid, empty, or not found", fontString); + } else { + FontRef previous = fontSourceToRef.putIfAbsent(fontFamily, ref); + if (previous != null) { + Tui.instance().warnf("Font '%s' is already defined as '%s'", fontString, previous); + } + } + } + + static void addFont(String fontString) { + String fontFamily = FontRef.fontFamily(fontString); + addFont(fontFamily, fontString); + } + + public static String getFontReference(String fontString) { + String fontFamily = FontRef.fontFamily(fontString); + FontRef ref = fontSourceToRef.get(fontFamily); + if (ref == null) { + return null; + } + ref.addTextReference(); + return fontFamily; + } + final boolean srd; final boolean basicRules; final Tools5eIndexType type; diff --git a/src/main/java/dev/ebullient/convert/tools/dnd5e/qute/QuteMonster.java b/src/main/java/dev/ebullient/convert/tools/dnd5e/qute/QuteMonster.java index 4c01c7ccb..0b56d5d27 100644 --- a/src/main/java/dev/ebullient/convert/tools/dnd5e/qute/QuteMonster.java +++ b/src/main/java/dev/ebullient/convert/tools/dnd5e/qute/QuteMonster.java @@ -369,7 +369,11 @@ public String toString() { /** Formatted description: renders all attributes (other than name) */ public String getDesc() { - List text = new ArrayList<>(headerEntries); + List text = new ArrayList<>(); + if (!headerEntries.isEmpty()) { + text.addAll(headerEntries); + text.add(""); + } appendList(text, "At will", will); if (daily != null && !daily.isEmpty()) { @@ -378,8 +382,10 @@ public String getDesc() { if (spells != null && !spells.isEmpty()) { spells.forEach((k, v) -> appendList(text, spellToTitle(k, v), v.spells)); } - - text.addAll(footerEntries); + if (!footerEntries.isEmpty()) { + text.add(""); + text.addAll(footerEntries); + } return String.join("\n", text); } diff --git a/src/main/resources/templates/tools5e/css-font.txt b/src/main/resources/templates/tools5e/css-font.txt new file mode 100644 index 000000000..f17310fcb --- /dev/null +++ b/src/main/resources/templates/tools5e/css-font.txt @@ -0,0 +1,7 @@ +@font-face { + font-family: "{fontFamily}"; + font-style: normal; + font-display: swap; + font-weight: 400; + src: url("data:font/{type};charset=utf8;base64,{encoded}"); +} diff --git a/src/scss/dnd5e/_float-images.scss b/src/scss/dnd5e/_float-images.scss index 016bbf24b..4ffe6b7b4 100644 --- a/src/scss/dnd5e/_float-images.scss +++ b/src/scss/dnd5e/_float-images.scss @@ -13,7 +13,8 @@ .json5e-reward, .json5e-spell, .json5e-vehicle { - div { + div, + img { &[src$="#center"] { text-align: center; } diff --git a/src/test/java/dev/ebullient/convert/TestUtils.java b/src/test/java/dev/ebullient/convert/TestUtils.java index 1e368b16d..cbf9c54e8 100644 --- a/src/test/java/dev/ebullient/convert/TestUtils.java +++ b/src/test/java/dev/ebullient/convert/TestUtils.java @@ -287,6 +287,7 @@ static List checkDirectoryContents(Path directory, Tui tui, if (!p.toString().endsWith(".md")) { if (!p.toString().endsWith(".png") && !p.toString().endsWith(".jpg") + && !p.toString().endsWith(".css") && !p.toString().endsWith(".svg") && !p.toString().endsWith(".webp") && !p.toString().endsWith(".json") diff --git a/src/test/java/dev/ebullient/convert/Tools5eDataConvertTest.java b/src/test/java/dev/ebullient/convert/Tools5eDataConvertTest.java index 4f6b31432..f7219b316 100644 --- a/src/test/java/dev/ebullient/convert/Tools5eDataConvertTest.java +++ b/src/test/java/dev/ebullient/convert/Tools5eDataConvertTest.java @@ -147,6 +147,7 @@ void testLiveData_5eHomebrew(QuarkusMainLauncher launcher) { TestUtils.TEST_RESOURCES.resolve("psion.json").toString(), TestUtils.TEST_RESOURCES.resolve("ermis-bg.json").toString(), TestUtils.HOMEBREW_PATH_5E.resolve("adventure/Anthony Joyce; The Blood Hunter Adventure.json").toString(), + TestUtils.HOMEBREW_PATH_5E.resolve("adventure/JVC Parry; Call from the Deep.json").toString(), TestUtils.HOMEBREW_PATH_5E.resolve("adventure/Kobold Press; Book of Lairs.json").toString(), TestUtils.HOMEBREW_PATH_5E.resolve("background/D&D Wiki; Featured Quality Backgrounds.json").toString(), TestUtils.HOMEBREW_PATH_5E.resolve("book/Ghostfire Gaming; Grim Hollow Campaign Guide.json").toString(), @@ -167,6 +168,8 @@ void testLiveData_5eHomebrew(QuarkusMainLauncher launcher) { TestUtils.HOMEBREW_PATH_5E .resolve("collection/MCDM Productions; The Talent and Psionics Open Playtest Round 2.json") .toString(), + TestUtils.HOMEBREW_PATH_5E.resolve("creature/Kobold Press; Creature Codex.json") + .toString(), TestUtils.HOMEBREW_PATH_5E.resolve("creature/MCDM Productions; Flee, Mortals!.json").toString(), TestUtils.HOMEBREW_PATH_5E.resolve("creature/Nerzugal Role-Playing; Nerzugal's Extended Bestiary.json") .toString(), diff --git a/src/test/resources/sources-homebrew.json b/src/test/resources/sources-homebrew.json index 1fb55ff79..92ddebfc3 100644 --- a/src/test/resources/sources-homebrew.json +++ b/src/test/resources/sources-homebrew.json @@ -4,6 +4,8 @@ "B:P", "BH2022", "BookofLairs", + "CallfromtheDeep", + "CCodex", "DM14", "DM:BFS", "DMG",