Added Abbreviate renderers
This commit is contained in:
parent
4dd5df1d61
commit
1aa2bc2681
8 changed files with 139 additions and 87 deletions
|
@ -36,6 +36,7 @@ This project provides a collection of useful template renderers.
|
||||||
|
|
||||||
| Renderer | Description |
|
| Renderer | Description |
|
||||||
|:--------------------------------------------------------------------------------------------------------------------|:-----------------------------------------------------------------|
|
|:--------------------------------------------------------------------------------------------------------------------|:-----------------------------------------------------------------|
|
||||||
|
| [rife.render.Abbreviate](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Abbreviate) | Abbreviates a template value |
|
||||||
| [rife.render.formatCreditcard](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.FormatCreditCard) | Formats a template credit card number value to the last 4 digits |
|
| [rife.render.formatCreditcard](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.FormatCreditCard) | Formats a template credit card number value to the last 4 digits |
|
||||||
| [rife.render.Mask](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Mask) | Masks characters of a template value |
|
| [rife.render.Mask](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Mask) | Masks characters of a template value |
|
||||||
| [rife.render.Normalize](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Normalize) | Normalizes a template value for inclusion in a URL path |
|
| [rife.render.Normalize](https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Normalize) | Normalizes a template value for inclusion in a URL path |
|
||||||
|
|
61
lib/src/main/java/rife/render/Abbreviate.java
Normal file
61
lib/src/main/java/rife/render/Abbreviate.java
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2023 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package rife.render;
|
||||||
|
|
||||||
|
import rife.template.Template;
|
||||||
|
import rife.template.ValueRenderer;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.StringReader;
|
||||||
|
import java.util.Properties;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Abbreviate a template value with ellipses.</p>
|
||||||
|
*
|
||||||
|
* <p>Usage:</p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* <!--v render:rife.render.Abbreviate:valueId/-->
|
||||||
|
* {{v render:rife.render.Abbreviate:valueId/}}
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
* @author <a href="https://erik.thauvin.net/">Erik C. Thauvin</a>
|
||||||
|
* @see <a href="https://github.com/rife2/rife2-template-renderers/wiki/rife.render.Abbreviate">rife.render.Abbreviate</a>
|
||||||
|
* @since 1.0
|
||||||
|
*/
|
||||||
|
public class Abbreviate implements ValueRenderer {
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String render(Template template, String valueId, String differentiator) {
|
||||||
|
var mark = "...";
|
||||||
|
var max = 0;
|
||||||
|
if (template.hasDefaultValue(valueId)) {
|
||||||
|
var properties = new Properties();
|
||||||
|
try {
|
||||||
|
properties.load(new StringReader(template.getDefaultValue(valueId)));
|
||||||
|
mark = properties.getProperty("mark", mark);
|
||||||
|
max = Integer.parseInt(properties.getProperty("max", String.valueOf(max)));
|
||||||
|
} catch (IOException | NumberFormatException ignore) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return RenderUtils.abbreviate(template.getValueOrAttribute(differentiator), max, mark);
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,7 +25,7 @@ import java.io.StringReader;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Masks characters of a template value.
|
* <p>Masks characters of a template value.</p>
|
||||||
*
|
*
|
||||||
* <p>Usage:</p>
|
* <p>Usage:</p>
|
||||||
*
|
*
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
|
|
||||||
package rife.render;
|
package rife.render;
|
||||||
|
|
||||||
import rife.tools.Localization;
|
|
||||||
import rife.tools.StringUtils;
|
import rife.tools.StringUtils;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -33,14 +32,40 @@ import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Collection of utility-type methods commonly used by the renderers.
|
* Collection of utility-type methods commonly used by the renderers.
|
||||||
|
*
|
||||||
|
* @author <a href="https://erik.thauvin.net/">Erik C. Thauvin</a>
|
||||||
|
* @since 1.0
|
||||||
*/
|
*/
|
||||||
public final class RenderUtils {
|
public final class RenderUtils {
|
||||||
private static final String DEFAULT_USER_AGENT = "Mozilla/5.0 (X11; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/111.0";
|
private static final String DEFAULT_USER_AGENT =
|
||||||
|
"Mozilla/5.0 (X11; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/111.0";
|
||||||
|
|
||||||
private RenderUtils() {
|
private RenderUtils() {
|
||||||
// no-op
|
// no-op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abbreviates a String to the given length using a replacement marker.
|
||||||
|
*
|
||||||
|
* @param src the source String
|
||||||
|
* @param max the maximum length of the resulting String
|
||||||
|
* @param marker the String used as a replacement marker
|
||||||
|
* @return the abbreviated String
|
||||||
|
*/
|
||||||
|
public static String abbreviate(String src, int max, String marker) {
|
||||||
|
if (src == null || src.isBlank()) {
|
||||||
|
return src;
|
||||||
|
}
|
||||||
|
|
||||||
|
var len = src.length();
|
||||||
|
|
||||||
|
if (len <= max || max < 0) {
|
||||||
|
return src;
|
||||||
|
}
|
||||||
|
|
||||||
|
return src.substring(0, max - marker.length()) + marker;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the Swatch Internet (.beat) Time for the give date-time.
|
* Returns the Swatch Internet (.beat) Time for the give date-time.
|
||||||
*
|
*
|
||||||
|
@ -49,7 +74,8 @@ public final class RenderUtils {
|
||||||
*/
|
*/
|
||||||
public static String beatTime(ZonedDateTime zonedDateTime) {
|
public static String beatTime(ZonedDateTime zonedDateTime) {
|
||||||
var zdt = zonedDateTime.withZoneSameInstant(ZoneId.of("UTC+01:00"));
|
var zdt = zonedDateTime.withZoneSameInstant(ZoneId.of("UTC+01:00"));
|
||||||
var beats = (int) ((zdt.get(ChronoField.SECOND_OF_MINUTE) + (zdt.get(ChronoField.MINUTE_OF_HOUR) * 60) + (zdt.get(ChronoField.HOUR_OF_DAY) * 3600)) / 86.4);
|
var beats = (int) ((zdt.get(ChronoField.SECOND_OF_MINUTE) + (zdt.get(ChronoField.MINUTE_OF_HOUR) * 60) +
|
||||||
|
(zdt.get(ChronoField.HOUR_OF_DAY) * 3600)) / 86.4);
|
||||||
return String.format("@%03d", beats);
|
return String.format("@%03d", beats);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -81,6 +107,29 @@ public final class RenderUtils {
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches the content (body) of a URL.
|
||||||
|
*
|
||||||
|
* @param url the URL string.
|
||||||
|
* @param defaultContent the default content to return if none fetched.
|
||||||
|
* @return the url content, or empty.
|
||||||
|
*/
|
||||||
|
public static String fetchUrl(String url, String defaultContent) {
|
||||||
|
try {
|
||||||
|
var connection = (HttpURLConnection) new URL(url).openConnection();
|
||||||
|
connection.setRequestProperty("User-Agent", DEFAULT_USER_AGENT);
|
||||||
|
var code = connection.getResponseCode();
|
||||||
|
if (code >= 200 && code <= 399) {
|
||||||
|
try (var inputStream = connection.getInputStream()) {
|
||||||
|
return new String(inputStream.readAllBytes(), StandardCharsets.UTF_8);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (IOException ignore) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
return defaultContent;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the last 4 digits a credit card number. The number must satisfy the Luhn algorithm.
|
* Returns the last 4 digits a credit card number. The number must satisfy the Luhn algorithm.
|
||||||
* Non-digits are stripped from the number.
|
* Non-digits are stripped from the number.
|
||||||
|
@ -204,23 +253,8 @@ public final class RenderUtils {
|
||||||
if (src == null || src.isBlank()) {
|
if (src == null || src.isBlank()) {
|
||||||
return src;
|
return src;
|
||||||
}
|
}
|
||||||
|
return fetchUrl(String.format("https://api.qrserver.com/v1/create-qr-code/?format=svg&size=%s&data=%s", size,
|
||||||
var svg = src;
|
StringUtils.encodeUrl(src.trim())), src);
|
||||||
try {
|
|
||||||
var connection = (HttpURLConnection) new URL(
|
|
||||||
String.format("https://api.qrserver.com/v1/create-qr-code/?format=svg&size=%s&data=%s", size,
|
|
||||||
StringUtils.encodeUrl(src.trim())))
|
|
||||||
.openConnection();
|
|
||||||
connection.setRequestProperty("User-Agent", DEFAULT_USER_AGENT);
|
|
||||||
if (validResponseCode(connection.getResponseCode())) {
|
|
||||||
try (var inputStream = connection.getInputStream()) {
|
|
||||||
svg = new String(inputStream.readAllBytes(), StandardCharsets.UTF_8);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (IOException ignore) {
|
|
||||||
// do nothing
|
|
||||||
}
|
|
||||||
return svg;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -274,22 +308,8 @@ public final class RenderUtils {
|
||||||
if (url == null || url.isBlank() || !url.matches("^[Hh][Tt][Tt][Pp][Ss]?://\\w.*")) {
|
if (url == null || url.isBlank() || !url.matches("^[Hh][Tt][Tt][Pp][Ss]?://\\w.*")) {
|
||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
return fetchUrl(String.format("https://is.gd/create.php?format=simple&url=%s",
|
||||||
var shorten = url;
|
StringUtils.encodeUrl(url.trim())), url);
|
||||||
try {
|
|
||||||
var connection = (HttpURLConnection) new URL(
|
|
||||||
String.format("https://is.gd/create.php?format=simple&url=%s", StringUtils.encodeUrl(url.trim())))
|
|
||||||
.openConnection();
|
|
||||||
connection.setRequestProperty("User-Agent", DEFAULT_USER_AGENT);
|
|
||||||
if (validResponseCode(connection.getResponseCode())) {
|
|
||||||
try (var inputStream = connection.getInputStream()) {
|
|
||||||
shorten = new String(inputStream.readAllBytes(), StandardCharsets.UTF_8);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (IOException ignore) {
|
|
||||||
// do nothing
|
|
||||||
}
|
|
||||||
return shorten;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -351,43 +371,6 @@ public final class RenderUtils {
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts the given String to a quoted-printable string.
|
|
||||||
*
|
|
||||||
* @param src the source String
|
|
||||||
* @return the quoted-printable String
|
|
||||||
*/
|
|
||||||
public static String toQuotedPrintable(String src) {
|
|
||||||
if (src == null || src.isEmpty()) {
|
|
||||||
return src;
|
|
||||||
}
|
|
||||||
|
|
||||||
var len = src.length();
|
|
||||||
var buff = new StringBuilder(len);
|
|
||||||
|
|
||||||
char c;
|
|
||||||
String hex;
|
|
||||||
for (var i = 0; i < len; i++) {
|
|
||||||
c = src.charAt(i);
|
|
||||||
|
|
||||||
if (((c > 47) && (c < 58)) || ((c > 64) && (c < 91)) || ((c > 96) && (c < 123))) {
|
|
||||||
buff.append(c);
|
|
||||||
} else {
|
|
||||||
hex = Integer.toString(c, 16);
|
|
||||||
|
|
||||||
buff.append('=');
|
|
||||||
|
|
||||||
if (hex.length() == 1) {
|
|
||||||
buff.append('0');
|
|
||||||
}
|
|
||||||
|
|
||||||
buff.append(hex.toUpperCase(Localization.getLocale()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return buff.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the formatted server uptime.
|
* Returns the formatted server uptime.
|
||||||
*
|
*
|
||||||
|
@ -439,14 +422,4 @@ public final class RenderUtils {
|
||||||
|
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks whether the specified HTTP response code is valid.
|
|
||||||
*
|
|
||||||
* @param code the HTTP response code.
|
|
||||||
* @return {@code true} if the response code is valid.
|
|
||||||
*/
|
|
||||||
public static boolean validResponseCode(int code) {
|
|
||||||
return code >= 200 && code <= 399;
|
|
||||||
}
|
|
||||||
}
|
}
|
|
@ -23,6 +23,18 @@ import rife.template.TemplateFactory;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
class TestFormat {
|
class TestFormat {
|
||||||
|
@Test
|
||||||
|
void testAbbreviate() {
|
||||||
|
var t = TemplateFactory.HTML.get("abbreviate");
|
||||||
|
t.setAttribute(TestCase.FOO, TestCase.SAMPLE_TEXT);
|
||||||
|
System.out.println(t.getContent());
|
||||||
|
assertThat(t.getContent()).as("max=12").endsWith("…").hasSize(12);
|
||||||
|
|
||||||
|
t = TemplateFactory.TXT.get("abbreviate");
|
||||||
|
t.setAttribute(TestCase.FOO, TestCase.SAMPLE_TEXT);
|
||||||
|
assertThat(t.getContent()).as("max=8").endsWith("...").hasSize(8);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void testFormatCreditCard() {
|
void testFormatCreditCard() {
|
||||||
var t = TemplateFactory.TXT.get("formatCreditCard");
|
var t = TemplateFactory.TXT.get("formatCreditCard");
|
||||||
|
|
4
lib/src/test/resources/templates/abbreviate.html
Normal file
4
lib/src/test/resources/templates/abbreviate.html
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
<!--v render:rife.render.Abbreviate:foo-->
|
||||||
|
mark=…
|
||||||
|
max=12
|
||||||
|
<!--/v-->
|
3
lib/src/test/resources/templates/abbreviate.txt
Normal file
3
lib/src/test/resources/templates/abbreviate.txt
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
{{v render:rife.render.Abbreviate:foo}}
|
||||||
|
max=8
|
||||||
|
{{/v}}
|
|
@ -1,3 +1 @@
|
||||||
<!--v render:rife.render.DateTimeIso-->
|
<!--v render:rife.render.DateTimeIso-->tz=UTC<!--/v-->
|
||||||
tz=UTC
|
|
||||||
<!--/v-->
|
|
Loading…
Add table
Add a link
Reference in a new issue