aboutsummaryrefslogtreecommitdiff
path: root/src/main/java/ftbsc/lll/proxies/impl/FieldProxy.java
blob: f445ec678f7ec972ba7318306cd3db0b1d582444 (plain) (blame)
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
package ftbsc.lll.proxies.impl;

import ftbsc.lll.proxies.AbstractProxy;
import ftbsc.lll.proxies.ProxyType;
import ftbsc.lll.proxies.QualifiableProxy;
import org.objectweb.asm.Type;

import java.lang.reflect.Field;

import static ftbsc.lll.utils.DescriptorBuilder.nameToDescriptor;

/**
 * A container for information about class fields to be used
 * in ASM patching.
 * @since 0.3.0
 */
public class FieldProxy extends AbstractProxy {
   /**
    * Protected constructor, called only from the builder.
    * @param name the name of the field
    * @param descriptor the descriptor of the field
    * @param modifiers the modifiers of the field
    * @param parent the {@link QualifiableProxy} for the parent
    */
   protected FieldProxy(String name, String descriptor, int modifiers, QualifiableProxy parent) {
      super(name, descriptor, modifiers, parent, ProxyType.FIELD);
   }

   /**
    * A public constructor, builds a proxy from a {@link Field}
    * obtained from reflection.
    * @param f the {@link Field} object corresponding to this.
    */
   public FieldProxy(Field f) {
      this(f.getName(), Type.getDescriptor(f.getType()), f.getModifiers(), TypeProxy.from(f.getDeclaringClass()));
   }

   /**
    * Returns a new instance of {@link FieldProxy.Builder}.
    * @param name the name of the field
    * @return the builder object for field proxies
    */
   public static Builder builder(String name) {
      return new Builder(name);
   }

   /**
    * Indicates whether the given object is a proxy for the same element as this.
    * @param obj the object to perform
    * @return true if it's equal
    */
   @Override
   public boolean equals(Object obj) {
      return obj instanceof FieldProxy && super.equals(obj);
   }

   /**
    * A builder object for {@link FieldProxy}.
    */
   public static class Builder extends AbstractProxy.Builder<FieldProxy> {
      /**
       * The constructor of the builder, used only internally.
       * @param name the name of the field
       */
      Builder(String name) {
         super(name);
      }

      /**
       * Sets the parent class of this field to the one described by the
       * fully qualified name and with the given modifiers.
       * @param parentFQN the fully qualified name of the parent
       * @param modifiers the modifiers of the parent
       * @return the builder's state after the change
       */
      public Builder setParent(String parentFQN, int modifiers) {
         super.setParent(TypeProxy.from(parentFQN, 0, modifiers));
         return this;
      }

      /**
       * Sets the parent class of this field to the one described by the
       * fully qualified name.
       * @param parentFQN the fully qualified name of the parent
       * @return the builder's state after the change
       */
      public Builder setParent(String parentFQN) {
         return this.setParent(parentFQN, 0);
      }

      /**
       * Sets the type of the field to the given {@link Class} object.
       * @param clazz the {@link Class} corresponding to the element
       * @return the current state of the builder
       */
      public Builder setType(Class<?> clazz) {
         super.setType(Type.getType(clazz));
         return this;
      }

      /**
       * Sets the type of the field to the given type.
       * @param fqn the fully qualified name of the parameter type
       * @param arrayLevel the array level
       * @return the builder's state after the change
       */
      public Builder setType(String fqn, int arrayLevel) {
         super.setType(Type.getType(nameToDescriptor(fqn, arrayLevel)));
         return this;
      }

      /**
       * Builds a {@link FieldProxy} of the given kind.
       * @return the built {@link FieldProxy}
       */
      @Override
      public FieldProxy build() {
         return new FieldProxy(this.name, this.descriptor, this.modifiers, this.parent);
      }
   }
}