There is no one line solution to this problem, but this worked for my use case. The problem is, the 'View(context, attrs, defStyle)' constructor does not refer to an actual style, it wants an attribute. So, we will:
- Define an attribute
- Create a style that you want to use
- Apply a style for that attribute on our theme
- Create new instances of our view with that attribute
In 'res/values/attrs.xml', define a new attribute:
<?xml version="1.0" encoding="utf-8"?>
<resources>
<attr name="customTextViewStyle" format="reference"/>
...
</resources>
In res/values/styles.xml' I'm going to create the style I want to use on my custom TextView
<style name="CustomTextView">
<item name="android:textSize">18sp</item>
<item name="android:textColor">@color/white</item>
<item name="android:paddingLeft">14dp</item>
</style>
In 'res/values/themes.xml' or 'res/values/styles.xml', modify the theme for your application / activity and add the following style:
<resources>
<style name="AppBaseTheme" parent="android:Theme.Light">
<item name="customTextViewStyle">@style/CustomTextView</item>
</style>
...
</resources>
Finally, in your custom TextView, you can now use the constructor with the attribute and it will receive your style. Here, instead of always
public class CustomTextView extends TextView {
public CustomTextView(Context context, int styleAttribute) {
super(context, null, styleAttribute);
}
// You could also just apply your default style if none is given
public CustomTextView(Context context) {
super(context, null, R.attr.customTextViewStyle);
}
}
With all of these components, you can now do an if/else statement to generate new views at runtime with the style you prefer
CustomTextView ctv;
if(useCustomStyles == true){
ctv = new CustomTextView(context, R.attr.customTextViewStyle);
}else{
ctv = new CustomTextView(context, R.attr.someOtherStyle);
}
It's worth noting that I repeatedly used customTextView in different variants and different places, but it is in no way required that the name of the view match the style or the attribute or anything. Also, this technique should work with any custom view, not just TextViews.