優質Android小部件:索尼滾動相冊

来源:http://www.cnblogs.com/willhua/archive/2016/09/11/5862386.html
-Advertisement-
Play Games

雖然騷尼手機賣的不怎麼樣,但是有些東西還是做的挺好的,工業設計就不用說了,索尼的相冊的雙指任意縮放功能也是尤其炫酷。其桌面小部件滾動相冊我覺得也挺好的,比谷歌原生的相冊牆功能好多了,網上搜了一下也沒發現有人寫這個,於是,下麵就介紹下我的高A貨。 首先是效果圖: 主要手勢操作有: 該小部件的主要優點: ...


  雖然騷尼手機賣的不怎麼樣,但是有些東西還是做的挺好的,工業設計就不用說了,索尼的相冊的雙指任意縮放功能也是尤其炫酷。其桌面小部件滾動相冊我覺得也挺好的,比谷歌原生的相冊牆功能好多了,網上搜了一下也沒發現有人寫這個,於是,下麵就介紹下我的高A貨。

首先是效果圖:

    

主要手勢操作有:

  1. 上/下滿速移動,可以上滑/下滑一張圖片
  2. 上/下快讀移動,則根據滑動速度,上滑/下滑多張圖片
  3. 單擊則請求系統圖庫展示該圖片

  該小部件的主要優點:在屏幕內的小範圍內提供一個很好的圖片選擇/瀏覽部件,尤其是切換圖片時有很強的靠近/遠離動畫感,增加好感。

代碼分析

  剛開始想這個小部件的時候以為是利用多個ImageView疊加實現的效果,例如谷歌原生的該部件就是利用多個ImageView疊加形成的,但是效果遠比不上這個。但覺得通過多個ImageView疊加可能會沒這麼流暢,性能上也不好。該效果本身也比較規律,應該可以通過一個View來實現,達到更好的性能。於是通過View Hierarchy分析,sony這個果然是通過一個View實現的,於是通過如下方式這個小部件。

  代碼主要由三個部分組成:

  • RollImageView:實際的View
  • CellCalculater:用來實時計算每張圖片的繪製區域以及透明度,這個是本小部件的核心部件。介面定義如下:  
    /**
     * get all rects for drawing image
     * @return
     */
    public Cell[] getCells();

    /**
     *
     * @param distance the motion distance during the period from ACTION_DOWN to this moment
     * @return 0 means no roll, positive number means roll forward and negative means roll backward
     */
    public int setStatus(float distance);


    /**
     * set the dimen of view
     * @param widht
     * @param height
     */
    public void setDimen(int widht, int height);

    /**
     * set to the status for static
     */
    public void setStatic();
View Code
  • ImageLoader:用來載入圖片,提供Bitmap給RollImageView繪製。介面定義如下:  
    /**
     * the images shown roll forward
     */
    public void rollForward();

    /**
     * the images shown roll backward
     */
    public void rollBackward();

    /**
     * get bitmaps
     * @return
     */
    public Bitmap[] getBitmap();

    /**
     * use invalidate to invalidate the view
     * @param invalidate
     */
    public void setInvalidate(RollImageView.InvalidateView invalidate);

    /**
     * set the dimen of view
     * @param width
     * @param height
     */
    public void setDimen(int width, int height);


    /**
     * the image path to be show
     * @param paths
     */
    public void setImagePaths(List<String> paths);

    /**
     * get large bitmap while static
     */
    public void loadCurrentLargeBitmap();
View Code

  下麵分析每個部分的核心代碼。

