使用 Gif Lib 加載 Gif 圖

一允坚、GIF 的文件格式

What's a GIF 文章

GIF 格式解析

GIF 實(shí)際是一種壓縮文件,采用 LZW 壓縮算法進(jìn)行編碼。GIF 格式的文件結(jié)構(gòu)整體上分為文件頭,GIF 數(shù)據(jù)流和文件結(jié)尾铐维。所有的顏色由一個(gè)全局顏色列表管理柬泽,當(dāng) GIF 要顯示顏色的時(shí)候,會(huì)通過索引查找顏色列表中的值.

[圖片上傳失敗...(image-e5d5e5-1580903437592)]

[圖片上傳失敗...(image-fbb60f-1580903437592)]

二嫁蛇、集成 GIF LIB

將 android 自帶的 gif_lib c 文件集成到工程中

cmake_minimum_required(VERSION 3.4.1)
aux_source_directory(. src_files)

add_library( # Sets the name of the library.
        native-lib
        SHARED
        src/main/cpp/gif_main.cpp
        ${src_files}
        )
find_library(
        jnigraphics-lib
        jnigraphics)

target_link_libraries(native-lib
        log
        ${jnigraphics-lib}
        )

native 代碼

//
// Created by Apple on 2020-02-04.
//

#include <jni.h>
#include <string>
#include "gif_lib.h"
#include <android/log.h>
#include <android/bitmap.h>

#define  LOG_TAG    "GifDecoder"
#define  LOGE(...)  __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)

#define  argb(a, r, g, b) ( ((a) & 0xff) << 24 ) | ( ((b) & 0xff) << 16 ) | ( ((g) & 0xff) << 8 ) | ((r) & 0xff)
typedef struct GifBean {
    // 當(dāng)前播放的幀
    int current_frame;
    // 總的幀數(shù)
    int total_frame;
    // 每一幀的間隔時(shí)間 指針數(shù)組
    int *dealys;
};

/**
 * 繪制一幀圖像
 * @param gifFileType
 * @param gifBean
 * @param info
 * @param pixels
 */
void drawFrame(GifFileType *gifFileType, GifBean *gifBean, AndroidBitmapInfo info, void *pixels) {
    ......
    // 當(dāng)前幀的圖像信息
    GifImageDesc imageInfo = savedImage.ImageDesc;
    // 拿到圖像數(shù)組的首地址
    int *px = (int *) pixels;
    // 圖像顏色表
    ColorMapObject *colorMap = imageInfo.ColorMap;
    if (colorMap == nullptr) {
        colorMap = gifFileType->SColorMap;
    }
    // y 方向偏移量
    px = (int *) ((char *) px + info.stride * imageInfo.Top);
    // 像素點(diǎn)的位置
    int pointPixel;
    GifByteType gifByteType;//壓縮數(shù)據(jù)
    //    每一行的首地址
    int *line;
    for (int y = imageInfo.Top; y < imageInfo.Top + imageInfo.Height; ++y) {
        line = px;
        for (int x = imageInfo.Left; x < imageInfo.Left + imageInfo.Width; ++x) {
            pointPixel = (y - imageInfo.Top) * imageInfo.Width + (x - imageInfo.Left);
            // 通過 LWZ 壓縮算法拿到當(dāng)前數(shù)組的值
            gifByteType = savedImage.RasterBits[pointPixel];
            GifColorType gifColorType = colorMap->Colors[gifByteType];
            // 將 color type 轉(zhuǎn)換成 argb 的值
            line[x] = argb(255, gifColorType.Red, gifColorType.Green, gifColorType.Blue);
        }
        // 更新到下一行
        px = (int *) ((char *) px + info.stride);
    }
}

