J'ai créé mon composant personnalisé simplement en assemblant quelques TextViews. Maintenant, je veux pouvoir lancer mon contrôle personnalisé directement à partir du code, en passant des tailles de texte indépendamment pour chacun des téléviseurs
Ma définition des attributs:
<resources>
<declare-styleable name="BasicGauge">
<attr name="valueTextSize" format="dimension" />
<attr name="titleTextSize" format="dimension" />
<attr name="unitsTextSize" format="dimension" />
</declare-styleable>
</resources>
Exemple d'initialisation du composant:
<pl.com.digita.BikeComputerUi.customviews.BasicGauge
Android:id="@+id/basicGauge1"
Android:layout_width="0dp"
Android:layout_height="wrap_content"
Android:layout_weight="1"
Android:padding="10dp"
valueTextSize="40sp">
</pl.com.digita.BikeComputerUi.customviews.BasicGauge>
Comment j'essaie de lire ces attributs dans le constructeur du composant:
final int N = typedArray.getIndexCount();
for (int i = 0; i < N; i++) {
int attribute = typedArray.getIndex(i);
switch (attribute) {
case R.styleable.BasicGauge_valueTextSize:
valueTextSize = typedArray.getString(attribute);
break;
case R.styleable.BasicGauge_titleTextSize:
titleTextSize = typedArray.getString(attribute);
break;
case R.styleable.BasicGauge_unitsTextSize:
unitsTextSize = typedArray.getString(attribute);
break;
}
typedArray.recycle();
}
Problème: Après la création, toutes mes valeurs sont toujours nulles. 40sp est exactement ma valeur souhaitée.
Quelques mots à dire:
Après cela, ce n'est pas une très bonne idée d'obtenir une chaîne, Android offre une solution plus puissante pour gérer les dimensions:
int unitsTextSize = typedArray.getDimensionPixelSize(R.styleable.BasicGauge_unitsTextSize, textSize);
puis il y a quelques subtilités:
Paint.setTextSize(unitTextSize):
textView.setTextSize(TypedValue.COMPLEX_UNIT_PX, unitTextSize);
La différence vient de ce qu'on appelle des "pixels bruts" (non mis à l'échelle en fonction de la densité, juste bruts) et des "pixels mis à l'échelle" (l'inverse).
Pour un peu plus de contexte à la réponse acceptée:
attrs.xml
Définissez l'attribut personnalisé au format dimension
.
<resources>
<declare-styleable name="CustomView">
<attr name="valueTextSize" format="dimension" />
<attr name="titleTextSize" format="dimension" />
<attr name="unitsTextSize" format="dimension" />
</declare-styleable>
</resources>
activity.xml
Référencez vos attributs avec xmlns:app=...
et définissez-les dans votre xml avec app:...
.
<RelativeLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
xmlns:app="http://schemas.Android.com/apk/res-auto"
Android:layout_width="match_parent"
Android:layout_height="match_parent">
<com.example.myproject.CustomView
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
app:valueTextSize="40sp"
app:titleTextSize="20sp"
app:unitsTextSize="24sp" />
</RelativeLayout>
CustomView.Java
Obtenez les valeurs du TypedArray à l'aide de getDimensionPixelSize
. Dans votre vue personnalisée, travaillez simplement avec des pixels. Voir cette réponse si vous devez vous convertir à une dimension différente.
public class CustomView extends View {
private float mValueTextSize; // pixels
private float mTitleTextSize; // pixels
private float mUnitsTextSize; // pixels
public CustomView(Context context, AttributeSet attrs) {
super(context, attrs);
TypedArray a = context.getTheme().obtainStyledAttributes(
attrs, R.styleable.CustomView, 0, 0);
try {
mValueTextSize = a.getDimensionPixelSize(R.styleable.CustomView_valueTextSize, 0);
mTitleTextSize = a.getDimensionPixelSize(R.styleable.CustomView_titleTextSize, 0);
mUnitsTextSize = a.getDimensionPixelSize(R.styleable.CustomView_unitsTextSize, 0);
} finally {
a.recycle();
}
}
// ...
}