如何定制snackBar的布局?

有没有任何方法来改变一个snackBar的自定义视图的布局?

现在它变黑了,我们可以改变背景颜色。 但我不知道正确的方式来夸大一个新的布局,并使其作为snackBars背景?

谢谢…

小吃店不允许您设置自定义布局。 然而,正如Primoz990build议你可以得到Snackbar的视图。 getView函数返回Snackbar.SnackbarLayout,它是一个水平的LinearLayout对象,它的子对象是一个TextView和一个Button。 要将自己的视图添加到Snackbar,只需要隐藏TextView,然后将视图添加到Snackbar.SnackbarLayout。

// Create the Snackbar Snackbar snackbar = Snackbar.make(containerLayout, "", Snackbar.LENGTH_LONG); // Get the Snackbar's layout view Snackbar.SnackbarLayout layout = (Snackbar.SnackbarLayout) snackbar.getView(); // Hide the text TextView textView = (TextView) layout.findViewById(android.support.design.R.id.snackbar_text); textView.setVisibility(View.INVISIBLE); // Inflate our custom view View snackView = mInflater.inflate(R.layout.my_snackbar, null); // Configure the view ImageView imageView = (ImageView) snackView.findViewById(R.id.image); imageView.setImageBitmap(image); TextView textViewTop = (TextView) snackView.findViewById(R.id.text); textViewTop.setText(text); textViewTop.setTextColor(Color.WHITE); // Add the view to the Snackbar's layout layout.addView(snackView, 0); // Show the Snackbar snackbar.show(); 

可以从Android支持库的25.1.0修订版开始

I.在您的值/布局文件夹中声明自定义布局。

 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content"> <Button android:id="@+id/snackbar_action" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginLeft="@dimen/design_snackbar_extra_spacing_horizontal" android:layout_marginStart="@dimen/design_snackbar_extra_spacing_horizontal" android:layout_gravity="center_vertical|right|end" android:paddingTop="@dimen/design_snackbar_padding_vertical" android:paddingBottom="@dimen/design_snackbar_padding_vertical" android:paddingLeft="@dimen/design_snackbar_padding_horizontal" android:paddingRight="@dimen/design_snackbar_padding_horizontal" android:visibility="gone" android:textColor="?attr/colorAccent" style="?attr/borderlessButtonStyle"/> <TextView android:gravity="center_vertical|right" android:id="@+id/snackbar_text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_weight="1" android:paddingTop="@dimen/design_snackbar_padding_vertical" android:paddingBottom="@dimen/design_snackbar_padding_vertical" android:paddingLeft="@dimen/design_snackbar_padding_horizontal" android:paddingRight="@dimen/design_snackbar_padding_horizontal" android:textAppearance="@style/TextAppearance.Design.Snackbar.Message" android:maxLines="@integer/design_snackbar_text_max_lines" android:layout_gravity="center_vertical|left|start" android:ellipsize="end"/> </LinearLayout> 

提示:

  • 使用@dimen/design_snackbar值来匹配材料devise准则。
  • 使用?attr/colorAccent将您的应用程序主题更改应用到Snackbar。

II。 扩展BaseTransientBottomBar类。

 public class final CustomSnackbar extends BaseTransientBottomBar<CustomSnackbar> { /** * Constructor for the transient bottom bar. * * @param parent The parent for this transient bottom bar. * @param content The content view for this transient bottom bar. * @param contentViewCallback The content view callback for this transient bottom bar. */ private CustomSnackbar(ViewGroup parent, View content, ContentViewCallback contentViewCallback) { super(parent, content, contentViewCallback); } } 

