summaryrefslogtreecommitdiff
path: root/src/main/java/org/javacs/ShortTypePrinter.java
blob: c20e44760a15fcd36e857adf9e42474922720e99 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
package org.javacs;

import java.util.StringJoiner;
import java.util.stream.Collectors;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.type.*;
import javax.lang.model.util.AbstractTypeVisitor8;

class ShortTypePrinter extends AbstractTypeVisitor8<String, Void> {
    private ShortTypePrinter() {}

    static String print(TypeMirror type) {
        // TODO allow packageContext which suppresses printing of those package prefixes
        return type.accept(new ShortTypePrinter(), null);
    }

    @Override
    public String visitIntersection(IntersectionType t, Void aVoid) {
        return t.getBounds().stream().map(ShortTypePrinter::print).collect(Collectors.joining(" & "));
    }

    @Override
    public String visitUnion(UnionType t, Void aVoid) {
        return t.getAlternatives().stream().map(ShortTypePrinter::print).collect(Collectors.joining(" | "));
    }

    @Override
    public String visitPrimitive(PrimitiveType t, Void aVoid) {
        return t.toString();
    }

    @Override
    public String visitNull(NullType t, Void aVoid) {
        return t.toString();
    }

    @Override
    public String visitArray(ArrayType t, Void aVoid) {
        return print(t.getComponentType()) + "[]";
    }

    @Override
    public String visitDeclared(DeclaredType t, Void aVoid) {
        var result = t.asElement().toString();

        if (!t.getTypeArguments().isEmpty()) {
            String params =
                    t.getTypeArguments().stream().map(ShortTypePrinter::print).collect(Collectors.joining(", "));

            result += "<" + params + ">";
        }

        if (result.startsWith("java.lang")) return result.substring("java.lang.".length());
        else if (result.startsWith("java.util")) return result.substring("java.util.".length());
        else return result;
    }

    @Override
    public String visitError(ErrorType t, Void aVoid) {
        return "???";
    }

    @Override
    public String visitTypeVariable(TypeVariable t, Void aVoid) {
        String result = t.asElement().toString();
        // TypeMirror upper = t.getUpperBound();
        // NOTE this can create infinite recursion
        // if (!upper.toString().equals("java.lang.Object"))
        //     result += " extends " + print(upper);

        return result;
    }

    @Override
    public String visitWildcard(WildcardType t, Void aVoid) {
        String result = "?";

        if (t.getSuperBound() != null) result += " super " + print(t.getSuperBound());

        if (t.getExtendsBound() != null) result += " extends " + print(t.getExtendsBound());

        return result;
    }

    @Override
    public String visitExecutable(ExecutableType t, Void aVoid) {
        return t.toString();
    }

    @Override
    public String visitNoType(NoType t, Void aVoid) {
        return t.toString();
    }

    static boolean missingParamNames(ExecutableElement e) {
        return e.getParameters().stream().allMatch(p -> p.getSimpleName().toString().matches("arg\\d+"));
    }

    private static String printArguments(ExecutableElement e) {
        var result = new StringJoiner(", ");
        var missingParamNames = missingParamNames(e);
        for (var p : e.getParameters()) {
            var s = new StringBuilder();
            s.append(ShortTypePrinter.print(p.asType()));
            if (!missingParamNames) {
                s.append(" ").append(p.getSimpleName());
            }
            result.add(s);
        }
        return result.toString();
    }

    static String printMethod(ExecutableElement m) {
        if (m.getSimpleName().contentEquals("<init>")) {
            return m.getEnclosingElement().getSimpleName() + "(" + printArguments(m) + ")";
        } else {
            var result = new StringBuilder();
            // static void foo
            if (m.getModifiers().contains(Modifier.STATIC)) result.append("static ");
            result.append(ShortTypePrinter.print(m.getReturnType())).append(" ");
            result.append(m.getSimpleName());
            // (int arg, String other)
            result.append("(").append(printArguments(m)).append(")");
            // throws Foo, Bar
            if (!m.getThrownTypes().isEmpty()) {
                result.append(" throws ");
                var types = new StringJoiner(", ");
                for (var t : m.getThrownTypes()) {
                    types.add(ShortTypePrinter.print(t));
                }
                result.append(types);
            }
            return result.toString();
        }
    }
}