SVG矢量图打造不规则自定义控件,可点击的中国地图

1、SVG概念:

SVG是一种图像文件格式,类似PNG,JPG。只不过PNG这种图片需要图像引擎加载,SVG则是由画布来加载,它的英文全称为Scalable Vector Graphics,意思为可缩放的矢量图形,可让你设计无损失、高分辨率的Web图形页面,用户可以直接使用代码来描绘图像;


2、SVG图像在Android中的使用

app图标:sdk23以后,app的图标都是由svg图像来表示

自定义控件:如不规则控件、复杂的交互控件、子控件重叠判断、图标等,都可以使用SVG图像实现

复杂动画:如根据用户滑动手势动态显示动画,路径动画等

3、实现中国地图的绘制,并且能正常点击省份

效果展示:


中国地图svg图像下载地址:http://www.amcharts.com/download/


点击第二个download按钮即可下载所有国家svg图像;

此下载比较费劲,我分享一个网盘地址:

链接 https://pan.baidu.com/s/1zbtejuTYhSL2ino8soOgRw  提取码:mung

下载成功后将文件导入工程res/raw/china.svg

自定义view代码:



package com.xxx.uidemo;

import android.annotation.SuppressLint;

import android.content.Context;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.Paint;

import android.graphics.Path;

import android.graphics.RectF;

import android.os.Handler;

import android.os.Looper;

import android.support.v4.graphics.PathParser;

import android.util.AttributeSet;

import android.view.MotionEvent;

import android.view.View;

import org.w3c.dom.Document;

import org.w3c.dom.Element;

import org.w3c.dom.NodeList;

import java.io.InputStream;

import java.util.ArrayList;

import java.util.List;

import javax.xml.parsers.DocumentBuilder;

import javax.xml.parsers.DocumentBuilderFactory;

public class ChinaMapView extends View {

    private int[] colorArrays = new int[]{0xFF239BD7, 0xFF30A9E5, 0xFF80CBF1, 0xFFFFFF00};

    private List<ProvinceItem> provinceItems = new ArrayList<>();// 所有省份

    private Paint paint;

    private ProvinceItem selectItem;// 点击选中的省份

    private RectF totalRectF;// 地图矩形

    private float scale = 1.0f;// 画布缩放系数

    public ChinaMapView(Context context) {

        this(context, null);

    }

    public ChinaMapView(Context context, AttributeSet attrs) {

        this(context, attrs, 0);

    }

    public ChinaMapView(Context context, AttributeSet attrs, int defStyleAttr) {

        super(context, attrs, defStyleAttr);

        init();

        parseSVG(context);

    }

