recyclerview滑动菜单两头大中间小 + 倒影

2023-10-13 18:59

本文主要是介绍recyclerview滑动菜单两头大中间小 + 倒影,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述
自定义了layoutManager+RecyclerView,直接上代码

public class MyLayoutManager extends RecyclerView.LayoutManager {private int mSumDx = 0;private int mTotalWidth = 0;private int mItemWidth, mItemHeight;private SparseArray<Rect> mItemRects = new SparseArray<>();/*** 记录Item是否出现过屏幕且还没有回收。true表示出现过屏幕上,并且还没被回收*/private SparseBooleanArray mHasAttachedItems = new SparseBooleanArray();@Overridepublic RecyclerView.LayoutParams generateDefaultLayoutParams() {return new RecyclerView.LayoutParams(RecyclerView.LayoutParams.WRAP_CONTENT,RecyclerView.LayoutParams.WRAP_CONTENT);}private int mIntervalWidth;private int mStartX;/*** RecyclerView的Item回收器*/private RecyclerView.Recycler mRecycle;/*** RecyclerView的状态器*/private RecyclerView.State mState;private ValueAnimator mAnimation;/*** 滑动的方向:左*/private static int SCROLL_LEFT = 1;/*** 滑动的方向:右*/private static int SCROLL_RIGHT = 2;@Overridepublic void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {if (getItemCount() == 0) {//没有Item,界面空着吧detachAndScrapAttachedViews(recycler);return;}mRecycle = recycler;mState = state;mHasAttachedItems.clear();mItemRects.clear();detachAndScrapAttachedViews(recycler);//将item的位置存储起来View childView = recycler.getViewForPosition(0);measureChildWithMargins(childView, 0, 0);mItemWidth = getDecoratedMeasuredWidth(childView);mItemHeight = getDecoratedMeasuredHeight(childView);mIntervalWidth = getIntervalWidth();mStartX = getWidth() / 2 - mItemWidth/2 ;//定义水平方向的偏移量int offsetX = 0;for (int i = 0; i < getItemCount(); i++) {Rect rect = new Rect(mStartX + offsetX, mItemWidth / 2, mStartX + offsetX + mItemWidth, mItemHeight);mItemRects.put(i, rect);mHasAttachedItems.put(i, false);offsetX += mIntervalWidth;}layoutItems(recycler, SCROLL_RIGHT);//如果所有子View的宽度和没有填满RecyclerView的宽度,// 则将宽度设置为RecyclerView的宽度mTotalWidth = Math.max(offsetX, getHorizontalSpace());}private int getHorizontalSpace() {return getWidth() - getPaddingLeft() - getPaddingRight();}@Overridepublic boolean canScrollHorizontally() {return true;}@Overridepublic int scrollHorizontallyBy(int dx, RecyclerView.Recycler recycler, RecyclerView.State state) {if (getChildCount() <= 0) {return dx;}int travel = dx;//如果滑动到最顶部if (mSumDx + dx < 0) {travel = -mSumDx;} else if (mSumDx + dx > getMaxOffset()) {//如果滑动到最底部travel = getMaxOffset() - mSumDx;}mSumDx += travel;layoutItems(recycler, SCROLL_RIGHT);return travel;}/**********************/private void layoutItems(RecyclerView.Recycler recycler,int scrollDirection){Rect visibleRect = getVisibleArea();//回收越界子Viewfor (int i=0 ; i < getChildCount(); i++) {View child = getChildAt(i);int position = getPosition(child);Rect rect = mItemRects.get(position);if (!Rect.intersects(rect, visibleRect)) {removeAndRecycleView(child, recycler);mHasAttachedItems.put(position, false);} else {layoutDecoratedWithMargins(child, rect.left - mSumDx, rect.top, rect.right - mSumDx, rect.bottom);handleChildView(child, rect.left - mStartX - mSumDx);mHasAttachedItems.put(position, true);}}int min = 0;int max = getItemCount();for (int i = min; i < max; i++) {Rect rect = mItemRects.get(i);if (Rect.intersects(visibleRect, rect) && !mHasAttachedItems.get(i)) {View child = recycler.getViewForPosition(i);if (scrollDirection == SCROLL_RIGHT) {addView(child, 0);} else {addView(child);}measureChildWithMargins(child, 0, 0);layoutDecoratedWithMargins(child, rect.left - mSumDx, rect.top, rect.right - mSumDx, rect.bottom);handleChildView(child, rect.left - mStartX - mSumDx );mHasAttachedItems.put(i, true);}}}/*** 获取可见的区域Rect** @return*/private Rect getVisibleArea() {Rect result = new Rect(getPaddingLeft() + mSumDx, getPaddingTop(), getWidth() - getPaddingRight() + mSumDx, getHeight() - getPaddingBottom());return result;}public int getIntervalWidth() {return mItemWidth + mItemWidth / 5;}public int getCenterPosition() {int pos = (int) (mSumDx / getIntervalWidth());int more = (int) (mSumDx % getIntervalWidth());if (more > getIntervalWidth() * 0.5f) pos++;return pos;}/*** 获取第一个可见的Item位置* <p>Note:该Item为绘制在可见区域的第一个Item,有可能被第二个Item遮挡*/public int getFirstVisiblePosition() {if (getChildCount() <= 0) {return 0;}View view = getChildAt(0);int pos = getPosition(view);return pos;}private View child;private int moveX;private void handleChildView(View child, int moveX) {this.child = child;this.moveX = moveX;float radio = computeScale(moveX);float rotation = computeRotationY(moveX);child.setPivotY(child.getHeight() / 4);child.setScaleX(radio);child.setScaleY(radio);if (isEnd && getPosition(child) == getCenterPosition()) {child.setRotationY(0);} else {child.setRotationY(rotation);}}@Overridepublic void smoothScrollToPosition(RecyclerView recyclerView, RecyclerView.State state, int position) {int finalOffset = calculateOffsetForPosition(position);if (mRecycle != null || mState != null) {//如果RecyclerView还没初始化完,先记录下要滚动的位置startScroll(mSumDx, finalOffset);}}/*** 滚动到指定X轴位置** @param from X轴方向起始点的偏移量* @param to   X轴方向终点的偏移量*/private void startScroll(int from, int to) {if (mAnimation != null && mAnimation.isRunning()) {mAnimation.cancel();}final int direction = from < to ? SCROLL_RIGHT : SCROLL_LEFT;mAnimation = ValueAnimator.ofFloat(from, to);mAnimation.setDuration(200);mAnimation.setInterpolator(new DecelerateInterpolator());mAnimation.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@Overridepublic void onAnimationUpdate(ValueAnimator animation) {mSumDx = Math.round((float) animation.getAnimatedValue());layoutItems(mRecycle,direction);}});mAnimation.start();}/*** 计算Item所在的位置偏移** @param position 要计算Item位置*/public int calculateOffsetForPosition(int position) {return Math.round(getIntervalWidth() * position);}/*** 计算Item缩放系数** @param x Item的偏移量* @return 缩放系数*/private float s = 0.8f;private float n = 0.1f;private float s2 = s + n;//0.94private float computeScale(int x) {
//        float scale = 1 - Math.abs(x * 2.0f / (8f * getIntervalWidth()));BigDecimal b = new BigDecimal(Math.abs(x * 2.0f / (11f * getIntervalWidth())) + s);float scale = b.setScale(2, BigDecimal.ROUND_HALF_DOWN).floatValue();if (scale < s2) scale = s2;
//        if (scale > 2) scale = 2;return scale;}/*** 获取最大偏移量*/private int getMaxOffset() {return (getItemCount() - 1) * getIntervalWidth();}private boolean isEnd = false;@Overridepublic void onScrollStateChanged(int state) {super.onScrollStateChanged(state);switch (state) {case RecyclerView.SCROLL_STATE_IDLE://滚动停止时isEnd = true;handleChildView(child, moveX);break;case RecyclerView.SCROLL_STATE_DRAGGING://拖拽滚动时break;case RecyclerView.SCROLL_STATE_SETTLING://动画滚动时break;}}/*** 最大Y轴旋转度数*/private float M_MAX_ROTATION_Y = 15.0f;private float computeRotationY(int x) {float rotationY;rotationY = -M_MAX_ROTATION_Y * x / getIntervalWidth();if (Math.abs(rotationY) > M_MAX_ROTATION_Y) {if (rotationY > 0) {rotationY = M_MAX_ROTATION_Y;} else {rotationY = -M_MAX_ROTATION_Y;}}return rotationY;}
}

滑动带惯性并自动选择居中

LinearSnapHelper().attachToRecyclerView(recyclerview)

倒影:在adapter中加入以下方法设置bitmap

fun compoundBitmap(context: Context, img: Int): Bitmap? {val originalBitmap = BitmapFactory.decodeResource(context.resources, img) //把资源图片变成一个Bitmap对象//生成下面的一半图片val matrix = Matrix()matrix.setScale(1f, -1f)//翻转val invertBitmap = Bitmap.createBitmap(originalBitmap, 0, 0, originalBitmap.width, originalBitmap.height, matrix, false)//创建一个空的位图val compoundBitmap = Bitmap.createBitmap(originalBitmap.width,//            originalBitmap.height + invertBitmap.height + 10,originalBitmap.height + invertBitmap.height,Bitmap.Config.ARGB_8888)val canvas = Canvas(compoundBitmap)canvas.drawBitmap(originalBitmap, 0f, 0f, null)//        canvas.drawBitmap(invertBitmap, 0f, (originalBitmap.height + 10).toFloat(), null)canvas.drawBitmap(invertBitmap, 0f, (originalBitmap.height-2).toFloat(), null)//上下俩个图片的间距val paint = Paint()// 设置渐变颜色val shader = LinearGradient(0f,//            (originalBitmap.height + 10).toFloat(),(originalBitmap.height).toFloat(),0f,compoundBitmap.height.toFloat(),0x70ffffff,0x00ffffff,Shader.TileMode.CLAMP)paint.shader = shaderpaint.xfermode = PorterDuffXfermode(PorterDuff.Mode.DST_IN)canvas.drawRect(0f,//            (originalBitmap.height + 5).toFloat(),(originalBitmap.height).toFloat(),originalBitmap.width.toFloat(),compoundBitmap.height.toFloat(),paint)return compoundBitmap}

这个时候基本可以了,但是在图片旋转的时候,会出现锯齿,继承ImageView

public class RotationImageView extends androidx.appcompat.widget.AppCompatImageView {public RotationImageView(Context context) {super(context);}public RotationImageView(Context context, @Nullable AttributeSet attrs) {super(context, attrs);}public RotationImageView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);}@Overrideprotected void onDraw(Canvas canvas) {if (getDrawable() instanceof BitmapDrawable) {//super.onDraw(canvas); 不走superPaint paint = new Paint();paint.setAntiAlias(true);//设置抗锯齿paint.setDither(true);Bitmap bitmap = ((BitmapDrawable) getDrawable()).getBitmap();BitmapShader shader = new BitmapShader(bitmap,Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);paint.setShader(shader);canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), paint);}}
}

要想点击左右箭头选择

