热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android-自定义类似excel表格,双向滑动的ListView

效果图:主要代码:***Createdbymontyon2017831.*publicclassPanelListLayoutextendsFrameLayout{

效果图:
列表效果图

主要代码:

/**
* Created by monty on 2017/8/31.
*/


public class PanelListLayout extends FrameLayout {
private static final String TAG = PanelListLayout.class.getSimpleName();
@BindView(R.id.tv_title)
TextView tvTitle;
@BindView(R.id.lv_lineNumber)
ListView lvLineNumber;
@BindView(R.id.ll_tableHeader)
LinearLayout llTableHeader;
@BindView(R.id.lv_content)
ListView lvContent;
@BindView(R.id.swipeRefreshLayout)
RefreshLayout swipeRefreshLayout;
@BindView(R.id.ll_lineNumber)
LinearLayout llLineNumber;

private Context mContext;
// private BaseAdapter mAdapter;

/*说明:这里只可设置表头高度和行号宽度是因为表头的宽度和行号高度由ContentListView中的Item来决定,由调用者在ListView的Item布局中控制*/
private int tableHeaderHeight = 30; // 表头高度
private int lineNumberWidth = 30; // 行号宽度

private int tableHeaderBackgroundColor = 0xff00bcd4; // 表头背景颜色
private int tableHeaderTextColor = 0xffffffff; // 表头文本颜色
private int tableHeaderTextSize = 14; // 表头文本大小

private int lineNumberBackgroundColor = 0xff00bcd4; // 行号背景颜色
private int lineNumberTextColor = 0xffffffff; // 行号文本颜色
private int lineNumberTextSize = 14; // 行号文本大小

private List tableHeaderTexts;


public PanelListLayout(Context context) {
this(context, null);
}

public PanelListLayout(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}

public PanelListLayout(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mCOntext= context;
inflate(getContext(), R.layout.panel_list_layout, this);
ButterKnife.bind(this);
initView();

}

private void initView() {
/* 去掉ListView滑动到头部或底部时的动画 */
lvLineNumber.setOverScrollMode(View.OVER_SCROLL_NEVER);
lvContent.setOverScrollMode(View.OVER_SCROLL_NEVER);

OnScrollListener OnScrollListener= new OnScrollListener();
lvLineNumber.setOnScrollListener(onScrollListener);
lvContent.setOnScrollListener(onScrollListener);
/**
* 解决SwipeRefreshLayout中多层嵌套ListView下拉事件冲突问题
*/

swipeRefreshLayout.setOnChildScrollUpCallback(new SwipeRefreshLayout.OnChildScrollUpCallback() {
@Override
public boolean canChildScrollUp(SwipeRefreshLayout parent, @Nullable View child) {
return ViewCompat.canScrollVertically(lvContent, -1);
}
});
}

public RefreshLayout getSwipeRefreshLayout() {
return swipeRefreshLayout;
}

/**
* 设置表头高度(单位:dp)
*
* @param height
*/

public PanelListLayout setTableHeaderHeight(int height) {
this.tableHeaderHeight = height;
return this;
}

/**
* 设置行号宽度(单位:dp)
*
* @param width
*/

public PanelListLayout setLineNumberWidth(int width) {
this.lineNumberWidth = width;
return this;
}

/**
* 设置表头背景颜色
*
* @param color
*/

public PanelListLayout setTableHeaderBackgroundColor(int color) {
this.tableHeaderBackgroundColor = color;
return this;
}

/**
* 设置行号文本颜色
*
* @param color
* @return
*/

public PanelListLayout setLineNumberTextColor(int color) {
this.lineNumberTextColor = color;
return this;
}

/**
* 设置行号文本颜色(单位:dp)
*
* @param textSize
* @return
*/

public PanelListLayout setLineNumberTextSize(int textSize) {
this.lineNumberTextSize = DisplayUtil.dp2px(textSize);
return this;
}

/**
* 设置行号背景颜色
*
* @param color
* @return
*/

public PanelListLayout setLineNumberBackgroundColor(int color) {
this.lineNumberBackgroundColor = color;
return this;
}

/**
* 设置表头文本
*
* @param text
* @return
*/

public PanelListLayout setTableHeaderTexts(String... text) {
this.tableHeaderTexts = Arrays.asList(text);
return this;
}

/**
* 设置内容Adapter
*
* @param adapter
*/

public void setAdapter(BaseAdapter adapter) {
if (adapter == null) {
throw new IllegalArgumentException("adapter is not allow to be null");
}

// this.mAdapter = adapter;
this.lvContent.setAdapter(adapter);
/****************************************
* 监听Adapter,当ListView中的内容发生改变时重新渲染(主要作用,第一次设置Adapter之后,没有数据的情况下,
* 从网络获取数据之后刷新Adapter之后需要重新绘制表头)
****************************************/

adapter.registerDataSetObserver(new DataSetObserver() {
@Override
public void onChanged() {
Log("DataSetObserver ******** onChanged");
applyView();
}

@Override
public void onInvalidated() {
Log("DataSetObserver ******** onInvalidated");
super.onInvalidated();
applyView();
}
});

applyView();
}

/**
* ************************************核心代码**********************************
*


* 填充表格
*/
private void applyView() {
/*设置行号宽度*/
ViewGroup.LayoutParams layoutParams = llLineNumber.getLayoutParams();
layoutParams.width = lineNumberWidth;

this.lvContent.post(new Runnable() {
@Override
public void run() {
Log("post******************************");
View childItem = lvContent.getChildAt(lvContent.getFirstVisiblePosition());
Log("post**********填充行号*************");
applyLineNumberListView();
Log("post**********填充表头*************");
applyTableHeader(childItem);
}

/**
* 填充表头
* @param childItem
*/

private void applyTableHeader(View childItem) {
llTableHeader.setBackgroundColor(tableHeaderBackgroundColor);
llTableHeader.getLayoutParams().height = tableHeaderHeight;
llTableHeader.setShowDividers(LinearLayout.SHOW_DIVIDER_MIDDLE | LinearLayout.SHOW_DIVIDER_BEGINNING);
llTableHeader.setDividerDrawable(ContextCompat.getDrawable(getContext(), R.drawable.row_item_divider));
tvTitle.getLayoutParams().height = llTableHeader.getLayoutParams().height;
LinearLayout cOntentItem= (LinearLayout) childItem;
llTableHeader.removeAllViews();

for (int i = 0; i View v = contentItem.getChildAt(i);
TextView textView = new TextView(mContext);
//根据ContentListView的Item中列的宽度来动态设置表头宽度
textView.setLayoutParams(new LinearLayout.LayoutParams(v.getWidth(), ViewGroup.LayoutParams.MATCH_PARENT));
textView.setText(tableHeaderTexts.get(i));
textView.setTextColor(tableHeaderTextColor);
textView.setTextSize(tableHeaderTextSize);
textView.setGravity(Gravity.CENTER);
llTableHeader.addView(textView);
}
}

/**
* 填充行号
*/

private void applyLineNumberListView() {
lvLineNumber.setAdapter(new LineNumberAdapter(lvContent.getCount()));
}

});
}

private class LineNumberAdapter extends BaseAdapter {
private int mLineNumberCount;
private String[] mLineNumbers;

public LineNumberAdapter(int lineNumberCount) {
this.mLineNumberCount = lineNumberCount;
this.mLineNumbers = generateLineNumbers();
}

@Override
public int getCount() {
return mLineNumbers.length;
}

@Nullable
@Override
public Object getItem(int position) {
return mLineNumbers[position];
}

@Override
public long getItemId(int position) {
return position;
}

@NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
View view;
if (cOnvertView== null) {
view = LayoutInflater.from(mContext).inflate(android.R.layout.simple_list_item_1, parent, false);
} else {
view = convertView;
}
View cOntentItemView= lvContent.getAdapter().getView(position, null, lvContent);
if (contentItemView != null) { // 动态测量ContentListView中Item的高度,并将行号的高度与其保持一致
contentItemView.measure(0, 0);
view.setLayoutParams(new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, contentItemView.getMeasuredHeight()));
}
TextView textView = (TextView) view;
textView.setBackgroundColor(lineNumberBackgroundColor);
textView.setText(this.mLineNumbers[position]);
textView.setTextColor(lineNumberTextColor);
textView.setTextSize(lineNumberTextSize);
textView.setPadding(0, 0, 0, 0);
textView.setGravity(Gravity.CENTER);
return view;
}

private String[] generateLineNumbers() {
String[] lineNumbers = new String[this.mLineNumberCount];
for (int i = 0; i <this.mLineNumberCount; i++) {
lineNumbers[i] = "" + (i + 1);
}
return lineNumbers;
}

}