RollImageView

  View的主要職責是draw各個bitmap以及響應用戶的手勢操作,相對比較簡單。

  繪製部分就是把從ImageLoader獲得的的各個Bitmap按照從CellCalculater中獲得的繪製區域以及透明度繪製到屏幕上,目前本代碼實現的比較簡單,沒有考慮不同尺寸的圖片需要進行一些更加協調的顯示方式,比如像ImageView.ScaleType中定義的一些顯示方式。  

    @Override
    public void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        Bitmap[] bitmaps = mImageLoader.getBitmap();
        Cell[] cells = mCellCalculator.getCells();  //得到每張Image的顯示區域與透明度
        canvas.translate(getWidth() / 2, 0);
        for (int i = SHOW_CNT - 1; i >= 0; i--) { //從最底層的Image開始繪製
            Bitmap bitmap = bitmaps[i];
            Cell cell = cells[i];
            if (bitmap != null && !bitmap.isRecycled()) {
                mPaint.setAlpha(cell.getAlpha());
                LOG("ondraw " + i + bitmap.getWidth() + " " + cell.getRectF() + " alpha " + cell.getAlpha());
                canvas.drawBitmap(bitmap, null, cell.getRectF(), mPaint);
            }
        }
    }

  手勢部分採用了GestureListener,主要代碼如下:  

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (event.getPointerCount() > 1) {
            return false;
        }
        mGestureDetector.onTouchEvent(event);
        switch (event.getAction()) {
            case MotionEvent.ACTION_UP:  //這裡主要用於處理沒有觸發Fling事件時,使界面保持沒有移動的狀態
                if(!mIsFling){
                    if(mRollResult == CellCalculator.ROLL_FORWARD){
                        mImageLoader.rollForward();
                    } else if (mRollResult == CellCalculator.ROLL_BACKWARD && !mScrollRollBack){
                        mImageLoader.rollBackward();
                    }
                    LOG("OnGestureListener ACTION_UP setstatic " );
                    mCellCalculator.setStatic();
                    mImageLoader.loadCurrentLargeBitmap();
                }
                break;
            default:
                break;
        }
        return true;
    }

    
    //緩慢拖動
    @Override
    public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
        mScrollDistance += distanceY;
        if(mScrollDistance > 0 && !mScrollRollBack){
            mImageLoader.rollBackward();
            mScrollRollBack = true;
        } else if(mScrollDistance < 0 && mScrollRollBack){
            mImageLoader.rollForward();
            mScrollRollBack = false;
        }
        LOG("OnGestureListener onScroll " + distanceY + " all" + mScrollDistance);
        mRollResult = mCellCalculator.setStatus(-mScrollDistance);
        invalidate();
        return true;
    }
    
    //快速拖動
    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
        if (Math.abs(velocityY) > MIN_FLING) {
            LOG("OnGestureListener onFling " + velocityY);
            if (mExecutorService == null) {
                mExecutorService = Executors.newSingleThreadExecutor();
            }
            mIsFling = true;
            mExecutorService.submit(new FlingTask(velocityY));
        }
        return true;
    }
    
    //利用一個非同步任務來處理滾動多張Images
    private class FlingTask implements Runnable {

        float mVelocity;
        float mViewHeight;
        int mSleepTime;
        boolean mRollBackward;

        FlingTask(float velocity) {
            mRollBackward = velocity < 0 ? true : false;
            mVelocity = Math.abs(velocity / 4);
            mViewHeight = RollImageView.this.getHeight() / 2;
            mSleepTime = (int)(4000 / Math.abs(velocity) * 100); //the slower velocity of fling, the longer interval for roll
        }

        @Override
        public void run() {
            int i = 0;
            try{
                while (mVelocity > mViewHeight) {
                    mCellCalculator.setStatus(mRollBackward ? -mViewHeight : mViewHeight);
                    mHandler.sendEmptyMessage(MSG_INVALATE);
                    //determines the count of roll. The using of mViewHeight has no strictly logical
                    mVelocity -= mViewHeight;
                    if (((i++) & 1) == 0) { //roll forward once for every two setStatus
                        if(mRollBackward){
                            mImageLoader.rollBackward();
                        }else {
                            mImageLoader.rollForward();
                        }
                    }
                    Thread.sleep(mSleepTime);
                }
                mCellCalculator.setStatic();
                mImageLoader.loadCurrentLargeBitmap();
                mHandler.sendEmptyMessage(MSG_INVALATE);
            } catch(Exception e){

            } finally{

            }
        }
    }