extern "C"
JNIEXPORT jlong JNICALL
Java_com_baidu_crazyorange_gifdecoder_gif_GifDecoder_loadGIF(JNIEnv *env, jclass type,
                                                             jstring path_) {
    const char *path = env->GetStringUTFChars(path_, 0);
    int errorCode = 0;
    // 根據(jù)路徑獲取 Gif 文件的結(jié)構(gòu)
    GifFileType *gifFileType = DGifOpenFileName(path, &errorCode);
    // Gif 結(jié)構(gòu)初始化锨并,會(huì)填充上面讀取的內(nèi)容到 GifFileType 對象中
    DGifSlurp(gifFileType);
    // 給結(jié)構(gòu)體 分配內(nèi)存空間
    GifBean *gifBean = (GifBean *) malloc(sizeof(GifBean));
    memset(gifBean, 0, sizeof(GifBean));

    /**
     * 給 GifBean 賦值
     */
    .....
    // 圖形拓展塊
    ExtensionBlock *extensionBlock;

    for (int i = 0; i < gifFileType->ImageCount; ++i) {
        // 取出 GIF 中的每一幀
        SavedImage frame = gifFileType->SavedImages[i];
        // 拿到每一幀的拓展塊
        for (int j = 0; j < frame.ExtensionBlockCount; ++j) {
            if (frame.ExtensionBlocks[j].Function == GRAPHICS_EXT_FUNC_CODE) {
                extensionBlock = &frame.ExtensionBlocks[j];
                break;
            }
            if (extensionBlock) {
                // 拿到圖形控制拓展塊和當(dāng)前幀的延時(shí)時(shí)間
                // 因?yàn)樗?Bytes 是小端字節(jié)序 延時(shí)單位是 10 ms
                // Bytes[0] 是保留字段
                // Bytes[1] 是低 8 位
                // Bytes[2] 表示高 8 位
                int frame_delay = (extensionBlock->Bytes[2] << 8 | extensionBlock->Bytes[1]) * 10;
                gifBean->dealys[i] = frame_delay;
            }
        }
    }
    gifBean->total_frame = gifFileType->ImageCount;
    // 把這個(gè)數(shù)據(jù)交給 gifFileType 保存
    gifFileType->UserData = gifBean;

    env->ReleaseStringUTFChars(path_, path);
    return (jlong) gifFileType;
}
extern "C"
JNIEXPORT jint JNICALL
Java_com_baidu_crazyorange_gifdecoder_gif_GifDecoder_getGifWidth(JNIEnv *env, jclass type,
                                                                 jlong gifPointer) {

    GifFileType *gifFileType = (GifFileType *) (gifPointer);

    return gifFileType->SWidth;
}
extern "C"
JNIEXPORT jint JNICALL
Java_com_baidu_crazyorange_gifdecoder_gif_GifDecoder_getGifHeight(JNIEnv *env, jclass type,
                                                                  jlong gifPointer) {

    GifFileType *gifFileType = (GifFileType *) (gifPointer);

    return gifFileType->SHeight;
}


