PopupWindow RecyclerView实现下拉选择Spinner示例解析

 更新时间:2023年07月19日 09:41:43   作者:沫小北  
这篇文章主要介绍了PopupWindow RecyclerView实现下拉选择Spinner示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

对Spinner进行封装

当我们在Android开发中需要实现下拉选择功能时,可以使用自定义的Spinner控件来实现。Spinner控件是一个下拉列表框,可以显示多个选项供用户选择,并在用户选择后显示所选项的文本。

自定义的Spinner控件

为了方便使用和扩展,我们可以对Spinner进行封装,创建一个自定义的Spinner控件。自定义Spinner可以具备以下特性:

  • 点击展开和收起:通过点击Spinner,可以展开或收起下拉列表框。
  • 自定义样式:可以根据项目需求,自定义Spinner的外观样式,如背景颜色、字体颜色、箭头图标等。
  • 支持数据源:可以传入数据源,将数据显示在下拉列表框中供用户选择。
  • 默认选择项:可以指定一个默认选择项,在下拉列表框展开时,该选项将被高亮显示。
  • 选择监听:可以设置选择监听器,监听用户的选择操作,并进行相应的处理。

通过封装Spinner,我们可以将其功能与外观进行统一管理,并提供更加简洁和易用的接口供其他开发者使用。这样,其他开发者在使用时只需关注数据源和监听回调即可,无需关心Spinner的内部实现细节。

自定义Spinner的封装可以提高代码的可维护性和可复用性,减少重复代码的编写,同时也使代码结构更加清晰和易于理解。

总之,封装Spinner可以帮助我们更高效地实现下拉选择功能,并提供灵活性和可扩展性,使代码更加优雅和易于维护。

使用第三方控件:implementation 'com.github.CymChad:BaseRecyclerViewAdapterHelper:2.9.47'

自定义控件封装

定义类CustomSpinner继承LinearLayout以下是示例:

package com.example.demo.view;
import android.content.Context;
import android.graphics.Color;
import android.graphics.drawable.ColorDrawable;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.PopupWindow;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.core.content.ContextCompat;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.chad.library.adapter.base.BaseViewHolder;
import com.example.demo.R;
import java.util.Collection;
import java.util.List;
/**
 * @author: xtxiaolu
 * @date: 2023/7/7
 * 描述: 自定义公共 下拉选择控件
 */
public class CustomSpinner<T> extends LinearLayout {
    private TextView textView;
    private ImageView imageView;
    private PopupWindow popupWindow;
    private List<T> dataList;
    private boolean isExpanded = false;
    private OnItemSelectedListener itemSelectedListener;
    public CustomSpinner(Context context) {
        super(context);
        init();
    }
    public CustomSpinner(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }
    public CustomSpinner(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }
    private void init() {
        LayoutInflater.from(getContext()).inflate(R.layout.custom_spinner_layout, this, true);
        textView = findViewById(R.id.ll_list_default_txt);
        imageView = findViewById(R.id.ll_list_default_icon);
        setClickable(true);
        setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isExpanded) {
                    collapse();
                } else {
                    expand();
                }
            }
        });
    }
    public void setData(List<T> dataList) {
        this.dataList = dataList;
    }
    /**
     * 刷新页面
     *
     * @param data
     */
    public void replaceData(@NonNull Collection<? extends T> data) {
        // 不是同一个引用才清空列表
        if (data != dataList) {
            dataList.clear();
            dataList.addAll(data);
        }
    }
    public void setTextViewValue(String value) {
        textView.setText(value);
    }
    public TextView getTextViewValue() {
        return textView;
    }
    private void expand() {
        if (dataList == null || dataList.isEmpty()) {
            return;
        }
        textView.setTextColor(ContextCompat.getColor(getContext(), R.color.text_order_black));
        imageView.setImageResource(R.drawable.expand_arrows_fold);
        View popupView = LayoutInflater.from(getContext()).inflate(R.layout.popup_selector, null);
        RecyclerView recyclerView = popupView.findViewById(R.id.recyclerView);
        recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));
        CustomAdapter popAdapter = new CustomAdapter<>(R.layout.item_listview_popwin, dataList);
        popAdapter.setSelectedPosition(0);
        popAdapter.setOnItemClickListener(new CustomAdapter.OnItemClickListener() {
            @Override
            public void onItemClick(int position) {
                T selectedItem = dataList.get(position);
                if (itemSelectedListener != null) {
                    itemSelectedListener.onItemSelected(position, selectedItem);
                }
                popupWindow.dismiss();
            }
            @Override
            public void convertView(BaseViewHolder holder, Object item, boolean isSelected) {
                if (itemSelectedListener != null) {
                    itemSelectedListener.onItemCallBackData(holder, item);
                }
            }
        });
        recyclerView.setAdapter(popAdapter);
        int width = ViewGroup.LayoutParams.MATCH_PARENT;
        int height = ViewGroup.LayoutParams.WRAP_CONTENT;
        popupWindow = new PopupWindow(popupView, width, height);
        popupWindow.setBackgroundDrawable(new ColorDrawable(Color.WHITE));
        popupWindow.setFocusable(true);
        popupWindow.setOutsideTouchable(true);
        popupWindow.setOnDismissListener(new PopupWindow.OnDismissListener() {
            @Override
            public void onDismiss() {
                collapse();
            }
        });
        int[] location = new int[2];
        getLocationOnScreen(location);
        int x = location[0];
        int y = location[1] + getHeight();
        popupWindow.showAtLocation(this, Gravity.NO_GRAVITY, x, y);
        isExpanded = true;
    }
    private void collapse() {
        textView.setTextColor(ContextCompat.getColor(getContext(), R.color.colorTextBlue));
        imageView.setImageResource(R.drawable.expand_arrows_unfold);
        if (popupWindow != null && popupWindow.isShowing()) {
            popupWindow.dismiss();
        }
        isExpanded = false;
    }
    public void setOnItemSelectedListener(OnItemSelectedListener listener) {
        this.itemSelectedListener = listener;
    }
    public interface OnItemSelectedListener {
        void onItemSelected(int position, Object item);
        void onItemCallBackData(BaseViewHolder holder, Object item);
    }
}

