Android UI--开源项目IndexableListView(字母索引)(1)

}

// 获得韩文初始化字符数组里面的一个字符

private static char getInitialSound(char c) {

return KOREAN_INITIAL[(c - KOREAN_UNICODE_START) / KOREAN_UNIT];

}

}

自定义索引列表

/*

  • Copyright 2011 woozzu

  • Licensed under the Apache License, Version 2.0 (the “License”);

  • you may not use this file except in compliance with the License.

  • You may obtain a copy of the License at

  • http://www.apache.org/licenses/LICENSE-2.0
    
  • Unless required by applicable law or agreed to in writing, software

  • distributed under the License is distributed on an “AS IS” BASIS,

  • WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

  • See the License for the specific language governing permissions and

  • limitations under the License.

*/

package com.wwj.indexableListView.widget;

import android.content.Context;

import android.graphics.Canvas;

import android.util.AttributeSet;

import android.view.GestureDetector;

import android.view.MotionEvent;

import android.widget.ListAdapter;

import android.widget.ListView;

/**

  • 自定义索引列表

  • @author by 佚名

*/

public class IndexableListView extends ListView {

private boolean mIsFastScrollEnabled = false;

private IndexScroller mScroller = null;

private GestureDetector mGestureDetector = null;

public IndexableListView(Context context) {

super(context);

}

public IndexableListView(Context context, AttributeSet attrs) {

super(context, attrs);

}

public IndexableListView(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

}

@Override

public boolean isFastScrollEnabled() {

return mIsFastScrollEnabled;

}

@Override

public void setFastScrollEnabled(boolean enabled) {

mIsFastScrollEnabled = enabled;

if (mIsFastScrollEnabled) {

if (mScroller == null)

mScroller = new IndexScroller(getContext(), this);

} else {

if (mScroller != null) {

mScroller.hide();

mScroller = null;

}

}

}

@Override

public void draw(Canvas canvas) {

super.draw(canvas);

// Overlay index bar

if (mScroller != null)

mScroller.draw(canvas);

}

@Override

public boolean onTouchEvent(MotionEvent ev) {

// Intercept ListView’s touch event

if (mScroller != null && mScroller.onTouchEvent(ev))

return true;

if (mGestureDetector == null) {

// 创建一个GestureDetector(手势探测器)

mGestureDetector = new GestureDetector(getContext(),

new GestureDetector.SimpleOnGestureListener() {

@Override

public boolean onFling(MotionEvent e1, MotionEvent e2,

float velocityX, float velocityY) {

// If fling happens, index bar shows

// 显示索引条

mScroller.show();

return super.onFling(e1, e2, velocityX, velocityY);

}

});

}

mGestureDetector.onTouchEvent(ev);

return super.onTouchEvent(ev);

}

@Override

public boolean onInterceptTouchEvent(MotionEvent ev) {

return true;

}

@Override

public void setAdapter(ListAdapter adapter) {

super.setAdapter(adapter);

if (mScroller != null)

mScroller.setAdapter(adapter);

}

@Override

protected void onSizeChanged(int w, int h, int oldw, int oldh) {

super.onSizeChanged(w, h, oldw, oldh);

if (mScroller != null)

mScroller.onSizeChanged(w, h, oldw, oldh);

}

}

索引条

/*

  • Copyright 2011 woozzu

  • Licensed under the Apache License, Version 2.0 (the “License”);

  • you may not use this file except in compliance with the License.

  • You may obtain a copy of the License at

  • http://www.apache.org/licenses/LICENSE-2.0
    
  • Unless required by applicable law or agreed to in writing, software

  • distributed under the License is distributed on an “AS IS” BASIS,

  • WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

  • See the License for the specific language governing permissions and

  • limitations under the License.

*/

package com.wwj.indexableListView.widget;

import android.content.Context;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.Paint;

import android.graphics.RectF;

import android.os.Handler;

import android.os.Message;

import android.os.SystemClock;

import android.view.MotionEvent;

import android.widget.Adapter;

import android.widget.ListView;

import android.widget.SectionIndexer;