		iv_left.setOnClickListener {//最小到哪个itemif (layoutManger.centerPosition > 0) {rv_menu.smoothScrollToPosition(layoutManger.centerPosition - 1)}}iv_right.setOnClickListener {//最大到哪个itemif (layoutManger.centerPosition < (rv_menu.adapter as MenuAdapter).itemCount - 1) {rv_menu.smoothScrollToPosition(layoutManger.centerPosition + 1)}}

这篇关于recyclerview滑动菜单两头大中间小 + 倒影的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/205225

相关文章

基于Redis有序集合实现滑动窗口限流的步骤

《基于Redis有序集合实现滑动窗口限流的步骤》滑动窗口算法是一种基于时间窗口的限流算法,通过动态地滑动窗口,可以动态调整限流的速率,Redis有序集合可以用来实现滑动窗口限流,本文介绍基于Redis... 滑动窗口算法是一种基于时间窗口的限流算法,它将时间划分为若干个固定大小的窗口,每个窗口内记录了该时间

禁止平板,iPad长按弹出默认菜单事件

通过监控按下抬起时间差来禁止弹出事件,把以下代码写在要禁止的页面的页面加载事件里面即可     var date;document.addEventListener('touchstart', event => {date = new Date().getTime();});document.addEventListener('touchend', event => {if (new

Windows如何添加右键新建菜单

Windows如何添加右键新建菜单 文章目录 Windows如何添加右键新建菜单实验环境缘起以新建`.md`文件为例第一步第二步第三步 总结 实验环境 Windows7 缘起 因为我习惯用 Markdown 格式写文本,每次新建一个.txt后都要手动修改为.md,真的麻烦。如何在右键新建菜单中添加.md选项呢? 网上有很多方法,这些方法我都尝试了,要么太麻烦,要么不凑效

专题二_滑动窗口_算法专题详细总结

目录 滑动窗口,引入: 滑动窗口,本质:就是同向双指针; 1.⻓度最⼩的⼦数组(medium) 1.解析:给我们一个数组nums,要我们找出最小子数组的和==target,首先想到的就是暴力解法 1)暴力: 2)优化,滑动窗口: 1.进窗口 2.出窗口 3.更新值 2.⽆重复字符的最⻓⼦串(medium) 1)仍然是暴力解法: 2)优化: 进窗口:hash[s[rig

hot100刷题第1-9题,三个专题哈希,双指针,滑动窗口

求满足条件的子数组,一般是前缀和、滑动窗口,经常结合哈希表; 区间操作元素,一般是前缀和、差分数组 数组有序,更大概率会用到二分搜索 目前已经掌握一些基本套路,重零刷起leetcode hot 100, 套路题按套路来,非套路题适当参考gpt解法。 一、梦开始的地方, 两数之和 class Solution:#注意要返回的是数组下标def twoSum(self, nums: Lis

【leetcode详解】考试的最大困扰度(滑动窗口典例)

实战总结: sum += answerKey[right] == c; 经典操作,将判断语句转化为0, 1接收来计数//大问题分解: 对'T'还是'F'做修改, 传参为c//滑动窗口: 遍历, 维护left& right指向 及 c的个数, 更新不知从何下手写代码时:考虑先写好第一次的,然后以此为基础补充代码以适后续情况 题面: 解题感受: 思路总体好想, 实现略有挑战。 思路分析:

【每日一题】LeetCode 2379.得到K个黑块的最少涂色次数(字符串、滑动窗口)

【每日一题】LeetCode 2379.得到K个黑块的最少涂色次数(字符串、滑动窗口) 题目描述 给定一个字符串 blocks,其中每个字符代表一个颜色块,可以是 ‘W’(白色)或 ‘B’(黑色)。你需要找到一个至少包含 k 个连续黑色块的子串。每次操作可以将一个白色块变成黑色块。你的任务是找到至少出现一次连续 k 个黑色块的最少操作次数。 和该题目类似:【每日一题】LeetCode 202

【视频教程】手把手AppWizard轻松制作一个emWin滑动主界面控制框架,任意跳转控制(2024-09-06)

现在的新版AppWizard已经比较好用,用户可以轻松的创建各种项目常规界面。 比如早期创建一个支持滑动的主界面框架,并且可以跳转各种子界面,仅仅界面布局和各种图片格式转换都要花不少时间,而现在使用AppWizard,可以说轻轻松松,毫不费力。 用户唯一要做的就是根据自己的芯片性能做一定的速度优化。 视频: https://www.bilibili.com/video/BV17Rp3eLE

Flutter-listview的item左右滑动,删除item

import 'package:flutter/material.dart';//列表左右滑动删除void main() =>runApp(MaterialApp(home: HomePage(),));class HomePage extends StatelessWidget {final List<String> items = List.generate(20, (index) =>

如何在Qt的widget上右键显示菜单

如何在Qt的widget上右键显示菜单 今天早上一来,我老大叫我在widget上点击右键加上一个菜单,并相应其响应的功能,因为我成刚接触Qt,所以看了下QtGUI编程这本书,做出来,记录下来,说不定哪天还用得上啊! 废话不多说,直接上代码: 方法一: m_text = QTextCodec::codecForLocale(); ui->tableWidget->addAction(ne