extern "C"
JNIEXPORT jint JNICALL
Java_com_baidu_crazyorange_gifdecoder_gif_GifDecoder_displayGif(JNIEnv *env, jclass type,
                                                                jobject bitmap, jlong gifPointer) {

    GifFileType *gifFileType = (GifFileType *) (gifPointer);
    GifBean *gifBean = (GifBean *) gifFileType->UserData;
    // 使用 AndroidBitmap 渲染 Bitmap
    AndroidBitmapInfo info;
    AndroidBitmap_getInfo(env, bitmap, &info);
    // 鎖定 Bitmap pixels 是創(chuàng)建一個(gè)像素?cái)?shù)組,用來裝 Gif 中的像素元素
    void *pixels;
    AndroidBitmap_lockPixels(env, bitmap, &pixels);
    drawFrame(gifFileType, gifBean, info, pixels);
    gifBean->current_frame += 1; // 繪制當(dāng)前幀后加 1
    // 如果當(dāng)前幀已經(jīng)是最后一幀睬棚,代表它已經(jīng)播完了琳疏,繼續(xù)播放
    ......
    AndroidBitmap_unlockPixels(env, bitmap);
    return gifBean->dealys[gifBean->current_frame];
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市闸拿,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌书幕,老刑警劉巖新荤,帶你破解...
    沈念sama閱讀 222,590評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異台汇,居然都是意外死亡苛骨,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,157評論 3 399
  • 文/潘曉璐 我一進(jìn)店門苟呐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來痒芝,“玉大人,你說我怎么就攤上這事牵素⊙铣模” “怎么了?”我有些...
    開封第一講書人閱讀 169,301評論 0 362
  • 文/不壞的土叔 我叫張陵笆呆,是天一觀的道長请琳。 經(jīng)常有香客問我,道長赠幕,這世上最難降的妖魔是什么俄精? 我笑而不...
    開封第一講書人閱讀 60,078評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮榕堰,結(jié)果婚禮上竖慧,老公的妹妹穿的比我還像新娘。我一直安慰自己逆屡,他們只是感情好圾旨,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,082評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著康二,像睡著了一般碳胳。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上沫勿,一...
    開封第一講書人閱讀 52,682評論 1 312
  • 那天挨约,我揣著相機(jī)與錄音味混,去河邊找鬼。 笑死诫惭,一個(gè)胖子當(dāng)著我的面吹牛翁锡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播夕土,決...
    沈念sama閱讀 41,155評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼馆衔,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了怨绣?” 一聲冷哼從身側(cè)響起角溃,我...
    開封第一講書人閱讀 40,098評論 0 277
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎篮撑,沒想到半個(gè)月后减细,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,638評論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡赢笨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,701評論 3 342
  • 正文 我和宋清朗相戀三年未蝌,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片茧妒。...
    茶點(diǎn)故事閱讀 40,852評論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡萧吠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出桐筏,到底是詐尸還是另有隱情纸型,我是刑警寧澤,帶...
    沈念sama閱讀 36,520評論 5 351
  • 正文 年R本政府宣布九昧,位于F島的核電站绊袋,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏铸鹰。R本人自食惡果不足惜癌别,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,181評論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望蹋笼。 院中可真熱鬧展姐,春花似錦、人聲如沸剖毯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,674評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽逊谋。三九已至擂达,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間胶滋,已是汗流浹背板鬓。 一陣腳步聲響...
    開封第一講書人閱讀 33,788評論 1 274
  • 我被黑心中介騙來泰國打工悲敷, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人俭令。 一個(gè)月前我還...
    沈念sama閱讀 49,279評論 3 379
  • 正文 我出身青樓后德,卻偏偏與公主長得像,于是被迫代替她去往敵國和親抄腔。 傳聞我的和親對象是個(gè)殘疾皇子瓢湃,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,851評論 2 361

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

  • 版本記錄 前言 只要是做圖片的或者與圖片相關(guān)的悟耘,那么圖片的格式就是一個(gè)不可以繞過的問題藏雏,我們見過很多的圖片格式,但...
    刀客傳奇閱讀 4,908評論 0 7
  • 前言 本文參考gif 格式圖片詳細(xì)解析作煌。加入了一些自己的理解和解析方面的示例。 GIF格式解析 圖像互換格式(GI...
    oceanLong閱讀 16,541評論 4 26
  • [[圖片上傳失敗...(image-cae8e3-1541926581609)]](file://C:\Users...
    沉默的大多數(shù)1876閱讀 8,869評論 0 12
  • 身邊的朋友都考了科二,自己特別害怕報(bào)名起意,也害怕考試鹰服,但是又不能不考 頭一次對考試這件事這么緊張,以前在學(xué)校每周都有...
    陳培巖閱讀 155評論 1 0
  • 窗外的風(fēng)輕柔,有些許的陽光透進(jìn)房間亲善,條線似的跟琴弦一樣设易。風(fēng),無色無味蛹头,陽光的琴弦也奏不出聲音顿肺,他發(fā)掘所有的潛能,努...
    瘋哥哥l閱讀 4,190評論 115 80