所谓赠人玫瑰,手留余香!非常感谢无私奉献的前辈们。之前在练习安卓的底部状态栏的时候,看到前辈的一个帖子很好的知道了我的实践。 地址在:这里
但是后面我觉得这样用起来不是很舒适,因为底部数量是固定的。 能不能根据后台的数据来动态的设置呢? 于是开始实践,最终的效果图是:
一个按钮的效果:
二个按钮的效果:
三个按钮的效果:
五个按钮的效果:
五个以上按钮的效果:
容器布局文件:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clipChildren="false">
<LinearLayout
android:id="@+id/main_container"
android:layout_width="match_parent"
android:layout_height="500dp"
android:orientation="vertical"></LinearLayout>
<HorizontalScrollView
android:layout_width="match_parent"
android:layout_height="60dp"
android:layout_alignParentBottom="true"
android:layout_alignParentStart="true"
android:clipChildren="false"
android:orientation="horizontal">
<com.automannn.meimeijiong.activity.view.api.baseApi.LinearLayoutListView
android:id="@+id/main_bottom_container"
android:layout_width="match_parent"
android:layout_height="60dp"
android:clipChildren="false"
android:layout_gravity="bottom"
android:gravity="bottom"
android:orientation="horizontal" />
</HorizontalScrollView>
</RelativeLayout>
子视图布局文件:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clipChildren="false">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="bottom"
android:layout_weight="1">
<ImageView
android:id="@+id/icon"
android:layout_width="40dp"
android:layout_height="40dp"
android:layout_above="@id/icon_text"
android:layout_alignParentTop="true"
android:layout_centerInParent="true"
android:src="@drawable/xiaoxi" />
<TextView
android:id="@+id/icon_text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:layout_alignParentBottom="true"
android:padding="2dp"
android:textColor="#000"
android:text="消息"
android:textSize="10sp" />
</RelativeLayout>
</LinearLayout>
基本思路是,页面整体是一个绝对定位。 分为上下两个部分。 上部分呈现fragment内容。 底部呈现所有的菜单按钮。 需要注意的是,底部呈现的按钮被放置在一个水平滑动的ScrollView容器中。这样的话,就能满足动态菜单的要求。
由于底部菜单需要实现突出效果,因此要在突出的元素的父节点的父节点设置一个属性: clipChildren="false"。
这里有一个自定义的视图 LinearLayoutListView,我当时的想法是它要能满足类似于ListView的那种ItemClick的功能,因为这样的话,就能很好的满足我们针对动态菜单设置自定义的监听事件。
该自定义视图代码,以及它的适配器代码如下:
适配器代码:(继承Android的BaseAdapter的目的是为了与其它的接口统一标准。也可以不继承,copy这些个方法就可以了。)
package com.automannn.meimeijiong.adapter;
import android.util.DisplayMetrics;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.BaseAdapter;
import android.widget.LinearLayout;
import com.automannn.meimeijiong.app.BaseApplication;
import com.automannn.meimeijiong.activity.view.api.baseApi.LinearLayoutAdapterView;
import java.util.ArrayList;
import java.util.List;
public abstract class MyLinearLayoutAdapter<T> extends BaseAdapter {
//需要适配的数据集
private final List<T> dataList;
//适配的每一项的视图资源id
private final int viewframeRes;
//代表当前视图的父容器
private final ViewGroup viewGroup;
public MyLinearLayoutAdapter(List<T> datalist,int viewframeres,ViewGroup viewgroup) {
dataList = datalist;
viewframeRes =viewframeres;
viewGroup = viewgroup;
}
@Override
public int getCount() {
return dataList.size();
}
@Override
public Object getItem(int i) {
return dataList.get(i);
}
@Override
public long getItemId(int i) {
throw new RuntimeException("Stub!");
}
@Override
public View getView(int i, View view, ViewGroup viewGroup) {
throw new RuntimeException("Stub!");
}
//用于将实体对象与视图做绑定
public abstract ViewHolder bindView(T obj, View viewFrame);
//用于获得窗口的相关属性,动态计算布局
public abstract WindowManager getWindowManager();
@Override
public void notifyDataSetChanged() {
WindowManager windowManager = getWindowManager();
DisplayMetrics outMetrics = new DisplayMetrics();
windowManager.getDefaultDisplay().getMetrics(outMetrics);
int width = outMetrics.widthPixels;
if (getCount()==0)throw new IllegalArgumentException("未正确使用,请设置非空数据源");
//当数据源大于五个的时候,以五个菜单标准平分该屏幕
if (getCount()>=5){
width = width/5;
}else {
//当数据源小于5个的时候,以所有的菜单平分该屏幕
width=width/getCount();
}
for (T t:dataList){
//对于绝对布局充当跟布局的时候,所有的宽高属性会失效,需动态设置
View viewFrame=LayoutInflater.from(BaseApplication.getContext()).inflate(viewframeRes,null,false);
View view =bindView(t,viewFrame).getRootView();
LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(width, ViewGroup.LayoutParams.MATCH_PARENT);
lp.gravity= Gravity.BOTTOM;
view.setLayoutParams(lp);
viewGroup.addView(view);
}
viewGroup.requestLayout();
//强制父容器刷新
viewGroup.invalidate();
}
//该方法是关键,用于设置每个item的监听。
public void setOnItemClick(final LinearLayoutAdapterView.OnItemClickDo onItemClickDo){
final List sizeList =new ArrayList();
//本质是通过循环分别为每个子视图设置监听,然后通过抽象实现它的扩展。
for (int i=0;i<getCount();i++){
final int j = i;
View view = viewGroup.getChildAt(i);
view.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
onItemClickDo.run(view,j);
sizeList.add(false);
}
});
}
}
}
视图代码(第一层抽象层):
package com.automannn.meimeijiong.activity.view.api.baseApi;
import android.content.Context;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Adapter;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.RelativeLayout;
import android.widget.TextView;
import com.automannn.meimeijiong.R;
import com.automannn.meimeijiong.app.BaseApplication;
import com.automannn.meimeijiong.util.DbPxUtils;
//改类的本质任然是一个ViewGroup, 通过继承LinearLayout进行扩展
public abstract class LinearLayoutAdapterView<T extends Adapter> extends LinearLayout{
//与该视图匹配的视图适配器
T currAdapter;
public LinearLayoutAdapterView(Context context) {
super(context);
}
public LinearLayoutAdapterView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public LinearLayoutAdapterView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
//该抽象方法用于设置适配器
public abstract void setAdapter(T var1);
//该抽象方法用于设置每个子视图的点击事件监听,通过自定义的静态抽象内部类作为参数进行扩展
public abstract void setOnItemClickListener(OnItemClickDo onItemClickDo);
public abstract static class OnItemClickDo {
//该参数用于辅助记录上一次点击的按钮位置
private Integer currPosition;
//该抽象函数用于完成自定一的业务逻辑
public abstract void exec(View view,int position);
//该函数用于执行最终的操作,是一个模板方法,具有两个步骤,第一个是完成固定的视图刷新逻辑,第二个是完成扩展的业务逻辑控制
public void run(View view,int position){
changeView(view,position);
exec(view,position);
}
//完成视图切换的逻辑,是固定的,用户不可见
private void changeView(View view,int position){
//当上一次没有进行操作的时候
if (currPosition==null){
//找到要扩大的图片
ImageView imageView= view.findViewById(R.id.icon);
//一些缩放逻辑。。 由于基础不行,生产了很多的坑
view.setLayoutParams(new LinearLayout.LayoutParams(view.getLayoutParams().width,210 ));
RelativeLayout.LayoutParams lp = new RelativeLayout.LayoutParams(LayoutParams.MATCH_PARENT, 150);
imageView.setLayoutParams(lp);
TextView textView = view.findViewById(R.id.icon_text);
textView.setTextSize(15.0f);
currPosition=position;
}else {
//当上一次的点击与这一次的点击相同的时候
if (currPosition == position){
//do nothing
}else {
//先撤销该操作
ViewGroup viewGroup = (ViewGroup) view.getParent();
View preView= viewGroup.getChildAt(currPosition);
((TextView)preView.findViewById(R.id.icon_text)).setTextSize(11.0f);
RelativeLayout.LayoutParams lbb=new RelativeLayout.LayoutParams(LayoutParams.MATCH_PARENT, DbPxUtils.dip2px(BaseApplication.getContext(),40));
((ImageView) preView.findViewById(R.id.icon)).setLayoutParams(lbb);
LinearLayout.LayoutParams ly= (LayoutParams) view.getLayoutParams();
ly.gravity= Gravity.CENTER_VERTICAL;
preView.setLayoutParams(new LinearLayout.LayoutParams(ly));
//之后将新的换上
ImageView imageView= view.findViewById(R.id.icon);
view.setLayoutParams(new LinearLayout.LayoutParams(view.getLayoutParams().width,210 ));
RelativeLayout.LayoutParams lp = new RelativeLayout.LayoutParams(LayoutParams.MATCH_PARENT, 150);
imageView.setLayoutParams(lp);
TextView textView = view.findViewById(R.id.icon_text);
textView.setTextSize(15.0f);
currPosition=position;
}
}
}
}
}
视图代码(第二层实现层):
package com.automannn.meimeijiong.activity.view.api.baseApi;
import android.content.Context;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import com.automannn.meimeijiong.adapter.MyLinearLayoutAdapter;
public class LinearLayoutListView extends LinearLayoutAdapterView<MyLinearLayoutAdapter> {
public LinearLayoutListView(Context context) {
super(context);
}
public LinearLayoutListView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public LinearLayoutListView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
//设置适配器后,默认让其自动刷新。 注意这里的刷新逻辑也是完全自己定义的
//其大致的过程就是: linearLayout.addView(childrenView); 根据数据源中的大小自动适配
@Override
public void setAdapter(MyLinearLayoutAdapter var1) {
currAdapter = var1;
currAdapter.notifyDataSetChanged();
}
//该方法实际就是作为了视图与适配器的中转中心,视图时逻辑的入口,逻辑的实现位于适配器中
@Override
public void setOnItemClickListener(OnItemClickDo onItemClickDo) {
currAdapter.setOnItemClick(onItemClickDo);
}
}
万事具备,只欠东风。定义好自己的视图与适配器之后,就可以开手练习了。
用到的核心第三方依赖:
implementation 'com.alibaba:fastjson:1.2.49'
implementation 'com.github.bumptech.glide:glide:4.8.0'
implementation 'com.jakewharton:butterknife:8.8.1'
implementation 'com.jakewharton:butterknife-compiler:8.8.1'
implementation 'com.squareup.retrofit2:retrofit:2.0.2'
implementation 'com.squareup.retrofit2:converter-gson:2.0.2'
核心代码:(presenter中)
package automannn.com.testmvp.presenter;
import android.content.Intent;
import android.view.View;
import android.widget.Toast;
import com.alibaba.fastjson.JSONArray;
import java.util.List;
import automannn.com.testmvp.MainActivity;
import automannn.com.testmvp.R;
import automannn.com.testmvp.entity.MainBottomItem;
import automannn.com.testmvp.entity.UnicodeResponse;
import automannn.com.testmvp.model.MainBottomItemModel;
import automannn.com.testmvp.retrofitApi.MainBottomApi;
import automannn.com.testmvp.retrofitApi.RetrofitConfigUtil;
import automannn.com.testmvp.view.api.LinearLayoutAdapterView;
import automannn.com.testmvp.widget.FabuPopupWindow;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class MainActivityPresenter extends BasePresenter<MainBottomItemModel,MainActivity> {
public MainActivityPresenter(MainActivity view) {
super(view);
currModel = new MainBottomItemModel();
}
@Override
public void init() {
currModel.setRootView(currView.getLinearLayoutListComponent());
currModel.setWindowManager(currView.getWindowManager());
Retrofit retrofit = new Retrofit.Builder().baseUrl(RetrofitConfigUtil.TEST_SERVER_LOCATION).addConverterFactory(GsonConverterFactory.create()).build();
MainBottomApi mainBottomApi= retrofit.create(MainBottomApi.class);
Call<UnicodeResponse> unicodeResponseCall= mainBottomApi.chaxun(new MainBottomItem(),5,0);
unicodeResponseCall.enqueue(new Callback<UnicodeResponse>() {
@Override
public void onResponse(Call<UnicodeResponse> call, Response<UnicodeResponse> response) {
if (response.body().getState()>500){
}else {
List<MainBottomItem> list = new JSONArray((List<Object>) response.body().getData()).toJavaList(MainBottomItem.class);
currModel.setDataList(list);
currView.setAdapter(currModel.getAdapter());
currView.linearLayoutListView.setOnItemClickListener(new LinearLayoutAdapterView.OnItemClickDo() {
@Override
public void exec(View view,int position) {
if (position==2){
FabuPopupWindow fabuPopupWindow =new FabuPopupWindow(currView, new FabuPopupWindow.OnPopWindowClickListener() {
@Override
public void onPopWindowClick(View view) {
switch (view.getId()){
case R.id.point_end_youji_img:
Toast.makeText(currView,"点击了游记",Toast.LENGTH_SHORT).show();
break;
case R.id.point_end_gonglue_img:
Toast.makeText(currView,"点击了攻略",Toast.LENGTH_SHORT).show();
break;
}
}
});
fabuPopupWindow.show();
}
}
});
}
}
@Override
public void onFailure(Call<UnicodeResponse> call, Throwable t) {
}
});
}
}
附带的一些实体,工具类,抽象就不贴代码了。 demo在github的位置:这里