Android中使用SVG矢量圖打造多邊形圖形框架

Android中使用SVG矢量圖打造多邊形圖形框架#

最后要實現(xiàn)的最終效果如下圖:


1.準備工作

(1) 因為項目中需要用到SVG矢量圖,所以我們就需要一個將SVG圖像轉換為Bitmap的工具類:

CSDN資源下載[jar包]:http://www.123.com

Github源碼下載:https://github.com/pents90/svg-android

如果下載的jar包直接在項目中引用就行,如果是源碼只需要將源碼放在項目中的包下即可,看看我AndroidStudio項目中源碼文件:

(2) 準備SVG矢量圖片,不清楚SVG圖片的童鞋可以到W3CSchool上面去了解下; 如果找不到SVG矢量圖的素材可以在 SVG在線編輯 上自己編輯一個SVG圖片。###

準備好的SVG圖片可以放在 res/raw 中(沒有就手動創(chuàng)建),也可以放在項目的asset中(沒有手動創(chuàng)建)

目錄 對應獲取SVG圖片的方法
raw SVGParser.getSVGFromInputStream()
asset SVGParser.getSVGFromAsset()

2.SVG加載到Android中

現(xiàn)在開始正式擼代碼 首先在項目中創(chuàng)建一個工具類:SvgBitmapUtils

在類中創(chuàng)建靜態(tài)方法getSvgBitmap

    /**
     * 主要是用來加載SVG矢量圖片,最后轉換成我們屬性的Bitmap格式
     * @param context
     * @param width SVG圖片的寬
     * @param height SVG圖片的高
     * @param resId SVG圖片的ID
     * @return Bitmap
     */
    public static Bitmap getSvgBitmap(Context context , int width,int height ,int resId){
        //1.創(chuàng)建一個空的圖片
        Bitmap bitmap = Bitmap.createBitmap(width,height, Bitmap.Config.ARGB_8888);
        //2.拿到空圖片的畫布:目前來講相當于一張白紙
        Canvas canvas = new Canvas(bitmap);
        //創(chuàng)建一直畫筆
        Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        paint.setDither(true);//防止畫筆抖動
        paint.setColor(Color.BLACK);//畫筆顏色為黑色'

        //當傳入的id是一個正常的ID
        if (resId>0){
            // 3.通過SVGParser來加載一個SVG圖片
            // 加載raw目錄下的SVG文件
            SVG svg = SVGParser.getSVGFromInputStream(context.getResources().openRawResource(resId),width,height);
            // 4.將通過svg對象的getPicture()方法得到一個Picture對象,然后將Picture畫在畫布上
            canvas.drawPicture(svg.getPicture());
        }else {
            //如果沒有找到SVG圖片就畫一個矩形
            canvas.drawRect(new Rect(0,0,width,height),paint);
        }

        return bitmap;
    }

接下來是Svg圖片與妹子的合并,所以首先準備一張妹子圖,因為我們合并和得到一張正方形的合并圖片,所以要先根據(jù)給定的妹子圖得到合并后圖片的寬(size),很明顯就是妹子圖片的寬或高中最小的一邊的長度:

int size = Math.min(fg.getWidth(),fg.getHeight());

然后根據(jù)這個寬(size)從妹子的原圖中切出一張正方形圖片,從原圖中切出一個正方形圖片需要知道切圖的起始位置(<font color="#ff3456">相對與圖片的左上角的位置:x,y</font>)

    //2.確定截取正方形左上角的坐標
    int x = (fg.getWidth()-size)/2;
    int y = (fg.getHeight()-size)/2;

有了開始的Svg矢量圖和現(xiàn)在切出來的妹子圖;就可以開始合并圖片,可是要怎么樣才能合并出上面的那樣的效果;

先來看看官方給出圖片合并模式 來自ApiDemos/Graphics/XferModes):

從上圖可以看到PorterDuff.Mode為枚舉類,一共有16個枚舉值:

枚舉值 對應效果
1.PorterDuff.Mode.CLEAR 所繪制不會提交到畫布上。
2.PorterDuff.Mode.SRC 顯示上層繪制圖片。
3.PorterDuff.Mode.DST 顯示下層繪制圖片。
4.PorterDuff.Mode.SRC_OVER 正常繪制顯示,上下層繪制疊蓋。。
5.PorterDuff.Mode.DST_OVER 上下層都顯示。下層居上顯示。。
6.PorterDuff.Mode.SRC_IN 取兩層繪制交集。顯示上層。。
7.PorterDuff.Mode.DST_IN 取兩層繪制交集。顯示下層。。
8.PorterDuff.Mode.SRC_OUT 取上層繪制非交集部分。。
9.PorterDuff.Mode.DST_OUT 取下層繪制非交集部分。。
10.PorterDuff.Mode.SRC_ATOP 取下層非交集部分與上層交集部分。
11.PorterDuff.Mode.DST_ATOP 取上層非交集部分與下層交集部分。
12.PorterDuff.Mode.XOR 所異或:去除兩圖層交集部分。
13.PorterDuff.Mode.DARKEN 取兩圖層全部區(qū)域,交集部分顏色加深。
14.PorterDuff.Mode.LIGHTEN 取兩圖層全部,點亮交集部分顏色。
15.PorterDuff.Mode.MULTIPLY 取兩圖層交集部分疊加后顏色。
16.PorterDuff.Mode.SCREEN 取兩圖層全部區(qū)域,交集部分變?yōu)橥该魃?/td>