/**

  • 右侧的索引条

  • @author by 佚名

*/

public class IndexScroller {

private float mIndexbarWidth; // 索引条宽度

private float mIndexbarMargin; // 索引条外边距

private float mPreviewPadding; //

private float mDensity; // 密度

private float mScaledDensity; // 缩放密度

private float mAlphaRate; // 透明度

private int mState = STATE_HIDDEN; // 状态

private int mListViewWidth; // ListView宽度

private int mListViewHeight; // ListView高度

private int mCurrentSection = -1; // 当前部分

private boolean mIsIndexing = false; // 是否正在索引

private ListView mListView = null;

private SectionIndexer mIndexer = null;

private String[] mSections = null;

private RectF mIndexbarRect;

// 4种状态(已隐藏、正在显示、已显示、正在隐藏)

private static final int STATE_HIDDEN = 0;

private static final int STATE_SHOWING = 1;

private static final int STATE_SHOWN = 2;

private static final int STATE_HIDING = 3;

public IndexScroller(Context context, ListView lv) {

mDensity = context.getResources().getDisplayMetrics().density;

mScaledDensity = context.getResources().getDisplayMetrics().scaledDensity;

mListView = lv;

setAdapter(mListView.getAdapter());

mIndexbarWidth = 20 * mDensity; // 索引条宽度

mIndexbarMargin = 10 * mDensity;// 索引条间距

mPreviewPadding = 5 * mDensity; // 内边距

}

public void draw(Canvas canvas) {

if (mState == STATE_HIDDEN)

return;

// mAlphaRate determines the rate of opacity

Paint indexbarPaint = new Paint();

indexbarPaint.setColor(Color.BLACK);

indexbarPaint.setAlpha((int) (64 * mAlphaRate));

indexbarPaint.setAntiAlias(true);

// 画右侧字母索引的圆矩形

canvas.drawRoundRect(mIndexbarRect, 5 * mDensity, 5 * mDensity,

indexbarPaint);

if (mSections != null && mSections.length > 0) {

// Preview is shown when mCurrentSection is set

if (mCurrentSection >= 0) {

Paint previewPaint = new Paint(); // 用来绘画所以条背景的画笔

previewPaint.setColor(Color.BLACK);// 设置画笔颜色为黑色

previewPaint.setAlpha(96); // 设置透明度

previewPaint.setAntiAlias(true);// 设置抗锯齿

previewPaint.setShadowLayer(3, 0, 0, Color.argb(64, 0, 0, 0)); // 设置阴影层

Paint previewTextPaint = new Paint(); // 用来绘画索引字母的画笔

previewTextPaint.setColor(Color.WHITE); // 设置画笔为白色

previewTextPaint.setAntiAlias(true); // 设置抗锯齿

previewTextPaint.setTextSize(50 * mScaledDensity); // 设置字体大小

// 文本的宽度

float previewTextWidth = previewTextPaint

.measureText(mSections[mCurrentSection]);

float previewSize = 2 * mPreviewPadding

  • previewTextPaint.descent()
  • previewTextPaint.ascent();

RectF previewRect = new RectF(

(mListViewWidth - previewSize) / 2,

(mListViewHeight - previewSize) / 2,

(mListViewWidth - previewSize) / 2 + previewSize,

(mListViewHeight - previewSize) / 2 + previewSize);

// 中间索引的那个框

canvas.drawRoundRect(previewRect, 5 * mDensity, 5 * mDensity,

previewPaint);

// 绘画索引字母

canvas.drawText(

mSections[mCurrentSection],

previewRect.left + (previewSize - previewTextWidth) / 2

  • 1,

previewRect.top + mPreviewPadding

  • previewTextPaint.ascent() + 1,

previewTextPaint);

}

// 绘画右侧索引条的字母

Paint indexPaint = new Paint();

indexPaint.setColor(Color.WHITE);

indexPaint.setAlpha((int) (255 * mAlphaRate));

indexPaint.setAntiAlias(true);

indexPaint.setTextSize(12 * mScaledDensity);

float sectionHeight = (mIndexbarRect.height() - 2 * mIndexbarMargin)

/ mSections.length;

float paddingTop = (sectionHeight - (indexPaint.descent() - indexPaint

.ascent())) / 2;

for (int i = 0; i < mSections.length; i++) {

float paddingLeft = (mIndexbarWidth - indexPaint

.measureText(mSections[i])) / 2;

canvas.drawText(mSections[i], mIndexbarRect.left + paddingLeft,

mIndexbarRect.top + mIndexbarMargin + sectionHeight * i

  • paddingTop - indexPaint.ascent(), indexPaint);

}

}

}

public boolean onTouchEvent(MotionEvent ev) {

switch (ev.getAction()) {

case MotionEvent.ACTION_DOWN: // 按下,开始索引

// If down event occurs inside index bar region, start indexing

if (mState != STATE_HIDDEN && contains(ev.getX(), ev.getY())) {

setState(STATE_SHOWN);

// It demonstrates that the motion event started from index bar

mIsIndexing = true;

// Determine which section the point is in, and move the list to

// that section

mCurrentSection = getSectionByPoint(ev.getY());

mListView.setSelection(mIndexer

.getPositionForSection(mCurrentSection));

return true;

}

break;

case MotionEvent.ACTION_MOVE: // 移动

if (mIsIndexing) {

// If this event moves inside index bar

if (contains(ev.getX(), ev.getY())) {

// Determine which section the point is in, and move the

// list to that section

mCurrentSection = getSectionByPoint(ev.getY());

mListView.setSelection(mIndexer

.getPositionForSection(mCurrentSection));

}

return true;

}

break;

case MotionEvent.ACTION_UP: // 抬起

if (mIsIndexing) {

mIsIndexing = false;

mCurrentSection = -1;

}

if (mState == STATE_SHOWN)

setState(STATE_HIDING);

break;

}

return false;

}

public void onSizeChanged(int w, int h, int oldw, int oldh) {

mListViewWidth = w;

mListViewHeight = h;

mIndexbarRect = new RectF(w - mIndexbarMargin - mIndexbarWidth,

mIndexbarMargin, w - mIndexbarMargin, h - mIndexbarMargin);

}

// 显示

public void show() {

if (mState == STATE_HIDDEN)

setState(STATE_SHOWING);

else if (mState == STATE_HIDING)

setState(STATE_HIDING);

}

// 隐藏

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

尾声

面试成功其实都是必然发生的事情,因为在此之前我做足了充分的准备工作,不单单是纯粹的刷题,更多的还会去刷一些Android核心架构进阶知识点,比如:JVM、高并发、多线程、缓存、热修复设计、插件化框架解读、组件化框架设计、图片加载框架、网络、设计模式、设计思想与代码质量优化、程序性能优化、开发效率优化、设计模式、负载均衡、算法、数据结构、高级UI晋升、Framework内核解析、Android组件内核等。

不仅有学习文档,视频+笔记提高学习效率,还能稳固你的知识,形成良好的系统的知识体系。这里,笔者分享一份从架构哲学的层面来剖析的视频及资料分享给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。

Android进阶学习资料库

一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!

image

大厂面试真题

PS:之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

《2017-2020字节跳动Android面试历年真题解析》

如果需要PDF版本可以自行领取!

[外链图片转存中…(img-woQVj8Xv-1711293652358)]

不仅有学习文档,视频+笔记提高学习效率,还能稳固你的知识,形成良好的系统的知识体系。这里,笔者分享一份从架构哲学的层面来剖析的视频及资料分享给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。

[外链图片转存中…(img-8BHPbI6r-1711293652358)]

Android进阶学习资料库

一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!

[外链图片转存中…(img-4Sj7Re8z-1711293652358)]

大厂面试真题

PS:之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

[外链图片转存中…(img-t6AhJEKw-1711293652359)]

《2017-2020字节跳动Android面试历年真题解析》

[外链图片转存中…(img-jO4vVLws-1711293652359)]

如果需要PDF版本可以自行领取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值