上面是控件代码

适配器代码

下面是适配器代码都是使用范型来传输数据这样方便通用!

package com.example.demo.view;
import android.view.View;
import androidx.annotation.LayoutRes;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import com.chad.library.adapter.base.BaseQuickAdapter;
import com.chad.library.adapter.base.BaseViewHolder;
import java.util.List;
/**
 * @author: xtxiaolu
 * @date: 2023/7/7
 * 描述:
 */
public class CustomAdapter<T> extends BaseQuickAdapter<T, BaseViewHolder> {
    private int selectedPosition = -1;
    private OnItemClickListener itemClickListener;
    public CustomAdapter(@LayoutRes int layoutResId, @Nullable List<T> data) {
        super(layoutResId, data);
    }
    public void setSelectedPosition(int position) {
        this.selectedPosition = position;
        notifyDataSetChanged();
    }
    public void setOnItemClickListener(OnItemClickListener listener) {
        this.itemClickListener = listener;
    }
    @Override
    protected void convert(BaseViewHolder holder, T item) {
        if (itemClickListener != null) {
            itemClickListener.convertView(holder, item, selectedPosition == holder.getBindingAdapterPosition());
        }
    }
    @Override
    public void onBindViewHolder(@NonNull BaseViewHolder holder, int position) {
        super.onBindViewHolder(holder, position);
        holder.itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (itemClickListener != null) {
                    itemClickListener.onItemClick(holder.getBindingAdapterPosition());
                }
            }
        });
    }
    public interface OnItemClickListener {
        void onItemClick(int position);
        void convertView(BaseViewHolder holder, Object item, boolean isSelected);
    }
}

下面是代码链接:https://gitee.com/xtxiaolu/XiaoluDemo.git

以上就是PopupWindow RecyclerView实现下拉选择Spinner示例解析的详细内容,更多关于PopupWindow RecyclerView下拉选择Spinner的资料请关注脚本之家其它相关文章!

相关文章

  • Android定制自己的EditText轻松改变底线颜色

    Android定制自己的EditText轻松改变底线颜色

    这篇文章主要介绍了Android如何定制自己的EditText,如何轻松改变底线颜色,感兴趣的小伙伴们可以参考一下
    2015-12-12
  • ActivityManagerService之Service启动过程解析

    ActivityManagerService之Service启动过程解析

    这篇文章主要为大家介绍了ActivityManagerService之Service启动过程解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Mac Android Studio 3.0 Terminal 中文乱码问题处理

    Mac Android Studio 3.0 Terminal 中文乱码问题处理

    本文给大家分享的是在更新Android Studio 3.0之后,使用Terminal时,发现 git log 命令查看历史 log会乱码,以及最后的解决方法,推荐给小伙伴们
    2017-11-11
  • android 获取屏幕尺寸

    android 获取屏幕尺寸

    这篇文章主要介绍了android 获取屏幕尺寸的方法,需要的朋友可以参考下
    2015-03-03
  • Android AS创建自定义布局案例详解

    Android AS创建自定义布局案例详解

    这篇文章主要介绍了Android AS创建自定义布局案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • Android多线程及异步处理问题详细探讨

    Android多线程及异步处理问题详细探讨

    究其为啥需要多线程的本质就是异步处理,直观一点说就是不要让用户感觉到“很卡”为了提高用户体验那是必须要使用的
    2013-06-06
  • Android中Socket的应用分析

    Android中Socket的应用分析

    这篇文章主要介绍了Android中Socket的应用,结合实例形式分析了Android中socket通信的实现技巧与相关注意事项,需要的朋友可以参考下
    2016-10-10
  • Flutter 插件url_launcher简介

    Flutter 插件url_launcher简介

    最近项目需求是打开一个连接跳转到安卓或苹果默认的浏览器。虽然开始一个简单的要求,其中的一个细节就是执行打开网页这一操作后,不能看上去像在应用内部打开,看上去要在应用外部打开,今天小编给大家介绍Flutter 插件url_launcher的相关知识,感兴趣的朋友一起看看吧
    2020-04-04
  • Android TagCloudView云标签的使用方法

    Android TagCloudView云标签的使用方法

    这篇文章主要为大家详细介绍了Android TagCloudView云标签的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • Android实现搜索保存历史记录功能

    Android实现搜索保存历史记录功能

    这篇文章主要介绍了Android实现搜索保存历史记录功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05

最新评论