热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android中PopuWindow实现下拉列表实例

前言 之前讲过一篇关于PopuWindow的基本使用的文章,想了解的同学可以参考:PopupWindow的基本使用 其实,下拉列表S

前言

之前讲过一篇关于PopuWindow的基本使用的文章,想了解的同学可以参考:PopupWindow的基本使用

其实,下拉列表Spanner(不知道控件拼写是否正确)就能实现,但是基于ui美化方面的考虑,用popuwindow实现是一个不错的选择,今天就来讲讲PopuWindow实现下拉列表的具体实现吧。

正文

文章可能会有点长,大家将就着看吧。先上波效果图才是厚道的:


下面开始正式讲解。

基础依赖,由于下拉列表是用RecycleView实现,然后控件初始化用到ButterKnife,所以要在app的gradle中添加相关依赖:

 //RecycleView
  compile 'com.android.support:recyclerview-v7:25.2.0'
  //butterKnife
  compile 'com.jakewharton:butterknife:8.5.1'
  //这条千万不能忘记!!
  annotationProcessor 'com.jakewharton:butterknife-compiler:8.5.1'

第一步,不言而喻,肯定是上BasePopupWindow代码:

/***
 * PopupWindow基类
 * 
 * @author pei
 * @version 1.0
 * @cretae 2016-7-21
 * @注:若要popwindow点击外部消失,则设置 this.setFocusable(true)
 *   若要popwindow点击外部不消失,不做setFocusable设置,也不要设置成this.setFocusable(false)
 * 
 */
public abstract class BasePopupWindow extends PopupWindow {

  protected View mLayoutView;
  protected int mLayoutId;
  protected Context mContext;
  protected int mWidth;
  protected int mHeight;

  public BasePopupWindow(int width, int height, int layoutId, Context context) {
    this.mWidth = width;
    this.mHeight = height;
    this.mLayoutId = layoutId;
    this.mCOntext= context;
    mLayoutView = LayoutInflater.from(context).inflate(mLayoutId, null);
    setWindow();
  }

  /** PopupWindow基本设置 **/
  protected void setWindow() {
    this.setContentView(mLayoutView);
    this.setWidth(mWidth);
    this.setHeight(mHeight);
    // this.setFocusable(true);// 可点击
    // 实例化一个ColorDrawable颜色为半透明(半透明遮罩颜色代码#66000000)
    ColorDrawable dw = new ColorDrawable(Color.TRANSPARENT);
    this.setBackgroundDrawable(dw);
  }

  /** PopupWindow背景设置 **/
  protected void setBackground(int color) {
    // 实例化一个ColorDrawable颜色为半透明
    ColorDrawable dw = new ColorDrawable(color);
    this.setBackgroundDrawable(dw);
  }

  protected abstract void initView();
  protected abstract void initData();
  protected abstract void setListener();

