• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            專注于服務器編程、網絡編程

            ~~保持一顆平常心~~持之以恒~~
            posts - 18, comments - 7, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            MTK GDI layer 收藏

            Posted on 2009-12-18 22:02 ~William~ 閱讀(825) 評論(0)  編輯 收藏 引用

            MTK GDI layer 收藏
            在某些頻繁更新的界面中,如果某些顯示元素一直沒有變化,我們就可以將這些元素提取出來畫到一個模擬的屏幕中,而將一些需要更新的元素畫到另外的模擬屏幕,而后將兩個模擬屏幕合并到真正的屏幕上,這樣我們就節省了不變元素的重畫時間,從而減輕了系統負擔及加速畫面更新。我們把這樣的模擬屏幕就叫層,也可以說層就是屏幕的緩沖空間。


                例如,如果我們用動畫做為背景,將其他的一些元素也畫到這一層中,就會出現當動畫跳到第二幀后,動畫上面的文本及圖象都會被蓋住。而有了層以后,我們就可以將不變的文本及圖象放到新建的一個層中,將動畫放到背景層中,每當有動畫換幀時,只需將新幀畫到背景層中,然后合并兩個屏幕(動畫刷新時會自動合并),這樣上層的文本等就不會被蓋住了。


               另外,因為層的格式簡單且統一,并且一般的圖形系統都會用硬件加速合并,所以在層合并時加上些特效會很方便,如通透、半透明、剪切等。


            下面將以mmi_phoart_entry_main_screen()為例說明關于層的一些內容。
            1.首先,因為用到兩個層,所以我們要開啟多層
               gdi_layer_multi_layer_enable();
            等到退出該屏幕的時候需要gdi_layer_multi_layer_disable();

            2.創建層:創建層需要先建一個層句柄,我們可以通過該句炳來控制層,函數gdi_layer_create用來創建層
            gdi_layer_create(OFFSET_X,OFFSET_Y,WIDTH,HEIGHT,HANDLE_PTR);
            OFFSET_X,OFFSET_Y是層的位置坐標,WIDTH,HEIGHT是層的大??;HANDLE_PTR是層句炳,用于返回所創建的層。不過,因為創建層時系統要為其分配動態內存空間,而系統保留的內存一般只夠創建一個UI_device_width*UI_device_height大小的層,所以如果已有兩個層,而需要再創建新層時,就需要用到函數gdi_layer_create_using_outside_memory(X,Y,WIDTH,HEIGHT,HANDLE_PTR,OUTMEM_PTR,OUTMEM_SIZE)
            前5個參數和gdi_layer_create的參數相同,OUTMEM_PTR是存放層的BUFFER,OUTMEM_SIZE是層的大小。

            3.激活層:在我們的圖形系統中,任何時刻有且只能有一個層處于激活狀態,所有的繪畫都會默認畫到這個激活層中,所以想要在層上繪畫必須先將其激活。激活函數是gdi_layer_set_active(gdi_handle handle);不過,由于在多層的處理中需要在各個層之間切換激活,所以我們經常用到的是gdi_layer_push_and_set_active(gdi_handle handle),此函數會把當前的激活層入棧而激活參數層,等到下次需要激活棧中的層時,只需要用函數gdi_layer_pop_and_restore_active()激活就可以了。

            4.基礎層:系統開機的時候會為每個硬件屏幕創建一個基礎層,基礎層有以下幾個特點:
            (1)基礎層由系統創建,無法刪除。
            (2)與硬件屏幕完全重合。
            (3)系統默認的激活層,EntryNewScreen時系統會紫銅將基礎層激活。
            (4)顯示更加快速,基礎層存儲于芯片內的flash中,所以在其上面繪畫極快,一般我們會將刷新頻繁的內容放在基礎層上。
                基于以上幾點,通常在不使用多層的情況下,我們完全可以將基礎層當成硬件屏幕來看待,也就是說普通程序完全可以忽略層的概念。另外,因為系統一般只在EntryNewScreen時才會自動將基礎層激活,為避免特殊情況下使用層混亂,通常在新層上繪畫完畢后我們會主動將基礎層還原為激活狀態。
            其實,對于我們上面說到的層之間的切換激活而言,大多說情況下是基礎層和新層之間的切換。為此需要用到gdi_layer_get_active(gdi_handle *handle_ptr)得到當前激活層句柄(多數情況下是基礎層句柄),這樣我們就可以切換激活層了。

            5.合并:有了多個層,當然要合并到一起了。合并層的函數是gdi_layer_blt_previous(S32 x1, S32 y1, S32 x2, S32 y2),gui_BLT_double_buffer也具有同樣的效果。另外,在合并前,我們還需要用函數gdi_layer_set_blt_layer(H1,H2,H3,H4) 來指明需要合并的層。

            6.釋放:由于空間的問題,我們創建的新層在用完后一定要釋放,釋放函數是:
            gdi_layer_free(gdi_handle handle)。注意:層一般是在退出函數中釋放。

            說到這兒,我們就可以建立一個多層的屏幕了。下面我們在看看層的一些特效:
            1.剪切:所謂剪切,就是在層中設一個限制區域,只有在這個區域中的繪畫才是有效的,否則就會被自動忽略。剪切有兩個顯著的特點,一是每個層一定而且只能有一個剪切區域。二是剪切區域一經設置將永遠生效。所以剪切區域用完后最好用gdi_layer_reset_clip()還原。剪切用函數gdi_layer_set_clip()來實現。

            2.通透:所謂通透,就是如果我們將某種顏色設為通透色,在層合并的時候,系統會自動將層中與通透色相同的顏色忽略掉,這樣我們在這一點上看到的是其下層的顏色。設置通透的函數是gdi_layer_set_source_key。另外在設置通透前我們通常用gdi_layer_clear_background將該層刷上某種顏色。

            3.透明:gdi_layer_set_opacity(BOOL opacity_enable, U8 opacity_value)
            opacity_enable指通明是否開啟,opacity_value指透明度,范圍是0至255,值越小越透明。

            4.鎖屏:由于某些元素要頻繁的刷新,而如果我們每次都合并就會很浪費時間,因此我們可以設置兩個計數器,一個用來加,一個用來減,當計數器為0時,我們再合并。這兩個計數器就是  gdi_layer_lock_frame_buffer()和  gdi_layer_unlock_frame_buffer();

            請參考以下進入一個新屏的函數:
            view plaincopy to clipboardprint?
            static void mmi_entry_new_screen(void)  
            {  
                  S8 buf_filename[FMGR_PATH_BUFFER_SIZE];  
                  S32 image_width;  
                  S32 image_height;  
                  PU8 buf_ptr;  
                  U8 *gui_buffer;  
                  GDI_RESULT result;  
                  S32 offset_x;  
                  S32 offset_y;  
                  U16 img_type;  
                  UI_string_type title_string;  
                  S32 str_width;  
                  S32 str_height;  
             
             
                 EntryNewScreen  (SCR_ID_PHOART_MAIN,mmi_phoart_exit_main_screen,mmi_phoart_entry_main_screen, NULL);  
               
                 gdi_layer_reset_clip();//設置剪切區域  
                 gdi_layer_reset_text_clip();//設置剪切區域  
               
                 //gui_set_font(&MMI_medium_font);  
               
                 gui_buffer = GetCurrGuiBuffer(SCR_ID_PHOART_MAIN);  
               
                 entry_full_screen();  
               
                 gdi_layer_multi_layer_enable();//開啟多層  
               
                 gdi_layer_get_active(&g_phoart_cntx.base_layer_handle);//得到當前活動層  
               
                 gdi_layer_get_source_key(&g_phoart_cntx.was_source_key_enable, &g_phoart_cntx.old_source_key);//得到當前活動層的通透屬性  
                 gdi_layer_set_source_key(FALSE, g_phoart_cntx.old_source_key);//設置通透  
               
                 gdi_layer_create(0, 0, UI_device_width, UI_device_height, &g_phoart_cntx.osd_layer_handle);//創建新層  
                 gdi_layer_push_and_set_active(g_phoart_cntx.osd_layer_handle);//激活新層  
                 gdi_layer_clear_background(GDI_COLOR_TRANSPARENT);//刷色  
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_TRANSPARENT);//設置通透  
             
                 //base layer   
                 gdi_layer_pop_and_restore_active();//激活棧中的層  
                 //gdi_layer_reset_clip();  
                 //gdi_layer_reset_text_clip();  
                 gdi_layer_clear_background(GDI_COLOR_RED);  
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_RED);  
             
               
                 gdi_image_stop_animation_all();  
                 gdi_image_draw_animation_id(50, 100,  IMG_ID_PHOART_ICON_6, NULL);  
             
                //new layer  
                 gdi_layer_push_and_set_active(g_phoart_cntx.osd_layer_handle);  
                // gdi_layer_reset_clip();  
                 //gdi_layer_reset_text_clip();  
                 gdi_layer_set_clip(0,0,50,50);  
               
                 gdi_layer_clear_background(GDI_COLOR_TRANSPARENT);  
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_TRANSPARENT);  
             
                gui_set_font(&MMI_medium_font);  
                gui_set_text_color(g_phoart_cntx.text_color);  
                gui_set_text_border_color(g_phoart_cntx.text_border_color);  
             
                title_string = (UI_string_type) get_string(STR_ID_PHOART_APP);  
                gui_measure_string(title_string, &str_width, &str_height);  
             
                 gdi_layer_set_clip((UI_device_width - str_width) >> 1,(MMI_title_height - str_height) >> 1,200,100);//設置剪切區域  
                gui_move_text_cursor((UI_device_width - str_width) >> 1, (MMI_title_height - str_height) >> 1);  
                gui_print_bordered_text(title_string);  
             
                mmi_phoart_draw_softkey((PS8) get_string(STR_GLOBAL_OPTIONS), (PS8) get_string(STR_GLOBAL_BACK), FALSE);  
             
                 gdi_layer_pop_and_restore_active();  
                 gdi_layer_set_blt_layer(g_phoart_cntx.base_layer_handle,  g_phoart_cntx.osd_layer_handle, 0, 0);//指明需要合并的層  
                 gdi_layer_blt_previous(0, 0, UI_device_width - 1, UI_device_height - 1);//合并層  

            static void mmi_entry_new_screen(void)
            {
                  S8 buf_filename[FMGR_PATH_BUFFER_SIZE];
                  S32 image_width;
                  S32 image_height;
                  PU8 buf_ptr;
                  U8 *gui_buffer;
                  GDI_RESULT result;
                  S32 offset_x;
                  S32 offset_y;
                  U16 img_type;
                  UI_string_type title_string;
                  S32 str_width;
                  S32 str_height;


                 EntryNewScreen  (SCR_ID_PHOART_MAIN,mmi_phoart_exit_main_screen,mmi_phoart_entry_main_screen, NULL);
             
                 gdi_layer_reset_clip();//設置剪切區域
                 gdi_layer_reset_text_clip();//設置剪切區域
             
                 //gui_set_font(&MMI_medium_font);
             
                 gui_buffer = GetCurrGuiBuffer(SCR_ID_PHOART_MAIN);
             
                 entry_full_screen();
             
                 gdi_layer_multi_layer_enable();//開啟多層
             
                 gdi_layer_get_active(&g_phoart_cntx.base_layer_handle);//得到當前活動層
             
                 gdi_layer_get_source_key(&g_phoart_cntx.was_source_key_enable, &g_phoart_cntx.old_source_key);//得到當前活動層的通透屬性
                 gdi_layer_set_source_key(FALSE, g_phoart_cntx.old_source_key);//設置通透
             
                 gdi_layer_create(0, 0, UI_device_width, UI_device_height, &g_phoart_cntx.osd_layer_handle);//創建新層
                 gdi_layer_push_and_set_active(g_phoart_cntx.osd_layer_handle);//激活新層
                 gdi_layer_clear_background(GDI_COLOR_TRANSPARENT);//刷色
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_TRANSPARENT);//設置通透

                 //base layer
                 gdi_layer_pop_and_restore_active();//激活棧中的層
                 //gdi_layer_reset_clip();
                 //gdi_layer_reset_text_clip();
                 gdi_layer_clear_background(GDI_COLOR_RED);
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_RED);

             
                 gdi_image_stop_animation_all();
                 gdi_image_draw_animation_id(50, 100,  IMG_ID_PHOART_ICON_6, NULL);

                //new layer
                 gdi_layer_push_and_set_active(g_phoart_cntx.osd_layer_handle);
                // gdi_layer_reset_clip();
                 //gdi_layer_reset_text_clip();
                 gdi_layer_set_clip(0,0,50,50);
             
                 gdi_layer_clear_background(GDI_COLOR_TRANSPARENT);
                 gdi_layer_set_source_key(TRUE, GDI_COLOR_TRANSPARENT);

                gui_set_font(&MMI_medium_font);
                gui_set_text_color(g_phoart_cntx.text_color);
                gui_set_text_border_color(g_phoart_cntx.text_border_color);

                title_string = (UI_string_type) get_string(STR_ID_PHOART_APP);
                gui_measure_string(title_string, &str_width, &str_height);

                 gdi_layer_set_clip((UI_device_width - str_width) >> 1,(MMI_title_height - str_height) >> 1,200,100);//設置剪切區域
                gui_move_text_cursor((UI_device_width - str_width) >> 1, (MMI_title_height - str_height) >> 1);
                gui_print_bordered_text(title_string);

                mmi_phoart_draw_softkey((PS8) get_string(STR_GLOBAL_OPTIONS), (PS8) get_string(STR_GLOBAL_BACK), FALSE);

                 gdi_layer_pop_and_restore_active();
                 gdi_layer_set_blt_layer(g_phoart_cntx.base_layer_handle,  g_phoart_cntx.osd_layer_handle, 0, 0);//指明需要合并的層
                 gdi_layer_blt_previous(0, 0, UI_device_width - 1, UI_device_height - 1);//合并層
            }
             

            ==================================================

            最近在搞MTK的GDI,這篇文章說的很詳細,受益匪淺。

            在MTK0812中合并層用gdi_layer_blt就可以了,文中所用的操作要簡單一些。

             

            本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/QlDoors/archive/2009/09/19/4569979.aspx

            欧美大战日韩91综合一区婷婷久久青草| 久久精品国产99国产精品亚洲 | 亚洲国产成人精品91久久久| 久久中文字幕视频、最近更新 | 精品熟女少妇AV免费久久| 久久精品无码一区二区无码| 女人香蕉久久**毛片精品| 亚洲精品成人网久久久久久| 丰满少妇人妻久久久久久| 伊人久久综合热线大杳蕉下载| 免费无码国产欧美久久18| 国产情侣久久久久aⅴ免费| 久久国产影院| 久久99精品久久只有精品| 欧美色综合久久久久久| 精品久久久久久亚洲| 精品久久人人爽天天玩人人妻| 久久国产乱子精品免费女| 久久久久久精品无码人妻| 久久久精品久久久久久| 国产成人综合久久综合| 精品国产乱码久久久久久呢| 久久精品国产精品亚洲艾草网美妙| 色婷婷综合久久久久中文| 伊人精品久久久久7777| 九九久久精品国产| 99久久精品免费国产大片| 97精品国产91久久久久久| 久久精品人人做人人妻人人玩| 国产香蕉久久精品综合网| 久久人人爽人人爽AV片| 久久天天日天天操综合伊人av| 色综合久久中文色婷婷| 久久精品国产99国产精偷| 99久久精品国产麻豆| 国产成人精品久久二区二区| 99久久综合狠狠综合久久止| 69久久夜色精品国产69| 国产精品18久久久久久vr| 日本久久久久久中文字幕| 99久久精品国产综合一区|