org.jline.style.StyleBundleInvocationHandler Maven / Gradle / Ivy
/*
* Copyright (c) 2002-2017, the original author or authors.
*
* This software is distributable under the BSD license. See the terms of the
* BSD license in the documentation provided with this software.
*
* https://opensource.org/licenses/BSD-3-Clause
*/
package org.jline.style;
import org.jline.style.StyleBundle.DefaultStyle;
import org.jline.style.StyleBundle.StyleGroup;
import org.jline.style.StyleBundle.StyleName;
import org.jline.utils.AttributedString;
import org.jline.utils.AttributedStyle;
import javax.annotation.Nullable;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.logging.Level;
import java.util.logging.Logger;
import static java.util.Objects.requireNonNull;
/**
* {@link StyleBundle} proxy invocation-handler to convert method calls into string styling.
*
* @see StyleBundle
* @since 3.4
*/
class StyleBundleInvocationHandler
implements InvocationHandler {
private static final Logger log = Logger.getLogger(StyleBundleInvocationHandler.class.getName());
private final Class extends StyleBundle> type;
private final StyleResolver resolver;
public StyleBundleInvocationHandler(final Class extends StyleBundle> type, final StyleResolver resolver) {
this.type = requireNonNull(type);
this.resolver = requireNonNull(resolver);
}
/**
* Throws {@link InvalidStyleBundleMethodException} if given method is not suitable.
*/
private static void validate(final Method method) {
// All StyleBundle methods must take exactly 1 parameter
if (method.getParameterCount() != 1) {
throw new InvalidStyleBundleMethodException(method, "Invalid parameters");
}
// All StyleBundle methods must return an AttributeString
if (method.getReturnType() != AttributedString.class) {
throw new InvalidStyleBundleMethodException(method, "Invalid return-type");
}
}
@Nullable
private static String emptyToNull(@Nullable final String value) {
if (value == null || value.isEmpty()) {
return null;
}
return value;
}
/**
* Returns the style group-name for given type, or {@code null} if unable to determine.
*/
@Nullable
private static String getStyleGroup(final Class> type) {
StyleGroup styleGroup = type.getAnnotation(StyleGroup.class);
return styleGroup != null ? emptyToNull(styleGroup.value().trim()) : null;
}
//
// Helpers
//
/**
* Returns the style-name for given method, or {@code null} if unable to determine.
*/
private static String getStyleName(final Method method) {
StyleName styleName = method.getAnnotation(StyleName.class);
return styleName != null ? emptyToNull(styleName.value().trim()) : method.getName();
}
/**
* Returns the default-style for given method, or {@code null} if unable to determine.
*/
@Nullable
private static String getDefaultStyle(final Method method) {
DefaultStyle defaultStyle = method.getAnnotation(DefaultStyle.class);
// allow whitespace in default-style.value, but disallow empty-string
return defaultStyle != null ? emptyToNull(defaultStyle.value()) : null;
}
/**
* Internal factory-method.
*
* @see Styler#bundle(Class)
*/
@SuppressWarnings("unchecked")
static T create(final StyleResolver resolver, final Class type) {
requireNonNull(resolver);
requireNonNull(type);
if (log.isLoggable(Level.FINEST)) {
log.finest(String.format("Using style-group: %s for type: %s", resolver.getGroup(), type.getName()));
}
StyleBundleInvocationHandler handler = new StyleBundleInvocationHandler(type, resolver);
return (T) Proxy.newProxyInstance(type.getClassLoader(), new Class>[]{type}, handler);
}
/**
* Internal factory-method.
*
* @see Styler#bundle(String, Class)
*/
static T create(final StyleSource source, final Class type) {
requireNonNull(type);
String group = getStyleGroup(type);
if (group == null) {
throw new InvalidStyleGroupException(type);
}
return create(new StyleResolver(source, group), type);
}
//
// Factory access
//
@Override
public Object invoke(final Object proxy, final Method method, final Object[] args) throws Throwable {
// Allow invocations to Object methods to pass-through
if (method.getDeclaringClass() == Object.class) {
return method.invoke(this, args);
}
// or validate StyleBundle method is valid
validate(method);
// resolve the style-name for method
String styleName = getStyleName(method);
// resolve the sourced-style, or use the default
String style = resolver.getSource().get(resolver.getGroup(), styleName);
if (log.isLoggable(Level.FINEST)) {
log.finest(String.format("Sourced-style: %s -> %s", styleName, style));
}
if (style == null) {
style = getDefaultStyle(method);
// if sourced-style was missing and default-style is missing barf
if (style == null) {
throw new StyleBundleMethodMissingDefaultStyleException(method);
}
}
String value = String.valueOf(args[0]);
if (log.isLoggable(Level.FINEST)) {
log.finest(String.format("Applying style: %s -> %s to: %s", styleName, style, value));
}
AttributedStyle astyle = resolver.resolve(style);
return new AttributedString(value, astyle);
}
/**
* Slightly better logging for proxies.
*/
@Override
public String toString() {
return type.getName();
}
//
// Exceptions
//
/**
* Thrown when {@link StyleBundle} method has missing {@link DefaultStyle}.
*/
// @VisibleForTesting
static class StyleBundleMethodMissingDefaultStyleException
extends RuntimeException {
private static final long serialVersionUID = 1L;
public StyleBundleMethodMissingDefaultStyleException(final Method method) {
super(String.format("%s method missing @%s: %s",
StyleBundle.class.getSimpleName(),
DefaultStyle.class.getSimpleName(),
method
));
}
}
/**
* Thrown when processing {@link StyleBundle} method is found to be invalid.
*/
// @VisibleForTesting
static class InvalidStyleBundleMethodException
extends RuntimeException {
private static final long serialVersionUID = 1L;
public InvalidStyleBundleMethodException(final Method method, final String message) {
super(message + ": " + method);
}
}
/**
* Thrown when looking up {@link StyleGroup} on a type found to be missing or invalid.
*/
// @VisibleForTesting
static class InvalidStyleGroupException
extends RuntimeException {
private static final long serialVersionUID = 1L;
public InvalidStyleGroupException(final Class> type) {
super(String.format("%s missing or invalid @%s: %s",
StyleBundle.class.getSimpleName(),
StyleGroup.class.getSimpleName(),
type.getName()
));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy