All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.fxmisc.richtext.TextFlowExt Maven / Gradle / Ivy

package org.fxmisc.richtext;

import static org.fxmisc.richtext.model.TwoDimensional.Bias.*;
import java.lang.reflect.Array;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;

import org.fxmisc.richtext.model.TwoLevelNavigator;

import javafx.scene.control.IndexRange;
import javafx.scene.shape.PathElement;
import javafx.scene.text.TextFlow;
import javafx.scene.shape.LineTo;
import javafx.scene.shape.MoveTo;

/**
 * Adds additional API to {@link TextFlow}.
 */
class TextFlowExt extends TextFlow {

    private static Method mGetTextLayout;
    static {
        try {
            mGetTextLayout = TextFlow.class.getDeclaredMethod("getTextLayout");
        } catch (NoSuchMethodException | SecurityException e) {
            throw new RuntimeException(e);
        }
        mGetTextLayout.setAccessible(true);
    }

    private static Object invoke(Method m, Object obj, Object... args) {
        try {
            return m.invoke(obj, args);
        } catch (IllegalAccessException | IllegalArgumentException
                | InvocationTargetException e) {
            throw new RuntimeException(e);
        }
    }


    int getLineCount() {
        return getLines().length;
    }

    int getLineStartPosition(int charIdx) {
        TextLine[] lines = getLines();
        TwoLevelNavigator navigator = new TwoLevelNavigator(
                () -> lines.length,
                i -> lines[i].getLength());
        int currentLineIndex = navigator.offsetToPosition(charIdx, Forward).getMajor();
        return navigator.position(currentLineIndex, 0).toOffset();
    }

    int getLineEndPosition(int charIdx) {
        TextLine[] lines = getLines();
        TwoLevelNavigator navigator = new TwoLevelNavigator(
                () -> lines.length,
                i -> lines[i].getLength());
        int currentLineIndex = navigator.offsetToPosition(charIdx, Forward).getMajor();
        int minor = currentLineIndex == lines.length - 1 ? 0 : -1;
        return navigator.position(currentLineIndex + 1, minor).toOffset();
    }

    int getLineOfCharacter(int charIdx) {
        TextLine[] lines = getLines();
        TwoLevelNavigator navigator = new TwoLevelNavigator(
                () -> lines.length,
                i -> lines[i].getLength());
        return navigator.offsetToPosition(charIdx, Forward).getMajor();
    }

    PathElement[] getCaretShape(int charIdx, boolean isLeading) {
// use if Java 9 becomes minimum requirement
//        return caretShape(charIdx, isLeading);
        return textLayout().getCaretShape(charIdx, isLeading, 0, 0);
    }

    PathElement[] getRangeShape(IndexRange range) {
        return getRangeShape(range.getStart(), range.getEnd());
    }

    PathElement[] getRangeShape(int from, int to) {
// use if Java 9 becomes minimum requirement
//        return rangeShape(from, to);
        return textLayout().getRange(from, to, TextLayout.TYPE_TEXT, 0, 0);
    }

    PathElement[] getUnderlineShape(IndexRange range) {
        return getUnderlineShape(range.getStart(), range.getEnd());
    }

    /**
     * @param from The index of the first character.
     * @param to The index of the last character.
     * @return An array with the PathElement objects which define an
     *         underline from the first to the last character.
     */
    PathElement[] getUnderlineShape(int from, int to) {
        // get a Path for the text underline
        PathElement[] shape = textLayout().getRange(from, to, TextLayout.TYPE_UNDERLINE, 0, 0);

        // The shape is returned as a closed Path (a thin rectangle).
        // If we use the Path as it is, this causes rendering issues.
        // Hence we only use the MoveTo and the succeeding LineTo elements for the result
        // so that simple line segments instead of rectangles are returned.
        List result = new ArrayList<>();

        boolean collect = false;
        for (PathElement elem : shape) {
            if (elem instanceof MoveTo) {   // There seems to be no API to get the type of the PathElement
                result.add(elem);
                collect = true;
            } else if (elem instanceof LineTo) {
                if (collect) {
                    result.add(elem);
                    collect = false;
                }
            }
        }

       return result.toArray(new PathElement[0]);
    }

    CharacterHit hitLine(double x, int lineIndex) {
        return hit(x, getLineCenter(lineIndex));
    }

    CharacterHit hit(double x, double y) {
// use if Java 9 becomes minimum requirement
//        HitInfo hit = hitTest(new Point2D(x, y));
        HitInfo hit = textLayout().getHitInfo((float) x, (float) y);
        int charIdx = hit.getCharIndex();
        boolean leading = hit.isLeading();

        int lineIdx = getLineIndex((float) y);
        if(lineIdx >= getLineCount()) {
            return CharacterHit.insertionAt(getCharCount());
        }

        TextLine[] lines = getLines();
        TextLine line = lines[lineIdx];
        RectBounds lineBounds = line.getBounds();

        // If this is a wrapped paragraph and hit character is at end of hit line,
        // make sure that the "character hit" stays at the end of the hit line
        // (and not at the beginning of the next line).
        if(lines.length > 1 &&
            lineIdx < lines.length - 1 &&
            charIdx + 1 >= line.getStart() + line.getLength() &&
            !leading)
        {
            leading = true;
        }

        if(x < lineBounds.getMinX() || x > lineBounds.getMaxX()) {
            if(leading) {
                return CharacterHit.insertionAt(charIdx);
            } else {
                return CharacterHit.insertionAt(charIdx + 1);
            }
        } else {
            if(leading) {
                return CharacterHit.leadingHalfOf(charIdx);
            } else {
                return CharacterHit.trailingHalfOf(charIdx);
            }
        }
    }

    private float getLineY(int index) {
        TextLine[] lines = getLines();
        float spacing = (float) getLineSpacing();
        float lineY = 0;
        for(int i = 0; i < index; ++i) {
            lineY += lines[i].getBounds().getHeight() + spacing;
        }
        return lineY;
    }

    private float getLineCenter(int index) {
        return getLineY(index) + getLines()[index].getBounds().getHeight() / 2;
    }

    private TextLine[] getLines() {
        return textLayout().getLines();
    }

    private int getLineIndex(float y) {
        return textLayout().getLineIndex(y);
    }

    private int getCharCount() {
        return textLayout().getCharCount();
    }

    private TextLayout textLayout() {
        return GenericIceBreaker.proxy(TextLayout.class, invoke(mGetTextLayout, this));
    }

    /* ********************************************************************** *
     *                                                                        *
     * GenericIceBreaker                                                      *
     *                                                                        *
     * ********************************************************************** */

    private static class GenericIceBreaker implements InvocationHandler {
        private final Object delegate;

        private GenericIceBreaker(Object delegate) {
            this.delegate = delegate;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            Method delegateMethod = getDeclaredMethod(delegate.getClass(), method.getName(), method.getParameterTypes());

            Object delegateMethodReturn = null;
            try {
                delegateMethodReturn = delegateMethod.invoke(delegate, args);
            } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
                throw new RuntimeException("problems invoking " + method.getName());
            }
            if (delegateMethodReturn == null) {
                return null;
            }

            if (method.getReturnType().isArray()) {
                if (method.getReturnType().getComponentType().isInterface()
                        && !method.getReturnType().getComponentType().equals(delegateMethod.getReturnType().getComponentType())) {

                    int arrayLength = Array.getLength(delegateMethodReturn);
                    Object retArray = Array.newInstance(method.getReturnType().getComponentType(), arrayLength);
                    for (int i = 0; i < arrayLength; i++) {
                        Array.set(retArray,
                                i,
                                proxy(
                                        method.getReturnType().getComponentType(),
                                        Array.get(delegateMethodReturn, i)));
                    }

                    return retArray;
                }
            }

            if (method.getReturnType().isInterface()
                    && !method.getReturnType().equals(delegateMethod.getReturnType())) {
                return proxy(method.getReturnType(), delegateMethodReturn);
            }

            return delegateMethodReturn;
        }

        @SuppressWarnings("unchecked")
        static  T proxy(Class iface, Object delegate) {
            return (T) Proxy.newProxyInstance(
                    iface.getClassLoader(),
                    new Class[]{iface},
                    new GenericIceBreaker(delegate));
        }

        private static final HashMap declaredMethodCache = new HashMap<>();

        private static synchronized Method getDeclaredMethod(Class cls, String name, Class... paramTypes)
                throws NoSuchMethodException, SecurityException
        {
            MethodCacheKey methodCacheKey = new MethodCacheKey(cls, name, paramTypes);

            Method m = declaredMethodCache.get(methodCacheKey);
            if (m == null) {
                m = cls.getDeclaredMethod(name, paramTypes);
                m.setAccessible(true);
                declaredMethodCache.put(methodCacheKey, m);
            }
            return m;
        }
    }

    private static class MethodCacheKey {
        final Class cls;
        final String name;
        final Class[] paramTypes;

        MethodCacheKey(Class cls, String name, Class... paramTypes) {
            this.cls = cls;
            this.name = name;
            this.paramTypes = paramTypes;
        }

        @Override
        public boolean equals(Object obj) {
            if (!(obj instanceof MethodCacheKey))
                return false;

            MethodCacheKey key2 = (MethodCacheKey) obj;
            return cls == key2.cls && name.equals(key2.name) && Arrays.equals(paramTypes, key2.paramTypes);
        }

        @Override
        public int hashCode() {
            return cls.hashCode() + name.hashCode() + Arrays.hashCode(paramTypes);
        }
    }

    /* ********************************************************************** *
     *                                                                        *
     * Proxy interfaces                                                       *
     *                                                                        *
     * ********************************************************************** */

    private interface TextLayout
    {
        static final int TYPE_TEXT           = 1 << 0;
        static final int TYPE_UNDERLINE      = 1 << 1;

        TextLine[] getLines();
        int getLineIndex(float y);
        int getCharCount();

        HitInfo getHitInfo(float x, float y);
        PathElement[] getCaretShape(int offset, boolean isLeading, float x, float y);
        PathElement[] getRange(int start, int end, int type, float x, float y);
    }

    private interface TextLine
    {
        int getLength();
        RectBounds getBounds();
        int getStart();
    }

    private interface RectBounds
    {
        float getMinX();
        float getMaxX();
        float getHeight();
    }

    private interface HitInfo
    {
        int getCharIndex();
        boolean isLeading();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy