1. 程式人生 > >SDI實現多檢視並切換檢視

SDI實現多檢視並切換檢視

               

    Visual Studio 2010的新特性裡面Ribbon介面是一個很重要的東西,Ribbon介面非常易用和高效,也不去強調和證明它將成為未來應用程式介面的一種趨勢。為了讓我們的應用程式“Ready for Windows 7”,我們有必要根據實際的情況需要,為我們的應用程式新增Ribbon介面。

 為了推廣Ribbon介面,為了討好程式設計師們,讓程式設計師可以輕鬆地為各種應用程式建立Ribbon介面,微軟真是不遺餘力,不斷地推出新的Ribbon介面介面。為了可以讓所有基於C++的普通Windows應用程式都可以新增Ribbon介面,微軟在Windows 7 SDK中,又推出了基於COM的介面——Scenic Ribbon API。

  利用Scenic Ribbon API,我們可以更加靈活方便地為各種Windows應用程式新增Ribbon介面,改善應用程式的使用者體驗。在接下來的文章中,我們就介紹一下如何利用這種新的方式為普通Windows應用程式新增Ribbon介面。

  我們都知道,Ribbon介面是隨著Office 2007一同釋出的。在過去的幾年中,微軟不斷地吸取使用者對Ribbon介面的反饋,不斷地對其進行改進,到了Windows 7,微軟正式加入了命名為Scenic Ribbon的第二代Ribbon介面,即取代第一代Ribbon的升級版本。隨著Scenic Ribbon成為Windows 7的一部分以及微軟所提供的Scenic Ribbon API,這意味著今後第三方程式開發者能在自己開發的應用程式中免費用上這種介面形式。Scenic Ribbon作為第二代Ribbon介面,所適用的是所有的Windows應用程式,和前代有所不同是理所當然的。例如,由於最初Ribbon介面是Office專用,因此使用Ribbon介面的Office 2007程式視窗左上角都有一個比較顯眼的Office圖示按鈕。當應用到其他程式之後,這個圖示顯然不能繼續存在,因此在Scenic Ribbon中,這個位置就被相應應用程式自身的圖示所取代。另外,為了滿足各種應用程式的需要,Scenic Ribbon還添加了部分Ribbon控制元件,使得Ribbon介面的控制元件更加豐富,能夠應對各種應用程式對Ribbon介面控制元件的需要。值得一提的是,Office作為Ribbon介面的開山鼻祖,到了Office的下一個版本Office 2010,也全面使用了Scenic Ribbon以替代第一代Ribbon介面。

   同時微軟在VisualC++2010也提供了Ribbon資源設計器,使Ribbon開發更加簡單快捷。

 Scenic Ribbon API是一套基於COM的非託管API。通過這套API,我們可以為所有基於Windows的非託管應用程式建立Ribbon介面面板,新增Ribbon控制元件,處理Ribbon控制元件的各種動作,執行某種業務邏輯等等。而至於Ribbon介面的佈局,各個控制元件的位置和屬性等,則是通過基於XAML的xml檔案來進行定義和描述的。編譯器會將這個XML檔案編譯成相應的資原始檔,標頭檔案和二進位制的BML檔案。而Scenic Ribbon API最終會讀取這些描述控制元件資訊的檔案並建立Ribbon介面。在業務邏輯層,Scenic Ribbon API提供了相應的Execute函式來處理所有控制元件的訊息,在其中我們可以完成相應的業務邏輯。

在Windows 7中,微軟還為我們提供了用於為非託管程式碼(Native)應用程式建立Ribbon介面的Scenic Ribbon API的解決方案。另外,針對基於WPF的託管程式碼應用程式,微軟還將提供另外的基於WPF的Ribbon介面解決方案。

Firefox 3.7已經採用Windows7的Ribbon介面。

   下面我們親自來動手構建一個基於VisualC++2010開發windows7的tabelt pc應用的WindowsRibbon範例

    1.帥氣的VS2010啟動畫面,啟動VS2010

    2.建立inkdoc的MFC工程

3.MFC嚮導1

4.MFC嚮導2,選擇中文介面,多文件程式,以及windows7風格

5.MFC嚮導3,跳過

6.MFC嚮導4,建立文件格式,檔名

7.MFC嚮導5。跳過

8.MFC嚮導6,選擇Ribbon

9.MFC嚮導7,設定高階選項

10.MFC設定8,設定CFORMVIEW,

11.列印提示,不理睬,直接跳過

12.VisualC++2010自帶的Ribbon編輯器

 13.操作與工具箱控制元件一樣操作,設定按鈕屬性

 14.設定按鈕

15.設定按鈕屬性,

16.為按鈕新增事件

17.類似選單的操作

18.在Formview,選擇對話方塊,插入flash控制元件,然後為flash新增MFC   activeX  類嚮導

19.插入flash控制元件類

帥的驚動了CSDN的windows7的tabelt pc應用的WindowsRibbon範例的效果圖

tablet PC的軌跡核心程式碼如下

