Android:以编程方式设置视图样式

这是XML:

<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
style="@style/LightStyle"
android:layout_width="fill_parent"
android:layout_height="55dip"
android:clickable="true"
android:orientation="horizontal" >


<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal" />


</RelativeLayout>

如何以编程方式设置style属性?

341242 次浏览

更新:在回答这个问题的时候(2012年年中,API级别14-15),以编程方式设置视图还不是一个选项(即使有一些重要的变通方法),而在最近的API发布之后,这已经成为可能。详见@Blundell的回答。

旧的回答:

不能还可以通过编程方式设置视图的样式,但你可能会发现这个线程很有用。

您可以创建包含所需样式布局的xml,然后更改视图的背景资源,如

从技术上讲,你可以用自定义视图以编程方式应用样式:

private MyRelativeLayout extends RelativeLayout {
public MyRelativeLayout(Context context) {
super(context, null, R.style.LightStyle);
}
}

一个参数构造函数是在以编程方式实例化视图时使用的构造函数。

将这个构造函数链接到带有样式形参的super。

RelativeLayout someLayout = new MyRelativeLayout(new ContextThemeWrapper(this,R.style.RadioButton));

或者就像@Dori简单指出的那样:

RelativeLayout someLayout = new RelativeLayout(new ContextThemeWrapper(activity,R.style.LightStyle));

现在在Kotlin:

class MyRelativeLayout @JvmOverloads constructor(
context: Context,
attributeSet: AttributeSet? = null,
defStyleAttr: Int = R.style.LightStyle,
) : RelativeLayout(context, attributeSet, defStyleAttr)

 val rl = RelativeLayout(ContextThemeWrapper(activity, R.style.LightStyle))

我在我的复合ViewGroup中使用XML定义的视图,将它们添加到ViewGroup中。这样我就不能动态地更改样式,但是我可以进行一些样式自定义。我的组合:

public class CalendarView extends LinearLayout {


private GridView mCalendarGrid;
private LinearLayout mActiveCalendars;


private CalendarAdapter calendarAdapter;


public CalendarView(Context context) {
super(context);


}


public CalendarView(Context context, AttributeSet attrs) {
super(context, attrs);


}


@Override
protected void onFinishInflate() {
super.onFinishInflate();
init();
}


private void init() {
mCalendarGrid = (GridView) findViewById(R.id.calendarContents);
mCalendarGrid.setNumColumns(CalendarAdapter.NUM_COLS);


calendarAdapter = new CalendarAdapter(getContext());
mCalendarGrid.setAdapter(calendarAdapter);
mActiveCalendars = (LinearLayout) findViewById(R.id.calendarFooter);
}

在XML视图中,我可以指定样式:

<com.mfitbs.android.calendar.CalendarView
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/calendar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:orientation="vertical"
>


<GridView
android:id="@+id/calendarContents"
android:layout_width="match_parent"
android:layout_height="wrap_content" />


<LinearLayout
android:id="@+id/calendarFooter"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
/>

你可以通过以下方式将样式应用到你的活动:

super.setTheme( R.style.MyAppTheme );

或Android默认设置:

super.setTheme( android.R.style.Theme );

在你的活动setContentView()之前。

对我有用的是:

Button b = new Button(new ContextThemeWrapper(this, R.style.ButtonText), null, 0);
  • 使用ContextThemeWrapper

  • 使用3个参数的构造函数(没有这个就不行)

提供的答案没有一个是正确的。

您可以通过编程方式设置样式。

简单的回答是看一下http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/5.1.1_r1/android/content/Context.java#435

< p >长答案。 这是我的片段,以编程方式设置自定义样式到您的视图

1)在styles.xml文件中创建一个样式

 <style name="MyStyle">
<item name="customTextColor">#39445B</item>
<item name="customDividerColor">#8D5AA8</item>
</style>

不要忘记在attrs.xml文件中定义您的自定义属性

我的attrsl.xml文件:

<declare-styleable name="CustomWidget">
<attr name="customTextColor" format="color" />
<attr name="customDividerColor" format="color" />
</declare-styleable>

注意,您可以为您的可样式(我的CustomWidget)使用任何名称

现在让我们以编程方式设置小部件的样式 这是我的简单小部件:

public class StyleableWidget extends LinearLayout {


private final StyleLoader styleLoader = new StyleLoader();


private TextView textView;
private View divider;


public StyleableWidget(Context context) {
super(context);
init();
}


private void init() {
inflate(getContext(), R.layout.widget_styleable, this);
textView = (TextView) findViewById(R.id.text_view);
divider = findViewById(R.id.divider);
setOrientation(VERTICAL);
}


protected void apply(StyleLoader.StyleAttrs styleAttrs) {
textView.setTextColor(styleAttrs.textColor);
divider.setBackgroundColor(styleAttrs.dividerColor);
}


public void setStyle(@StyleRes int style) {
apply(styleLoader.load(getContext(), style));
}
}
< p >布局:

<TextView
android:id="@+id/text_view"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="22sp"
android:layout_gravity="center"
android:text="@string/styleble_title" />


<View
android:id="@+id/divider"
android:layout_width="match_parent"
android:layout_height="1dp"/>


</merge>

最后是StyleLoader类实现

public class StyleLoader {


public StyleLoader() {


}


public static class StyleAttrs {
public int textColor;
public int dividerColor;
}


public StyleAttrs load(Context context, @StyleRes int styleResId) {
final TypedArray styledAttributes = context.obtainStyledAttributes(styleResId, R.styleable.CustomWidget);
return load(styledAttributes);
}


@NonNull
private StyleAttrs load(TypedArray styledAttributes) {
StyleAttrs styleAttrs = new StyleAttrs();
try {
styleAttrs.textColor = styledAttributes.getColor(R.styleable.CustomWidget_customTextColor, 0);
styleAttrs.dividerColor = styledAttributes.getColor(R.styleable.CustomWidget_customDividerColor, 0);
} finally {
styledAttributes.recycle();
}
return styleAttrs;
}
}

你可以在https://github.com/Defuera/SetStylableProgramatically找到完整的工作示例

如果您想继续使用XML(公认的答案不允许您这样做)并在创建视图后设置样式,那么您可以使用Paris库,它支持所有可用属性的一个子集。

因为你正在从XML扩展视图,你需要在布局中指定一个id:

<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/my_styleable_relative_layout"
style="@style/LightStyle"
...

然后,当你需要以编程方式改变样式时,在布局被膨胀后:

// Any way to get the view instance will do
RelativeLayout myView = findViewById(R.id.my_styleable_relative_layout);


// This will apply all the supported attribute values of the style
Paris.style(myView).apply(R.style.LightStyle);

更多信息:支持的视图类型和属性列表(包括背景,填充,空白等,可以很容易地扩展)和安装说明和附加文档

声明:我是该库的原始作者。

这是我的简单示例,关键是ContextThemeWrapper包装器,没有它,我的样式将无法工作,并使用视图的三个参数构造函数。

ContextThemeWrapper themeContext = new ContextThemeWrapper(this, R.style.DefaultLabelStyle);
TextView tv = new TextView(themeContext, null, 0);
tv.setText("blah blah ...");
layout.addView(tv);

对于一个新的按钮/TextView:

Button mMyButton = new Button(new ContextThemeWrapper(this, R.style.button_disabled), null, 0);

对于已存在的实例:

mMyButton.setTextAppearance(this, R.style.button_enabled);

对于图像或布局:

Image mMyImage = new ImageView(new ContextThemeWrapper(context, R.style.article_image), null, 0);

我不建议使用ContextThemeWrapper,因为它这样做:

指定的主题将被应用于

什么会在应用程序中产生不必要的结果。相反,我建议Airbnb的工程师们建造一个名为“paris”的图书馆:

https://github.com/airbnb/paris

以编程方式定义并应用样式到Android视图。

但在使用了一段时间后,我发现它实际上是相当有限的,我停止使用它,因为它不支持很多属性,我需要开箱即用,所以一个人必须检查并决定一如既往。

简单的方法是通过构造函数

RadioButton radioButton = new RadioButton(this,null,R.style.radiobutton_material_quiz);

这是一个相当老的问题,但解决方案,对我来说,现在是使用构造函数defStyleRes的第四个参数-如果可用..视图…设置风格

以下工作为我的目的(kotlin):

val textView = TextView(context, null, 0, R.style.Headline1)

如果在自定义视图中: val editText = TextInputEditText(context, attrs, defStyleAttr)

. val editText = TextInputEditText(context, attrs, defStyleAttr

我发现最好的简单解决方案,使用alertDialog自定义布局,是:

val mView = LayoutInflater.from(context).inflate(layoutResId, null)


val dialog = AlertDialog.Builder(context, R.style.CustomAlertDialog)
.setView(mView)
.setCancelable(false)
.create()

风格在哪里

<style name="CustomAlertDialog" parent="Theme.AppCompat.Light.Dialog.Alert">
<item name="android:background">@drawable/bg_dialog_white_rounded</item>
</style>

bg_dialog_white_rounded.xml是

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<corners android:radius="16dp" />


<solid android:color="@Color/white" />
</shape>

layoutResId是任何必须将主题设置为“@style/CustomAlertDialog"”的布局的资源id,例如:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/root"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginStart="@dimen/wdd_margin_medium"
android:theme="@style/CustomAlertDialog"
android:layout_marginEnd="@dimen/wdd_margin_medium">


..... etc...
</androidx.constraintlayout.widget.ConstraintLayout>
int buttonStyle = R.style.your_button_style;
Button button = new Button(new ContextThemeWrapper(context, buttonStyle), null, buttonStyle);

只有这个答案对我有用。看到https://stackoverflow.com/a/24438579/5093308