III。 添加BaseTransientBottomBar.ContentViewCallback

 public class final CustomSnackbar ...{ ... private static class ContentViewCallback implements BaseTransientBottomBar.ContentViewCallback { // view inflated from custom layout private View content; public ContentViewCallback(View content) { this.content = content; } @Override public void animateContentIn(int delay, int duration) { // add custom *in animations for your views // eg original snackbar uses alpha animation, from 0 to 1 ViewCompat.setScaleY(content, 0f); ViewCompat.animate(content) .scaleY(1f).setDuration(duration) .setStartDelay(delay); } @Override public void animateContentOut(int delay, int duration) { // add custom *out animations for your views // eg original snackbar uses alpha animation, from 1 to 0 ViewCompat.setScaleY(content, 1f); ViewCompat.animate(content) .scaleY(0f) .setDuration(duration) .setStartDelay(delay); } } } 

IV。 使用自定义布局和方法添加方法来创buildSnackbar来填充它。

 public class final CustomSnackbar ...{ ... public static CustomSnackbar make(ViewGroup parent, @Duration int duration) { // inflate custom layout LayoutInflater inflater = LayoutInflater.from(parent.getContext()); View content = inflater.inflate(R.layout.snackbar_view, parent, false); // create snackbar with custom view ContentViewCallback callback= new ContentViewCallback(content); CustomSnackbar customSnackbar = new CustomSnackbar(parent, content, callback); // set snackbar duration customSnackbar.setDuration(duration); return customSnackbar; } // set text in custom layout public CustomSnackbar setText(CharSequence text) { TextView textView = (TextView) getView().findViewById(R.id.snackbar_text); textView.setText(text); return this; } // set action in custom layout public CustomSnackbar setAction(CharSequence text, final OnClickListener listener) { Button actionView = (Button) getView().findViewById(R.id.snackbar_action); actionView.setText(text); actionView.setVisibility(View.VISIBLE); actionView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { listener.onClick(view); // Now dismiss the Snackbar dismiss(); } }); return this; } } 

V.创buildCustomSnackbar实例并调用show()方法。

 CustomSnackbar customSnackbar = CustomSnackbar.make(rooView, CustomSnackbar.LENGTH_INDEFINITE); customSnackbar.setText("No network connection!"); customSnackbar.setAction("Retry", new View.OnClickListener() { @Override public void onClick(View v) { // handle click here } }); customSnackbar.show(); 

在materialdoc.com上查看更多关于Snackbar及其定制的信息

完整的CustomSnackbar.class代码:

 import android.support.annotation.NonNull; import android.support.design.widget.BaseTransientBottomBar; import android.support.v4.view.ViewCompat; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; import android.widget.TextView; public class CustomSnackbar extends BaseTransientBottomBar<CustomSnackbar> { /** * Constructor for the transient bottom bar. * * @param parent The parent for this transient bottom bar. * @param content The content view for this transient bottom bar. * @param callback The content view callback for this transient bottom bar. */ private CustomSnackbar(ViewGroup parent, View content, ContentViewCallback callback) { super(parent, content, callback); } public static CustomSnackbar make(@NonNull ViewGroup parent, @Duration int duration) { final LayoutInflater inflater = LayoutInflater.from(parent.getContext()); final View content = inflater.inflate(R.layout.snackbar_view, parent, false); final ContentViewCallback viewCallback = new ContentViewCallback(content); final CustomSnackbar customSnackbar = new CustomSnackbar(parent, content, viewCallback); customSnackbar.setDuration(duration); return customSnackbar; } public CustomSnackbar setText(CharSequence text) { TextView textView = (TextView) getView().findViewById(R.id.snackbar_text); textView.setText(text); return this; } public CustomSnackbar setAction(CharSequence text, final View.OnClickListener listener) { Button actionView = (Button) getView().findViewById(R.id.snackbar_action); actionView.setText(text); actionView.setVisibility(View.VISIBLE); actionView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { listener.onClick(view); // Now dismiss the Snackbar dismiss(); } }); return this; } private static class ContentViewCallback implements BaseTransientBottomBar.ContentViewCallback { private View content; public ContentViewCallback(View content) { this.content = content; } @Override public void animateContentIn(int delay, int duration) { ViewCompat.setScaleY(content, 0f); ViewCompat.animate(content).scaleY(1f).setDuration(duration).setStartDelay(delay); } @Override public void animateContentOut(int delay, int duration) { ViewCompat.setScaleY(content, 1f); ViewCompat.animate(content).scaleY(0f).setDuration(duration).setStartDelay(delay); } } } 

XML方式:

用于Snackbar的原始布局xml文件是这个文件:

design_layout_snackbar_include.xml

  <?xml version="1.0" encoding="utf-8"?> <merge xmlns:android="http://schemas.android.com/apk/res/android"> <TextView android:id="@+id/snackbar_text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_weight="1" ... android:ellipsize="end"/> <Button android:id="@+id/snackbar_action" android:layout_width="wrap_content" android:layout_height="wrap_content" android:visibility="gone" ... android:textColor="?attr/colorAccent" style="?attr/borderlessButtonStyle"/> </merge> 

所以为了覆盖这个布局,你应该使用和这个一样的android:id来编写你自己的布局,在你的refs.xml文件中你应该添加下面的代码:

 <resources xmlns:tools="http://schemas.android.com/tools"> .... <item name="design_layout_snackbar_include" tools:override="true" type="layout"> @layout/my_layout_snackbar </item> .... </resources> 

答案是:不要自定义小吃店。 它不应该包含比短文本和一个动作更多的元素。 请参阅Google材料devise指南 。

更新:如果你反正想要自定义Snackbar这里是我在我的应用程序中实现的:

 //generate the snackbar Snackbar sb = Snackbar.make(rootView, snack.text, duration); //set te action button text color sb.setActionTextColor(mCurrentActivity.getResources().getColor(R.color.snack_text_action)); //Get the view of the snackbar View sbView = sb.getView(); //set background color sbView.setBackgroundColor(mCurrentActivity.getResources().getColor(backgroudResId)); //Get the textview of the snackbar text TextView textView = (TextView) sbView.findViewById(android.support.design.R.id.snackbar_text); //set text color textView.setTextColor(mCurrentActivity.getResources().getColor(R.color.snack_text)); //increase max lines of text in snackbar. default is 2. textView.setMaxLines(10); 

我从来没有尝试过,但获取Snackbar的根视图,你可以编程添加新的视图到Snackbar。

尝试下面的代码。

 Snackbar snackbar = Snackbar.make(container, "No Internet Connection", Snackbar.LENGTH_LONG); View sbView = snackbar.getView(); sbView.setBackgroundColor(ContextCompat.getColor(this, R.color.colorPrimary)); snackbar.show(); 

注意:

容器 – 布局的父视图。

你可以试试这个库。 这是一个Android默认小吃包的包装。 https://github.com/ChathuraHettiarachchi/CSnackBar

 Snackbar.with(this,null) .type(Type.SUCCESS) .message("Profile updated successfully!") .duration(Duration.SHORT) .show(); 

或者你甚至可以使用你自己的观点,

 View view = getLayoutInflater().inflate(R.layout.custom_view, null); Snackbar.with(this,null) .type(Type.UPDATE) .contentView(view, 76) .duration(Duration.SHORT) .show(); 

目前只有自定义布局的问题是,我们需要在dp中传递视图高度作为input

 private Snackbar showSnackbar() { // Create the Snackbar Snackbar snackbar = Snackbar.make(coordinatorLayout, "", length); // 15 is margin from all the sides for snackbar int marginFromSides = 15; float hei[![enter image description here][1]][1]ght = 100; //inflate view View snackView = getLayoutInflater().inflate(R.layout.snackbar_layout, null); // White background snackbar.getView().setBackgroundColor(Color.WHITE); // for rounded edges snackbar.getView().setBackground(getResources().getDrawable(R.drawable.round_edges)); Snackbar.SnackbarLayout snackBarView = (Snackbar.SnackbarLayout) snackbar.getView(); FrameLayout.LayoutParams parentParams = (FrameLayout.LayoutParams) snackBarView.getLayoutParams(); parentParams.setMargins(marginFromSides, 0, marginFromSides, marginFromSides); parentParams.height = (int) height; parentParams.width = FrameLayout.LayoutParams.MATCH_PARENT; snackBarView.setLayoutParams(parentParams); snackBarView.addView(snackView, 0); return snackbar; } final Snackbar snackbar = showSnackbar(mainLayout, 90, Snackbar.LENGTH_LONG, R.layout.snackbar_layout); snackbar.show(); View view = snackbar.getView(); TextView tv = (TextView) view.findViewById(R.id.snackbar_action); tv.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { snackbar.dismiss(); } }); 

要添加到Yakiv Mospan的答案,要使您的自定义BaseTransientBottomBar从底部显示像一个Snackbar ,从Snackbar类复制此方法find一个合适的父BaseTransientBottomBar构造函数。

 private static ViewGroup findSuitableParent(View view) { ViewGroup fallback = null; do { if (view instanceof CoordinatorLayout) { // We've found a CoordinatorLayout, use it return (ViewGroup) view; } else if (view instanceof FrameLayout) { if (view.getId() == android.R.id.content) { // If we've hit the decor content view, then we didn't find a CoL in the // hierarchy, so use it. return (ViewGroup) view; } else { // It's not the content view but we'll use it as our fallback fallback = (ViewGroup) view; } } if (view != null) { // Else, we will loop and crawl up the view hierarchy and try to find a parent final ViewParent parent = view.getParent(); view = parent instanceof View ? (View) parent : null; } } while (view != null); // If we reach here then we didn't find a CoL or a suitable content view so we'll fallback return fallback; } 
Interesting Posts