private void Log(String log) {
Log.d(TAG, log);
}

public class OnScrollListener implements AbsListView.OnScrollListener {
private int scrollState;

@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
Log("onScrollStateChanged - scrollState -> " + scrollState);
this.scrollState = scrollState;
}

@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
/*********************************************
* 解决无限同步LineNumberListView和ContentListView滚动位置的问题,SCROLL_STATE_IDLE是ListView滚动结束时的标记,
* 此处判断如果滚动结束了就不再进行同步,避免因为一直处于滚动状态而导致ListView的getView会一直调用的问题。
*********************************************/

if (scrollState == SCROLL_STATE_IDLE) {
return;
}
View subView = view.getChildAt(0);
if (subView != null) {
int top = subView.getTop();
if (view == lvContent) {
lvLineNumber.setSelectionFromTop(firstVisibleItem, top);
} else {
lvContent.setSelectionFromTop(firstVisibleItem, top);
}
}
}
}

}


推荐阅读
  • 在 Vue 应用开发中,页面状态管理和跨页面数据传递是常见需求。本文将详细介绍 Vue Router 提供的两种有效方式,帮助开发者高效地实现页面间的数据交互与状态同步,同时分享一些最佳实践和注意事项。 ... [详细]
  • 在处理木偶评估函数时,我发现可以顺利传递本机对象(如字符串、列表和数字),但每当尝试将JSHandle或ElementHandle作为参数传递时,函数会拒绝接受这些对象。这可能是由于这些句柄对象的特殊性质导致的,建议在使用时进行适当的转换或封装,以确保函数能够正确处理。 ... [详细]
  • 设计实战 | 10个Kotlin项目深度解析:首页模块开发详解
    设计实战 | 10个Kotlin项目深度解析:首页模块开发详解 ... [详细]
  • C#中实现高效UDP数据传输技术
    C#中实现高效UDP数据传输技术 ... [详细]
  • 在处理大规模并发请求时,传统的多线程或多进程模型往往无法有效解决性能瓶颈问题。尽管它们在处理小规模任务时能提升效率,但在高并发场景下,系统资源的过度消耗和上下文切换的开销会显著降低整体性能。相比之下,Python 的 `asyncio` 模块通过协程提供了一种轻量级且高效的并发解决方案。本文将深入解析 `asyncio` 模块的原理及其在实际应用中的优化技巧,帮助开发者更好地利用协程技术提升程序性能。 ... [详细]
  • 本文介绍如何在 Android 中自定义加载对话框 CustomProgressDialog,包括自定义 View 类和 XML 布局文件的详细步骤。 ... [详细]
  • 本文详细介绍了如何在Unity中实现一个简单的广告牌着色器,帮助开发者更好地理解和应用这一技术。 ... [详细]
  • 实验九:使用SharedPreferences存储简单数据
    本实验旨在帮助学生理解和掌握使用SharedPreferences存储和读取简单数据的方法,包括程序参数和用户选项。 ... [详细]
  • 在软件开发过程中,经常需要将多个项目或模块进行集成和调试,尤其是当项目依赖于第三方开源库(如Cordova、CocoaPods)时。本文介绍了如何在Xcode中高效地进行多项目联合调试,分享了一些实用的技巧和最佳实践,帮助开发者解决常见的调试难题,提高开发效率。 ... [详细]
  • 本文介绍了如何使用 Node.js 和 Express(4.x 及以上版本)构建高效的文件上传功能。通过引入 `multer` 中间件,可以轻松实现文件上传。首先,需要通过 `npm install multer` 安装该中间件。接着,在 Express 应用中配置 `multer`,以处理多部分表单数据。本文详细讲解了 `multer` 的基本用法和高级配置,帮助开发者快速搭建稳定可靠的文件上传服务。 ... [详细]
  • 本文详细探讨了 jQuery 中 `ajaxSubmit` 方法的使用技巧及其应用场景。首先,介绍了如何正确引入必要的脚本文件,如 `jquery.form.js` 和 `jquery-1.8.0.min.js`。接着,通过具体示例展示了如何利用 `ajaxSubmit` 方法实现表单的异步提交,包括数据的发送、接收和处理。此外,还讨论了该方法在不同场景下的应用,如文件上传、表单验证和动态更新页面内容等,提供了丰富的代码示例和最佳实践建议。 ... [详细]
  • 技术日志:使用 Ruby 爬虫抓取拉勾网职位数据并生成词云分析报告
    技术日志:使用 Ruby 爬虫抓取拉勾网职位数据并生成词云分析报告 ... [详细]
  • 本文深入探讨了Ajax的工作机制及其在现代Web开发中的应用。Ajax作为一种异步通信技术,改变了传统的客户端与服务器直接交互的模式。通过引入Ajax,客户端与服务器之间的通信变得更加高效和灵活。文章详细分析了Ajax的核心原理,包括XMLHttpRequest对象的使用、数据传输格式(如JSON和XML)以及事件处理机制。此外,还介绍了Ajax在提升用户体验、实现动态页面更新等方面的具体应用,并讨论了其在当前Web开发中的重要性和未来发展趋势。 ... [详细]
  • 本文详细解析了 Python 2.x 版本中 `urllib` 模块的核心功能与应用实例,重点介绍了 `urlopen()` 和 `urlretrieve()` 方法的使用技巧。其中,`urlopen()` 方法用于发送网络请求并获取响应内容,而 `urlretrieve()` 方法则用于下载文件并保存到本地。文章通过具体示例展示了这两个方法在实际开发中的应用场景,帮助读者更好地理解和掌握 `urllib` 模块的使用。 ... [详细]
  • 《精通 jQuery》第六章:深入解析与实战应用
    《精通 jQuery》第六章:深入解析与实战应用本章详细探讨了 Ajax 技术的核心机制及其实际应用。Ajax 通过 XMLHttpRequest 对象实现客户端与服务器之间的异步数据交换,从而在不重新加载整个页面的情况下更新部分内容。这种技术不仅提升了用户体验,还提高了应用的响应速度和效率。此外,本章还介绍了如何利用 jQuery 简化 Ajax 操作,并提供了多个实战案例,帮助读者更好地理解和掌握这一重要技术。 ... [详细]
author-avatar
手机用户2502870863
这个家伙很懒,什么也没留下!
Tags | 热门标签
RankList | 热门文章
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有