星空发现图(仿蜗牛梦话圈)

简介: 最终效果   直接使用继承 BaseAdapter 实现onCreateCenter(ViewGroup parent) onCreateChild(ViewGroup parent, T t)方法设置中心View,子View.

最终效果

 

 

 

直接使用

  1. 继承 BaseAdapter 实现onCreateCenter(ViewGroup parent) onCreateChild(ViewGroup parent, T t)方法设置中心View,子View.
  2. DreamLayout实例 调用 setAdapter(BaseAdapter baseAdapter) 方法.
  3. 调用BaseAdapteraddChild(T t),removeChild(T t)增删子View.

需求分解

  • 背景类水波纹的扩散效果

    • 初始化

    初始化没啥好说的,就是一些画笔设置,点位初始化,因为部分初始化与控件大小相关,因此要讲初始化放在尺寸确认后,了解一下View的关键生命周期构造函数() --> onFinishInflate() --> onAttachedToWindow() --> onMeasure() --> onSizeChanged() --> onLayout() --> onDraw() --> onDetackedFromWindow() 当onSizeChanged之后,布局大小确定下来,因此在此处初始化.

    • 环形渐变效果 使用RadialGradient
        //圆心为画布中心,半径为 mRadiusMax,半径80%处向外渐变为白色
        RadialGradient gradient =
                new RadialGradient(width / 2, height / 2, mRadiusMax,
                        new int[]{centerColor, centerColor, edgeColor}, new float[]{0f, .8f, 1.0f}, Shader.TileMode.MIRROR);
        mLayoutPaint.setShader(gradient);
    
    • 使用handler 定时,更新圆直径,模拟水波纹效果

    初始化三个大小不等的圆,每次放大2%,超出最大值,重新开始,模拟水波纹效果

    • 绘制图形,使用画布缩放绘制不同大小的圆环;

    原本直接想用drawCircle()绘制不同大小的圆,出现一个问题,需要为不同大小的圆重新 设置环形渐变,而onDraw()是一个频繁操作,不建议创建对象,因此想到通过缩放画布的方式来实现绘制不同大小的渐变圆环.

    	// scale 描述图片放大比例;
        // 将画布放大scale 比例后,绘制同样大小的圆,还原画布,即可得放大scale倍的圆
        canvas.save();
        canvas.scale(scale, scale, width / 2, height / 2);
        canvas.drawCircle(width / 2, height / 2, mRadiusMax, mLayoutPaint);
        canvas.restore();
    
  • 随机显示位置的子控件

    • 随机点位:保证不重叠;

    对于圆形图,不重叠即保证生成的新点与已存在的点距离大于两者半径和 随机生成点,计算其与已存在在的控件比较,不符合条件,重新生成,超出重试次数即认为无足够空间.

        /**
     * 获取随机点,根据父子空间半径判断随机点是否有足够空间可用
     * @param rad 子控件半径
     * @param radMax 父布局半径
     * @param centerX 父布局中点 x
     * @param centerY 父布局中点 y
     * @return 一个可用的随机点位
     */
    private Point getRandomPoint(int rad, int radMax, int centerX, int centerY) {
        int x = 0;
        int y = 0;
        int counter = 0;
        boolean conflict;
        Random random = new Random();
        while (counter < 100000) {
            counter++;
            x = (int) (getMeasuredWidth() / 2 - radMax + rad + random.nextInt( 2 * radMax - 2 * rad));
            y = (int) (getMeasuredHeight() / 2 - radMax + rad + random.nextInt( 2 * radMax - 2 * rad));
            conflict = false;
            // 判断与其他子控件是否冲突
            for (Point p : mChildrenPoints) {
                int dis = (int) Math.sqrt(Math.pow(p.x - x, 2) + Math.pow(p.y - y, 2));
                if (dis < 2 * rad * (1 + mBlankPer)) {//子控件大小一致,设置两个子控件中点距离最小为 半径和的(1+mBlankPer)倍
                    conflict = true;
                    break;
                }
            }
            if (!conflict) {// 判断与中心点冲突
                int dis = (int) Math.sqrt(Math.pow(centerX - x, 2) + Math.pow(centerY - y, 2));
                if (dis < (rad + mCenterRadius) * ((1 + mBlankPer))) conflict = true;
            }
            if (!conflict) return new Point(x, y);
        }
        Log.e(TAG, "getRandomPoint: 无剩余空间");
        return new Point(0, 0);
    }
    
    • 测量大小:交给子控件自己测量,保证最大值不超过布局的宽高较小值即可
     @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int contentWidth = MeasureSpec.getSize(widthMeasureSpec);
        int contentHeight = MeasureSpec.getSize(heightMeasureSpec);
        int len = Math.min(contentWidth, contentHeight);
        measureChildren(MeasureSpec.makeMeasureSpec(len, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(len, MeasureSpec.AT_MOST));
    }
    
    • 布局 onLayout()
            for (int i = 1; i < getChildCount(); i++) {
            View view = getChildAt(i);
            int rad = Math.max(view.getMeasuredWidth(), view.getMeasuredHeight()) / 2;
            
            Point position;
    		// 数据只支持末尾插入,可直接通过列表长度判断是否是新增点
            if (i > mChildrenPoints.size()) {// 新增的控件,设置随机点
                position = getRandomPoint(rad, mRadiusMax, getMeasuredWidth() / 2, getMeasuredHeight() / 2);
                mChildrenPoints.add(position);
            } else {//已存在控件,直接从列表取
                position = mChildrenPoints.get(i - 1);
            }
            int x = position.x;
            int y = position.y;
            view.layout(x - rad, y - rad, x + rad, y + rad);
        }
    

