• <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~ 閱讀(821) 評論(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

            国产精品美女久久久久久2018| 久久午夜综合久久| 久久综合88熟人妻| 日产精品99久久久久久| MM131亚洲国产美女久久| 欧美伊香蕉久久综合类网站| 国产真实乱对白精彩久久| 亚洲人成无码久久电影网站| 亚洲精品乱码久久久久66| 精品久久国产一区二区三区香蕉| 精品久久久久久久国产潘金莲 | 国产AV影片久久久久久| 青青久久精品国产免费看| 国产∨亚洲V天堂无码久久久| 久久久免费观成人影院| 麻豆一区二区99久久久久| 国产国产成人久久精品| 一本一本久久a久久综合精品蜜桃| 99久久人妻无码精品系列| 国内精品伊人久久久影院| 久久中文娱乐网| 久久久久亚洲av无码专区| 欧美日韩精品久久久久 | 国产伊人久久| 99久久国产热无码精品免费| 久久人人爽人人澡人人高潮AV| 久久国产一区二区| 久久亚洲精品国产精品| 久久人与动人物a级毛片| 午夜精品久久久久久| 久久99精品久久久久久9蜜桃| 亚洲国产精品久久久久久| 久久婷婷五月综合色高清| 精品无码久久久久国产动漫3d| 日韩久久久久中文字幕人妻| 久久天天躁狠狠躁夜夜不卡| 国产精品久久久久久久午夜片| 91超碰碰碰碰久久久久久综合| 国产精品女同久久久久电影院| 国产精品一久久香蕉国产线看 | 久久99精品久久久久久hb无码|