  /** PopupWindow点击间隙处理,根据实际情况重写 **/
  protected void onTouchdimiss() {
    // mMenuView添加OnTouchListener监听判断获取触屏位置如果在选择框外面则销毁弹出框
    mLayoutView.setOnTouchListener(new OnTouchListener() {
      @Override
      public boolean onTouch(View view, MotionEvent event) {
//        int height = mLayoutView.getTop();
//        int y = (int) event.getY();
//        if (event.getAction() == MotionEvent.ACTION_UP) {
//          if (y  T getView(int viewId) {
    View view = null;
    if (mLayoutView == null) {
      mLayoutView = LayoutInflater.from(mContext).inflate(mLayoutId, null);
    }
    view = mLayoutView.findViewById(viewId);
    return (T) view;
  }

}

第二步,写一个OrderPop继承于BasePopupWindow:

/**
 * Instruction:下拉列表Pop
 * 

* Author:pei * Date: 2017/6/28 * Description: */ public class OrderPop extends BasePopupWindow{ private RecyclerView mRecyclerView; private ListmDatas; private ManagerPopuAdapter managerPopuAdapter; public OrderPop(Context context, Listdatas) { super(ScreenUtil.dp2px(100,context), ScreenUtil.dp2px(150,context), R.layout.pop_order, context); this.mDatas=datas; initView(); initData(); setListener(); } @Override protected void initView() { mRecyclerView=getView(R.id.recycler_view); } @Override protected void initData() { setFocusable(true); setAnimationStyle(R.style.popuwindow_up_style);//popuwindow显示隐藏的动画 mRecyclerView.setHasFixedSize(true); mRecyclerView.setLayoutManager(new LinearLayoutManager(mContext)); managerPopuAdapter = new ManagerPopuAdapter(mContext, mDatas); mRecyclerView.setAdapter(managerPopuAdapter); } @Override protected void setListener(){ } public ManagerPopuAdapter getAdapter(){ return managerPopuAdapter; } public void notifyDataSetChanged(){ if(managerPopuAdapter!=null){ managerPopuAdapter.notifyDataSetChanged(); } } public void setCurrentIndex(int position){ if(managerPopuAdapter!=null){ managerPopuAdapter.setIndex(position); } } }

OrderPop类中涉及到的内容挺多,以下将细细讲解。

1.--- 声明中涉及到RecycleView的一个适配器ManagerPopuAdapter,其代码如下:

/**
 * Instruction: Orderpop的适配器
 * 

* Author:pei * Date: 2017/6/29 * Description: */ public class ManagerPopuAdapter extends RecyclerView.Adapter { protected Context mContext; protected View mLayoutView; protected List mData; private ViewHolder mViewHolder; protected OnRecyclerItemClickListener mOnRecyclerItemClickListener; private int mIndex; public void setOnRecyclerItemClickListener(OnRecyclerItemClickListener onRecyclerItemClickListener) { this.mOnRecyclerItemClickListener= onRecyclerItemClickListener; } public ManagerPopuAdapter(Context context, List data) { this.mCOntext= context; this.mData = data; } @Override public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { //注:不可使用view=LayoutInflater.from(mContext).inflate(R.layout.item_layout,null);不然会报错 mLayoutView = LayoutInflater.from(mContext).inflate(R.layout.item_popu_order_layout, parent, false); return new ViewHolder(mLayoutView); } @Override public int getItemViewType(int position) { return super.getItemViewType(position); } @Override public int getItemCount() { return mData == null ? 0 : mData.size(); } @Override public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { mViewHolder = ((ViewHolder) holder); initData(position); setListener(position); } private void initData(int position) { String cOntent=mData.get(position).toString(); mViewHolder.tvContent.setText(content); if(mIndex==position){ mViewHolder.tvContent.setSelected(true); }else{ mViewHolder.tvContent.setSelected(false); } } private void setListener(final int position) { mViewHolder.tvContent.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if (mOnRecyclerItemClickListener != null) { mOnRecyclerItemClickListener.onRecyclerClick(position); } } }); } public void setIndex(int index){ this.mIndex=index; } class ViewHolder extends RecyclerView.ViewHolder { TextView tvContent; public ViewHolder(View view) { super(view); tvCOntent=(TextView)view.findViewById(R.id.tv_content); } } public interface OnRecyclerItemClickListener { void onRecyclerClick(int position); } }

2.--- ManagerPopuAdapter.java对应的layout----- item_popu_order_layout.xml代码:

<&#63;xml version="1.0" encoding="utf-8"&#63;>


  



3.--- item_popu_order_layout.xml中android:background="@drawable/manager_fragment_popu_bg"对应的drawable文件为:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  
  

manager_fragment_popu_pressed和manager_fragment_popu_normal对应的其实都是纯颜色xml文件。

manager_fragment_popu_pressed.xml代码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  

manager_fragment_popu_normal.xml代码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  

也许有的同学会问android:background="@drawable/manager_fragment_popu_bg文件中为什恶魔不直接用color属性设置背景切换,而要用color写个drawable供调用,其实我一开始也是这样弄的,无奈报错,具体原因不详,知道的同学可以回复下,此处不做重点。

4.--- item_popu_order_layout.xml中android:textColor="@drawable/text_color_bg"对应的drawable文件如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  
  

5.---讲讲OrderPop的构造函数

 public OrderPop(Context context, Listdatas) {
    super(ScreenUtil.dp2px(100,context), ScreenUtil.dp2px(150,context), R.layout.pop_order, context);
    this.mDatas=datas;

    initView();
    initData();
    setListener();
  }

这里我其实是图方便,所以直接传了个固定宽度 ScreenUtil.dp2px(100,context) 进去了,实际开发中因为是点击某个控件然后在控件下面显示出来,那么应该传那个控件的宽度。

6.---OrderPop的layout布局pop_order.xml代码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>


  



其中,android:scrollbars="vertical"是设置滚动条方向,android:scrollbarThumbVertical="@color/blue"是设置滚动条颜色,android:scrollbar设置滚动条样式

7.---讲讲OrderPop的显隐动画问题

在OrderPop类中的initData()方法中涉及到这样一行代码:

代码如下:

setAnimationStyle(R.style.popuwindow_up_style);//popuwindow显示隐藏的动画

涉及到popuwindow的显隐动画问题,大家可以参考的前言中提到的popuwindow的基本使用文章,这里就不废话了。

第三步,OrderPop写好了,就该看看在MainActivity中是怎么调用的了,贴出MainActivity的代码:

/**
 * Created by Admin on 2017/5/19.
 */

public class MainActivity extends BaseActivity implements View.OnClickListener{

  @BindView(R.id.tv_order)
  TextView mTvOrder;

  private static final int DEFAULT_INDEX=0;
  private List mOrderList=new ArrayList<>();
  private OrderPop mOrderPop;

  @Override
  protected int getContentViewId() {
    return R.layout.activity_main;
  }

  @Override
  protected void initData() {
    initOrderTextBar();
  }

  /**订单列表**/
  private void initOrderTextBar(){
    mOrderList.add("野蛮人");
    mOrderList.add("圣骑士");
    mOrderList.add("亚马逊");
    mOrderList.add("死灵法师");
    mOrderList.add("法师");
    mOrderList.add("德鲁伊");
    mOrderList.add("刺客");
    mOrderPop=new OrderPop(mContext,mOrderList);
    setBarContent(mTvOrder,mOrderList,DEFAULT_INDEX);

    mOrderPop.setOnDismissListener(new PopupWindow.OnDismissListener() {
      @Override
      public void onDismiss(){
        mTvOrder.setSelected(false);
      }
    });
    //mOrderPop项点击事件
    mOrderPop.getAdapter().setOnRecyclerItemClickListener(new ManagerPopuAdapter.OnRecyclerItemClickListener() {
      @Override
      public void onRecyclerClick(int position) {
        showShortToast(mOrderList.get(position));
        //更新mTvOrder显示内容
        setBarContent(mTvOrder,mOrderList,position);
        //更新mOrderPop视图选中背景
        mOrderPop.setCurrentIndex(position);
        mOrderPop.notifyDataSetChanged();
      }
    });
  }


  @Override
  protected void setListener() {
    mTvOrder.setOnClickListener(this);
  }

  @Nullable
  @Override
  protected BasePresenter getPresenter() {
    return null;
  }

  @Override
  protected void onDestroy(){
    super.onDestroy();
  }

  @Override
  public void onClick(View v) {
    switch (v.getId()) {
      case R.id.tv_order:
        if(mOrderPop!=null&&!mOrderPop.isShowing()){
          mTvOrder.setSelected(true);//控制mTvOrder变色
          mOrderPop.showAtLocationGravityBottom(mTvOrder,3);//显示mOrderPop
          //更新mOrderPop视图选中背景
          mOrderPop.setCurrentIndex(getIndexByTag(mTvOrder));
          mOrderPop.notifyDataSetChanged();
        }
        break;
      default:
        break;
    }
  }

  private void setBarContent(TextView textView,Listdata,int position){
    textView.setTag(position);
    textView.setText(data.get(position).toString());
  }

  private int getIndexByTag(TextView textView){
    int index=DEFAULT_INDEX;
    Object obj=textView.getTag();
    if(obj!=null){
      try {
        index=Integer.valueOf(obj.toString());
      } catch (NumberFormatException e) {
        e.printStackTrace();
      }
    }
    return index;
  }
}

MainActivity对应的布局activity_main.xml代码如下:




  


android:drawableRight="@drawable/manager_fragment_order_bg"中manager_fragment_order_bg.xml对应的代码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  
  

ic_drop_up和ic_drop_down对应的分别是一张箭头向上的图片和一张箭头向下的图片,这里就不多说了。

android:textColor="@drawable/text_color_bg"的话是设置文字选中和未被选中时显示的颜色,在上面的第二步第四条已经讲过了,这里就不说了。

ok,整个实现过程大致就是这样的。今天关于PopuWindow实现下拉列表的知识就讲到这里了,谢谢诶。希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • HDFS2.x新特性
    一、集群间数据拷贝scp实现两个远程主机之间的文件复制scp-rhello.txtroothadoop103:useratguiguhello.txt推pushscp-rr ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 本文是关于自学Android的笔记,包括查看类的源码的方法,活动注册的必要性以及布局练习的重要性。通过学习本文,读者可以了解到在自学Android过程中的一些关键点和注意事项。 ... [详细]
  • iOS Swift中如何实现自动登录?
    本文介绍了在iOS Swift中如何实现自动登录的方法,包括使用故事板、SWRevealViewController等技术,以及解决用户注销后重新登录自动跳转到主页的问题。 ... [详细]
author-avatar
簕竹仔_591
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有