大功告成

至此为止,这个小布局的核心功能就完成了


 

相关文章
|
8月前
|
资源调度 前端开发 JavaScript
前端宝藏图:寻找技术之旅的星辰大海
前端宝藏图:寻找技术之旅的星辰大海
53 4
|
7月前
|
定位技术 图形学
【用unity实现100个游戏之1】制作类元气骑士、挺进地牢——俯视角射击游戏多种射击效果(一)(附源码)
【用unity实现100个游戏之1】制作类元气骑士、挺进地牢——俯视角射击游戏多种射击效果(一)(附源码)
175 0
|
7月前
|
Java 关系型数据库 MySQL
浪漫编码:手把手教你实现校园表白墙功能
浪漫编码:手把手教你实现校园表白墙功能
98 0
|
7月前
|
应用服务中间件 Apache 数据库
校园表白墙源码LoveWall
LoveWall V2.0Pro是款社区型表白墙,提供点赞、评论、发弹幕、多校区支持及分享功能。环境需Centos7+/Windows Server 2008+、宝塔面板、Apache或Nginx、PHP7.1+及数据库5.6+。
138 0
|
8月前
|
人工智能 Linux C#
八宫格丨九宫格丨淘金镇丨潮玩元宇宙大逃杀游戏系统开发指南步骤及稳定版
uSurvival - the new Multiplayer Survival Asset from the creator of uMMORPG.
|
JavaScript 前端开发 程序员
【中秋征文】手把手教你海面月亮升起中秋节特效制作
【中秋征文】手把手教你海面月亮升起中秋节特效制作
242 0
【中秋征文】手把手教你海面月亮升起中秋节特效制作
|
安全
在淘宝养神兽|这种小鸟比大熊猫还稀少,你竟供养了14只?
在淘宝养神兽|这种小鸟比大熊猫还稀少,你竟供养了14只?
162 0
在淘宝养神兽|这种小鸟比大熊猫还稀少,你竟供养了14只?
程序媛比码大赛,不秀美颜照,秀代码图
女人是上帝的艺术品,那么会写代码的女人是什么?
4038 0
|
新零售
案例酷 | 领克汽车:车市下滑,它凭什么逆势上扬?
2018年末开始,寒潮席卷中国汽车市场。热了20多年的中国车市,迎来首次销量下滑。 当整车销售从增量市场转为存量市场时,车企面临产能过剩和消费升级的双重挑战。一边是日益发展的科技社会和在此环境下迅速成长的新消费人群,一边却是传统的销售体系,市场这个看不见的手,正在逼迫车企转型。
523 0