    private void parseSVG(final Context context) {

        final InputStream inputStream = context.getResources().openRawResource(R.raw.china);

        new Thread(new Runnable() {

            @Override

            public void run() {

                try {

                    DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();

                    DocumentBuilder documentBuilder = factory.newDocumentBuilder();

                    Document parse = documentBuilder.parse(inputStream);

                    Element documentElement = parse.getDocumentElement();

                    NodeList g = documentElement.getElementsByTagName("path");

                    float left = -1;

                    float right = -1;

                    float top = -1;

                    float bottom = -1;

                    ArrayList<ProvinceItem> list = new ArrayList<>();

                    for (int i = 0; i < g.getLength(); i++) {

                        Element element = (Element) g.item(i);

                        String pathData = element.getAttribute("d");

                        @SuppressLint("RestrictedApi") Path path = PathParser.createPathFromPathData(pathData);

                        ProvinceItem provinceItem = new ProvinceItem(path);

                        provinceItem.setColor(colorArrays[i % 4]);

                        // 这里循环遍历每个省份path的边界,并求得最左边、右边、顶部、底部path的边界

                        RectF rectF = new RectF();

                        path.computeBounds(rectF, true);

                        left = left == -1 ? rectF.left : Math.min(left, rectF.left);

                        right = right == -1 ? rectF.right : Math.max(right, rectF.right);

                        top = top == -1 ? rectF.top : Math.min(top, rectF.top);

                        bottom = bottom == -1 ? rectF.bottom : Math.max(bottom, rectF.bottom);

                        list.add(provinceItem);

                    }

                    provinceItems = list;// 避免priviceItems集合并发操作,先使用临时集合,然后再重新赋值

                    totalRectF = new RectF(left, top, right, bottom);//保存实际地图大小

                    // 通知刷新界面

                    Handler handler = new Handler(Looper.getMainLooper());

                    handler.post(new Runnable() {

                        @Override

                        public void run() {

                            requestLayout();

                            invalidate();

                        }

                    });

                    //postInvalidate();

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

        }).start();

    }

    private void init() {

        paint = new Paint();

        paint.setColor(Color.BLACK);

        paint.setAntiAlias(true);

    }

    @Override

    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        // 获取画布原始大小

        int measuredWidth = MeasureSpec.getSize(widthMeasureSpec);

        int measuredHeight = MeasureSpec.getSize(heightMeasureSpec);

        // 计算缩放系数

        if (totalRectF != null) {

            float width = totalRectF.width();

            scale = measuredWidth / width;

        }

    }

    @Override

    protected void onDraw(Canvas canvas) {

        super.onDraw(canvas);

        if (provinceItems != null && provinceItems.size() > 0) {

            canvas.save();

            //按照缩放系数将画布进行缩放

            canvas.scale(scale, scale);

            for (ProvinceItem provinceItem : provinceItems) {

                if (provinceItem == selectItem) {

                    provinceItem.DrawItem(canvas, paint, true);

                } else {

                    provinceItem.DrawItem(canvas, paint, false);

                }

            }

        }

    }

    @Override

    public boolean onTouchEvent(MotionEvent event) {

        handleTouch(event.getX() / scale, event.getY() / scale);// 点击触摸时也需要将事件位置进行缩放,不然点击事件会受影响

        return super.onTouchEvent(event);

    }

    private void handleTouch(float x, float y) {

        if (provinceItems == null) {

            return;

        }

        ProvinceItem select = null;

        for (ProvinceItem provinceItem : provinceItems) {

            if (provinceItem.isTouch(x, y)) {

                select = provinceItem;

            }

        }

        if (select != null) {

            selectItem = select;

            postInvalidate();

        }

    }

}



package com.xxx.uidemo;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.Paint;

import android.graphics.Path;

import android.graphics.RectF;

import android.graphics.Region;

/**

* 省份

*/

public class ProvinceItem {

    private Path path;

    private int color;

    public ProvinceItem(Path path) {

        this.path = path;

    }

    public void setColor(int color) {

        this.color = color;

    }

    /**

    * 绘制自己

    *

    * @param canvas

    * @param paint

    * @param isSelect

    */

    public void DrawItem(Canvas canvas, Paint paint, boolean isSelect) {

        if (isSelect) {

            // 绘制内部的颜色

            paint.clearShadowLayer();

            paint.setStrokeWidth(1);

            paint.setStyle(Paint.Style.FILL);

            paint.setColor(color);

            canvas.drawPath(path, paint);

            // 绘制边界

            paint.setColor(0xFFD0E8F4);

            paint.setStyle(Paint.Style.STROKE);

            canvas.drawPath(path, paint);

        } else {

            // 绘制内部的颜色

            paint.clearShadowLayer();

            paint.setStrokeWidth(2);

            paint.setStyle(Paint.Style.FILL);

            paint.setColor(Color.BLACK);

            paint.setShadowLayer(8, 0, 0, 0xFFFFFF);

            canvas.drawPath(path, paint);

            // 绘制边界

            paint.setColor(color);

            paint.setStyle(Paint.Style.FILL);

            paint.setStrokeWidth(2);

            canvas.drawPath(path, paint);

        }

    }

    /**

    * 按下坐标坐标是否在path轨迹和path对应的矩形轨迹的交集中

    *

    * @param x

    * @param y

    * @return

    */

    public boolean isTouch(float x, float y) {

        RectF rectF = new RectF();

        path.computeBounds(rectF, true);//计算path轨迹对应的矩形界限

        Region region = new Region();

        // 计算path对应的轨迹和path轨迹对应的矩形界限的范围交集

        region.setPath(path, new Region((int) rectF.left, (int) rectF.top, (int) rectF.right, (int) rectF.bottom));

        return region.contains((int) x, (int) y);// 坐标值是否包含在此范围中

    }

}



使用布局加载地图

<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:app="http://schemas.android.com/apk/res-auto"

    android:layout_width="match_parent"

    android:layout_height="match_parent">

    <com.xxx.uidemo.ChinaMapView

        android:layout_width="match_parent"

        android:layout_height="match_parent" />

</RelativeLayout>



里面有详细备注,基本思路也就是先解析svg文件,然后依据解析结果画出地图;



关键步骤:

1、解析svg;

2、绘制自己

3、判断点击坐标是否属于此path轨迹范围内;

4、解析过程中获取地图实际大小;

5、计算画布和地图 缩放比并按比例缩放画布;

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容