Velocity ist eine tolle Sache. Viel toller als der Internet Explorer jedenfalls. Der IE < 9 beispielsweise crasht, wenn ein Ajax-Response zu groß ist. Zugegeben: Der fragliche Response war auch wirklich groß, so 6 MB oder mehr. Er besteht aber zum Glück größtenteils aus Whitespaces, die daraus resultierten, dass zig Velocity-Templates (jedes mit zig Schleifen) zusammengemerged wurden – und Velocity erhält Whitespace. Man könnte “einfach” alle Whitespaces aus den Templates nehmen, und hätte vermutlich viel gewonnen. Allerdings könnte dann niemand mehr die Templates warten (weil lesen), und das wäre wiederrum nicht so toll.
Deshalb die Idee, den Velocity-Output zu “säubern” (zu trimmen, zu stripen, …). Gesagt getan: Eine Funktion, die die entsprechenden Chars löscht, könnte beispielsweise so aussehen:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
public static String trimInnerWhiteSpacesToOne( String s ) { s = replaceAll(s, "rn", ""); s = replaceAll(s, "n", ""); s = replaceAll(s, "t", "␣"); // zur Visualisierung wurden ␣ eingesetzt s = replaceAll(s, "␣␣", "␣"); // zur Visualisierung wurden ␣ eingesetzt return s; } public static String replaceAll( String s, String search, String replace) { if( s != null ) { String tmp = s.replaceAll( search, replace ); if( s.equals(tmp) ) { return s; } else { return replaceAll( tmp, search, replace ); } } return null; } |
Die Rekursion ist deshalb nötig, weil ␣␣␣␣ sonst nur zu ␣␣ statt zu ␣ werden.
Das funktionert auch ganz gut. Blöd, wenn man ein SAP hinter seiner Webseite hat, denn SAP kennt IDs, die zum Beispiel so aussehen: “01101NA50021435513400011␣␣2012081499991231␣␣␣␣␣␣␣␣␣␣000”. Die stehen bei uns in data-Attributen, und müssen so erhalten bleiben. Deshalb musste eine Lösung her, die Whitespaces filtert, aber NICHT in data-Attributen (genauer gesagt heißen die Attribute data-entry-json).
Die Velocity-Jungs empfehlen für das “Whitespace Gobbling” JTidy, aber das funktioniert nicht (out of the box); die Whitespaces in data-entry-json-Attributen bleiben erhalten.
Besser ist der HtmlCompressor, der das zwar auch nicht out of the box kann, der aber die Möglichkeit bietet, eigene Patterns zu definieren, die erhalten bleiben sollen. Laut Doku ist das für bestimmte Tags, aber man kann auch freiere Patterns angeben:
1 2 3 4 5 |
ArrayList preserve = new ArrayList(); preserve.add(Pattern.compile("<[^>]*data-entry-json[^>]*>", Pattern.DOTALL | Pattern.CASE_INSENSITIVE)); HtmlCompressor compressor = new HtmlCompressor(); compressor.setPreservePatterns(preserve); String trimmed = compressor.compress(html); |
Das obige Pattern sucht nach Tags, die “data-entry-json” enthalten, und sagt dem HtmlCompressor, diese Tags zu erhalten.