根據(jù)我們的效果需要,很明顯選擇:

[10.PorterDuff.Mode.SRC_ATOP ] 取下層非交集部分與上層交集部分。;

現(xiàn)在我只需要把SVG圖片放下面,然后把妹子放上面,就能得到最終效果!

SVG跟妹子合并方法,也是我們最后在界面中使用的方法getSvgShapBitmap()

   /**
     * SVG跟妹子圖片進行合并
     * @param context
     * @param fg 原圖
     * @param resId 矢量圖的(SVG)ID
     * @return Bitmap
     */
    public static Bitmap getSvgShapBitmap(Context context,Bitmap fg,int resId){
        //1.確定截取原圖正方形的邊長 (以原圖中寬高最小的為邊長)
        int size = Math.min(fg.getWidth(),fg.getHeight());
        //2.確定截取正方形左上角的坐標
        int x = (fg.getWidth()-size)/2;
        int y = (fg.getHeight()-size)/2;
        //裁剪妹子圖片
        Bitmap girl = Bitmap.createBitmap(fg,x,y,size,size);

        //3.獲得和正方形同樣大小的SVG圖片
        Bitmap svg = getSvgBitmap(context,fg.getWidth(),fg.getHeight(),resId);
        //創(chuàng)建畫筆
        Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        paint.setDither(true);//防止抖動

        //4.SVG做底部圖片,妹子放上面,采用SRC_ATOP模式重疊
        Bitmap bitmap = Bitmap.createBitmap(size,size, Bitmap.Config.ARGB_8888);
        Canvas canvas = new Canvas(bitmap);
        canvas.drawBitmap(svg,new Matrix(),paint);
        //設置圖片重合模式
        paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_ATOP));
        canvas.drawBitmap(girl,new Matrix(),paint);
        return bitmap;
    }

3.界面使用,直接顯示到ImageView##

MainActivity中的代碼:

    public class MainActivity extends AppCompatActivity {
    private  Bitmap bitmap;
    private ImageView ivImage;
    private Button btnImage;

    ////加載多邊形SVG
    private static int[] resIds = new int[]{
            R.raw.ba,
            R.raw.fengye,
            R.raw.linxin,
            R.raw.liu,
            R.raw.meihua,
            R.raw.pingguo,
            R.raw.qi,
            R.raw.qizi,
            R.raw.tuzi,
            R.raw.wujiaoxing,
            R.raw.yezi,
    };
    
    int flag = 0;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ivImage = (ImageView) this.findViewById(R.id.iv_image);
        btnImage = (Button) this.findViewById(R.id.btn_image);
        
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.girl);//
        //為了效果先顯示原圖
        ivImage.setImageBitmap(bitmap);
        
        btnImage.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                flag = flag % (resIds.length);
                //得到合并后的圖片
                Bitmap svgShapBitmap = SvgBitmapUtils.getSvgShapBitmap(MainActivity.this, bitmap, resIds[flag]);
                ivImage.setImageBitmap(svgShapBitmap);
                flag++;
            }
        });
    }

代碼簡單,就不多說了,下面還是貼上XML代碼吧

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main"
    android:layout_width="match_parent" android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:background="@color/colorAccent"
    tools:context="zhengliang.com.svg_polygon_frame.MainActivity">
    <ImageView
        android:layout_width="250dp"
        android:layout_height="250dp"
        android:scaleType="centerCrop"
        android:layout_centerInParent="true"
        android:id="@+id/iv_image"/>
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:id="@+id/btn_image"
        android:background="#22FFFFFF"
        android:text="加載妹子"/>
</RelativeLayout>

大工告成,最后小提示(SVG圖片的path路徑盡量使用封閉路徑,不然會出現(xiàn)奇葩效果!!)

印象丶亮仔###

svg-android.jar下載

源碼下載

博客地址:http://blog.csdn.net/qq_23179075/article/details/53285852

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,401評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,011評論 3 413
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,263評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,543評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,323評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,874評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,968評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,095評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,605評論 1 331
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,551評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,720評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,242評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 43,961評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,358評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,612評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,330評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,690評論 2 370

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,580評論 25 707
  • 原文鏈接:https://github.com/opendigg/awesome-github-android-u...
    IM魂影閱讀 32,949評論 6 472
  • 發(fā)現(xiàn) 關注 消息 iOS 第三方庫、插件、知名博客總結 作者大灰狼的小綿羊哥哥關注 2017.06.26 09:4...
    肇東周閱讀 12,142評論 4 61
  • 陳爺爺高壽, 今年九十九歲了。可秋風劉起麒 陳爺爺老是感到心絞痛, 家人趕緊把他送迸了縣醫(yī)院. 主治醫(yī)師牛醫(yī)生說可...
    一路走來不會累閱讀 476評論 0 1
  • 創(chuàng)造,也可以指舞蹈中跳錯的那一步,石頭上鑿壞的那一鑿子。動作的成功與否不是主要的。這種努力在你看來是徒勞無益,這是...
    Sunny飛鏡閱讀 174評論 0 0