View Code

 CellCalculater分析

  首先闡明下向前移動/向後移動的概念。需要顯示的圖片路徑存儲為一個List,假設顯示在最前的圖片的索引為index,則當前顯示的圖片為[index,index+3],向前則表示index加1,向後則表示index減1.

  CellCalculater的計算情形主要在於用戶通過手勢操作,表達了需要向前或者向後移動一張圖片的意圖。在View中能夠獲取到的只是手勢移動的距離,所以在CellCalculater中需要對傳進來的移動距離進行處理,輸出移動結果。在我的實現中,當移動距離超過圖片高度一半的時候,就表示顯示的圖片需要移動一位,否則當手勢操作結束的時候就設置為static狀態。主要代碼如下:  

    public DefaultCellCalculator(int showCnt){
        mCnt = showCnt;
        mCells = new Cell[mCnt];
        mAlphas = new float[mCnt];
        STATIC_ALPHA = new int[mCnt];
        STATIC_ALPHA[mCnt - 1] = 0; //最後一張圖的透明度為0
        int alphaUnit = (255 - FIRST_ALPHA) / (mCnt - 2);
        for(int i = mCnt - 2; i >= 0; i--){  //定義靜態時每張圖的透明度
            STATIC_ALPHA[i] = FIRST_ALPHA + (mCnt - 2 - i) * alphaUnit;
        }
    }

    @Override
    public Cell[] getCells() {
        return  mCells;
    }
    
    //用戶手勢移動,distance表示移動距離,正負值分別意味著需要向前/向後移動
    @Override
    public int setStatus(float distance) {
        if(distance > 0){
            return calculateForward(distance);
        } else if(distance < 0){
            return calculateBackward(distance);
        } else{
            initCells();
        }
        return 0;
    }

    //設置RollImageView的尺寸,從而計算合適的顯示區域
    @Override
    public void setDimen(int widht, int height) {
        mViewWidth = widht;
        mViewHeight = height;
        mWidhtIndent = (int)(WIDHT_INDENT * mViewWidth);
        mWidths = new int[mCnt];
        for(int i = 0; i < mCnt; i++){
            mWidths[i] = mViewWidth - i * mWidhtIndent;
        }
        //每張圖片的高度。
        //假如顯示四張圖,那麼在上面會有三個高度落差,然後最底部保留一個高度落差,所以是mcnt-1
        mImageHeight = mViewHeight - (mCnt - 1) * HEIGHT_INDENT;
        LOG("mImageHeight " + mImageHeight);
        initCells();
    }

    
    //靜態時,即用戶手勢操作結束時
    @Override
    public void setStatic() {
        initCells();
    }

    //用戶有需要向前移動一位的趨勢
    private int calculateForward(float status){
        float scale = status / mImageHeight;
        LOG("scale " + scale + " mImageHeight " + mImageHeight + " status " + status);
        for(int i = 1; i < mCnt; i++){
            mCells[i].setWidth(interpolate(scale * 3, mWidths[i], mWidths[i - 1])); // *3 使得後面的寬度快速增大,經驗值
            mCells[i].moveVertical(interpolate(scale * 10, 0, HEIGHT_INDENT));  //*10使得後面的圖片迅速向前,向前的動畫感更強
            mCells[i].setAlpha((int)interpolate(scale, STATIC_ALPHA[i], STATIC_ALPHA[i - 1]));
        }
        mCells[0].moveVertical(status);
        mCells[0].setAlpha((int)interpolate(scale, 255, 0));
        if(status >= mImageHeight / 3){
            return ROLL_FORWARD;
        } else {
            return 0;
        }
    }

    //用戶有需要向後移動一位的趨勢
    private int calculateBackward(float status){
        float scale = Math.abs(status / mImageHeight);
        for(int i = 1; i < mCnt; i++){
            mCells[i].setWidth(interpolate(scale, mWidths[i - 1], mWidths[i]));
            mCells[i].moveVertical(-scale * HEIGHT_INDENT);
            mCells[i].setAlpha((int)interpolate(scale, STATIC_ALPHA[i - 1], STATIC_ALPHA[i]));
        }
        mCells[0].resetRect();
        mCells[0].setWidth(mWidths[0]);
        mCells[0].setHeight(mImageHeight);
        mCells[0].moveVertical(mImageHeight + status);
        mCells[0].setAlpha((int)interpolate(scale, 0, 255));
        if(-status >= mImageHeight / 3){
            return ROLL_BACKWARD;
        } else {
            return 0;
        }
    }

    /**
     * status without move
     */
    private void initCells(){
        int top = -HEIGHT_INDENT;
        for(int i = 0; i < mCnt; i++){
            RectF rectF = new RectF(0,0,0,0);
            rectF.top = top + (mCnt - 1 - i) * HEIGHT_INDENT;
            rectF.bottom = rectF.top + mImageHeight;
            mCells[i] = new Cell(rectF, STATIC_ALPHA[i]);
            mCells[i].setWidth(mWidths[i]);
        }
    }

    //計算差值
    private float interpolate(float scale, float start, float end){
        if(scale > 1){
            scale = 1;
        }
        return start + scale * (end - start);
    }
View Code