[cpp] view plain copy print?
  1. #include <windows.h>   
  2. #include <comdef.h>   
  3. #include <msinkaut.h>   
  4. #include <msinkaut_i.c>  //包含windows SDK  for windows7的核心tabletpc標頭檔案   
  5. #include "resource.h"   
  6. constTCHAR*    gc_szAppName = TEXT("Basic Recognition");     
  7. // 建立軌跡指標,識別指標,軌跡畫板指標   
  8. IInkCollector *     g_pIInkCollector    = NULL;     
  9. IInkDisp *          g_pIInkDisp         = NULL;     
  10. IInkRecognizerContext *   g_pIInkRecoContext  = NULL;     
  11. //定義初始化變數   
  12. const LPOLESTR gc_pwsInputScopes[] = {     
  13.     L"(!IS_DEFAULT)",     
  14.     L"(!IS_URL)",     
  15.     L"(!IS_FILE_FULLFILEPATH)",     
  16.     L"(!IS_FILE_FILENAME)",     
  17.     L"(!IS_EMAIL_USERNAME)",     
  18.     L"(!IS_EMAIL_SMTPEMAILADDRESS)",     
  19.     L"(!IS_LOGINNAME)",     
  20.     L"(!IS_PERSONALNAME_FULLNAME)",     
  21.     L"(!IS_PERSONALNAME_PREFIX)",     
  22.     L"(!IS_PERSONALNAME_GIVENNAME)",     
  23.     L"(!IS_PERSONALNAME_MIDDLENAME)",     
  24.     L"(!IS_PERSONALNAME_SURNAME)",     
  25.     L"(!IS_PERSONALNAME_SUFFIX)",     
  26.     L"(!IS_ADDRESS_FULLPOSTALADDRESS)",     
  27.     L"(!IS_ADDRESS_POSTALCODE)",     
  28.     L"(!IS_ADDRESS_STREET)",     
  29.     L"(!IS_ADDRESS_STATEORPROVINCE)",     
  30.     L"(!IS_ADDRESS_CITY)",     
  31.     L"(!IS_ADDRESS_COUNTRYNAME)",     
  32.     L"(!IS_ADDRESS_COUNTRYSHORTNAME)",     
  33.     L"(!IS_CURRENCY_AMOUNTANDSYMBOL)",     
  34.     L"(!IS_CURRENCY_AMOUNT)",     
  35.     L"(!IS_DATE_FULLDATE)",     
  36.     L"(!IS_DATE_MONTH)",     
  37.     L"(!IS_DATE_DAY)",     
  38.     L"(!IS_DATE_YEAR)",     
  39.     L"(!IS_DATE_MONTHNAME)",     
  40.     L"(!IS_DATE_DAYNAME)",     
  41.     L"(!IS_DIGITS)",     
  42.     L"(!IS_NUMBER)",     
  43.     L"(!IS_ONECHAR)",     
  44.     L"(!IS_TELEPHONE_FULLTELEPHONENUMBER)",     
  45.     L"(!IS_TELEPHONE_COUNTRYCODE)",     
  46.     L"(!IS_TELEPHONE_AREACODE)",     
  47.     L"(!IS_TELEPHONE_LOCALNUMBER)",     
  48.     L"(!IS_TIME_FULLTIME)",     
  49.     L"(!IS_TIME_HOUR)",     
  50.     L"(!IS_TIME_MINORSEC)",     
  51.     L"((0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?-? ?)?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?-? ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9)",     
  52.     L"(!IS_PERSONALNAME_FULLNAME)|((!IS_PERSONALNAME_PREFIX)? +(!IS_PERSONALNAME_GIVENNAME)+ +(!IS_PERSONALNAME_MIDDLENAME)* +(!IS_PERSONALNAME_SURNAME)+)",     
  53.     L"MN(0|1|2|3|4|5|6|7|8|9)(0|1|2|3|4|5|6|7|8|9)(A|B)(!IS_DIGITS)(X|Y)((0|1)*)"
  54. };     
  55. constLONG gc_lMaxInputScopeMenuItemLength = 40;     
  56. // 定義筆勢常量   
  57. const InkApplicationGesture gc_igtSingleStrokeGestures[] = {     
  58.     IAG_Scratchout, IAG_Triangle, IAG_Square, IAG_Star, IAG_Check,     
  59.     IAG_Circle, IAG_DoubleCircle, IAG_Curlicue, IAG_DoubleCurlicue,     
  60.     IAG_SemiCircleLeft, IAG_SemiCircleRight,     
  61.     IAG_ChevronUp, IAG_ChevronDown, IAG_ChevronLeft,     
  62.     IAG_ChevronRight, IAG_Up, IAG_Down, IAG_Left, IAG_Right, IAG_UpDown, IAG_DownUp,     
  63.     IAG_LeftRight, IAG_RightLeft, IAG_UpLeftLong, IAG_UpRightLong, IAG_DownLeftLong,     
  64.     IAG_DownRightLong, IAG_UpLeft, IAG_UpRight, IAG_DownLeft, IAG_DownRight, IAG_LeftUp,     
  65.     IAG_LeftDown, IAG_RightUp, IAG_RightDown, IAG_Tap     
  66. };     
  67. constUINT gc_nRecommendedForMixedMode[] = {     
  68.         0 /*Scratchout*/, 3/*Star*/, 6/*Double Circle*/,     
  69.         7 /*Curlicue*/, 8 /*Double Curlicue*/, 25 /*Down-Left Long*/ };     
  70. const InkApplicationGesture gc_igtMultiStrokeGestures[] = {     
  71.     IAG_ArrowUp, IAG_ArrowDown, IAG_ArrowLeft,     
  72.     IAG_ArrowRight, IAG_Exclamation, IAG_DoubleTap     
  73. };     
  74. const _ATL_FUNC_INFO IInkRecognitionEventsImpl<CAdvRecoApp>::mc_AtlFuncInfo =     
  75.         {CC_STDCALL, VT_EMPTY, 3, {VT_UNKNOWN, VT_VARIANT, VT_I4}};     
  76. const _ATL_FUNC_INFO IInkCollectorEventsImpl<CAdvRecoApp>::mc_AtlFuncInfo[2] = {     
  77.         {CC_STDCALL, VT_EMPTY, 3, {VT_UNKNOWN, VT_UNKNOWN, VT_BOOL|VT_BYREF}},     
  78.         {CC_STDCALL, VT_EMPTY, 4, {VT_UNKNOWN, VT_UNKNOWN, VT_VARIANT, VT_BOOL|VT_BYREF}}     
  79. };     
  80. constTCHAR gc_szAppName[] = TEXT("CSDN專家尹成的程式,http://blog.csdn.net/yincheng01");     
  81. /////////////////////////////////////////////////////////   
  82. //   
  83. //清除Com指標   
  84. /////////////////////////////////////////////////////////   
  85. void Cinkdoc::CleanUp()  // Release all objects   
  86. {     
  87.     if (g_pIInkRecoContext != NULL)     
  88.     {     
  89.         g_pIInkRecoContext->Release();     
  90.         g_pIInkRecoContext = NULL;     
  91.     }     
  92.     if (g_pIInkDisp != NULL)     
  93.     {     
  94.         g_pIInkDisp->Release();     
  95.         g_pIInkDisp = NULL;     
  96.     }     
  97.     if (g_pIInkCollector != NULL)     
  98.     {     
  99.         g_pIInkCollector->Release();     
  100.         g_pIInkCollector = NULL;     
  101.     }     
  102. }     
  103. void Cinkdoc::Reco()  //識別
  104. {     
  105.   //獲取識別結果,並顯示最優化識別結果   
  106.                 HCURSOR hCursor = ::SetCursor(::LoadCursor(NULL, IDC_WAIT));     
  107.                //獲取軌跡指標用於識別   
  108.                 IInkStrokes* pIInkStrokes = NULL;     
  109.                 HRESULT hr = g_pIInkDisp->get_Strokes(&pIInkStrokes);     
  110.                 if (SUCCEEDED(hr))     
  111.                 {     
  112.                     hr = g_pIInkRecoContext->putref_Strokes(pIInkStrokes);     
  113.                     if (SUCCEEDED(hr))     
  114.                     {    //利用識別指標獲取識別結果   
  115.                         IInkRecognitionResult* pIInkRecoResult = NULL;     
  116.                         InkRecognitionStatus RecognitionStatus;     
  117.                         hr = g_pIInkRecoContext->Recognize(&RecognitionStatus, &pIInkRecoResult);     
  118.                         if (SUCCEEDED(hr) && (pIInkRecoResult!= NULL))     
  119.                         {     
  120.                             BSTR bstrBestResult = NULL;     
  121.                             hr = pIInkRecoResult->get_TopString(&bstrBestResult);     
  122.                             pIInkRecoResult->Release();     
  123.                             pIInkRecoResult = NULL;     
  124.                             if (SUCCEEDED(hr) && bstrBestResult)     
  125.                             {     
  126.                                 MessageBoxW(hwnd, bstrBestResult,     
  127.                                             L"識別結果", MB_OK);     
  128.                                 SysFreeString(bstrBestResult);     
  129.                             }     
  130.                         }     
  131.                         g_pIInkRecoContext->putref_Strokes(NULL);     
  132.                     }     
  133.                     pIInkStrokes->Release();     
  134.                 }     
  135.                 ::SetCursor(hCursor);     
  136. }     
  137. void Cinkdoc::start()  // 開始建立  
  138. {     
  139.            //並建立com指標介面   
  140.             HRESULT hr;     
  141.             hr = CoCreateInstance(CLSID_InkRecognizerContext,     
  142.                                   NULL, CLSCTX_INPROC_SERVER,     
  143.                                   IID_IInkRecognizerContext,     
  144.                                   (void **) &g_pIInkRecoContext);     
  145.             if (FAILED(hr))     
  146.             {     
  147.                 return -1;     
  148.             }     
  149.             hr = CoCreateInstance(CLSID_InkCollector,     
  150.                                   NULL, CLSCTX_INPROC_SERVER,     
  151.                                   IID_IInkCollector,     
  152.                                   (void **) &g_pIInkCollector);     
  153.             if (FAILED(hr))     
  154.                 return -1;     
  155.             hr = g_pIInkCollector->get_Ink(&g_pIInkDisp);     
  156.             if (FAILED(hr))     
  157.                 return -1;     
  158.             hr = g_pIInkCollector->put_hWnd((long)this->hwnd);     
  159.             if (FAILED(hr))     
  160.                 return -1;     
  161.             hr = g_pIInkCollector->put_Enabled(VARIANT_TRUE);     
  162.             if (FAILED(hr))     
  163.                 return -1;     
  164. }    
  165. //筆勢事件   
  166. HRESULT Cinkdoc::OnGesture(     
  167.         IInkCursor* /*pIInkCursor*/,     
  168.         IInkStrokes* pInkStrokes,     
  169.         VARIANT vGestures,     
  170.         VARIANT_BOOL* pbCancel     
  171.         )     
  172. {     
  173.     if (((VT_ARRAY | VT_DISPATCH) != vGestures.vt) || (NULL == vGestures.parray))     
  174.         return E_INVALIDARG;     
  175.     if (0 == vGestures.parray->rgsabound->cElements)     
  176.         return E_INVALIDARG;     
  177.    / /陣列中的手勢應該是排序的承認     
  178.     / /置信水平。此示例拿起最高之一。     
  179.     / /注意:當在InkAndGesture收集模式,除了預期的手勢     
  180.     / /應用程式也有可能是帶著身份證IAG_NoGesture手勢物件     
  181.     / /此應用程式時,如果取消與ISG_NoGesture物件     
  182.     / /最高水平的信心陣列(第一項)。     
  183.     InkApplicationGesture idGesture = IAG_NoGesture;     
  184.     IDispatch** ppIDispatch;     
  185.     HRESULT hr = ::SafeArrayAccessData(vGestures.parray, (void HUGEP**)&ppIDispatch);     
  186.     if (SUCCEEDED(hr))     
  187.     {     
  188.         CComQIPtr<IInkGesture> spIInkGesture(ppIDispatch[0]);     
  189.         if (spIInkGesture != NULL)     
  190.         {     
  191.             hr = spIInkGesture->get_Id(&idGesture);     
  192.         }     
  193.         ::SafeArrayUnaccessData(vGestures.parray);     
  194.     }     
  195.   / /載入資源字串表的姿態名稱     
  196.     UINT idGestureName;     
  197.     bool bAccepted;     // will be true, if the gesture is known to this application   
  198.     if (IAG_NoGesture != idGesture)     
  199.     {     
  200.         bAccepted = GetGestureName(idGesture, idGestureName);     
  201.     }     
  202.     else
  203.     {     
  204.         bAccepted = false;     
  205.         idGestureName = 0;     
  206.     }     
  207.   //如果當前的收集模式ICM_GestureOnly或者如果我們接受   
  208.     //的姿態,手勢的中風將被刪除從墨跡物件,   
  209.     //因此,視窗需要在招'區更新。   
  210.     if (ID_MODE_GESTURES == m_nCmdMode || true == bAccepted)     
  211.     {     
  212.         // Get the rectangle to update.   
  213.         RECT rc;     
  214.         CComPtr<IInkRectangle> spIInkRect;     
  215.         if (m_spIInkRenderer != NULL     
  216.             && pInkStrokes != NULL     
  217.             && SUCCEEDED(pInkStrokes->GetBoundingBox(IBBM_Default, &spIInkRect))     
  218.             && SUCCEEDED(spIInkRect->GetRectangle(&rc.top, &rc.left,     
  219.                                                   &rc.bottom, &rc.right)))     
  220.         {     
  221.             HDC hdc = m_wndInput.GetDC();     
  222.             if (NULL != hdc)     
  223.             {     
  224.                 if (FAILED(m_spIInkRenderer->InkSpaceToPixel((long)hdc, &rc.left, &rc.top))     
  225.                     || FAILED(m_spIInkRenderer->InkSpaceToPixel((long)hdc, &rc.right,     
  226.                                                                 &rc.bottom)))     
  227.                 {     
  228.                     m_wndInput.GetClientRect(&rc);     
  229.                 }     
  230.                 ReleaseDC(hdc);     
  231.             }     
  232.             else
  233.             {     
  234.                 m_wndInput.GetClientRect(&rc);     
  235.             }     
  236.         }     
  237.         else
  238.         {     
  239.             m_wndInput.GetClientRect(&rc);     
  240.         }     
  241.         m_wndInput.InvalidateRect(&rc);     
  242.     }     
  243.     else// if something's failed,   
  244.     {     
  245.         *pbCancel = VARIANT_TRUE;     
  246.         idGestureName = IDS_GESTURE_UNKNOWN;     
  247.     }     
  248.     m_wndResults.SetGestureName(idGestureName);     
  249.     m_wndResults.Invalidate();     
  250.     return hr;     
  251. }     
  252. // Recognition event handlers 識別器事件   
  253. HRESULT Cinkdoc::OnRecognitionWithAlternates(     
  254.         IInkRecognitionResult* pIInkRecoResult,     
  255.         VARIANT /*vCustomParam*/,     
  256.         InkRecognitionStatus /*RecognitionStatus*/
  257.         )     
  258. {     
  259.     if (NULL == pIInkRecoResult)     
  260.         return E_INVALIDARG;     
  261.     m_wndResults.ResetResults();     
  262.     // Get the best lCount results   
  263.     HRESULT hr;     
  264.     CComPtr<IInkRecognitionAlternates> spIInkRecoAlternates;     
  265.     hr = pIInkRecoResult->AlternatesFromSelection(     
  266.         0,                                
  267.         -1,                               
  268.         CRecoOutputWnd::mc_iNumResults,      
  269.         &spIInkRecoAlternates             
  270.         );     
  271.     LONG lCount = 0;     
  272.     if (SUCCEEDED(hr) && SUCCEEDED(spIInkRecoAlternates->get_Count(&lCount)))     
  273.     {     
  274.         IInkRecognitionAlternate* pIInkRecoAlternate = NULL;     
  275.         for (LONG iItem = 0; (iItem < lCount) && (iItem < CRecoOutputWnd::mc_iNumResults); iItem++)     
  276.         {     
  277.             if (SUCCEEDED(spIInkRecoAlternates->Item(iItem, &pIInkRecoAlternate)))     
  278.             {     
  279.                 BSTR bstr = NULL;     
  280.                 if (SUCCEEDED(pIInkRecoAlternate->get_String(&bstr)))     
  281.                 {     
  282.                     m_wndResults.m_bstrResults[iItem].Attach(bstr);     
  283.                 }     
  284.                 pIInkRecoAlternate->Release();     
  285.             }     
  286.         }     
  287.     }     
  288.     // Update the output window with the new results   
  289.     m_wndResults.Invalidate();     
  290.     return S_OK;     
  291. }     
  292. //識別結果獲取   
  293. LRESULT Cinkdoc::OnRecognizer(     
  294.         WORD/*wNotifyCode*/,     
  295.         WORD wID,     
  296.         HWND/*hWndCtl*/,     
  297.         BOOL/*bHandled*/
  298.         )     
  299. {     
  300.     if (m_spIInkRecognizers == NULL || wID == m_nCmdRecognizer)     
  301.         return 0;     
  302.     HRESULT hr;     
  303.     CComPtr<IInkRecognizer> spIInkRecognizer;     
  304.     if (ID_RECOGNIZER_DEFAULT == wID)     
  305.     {     
  306.         hr = m_spIInkRecognizers->GetDefaultRecognizer(0, &spIInkRecognizer);//獲取預設識別器設定之   
  307.     }     
  308.     else
  309.     {     
  310.         hr = m_spIInkRecognizers->Item(wID - ID_RECOGNIZER_FIRST, &spIInkRecognizer);//否則獲取第一個識別器設定之   
  311.     }     
  312.     if (SUCCEEDED(hr) && UseRecognizer(spIInkRecognizer))     
  313.     {     
  314.         UpdateMenuRadioItems(mc_iSubmenuRecognizers, wID, m_nCmdRecognizer);     
  315.         m_bstrCurRecoName.Empty();     
  316.         spIInkRecognizer->get_Name(&m_bstrCurRecoName);     
  317.         UpdateStatusBar();     
  318.         m_nCmdRecognizer = wID;     
  319.     }     
  320.     return 0;     
  321. }     
  322. //輸入獲取   
  323. LRESULT Cinkdoc::OnInputScopeCoerce(     
  324.         WORD/*wNotifyCode*/,     
  325.         WORD wID,     
  326.         HWND/*hWndCtl*/,     
  327.         BOOL/*bHandled*/
  328.         )     
  329. {     
  330.     if (m_spIInkRecoContext == NULL )     
  331.         return 0;     
  332.     if (m_spIInkStrokes != NULL)     
  333.     {     
  334.         m_spIInkRecoContext->putref_Strokes(NULL);     
  335.     }     
  336.     if (FAILED(m_spIInkRecoContext->put_RecognitionFlags (m_bCoerceInputScope?IRM_None:IRM_Coerce)))     
  337.     {     
  338.         MessageBox(TEXT("獲取結果失敗!"),     
  339.                        gc_szAppName, MB_ICONERROR | MB_OK);     
  340.         // Re-attach the stroke collection to the context   
  341.         if (m_spIInkStrokes != NULL)     
  342.         {     
  343.             m_spIInkRecoContext->putref_Strokes(m_spIInkStrokes);     
  344.         }     
  345.             return 0;     
  346.         }     
  347.     m_bCoerceInputScope = !m_bCoerceInputScope;     
  348.     // 收集筆跡   
  349.     if (m_spIInkStrokes != NULL)     
  350.     {     
  351.         m_spIInkRecoContext->putref_Strokes(m_spIInkStrokes);     
  352.     }     
  353.     // 更新識別結果   
  354.     CComVariant vCustomData;    // no custom data   
  355.     m_spIInkRecoContext->BackgroundRecognizeWithAlternates(vCustomData);     
  356.     return 0;     
  357. }      
