-
Notifications
You must be signed in to change notification settings - Fork 727
Generating Models from View Annotations
Available in upcoming 2.3.0 release
If you use custom views, an EpoxyModel can be completely generated for each view based on annotations in the view. This removes much of the boilerplate of creating a model manually, and helps make sure you have an accurate, optimized model setup.
To start, first annotate your custom view with @ModelView
:
@ModelView(defaultLayout = R.layout.my_view)
public class MyView extends FrameLayout {
...
}
The provided layout file (R.layout.my_view
in this case) is the layout that will be inflated when Epoxy creates your view for use in the RecyclerView. It should contain only one child, whose type is your custom view.
In this case our layout might look like this:
<?xml version="1.0" encoding="utf-8"?>
<com.example.MyView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
If you have any styling to apply to the view you can do it in this layout.
Once, this annotation is added to the view and the code is compiled, a MyViewModel_
class will be generated in the same package. The name of this class is always the name of the view suffixed with Model_
to represent that it is generated.
This generated model can then be used in your EpoxyControllers.
The generated model binds data to your view for you. To allow this, annotate setter methods in your view with @ModelProp
.
@ModelView(defaultLayout = R.layout.my_view)
public class MyView extends FrameLayout {
...
@ModelProp
public void setImageUrl(String url){
...
}
}
This will result in an imageUrl
method being added to the MyViewModel_
class that you can use to set the url.
When a view scrolls onto screen in the RecyclerView, Epoxy will take the values of each prop set on the model and set them on the view.
If the models change while the view is on screen Epoxy will do an efficient partial update. For example, the MyViewModel_
checks the value of the url string and only updates the view when the url changes. If there are multiple props on the view and only a subset of them change, only the changed properties are updated.
Setters must have exactly one parameter, should have a void
return type, and cannot be static or private.
If there is demand we may support setters with multiple parameters (open an issue if you are interested in this).
Epoxy will generate methods to use Android String and Plural resources if the GenerateStringOverloads
option is included in the @ModelProp
annotation. For example:
@ModelProp(options = Option.GenerateStringOverloads)
public void setTitle(CharSequence text) {
}
If this is used the parameter type must be CharSequence
.
The generated model will have the following convenience methods for getting and setting the text:
// The getter requires a context to resolve string resources
public CharSequence getTitle(Context context) {
...
}
public TestViewModel_ title(CharSequence title) {
...
}
public TestViewModel_ title(@StringRes int stringRes) {
...
}
public TestViewModel_ title(@StringRes int stringRes, Object... formatArgs) {
...
}
public TestViewModel_ titleQuantityRes(@PluralsRes int pluralRes, int quantity,
Object... formatArgs) {
...
}
Any javadoc on a ModelProp method will be included on the generated setter on the view's model. Additionally, the model's setter will have javadoc that specifies whether the prop is required or optional, what the default value is, and a link to the view's setter method.
Any annotations included on a ModelProp setter parameter will be included on the generated model's method. This may be helpful for annotations such as @Nullable
, @DimenRes
, @FloatRange
, and other Android support library annotations.
For example:
@ModelProp
void setCount(@IntRange(from = 0, to = 10) int value) {
...
}
There are a few options for specifying default values for props in order to make them optional.
Props that are a subtype of Object
(aka non-primitive types) are required by default. They can be made optional by adding a @Nullable
annotation to the setter parameter. (Any annotation with the name "Nullable" will work)
@ModelProp
public void setImage(@Nullable MyObject param)
If this prop is not specified on the model then this method will be set to null when the view is bound.
Props that have a primitive type are always optional. If the user of the model does specify an explicit value for a prop then it will default to the default value of the primitive as specified by the Java Language.
For example, int
will default to 0 and boolean
will default to false.
It is not possible to make primitive props required.
If you would like the default value to be something specific you can use the defaultValue
annotation parameter for both Object and primitive prop types.
@ModelProp(defaultValue = "MY_DEFAULT")
public void setImage(MyObject param)
The string "MY_DEFAULT"
represents a constant in the class that is named MY_DEFAULT
. In this case Epoxy would expect that the view class define a field like static final MyObject MY_DEFAULT = foo;
The constant must be static, final, and not private so Epoxy can reference it. The type of the constant must also match the type of the Prop.
The name of the constant must be used instead of referencing the constant directly since objects are not valid annotation parameters.
If the prop is an object and has both an @Nullable
annotation and an explicit defaultValue
set then the behavior is as follows:
- If no value for the prop is set on the model then the custom
defaultValue
will be applied - If
null
is set for the prop value then null will be set on the view.
In the future we may support defining default values via method references so defaults can be dynamic and overridden by subclasses. Open an issue if this would be helpful for you.
Sometimes you may want multiple setters to represent the same prop. This is common for overloaded setters, for example, you may allow setting an image via a url or via a drawable resource. Normally the generated model always sets every prop when the view is bound, but in this case setting both the url and drawable resource would lead to one of the values being overridden.
The solution is to tell Epoxy that the props are in the same "group". Only one prop per group will ever be set on the view. If the consumer of the model sets values for multiple props in the same group, only the last prop that is set is used.
Setters with the same method name are automatically placed in the same group. For example, the following method signatures would result in both props being placed in the same group.
@ModelProp
public void setImage(String url)
@ModelProp
public void setImage(@DrawableRes int drawableRes)
You can manually specify groups with the group
parameter:
@ModelProp(group = "image")
public void setImageUrl(String url)
@ModelProp(group = "image")
public void setImageDrawable(@DrawableRes int drawableRes)
Since the drawableRes is a primitive, it has a default value of 0. Therefore, if no image is specified on the model, the view will have a default value set with setImageDrawable(0)
. However, you can specify an explicit default for the group with the standard default parameter:
@ModelProp(group = "image")
public void setImageUrl(String url)
@ModelProp(group = "image", defaultValue = "DEFAULT_DRAWABLE_RES")
public void setImageDrawable(@DrawableRes int drawableRes)
Only one prop per group can specify an explicit default value, otherwise an exception will be thrown at compile time.
If there are multiple primitive props in a group, and no explicit default value is set, one of the primitives will be set with its Java default value when the view is bound. (It is undefined which prop is chosen).
If one of the props in the group has a @Nullable
annotation on its parameter, (and no explicit default is set), then that prop will be set to null as the group's default value.
If there are no primitive props in the group, and no props are @Nullable
, then a value is required to be set for the group.
If there are primitive props in the group then it isn't possible to require a value to be set for the group, since primitives don't (yet) support required values.
Every prop must have a type that implements equals
and hashCode
. This is necessary so that Epoxy can determine when the model's state has changed so it knows when to update the view. If a type does not provide an equals
and hashCode
implementation other than the default implementation on the Object
class then an error will be thrown at compile time.
However, some common types, such as interfaces, do not implement equals
and hashCode
. A common example is View.OnClickListener
. In the click listener case it is unlikely that the listener needs to contribute to model state; When models are rebuilt, a new anonymous class is usually created for the listener but it would normally have the same implementation. This is a very common pattern, and generally we only need to update the view if the listener goes from not being set to being set (and vice versa).
To accomplish this you can use the DoNotHash
option.
@ModelProp(options = Option.DoNotHash)
public void setClickListener(@Nullable View.OnClickListener listener) {
}
Epoxy will not use the equals and hashCode method on the listener object to calculate the model's state. However, the presence of the listener will change the model state. That is, if the listener goes from null to non null Epoxy will update the view with the new listener.
A good way to think about this is: "If my view is on screen and the hashCode of my prop changes, do I want the view to be updated accordingly?" If so, make sure the object's type implements equals and hashCode correctly, otherwise you can use DoNotHash
Sometimes your prop must be a type of an interface or abstract class that doesn't yet implement equals or hashCode. However, you may know that at runtime the implementation of that type will implement equals and hashCode. In this case you still want Epoxy to use equals and hashCode to allow the prop to contribute to model state, but since the compile time type of the prop does not implement them Epoxy will throw an error.
To get around this you can use the option IgnoreRequireHashCode
to indicate to Epoxy that you know the equals and hashCode implementations are missing, but to ignore that and use them anyway.
@ModelProp(options = Option.IgnoreRequireHashCode)
public void setObject(MyInterface myInterface) {
...
}
An example case is generating objects with @AutoValue
, where only the generated class implements equals and hashCode. (Since AutoValue is a common case, if Epoxy sees a type annotated with it it will actually allow the type to not have equals and hashCode.)
Alternatively, instead of using IgnoreRequireHashCode
, you can simply add abstract stubs of equals
and hashCode
to your interface or abstract class to indicate to Epoxy that the runtime implementation of the prop type will correctly implement the methods.
When a view is scrolled off screen it is unbound from its model and recycled by RecyclerView. If you would like to clean up resources at this time you can annotate one or more methods in the view with @OnViewRecycled
.
@OnViewRecycled
public void clear(){
...
}
Any methods on the view with this annotations will be called when the view is unbound and recycled. This is a good time to release listeners (to prevent memory leaks) or large objects like bitmaps to free up memory. You could also use it to cancel running operations.
Methods with this annotation must not be private or static, and cannot have any parameters.
If a prop is annotated with @Nullable
then the NullOnRecycle
option can be set on the prop's annotation to tell Epoxy to call the setter with a null value when the view is recycled.
@ModelProp(options = Option.NullOnRecycle)
public void setTitle(@Nullable CharSequence title) {
...
}
This is a shortcut to manually doing this with OnViewRecycled
like:
@OnViewRecycled
public void clear(){
setTitle(null);
}
Both ways are equivalent, but the NullOnRecycle
option is a nice shortcut for nullable props that you would like cleared. This can be helpful for objects like listeners or images.
If your generated model should save view state (explained here) then you can set the saveViewState
parameter to true in the ModelView
annotation.
@ModelView(defaultLayout = R.layout.my_layout, saveViewState = true)
public class MyView extends View {
...
}
The generated model will then include this override
@Override
public boolean shouldSaveViewState() {
return true;
}
Otherwise shouldSaveViewState
will default to false.
By default, generated models will take up the full span count when used with a GridLayoutManager (see Grid Support for more details). If you would like them to instead take up a single span you can set the fullSpan
parameter to false in the ModelView
annotation.
@ModelView(defaultLayout = R.layout.my_layout, fullSpan = false)
public class MyView extends View {
...
}
If you would like more fine grain control over the model's span size you can set a SpanSizeOverrideCallback
on the model when you use it in your EpoxyController.
model.spanSizeOverride(new SpanSizeOverrideCallback() {
@Override
public int getSpanSize(int totalSpanCount, int position, int itemCount) {
return totalSpanCount / 2;
}
});