ImageLoader分析

  ImageLoader其實比較簡單,主要有如下兩點:

  • 響應手勢操作,處理對應的向前/向後移動時的Bitmap請求
  • 當手勢還在操作時,應該載入小圖,等手勢操作結束之後,應該載入大圖。因為只有緩慢移動時,需要清晰顯示,而快速移動時,顯示小圖即可,所以需要載入當前index以及向前向後一張圖即可。

  

    //載入當前index以及向前向後三張大圖
    @Override
    public void loadCurrentLargeBitmap() {
        for(int i = mCurrentIndex - 1; i < mCurrentIndex + 2; i++){
            if(i >= 0 && i < mImagesCnt - 1){
                mBitmapCache.getLargeBitmap(mAllImagePaths[i]);
            }
        }
    }

    //index向前移動一位
    @Override
    public void rollForward() {
        LOG("rollForward");
        mCurrentIndex++;
        if(mCurrentIndex > mImagesCnt - 1){
            mCurrentIndex = mImagesCnt - 1;
        }
        setCurrentPaths();
    }

    //index向後移動一位
    @Override
    public void rollBackward() {
        LOG("rollBackward");
        mCurrentIndex--;
        if(mCurrentIndex < 0){
            mCurrentIndex = 0;
        }
        setCurrentPaths();
    }

    @Override
    public Bitmap[] getBitmap() {
        if(mCurrentPaths != null){
            LOG("getBitmap paths nut null");
            for(int i = mCurrentIndex, j = 0; j < mShowCnt; j++, i++){
                if(i >= 0 && i < mImagesCnt){
                    mCurrentBitmaps[j] = mBitmapCache.getBimap(mAllImagePaths[i]);
                } else{
                    mCurrentBitmaps[j] = mBitmapCache.getBimap(NO_PATH);
                }
            }
        }
        return  mCurrentBitmaps;
    }
View Code

 

最後,所有源代碼:https://github.com/willhua/RollImage

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 記錄字元串的處理,不是一個簡單的工作。 NSString是代碼中隨處可見的類型,也是應用和處理繁多的對象,在此只記錄需要常備的方法,並且加以說明。 說明: 1.計算字元串尺寸的方法,sizeWithFont系列方法已經被廢物,建議改為boundingRectWithSize方法;NSAttribut ...
  • 一:原生傳遞參數給React Native 1:原生給React Native傳參 原生給JS傳數據,主要依靠屬性。 通過initialProperties,這個RCTRootView的初始化函數的參數來完成。 RCTRootView還有一個appProperties屬性,修改這個屬性,JS端會調用 ...
  • 前面有同學問到了iOS記憶體警告臨界值和工程項目里AppIcon的一些配置信息,相信對剛入行的同學來說,可能都會碰到類似的問題,記錄一下供後來者查詢。 1、先簡單說下AppIcon的圖標的配置信息 1)、在右邊的App Icon勾選項,暫時只選擇了支持iPhone iOS7.0 and Later, ...
  • 大家看到微信首頁切換效果有沒有覺得很炫,滑動切換,點擊底部bar瞬間切換,滑動切換漸變效果,線上效果圖: 之前也在博客上看到別人的實現,再次基礎上,我做了些優化。首先說下實現原理,大神略過,o(╯□╰)o 頁面上看到的三個頁面是三個Fragment, 左右滑動使用viewpager,相信大家也都是這 ...
  • 一、使用純代碼方式 initWithFrame:中添加子控制項 layoutSubViews中設置子控制項的fame 對外設置數據介面,重寫setter方法給子控制項設置數據顯示 在的viewController裡面使用init/initWithFrame:方法創建自定義類,並且給自定義類的frame賦值 ...
  • 二手交易平臺 我的畢業設計項目安卓源碼,二手交易平臺,dagger2+mvp+Bmob後臺雲搭建,集成了百度地圖,友盟三方登錄等 系統架構 Dagger2+MVP分層,完成了一次正常的retrofit下的天氣信息的請求,其餘請求後臺均基於Bmob雲後臺,圖片在水平方向可滾動 說明 使用請尊重本人技術 ...
  • 視圖基礎 視圖層次結構 任何應用有且只有一個 UIWindow 對象。 UIWindow 對象就像是一個容器,負責包含應用中的所有的視圖。應用需要在啟動時創建並設置 UIWindow 對象,然後為其添加其他視圖。 加入視窗的視圖會成為該視窗的子視圖。視窗的子視圖還可以有自己的子視圖,從而構成一個以  ...
  • 1、下拉列表框(Spinner) 項目佈局 添加相應代碼: 2、輸入內容自動完成文本框(AutoCompleteTextView) AutoCompleteTextView和EditText組件類似,都可以輸入文本。但AutoCompleteTextView組件可以和一個字元串數組或List對象綁定 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...