#include <windows.h>   #include <comdef.h>   #include <msinkaut.h>   #include <msinkaut_i.c>  //包含windows SDK  for windows7的核心tabletpc標頭檔案   #include "resource.h"       const TCHAR*    gc_szAppName = TEXT("Basic Recognition");     // 建立軌跡指標,識別指標,軌跡畫板指標   IInkCollector *     g_pIInkCollector    = NULL;   IInkDisp *          g_pIInkDisp         = NULL;   IInkRecognizerContext *   g_pIInkRecoContext  = NULL;       //定義初始化變數   const LPOLESTR gc_pwsInputScopes[] = {       L"(!IS_DEFAULT)",       L"(!IS_URL)",       L"(!IS_FILE_FULLFILEPATH)",       L"(!IS_FILE_FILENAME)",       L"(!IS_EMAIL_USERNAME)",       L"(!IS_EMAIL_SMTPEMAILADDRESS)",       L"(!IS_LOGINNAME)",       L"(!IS_PERSONALNAME_FULLNAME)",       L"(!IS_PERSONALNAME_PREFIX)",       L"(!IS_PERSONALNAME_GIVENNAME)",       L"(!IS_PERSONALNAME_MIDDLENAME)",       L"(!IS_PERSONALNAME_SURNAME)",       L"(!IS_PERSONALNAME_SUFFIX)",       L"(!IS_ADDRESS_FULLPOSTALADDRESS)",       L"(!IS_ADDRESS_POSTALCODE)",       L"(!IS_ADDRESS_STREET)",       L"(!IS_ADDRESS_STATEORPROVINCE)",       L"(!IS_ADDRESS_CITY)",       L"(!IS_ADDRESS_COUNTRYNAME)",       L"(!IS_ADDRESS_COUNTRYSHORTNAME)",       L"(!IS_CURRENCY_AMOUNTANDSYMBOL)",       L"(!IS_CURRENCY_AMOUNT)",       L"(!IS_DATE_FULLDATE)",       L"(!IS_DATE_MONTH)",       L"(!IS_DATE_DAY)",       L"(!IS_DATE_YEAR)",       L"(!IS_DATE_MONTHNAME)",       L"(!IS_DATE_DAYNAME)",       L"(!IS_DIGITS)",       L"(!IS_NUMBER)",       L"(!IS_ONECHAR)",       L"(!IS_TELEPHONE_FULLTELEPHONENUMBER)",       L"(!IS_TELEPHONE_COUNTRYCODE)",       L"(!IS_TELEPHONE_AREACODE)",       L"(!IS_TELEPHONE_LOCALNUMBER)",       L"(!IS_TIME_FULLTIME)",       L"(!IS_TIME_HOUR)",       L"(!IS_TIME_MINORSEC)",       L"((0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?-? ?)?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?-? ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9) ?(0|1|2|3|4|5|6|7|8|9)",       L"(!IS_PERSONALNAME_FULLNAME)|((!IS_PERSONALNAME_PREFIX)? +(!IS_PERSONALNAME_GIVENNAME)+ +(!IS_PERSONALNAME_MIDDLENAME)* +(!IS_PERSONALNAME_SURNAME)+)",       L"MN(0|1|2|3|4|5|6|7|8|9)(0|1|2|3|4|5|6|7|8|9)(A|B)(!IS_DIGITS)(X|Y)((0|1)*)"  };       const LONG gc_lMaxInputScopeMenuItemLength = 40;     // 定義筆勢常量   const InkApplicationGesture gc_igtSingleStrokeGestures[] = {       IAG_Scratchout, IAG_Triangle, IAG_Square, IAG_Star, IAG_Check,       IAG_Circle, IAG_DoubleCircle, IAG_Curlicue, IAG_DoubleCurlicue,       IAG_SemiCircleLeft, IAG_SemiCircleRight,       IAG_ChevronUp, IAG_ChevronDown, IAG_ChevronLeft,       IAG_ChevronRight, IAG_Up, IAG_Down, IAG_Left, IAG_Right, IAG_UpDown, IAG_DownUp,       IAG_LeftRight, IAG_RightLeft, IAG_UpLeftLong, IAG_UpRightLong, IAG_DownLeftLong,       IAG_DownRightLong, IAG_UpLeft, IAG_UpRight, IAG_DownLeft, IAG_DownRight, IAG_LeftUp,       IAG_LeftDown, IAG_RightUp, IAG_RightDown, IAG_Tap   };     const UINT gc_nRecommendedForMixedMode[] = {           0 /*Scratchout*/, 3/*Star*/, 6/*Double Circle*/,           7 /*Curlicue*/, 8 /*Double Curlicue*/, 25 /*Down-Left Long*/ };     const InkApplicationGesture gc_igtMultiStrokeGestures[] = {       IAG_ArrowUp, IAG_ArrowDown, IAG_ArrowLeft,       IAG_ArrowRight, IAG_Exclamation, IAG_DoubleTap   };       const _ATL_FUNC_INFO IInkRecognitionEventsImpl<CAdvRecoApp>::mc_AtlFuncInfo =           {CC_STDCALL, VT_EMPTY, 3, {VT_UNKNOWN, VT_VARIANT, VT_I4}};     const _ATL_FUNC_INFO IInkCollectorEventsImpl<CAdvRecoApp>::mc_AtlFuncInfo[2] = {           {CC_STDCALL, VT_EMPTY, 3, {VT_UNKNOWN, VT_UNKNOWN, VT_BOOL|VT_BYREF}},           {CC_STDCALL, VT_EMPTY, 4, {VT_UNKNOWN, VT_UNKNOWN, VT_VARIANT, VT_BOOL|VT_BYREF}}   };     const TCHAR gc_szAppName[] = TEXT("CSDN專家尹成的程式,http://blog.csdn.net/yincheng01");   /////////////////////////////////////////////////////////   //     //清除Com指標   /////////////////////////////////////////////////////////   void Cinkdoc::CleanUp()  // Release all objects   {       if (g_pIInkRecoContext != NULL)       {           g_pIInkRecoContext->Release();           g_pIInkRecoContext = NULL;       }         if (g_pIInkDisp != NULL)       {           g_pIInkDisp->Release();           g_pIInkDisp = NULL;       }         if (g_pIInkCollector != NULL)       {           g_pIInkCollector->Release();           g_pIInkCollector = NULL;       }   }   void Cinkdoc::Reco()  //識別{     //獲取識別結果,並顯示最優化識別結果                                    HCURSOR hCursor = ::SetCursor(::LoadCursor(NULL, IDC_WAIT));                  //獲取軌跡指標用於識別                   IInkStrokes* pIInkStrokes = NULL;                   HRESULT hr = g_pIInkDisp->get_Strokes(&pIInkStrokes);                   if (SUCCEEDED(hr))                   {                                          hr = g_pIInkRecoContext->putref_Strokes(pIInkStrokes);                       if (SUCCEEDED(hr))                       {    //利用識別指標獲取識別結果                                             IInkRecognitionResult* pIInkRecoResult = NULL;                           InkRecognitionStatus RecognitionStatus;                           hr = g_pIInkRecoContext->Recognize(&RecognitionStatus, &pIInkRecoResult);                           if (SUCCEEDED(hr) && (pIInkRecoResult!= NULL))                           {                                                        BSTR bstrBestResult = NULL;                               hr = pIInkRecoResult->get_TopString(&bstrBestResult);                               pIInkRecoResult->Release();                               pIInkRecoResult = NULL;                                                        if (SUCCEEDED(hr) && bstrBestResult)                               {                                   MessageBoxW(hwnd, bstrBestResult,                                               L"識別結果", MB_OK);                                   SysFreeString(bstrBestResult);                               }                           }                                               g_pIInkRecoContext->putref_Strokes(NULL);                       }                       pIInkStrokes->Release();                   }                             ::SetCursor(hCursor);   }   void Cinkdoc::start()  // 開始建立  {                 //並建立com指標介面               HRESULT hr;                                           hr = CoCreateInstance(CLSID_InkRecognizerContext,                                     NULL, CLSCTX_INPROC_SERVER,                                     IID_IInkRecognizerContext,                                     (void **) &g_pIInkRecoContext);               if (FAILED(hr))               {                                     return -1;               }                              hr = CoCreateInstance(CLSID_InkCollector,                                     NULL, CLSCTX_INPROC_SERVER,                                     IID_IInkCollector,                                     (void **) &g_pIInkCollector);               if (FAILED(hr))                   return -1;                               hr = g_pIInkCollector->get_Ink(&g_pIInkDisp);               if (FAILED(hr))                   return -1;                              hr = g_pIInkCollector->put_hWnd((long)this->hwnd);               if (FAILED(hr))                   return -1;                 hr = g_pIInkCollector->put_Enabled(VARIANT_TRUE);               if (FAILED(hr))                   return -1;      }  //筆勢事件   HRESULT Cinkdoc::OnGesture(           IInkCursor* /*pIInkCursor*/,           IInkStrokes* pInkStrokes,           VARIANT vGestures,           VARIANT_BOOL* pbCancel           )   {       if (((VT_ARRAY | VT_DISPATCH) != vGestures.vt) || (NULL == vGestures.parray))           return E_INVALIDARG;       if (0 == vGestures.parray->rgsabound->cElements)           return E_INVALIDARG;        / /陣列中的手勢應該是排序的承認       / /置信水平。此示例拿起最高之一。       / /注意:當在InkAndGesture收集模式,除了預期的手勢       / /應用程式也有可能是帶著身份證IAG_NoGesture手勢物件       / /此應用程式時,如果取消與ISG_NoGesture物件       / /最高水平的信心陣列(第一項)。       InkApplicationGesture idGesture = IAG_NoGesture;       IDispatch** ppIDispatch;       HRESULT hr = ::SafeArrayAccessData(vGestures.parray, (void HUGEP**)&ppIDispatch);       if (SUCCEEDED(hr))       {           CComQIPtr<IInkGesture> spIInkGesture(ppIDispatch[0]);           if (spIInkGesture != NULL)           {               hr = spIInkGesture->get_Id(&idGesture);           }           ::SafeArrayUnaccessData(vGestures.parray);       }       / /載入資源字串表的姿態名稱       UINT idGestureName;       bool bAccepted;     // will be true, if the gesture is known to this application       if (IAG_NoGesture != idGesture)       {           bAccepted = GetGestureName(idGesture, idGestureName);       }       else           {           bAccepted = false;           idGestureName = 0;       }       //如果當前的收集模式ICM_GestureOnly或者如果我們接受       //的姿態,手勢的中風將被刪除從墨跡物件,       //因此,視窗需要在招'區更新。       if (ID_MODE_GESTURES == m_nCmdMode || true == bAccepted)       {           // Get the rectangle to update.           RECT rc;           CComPtr<IInkRectangle> spIInkRect;           if (m_spIInkRenderer != NULL               && pInkStrokes != NULL               && SUCCEEDED(pInkStrokes->GetBoundingBox(IBBM_Default, &spIInkRect))               && SUCCEEDED(spIInkRect->GetRectangle(&rc.top, &rc.left,                                                     &rc.bottom, &rc.right)))           {                            HDC hdc = m_wndInput.GetDC();               if (NULL != hdc)               {                   if (FAILED(m_spIInkRenderer->InkSpaceToPixel((long)hdc, &rc.left, &rc.top))                       || FAILED(m_spIInkRenderer->InkSpaceToPixel((long)hdc, &rc.right,                                                                   &rc.bottom)))                   {                                            m_wndInput.GetClientRect(&rc);                   }                   ReleaseDC(hdc);               }               else              {                                m_wndInput.GetClientRect(&rc);               }           }           else          {                            m_wndInput.GetClientRect(&rc);           }             m_wndInput.InvalidateRect(&rc);       }       else // if something's failed,               {                    *pbCancel = VARIANT_TRUE;           idGestureName = IDS_GESTURE_UNKNOWN;       }               m_wndResults.SetGestureName(idGestureName);       m_wndResults.Invalidate();         return hr;   }     // Recognition event handlers 識別器事件   HRESULT Cinkdoc::OnRecognitionWithAlternates(           IInkRecognitionResult* pIInkRecoResult,           VARIANT /*vCustomParam*/,           InkRecognitionStatus /*RecognitionStatus*/          )   {       if (NULL == pIInkRecoResult)           return E_INVALIDARG;           m_wndResults.ResetResults();         // Get the best lCount results       HRESULT hr;       CComPtr<IInkRecognitionAlternates> spIInkRecoAlternates;       hr = pIInkRecoResult->AlternatesFromSelection(           0,                                      -1,                                     CRecoOutputWnd::mc_iNumResults,            &spIInkRecoAlternates                   );              LONG lCount = 0;       if (SUCCEEDED(hr) && SUCCEEDED(spIInkRecoAlternates->get_Count(&lCount)))       {                 IInkRecognitionAlternate* pIInkRecoAlternate = NULL;           for (LONG iItem = 0; (iItem < lCount) && (iItem < CRecoOutputWnd::mc_iNumResults); iItem++)           {                         if (SUCCEEDED(spIInkRecoAlternates->Item(iItem, &pIInkRecoAlternate)))               {                   BSTR bstr = NULL;                   if (SUCCEEDED(pIInkRecoAlternate->get_String(&bstr)))                   {                       m_wndResults.m_bstrResults[iItem].Attach(bstr);                   }                   pIInkRecoAlternate->Release();               }           }       }         // Update the output window with the new results       m_wndResults.Invalidate();         return S_OK;   }     //識別結果獲取   LRESULT Cinkdoc::OnRecognizer(           WORD /*wNotifyCode*/,           WORD wID,           HWND /*hWndCtl*/,           BOOL& /*bHandled*/          )   {       if (m_spIInkRecognizers == NULL || wID == m_nCmdRecognizer)           return 0;             HRESULT hr;       CComPtr<IInkRecognizer> spIInkRecognizer;       if (ID_RECOGNIZER_DEFAULT == wID)       {                     hr = m_spIInkRecognizers->GetDefaultRecognizer(0, &spIInkRecognizer);//獲取預設識別器設定之       }       else      {           hr = m_spIInkRecognizers->Item(wID - ID_RECOGNIZER_FIRST, &spIInkRecognizer);//否則獲取第一個識別器設定之       }              if (SUCCEEDED(hr) && UseRecognizer(spIInkRecognizer))       {                   UpdateMenuRadioItems(mc_iSubmenuRecognizers, wID, m_nCmdRecognizer);           m_bstrCurRecoName.Empty();           spIInkRecognizer->get_Name(&m_bstrCurRecoName);           UpdateStatusBar();                   m_nCmdRecognizer = wID;       }         return 0;   }   //輸入獲取