• 相关软件
    >Visual C++编程技巧之一 创建者:webmaster 更新时间:2005-07-02 01:23

    1. 如何获取应用程序的实例句柄?


    应用程序的 实例句柄保存在CWinAppIm_hInstance ,可以这么调用


    AfxGetInstancdHandle获得句柄.


    Example: HANDLE hInstance=AfxGetInstanceHandle();


    2. 如何通过代码获得应用程序主窗口的指针?


    主窗口的 指针保存在CWinThread::m_pMainWnd,调用 AfxGetMainWnd实现。


    AfxGetMainWnd() ->ShowWindow(SW_SHOWMAXMIZED); //使程序最大化.


    3.如何在程序中获得其他程序的 图标?


    两种方法:


    (1) SDK函数 SHGetFileInfo 或使用 ExtractIcon获得图标资源的 handle,


    (2) SDK函数 SHGetFileInfo获得有关文件的 很多信息,如大小图标,属性,类型等.


    Example(1): 在程序窗口左上角显示 NotePad图标.


    void CSampleView: OnDraw(CDC * pDC)


    {


    if( :: SHGetFileInfo(_T("c:\\pwin95\\notepad.exe"),0,


    &stFileInfo,sizeof(stFileInfo),SHGFI_ICON))


    {


    pDC ->DrawIcon(10,10,stFileInfo.hIcon);


    }


    }


    Example(2):同样功能,Use ExtractIcon Function


    void CSampleView:: OnDraw(CDC *pDC)


    {


    HICON hIcon=:: ExtractIcon(AfxGetInstanceHandle(),_T


    ("NotePad.exe"),0);


    if (hIcon &&hIcon!=(HICON)-1)


    pDC->DrawIcon(10,10,hIcon);


    }


    说明: 获得notepad.exe的路径正规上来说用GetWindowsDirectory 函数得到,如果是调用 win95下的画笔,应该用访问注册表的方法获得其路径,要作成一个比较考究的程序,考虑应该全面点.


    4.如何编程结束应用程序?如何编程控制windows的重新引导?


    这是个很简单又是编程中经常要遇到的问题.


    第一问,向窗口发送 WM_CLOSE消息,调用 CWnd::OnClose成员函数.允许对用户提示


    是否保存修改过的数据.


    Example: AfxGetMainWindow()->SendMessage(WM_CLOSE);


    还可以创建一个自定义的函数 Terminate Window


    void Terminate Window(LPCSTR pCaption)


    {


    CWnd *pWnd=Cwnd::FindWindow(NULL,pCaption);


    if (pWnd)


    pWnd ->SendMessage(WM_CLOSE);


    }


    说明: FindWindow函数不是提倡的做法,因为它无法处理标题栏自动改变,比如我们要检测 Notepad是不是已运行而事先不知道Notepad的标题栏,这时 FindWindow就无能为力了,可以通过枚举 windows任务列表的办法来实现。 在 机械出版社"Windows 95 API开发人员指南"一书有比较详细的介绍,这里就不再多说乐。


    第二问,Use ExitWindowsEx Function函数控制系统是重新引导,还是重启 windows.前面已经有人讲过乐,就不再提了。


    5.怎样加栽其他的应用程序?


    我记得这好象是出场频度很高的问题。


    三个SDK函数 winexec, shellexecute,createprocess可以使用。


    WinExec最简单,两个参数,前一个指定路径,后一个指定显示方式.后一个参数值得说一下,比如泥用 SW_SHOWMAXMIZED方式去加栽一个无最大化按钮的程序,呵呵就是Neterm,calc等等,就不会出现正常的窗体,但是已经被加到任务列表里了。ShellExecuteWinExex灵活一点,可以指定工作目录,下面的 Example就是直接打开 c:\temp\1.txt,而不用加栽与 txt文件关联的应用程序,很多安装程序完成后都会打开一个窗口,来显示Readme or Faq,偶猜就是这么作的啦.


    ShellExecute(NULL,NULL,_T("1.txt"),NULL,_T("c:\\temp"),SW_SHOWMAXMIZED);


    CreateProcess最复杂,一共有十个参数,不过大部分都可以用NULL代替,它可以指定进程的安全属性,继承信息,类的优先级等等.来看个很简单的 Example:


    STARTUPINFO stinfo; //启动窗口的信息


    PROCESSINFO procinfo; //进程的信息


    CreateProcess(NULL,_T("notepad.exe"),NULL,NULL.FALSE, NORMAL_PRIORITY_


    CLASS,NULL,NULL, &stinfo,&procinfo);


    6. 确定应用程序的 路径


    前些天好象有人问过这个问题.


    Use GetModuleFileName 获得应用程序的路径,然后去掉可执行文件名。


    Example:


    TCHAR exeFullPath[MAX_PATH]; // MAX_PATHAPI中定义了吧,好象是128


    GetModuleFileName(NULL,exeFullPath,MAX_PATH)


    7. 获得各种目录信息


    Windows目录: Use "GetWindowsDirectory


    Windows下的system目录: Use "GetSystemDirectory"


    temp目录: Use "GetTempPath "


    当前目录: Use "GetCurrentDirectory"


    请注意前两个函数的第一个参数为目录变量名,后一个为缓冲区; 后两个相反.


    8. 如何自定义消息


    也有人问过的,其实不难。


    (1) 手工定义消息,可以这么写 #define WM_MY_MESSAGE(WM_USER+100),


    MS 推荐的至少是 WM_USER+100;


    (2)写消息处理函数,WPARAM,LPARAM返回LRESULT.


    LRESULT CMainFrame::OnMyMessage(WPARAM wparam,LPARAM lParam)


    {


    //加入你的处理函数


    }


    (3) 在类的 AFX_MSG处进行声明,也就是常说的"宏映射"


    9. 如何改变窗口的图标?


    向窗口发送 WM_SECTION消息。


    Example:


    HICON hIcon=AfxGetApp() ->LoadIcon(IDI_ICON);


    ASSERT(hIcon);


    AfxGetMainWnd() ->SendMessage(WM_SECTION,TRUE,(LPARAM) hIcon);


    10. 如何改变窗口的缺省风格?


    重栽 CWnd:: PreCreateWindow 并修改CREATESTRUCT结构来指定窗口风格和其他


    创建信息.


    Example: Delete "Max" Button and Set Original Window's Position and Size


    BOOL CMainFrame:: PreCreateWindow (CREATESTRUCT &cs)


    {


    cs.style &=~WS_MAXINIZEMOX;


    cs.x=cs.y=0;


    cs.cx=GetSystemMetrics(SM_CXSCREEN/2);


    cs.cy=GetSystemMetrics(SM_CYSCREEN/2);


    return CMDIFramewnd ::PreCreateWindow(cs);


    }


    11. 如何将窗口居中显示?


    Easy, Call Function CWnd:: Center Windows


    Example(1): Center Window( ); //Relative to it's parent


    // Relative to Screen


    Example(2): Center Window(CWnd:: GetDesktopWindow( ));


    //Relative to Application's MainWindow


    AfxGetMainWnd( ) -> Center Window( );


    12. 如何让窗口和 MDI窗口一启动就最大化和最小化?


    先说窗口。


    InitStance 函数中设定 m_nCmdShow的 取值.


    m_nCmdShow=SW_SHOWMAXMIZED ; //最大化


    m_nCmdShow=SW_SHOWMINMIZED ; //最小化


    m_nCmdShow=SW_SHOWNORMAL ; //正常方式


    MDI窗口:


    如果是创建新的应用程序,可以用 MFC AppWizard Advanced 按钮并在


    MDI子窗口风格组中检测最大化或最小化; 还可以重载 MDI Window


    PreCreateWindow函数,设置WS_MAXMIZE or WS_MINMIZE;


    如果从 CMDIChildWnd 派生,调用 OnInitialUpdate函数中的 CWnd::Show


    Window来指定 MDI Child Window的 风格。


    13. 如何使程序保持极小状态?


    很有意思的 问题


    这么办: 在恢复程序窗体大小时, Windows会发送WM_QUERY-OPEN消息,


    ClassWizard设置成员函数 OnQueryOpen() ,add following code:


    Bool CMainFrame:: OnQueryOpen( )


    {


    Return false;


    }


    14. 如何限制窗口的大小?


    也就是 FixedDialog形式。 Windows 发送 WM_GETMAXMININFO消息来跟踪,


    响应它,OnGetMAXMININFO 中写代码:


    15. 如何使窗口不可见?


    很简单,SW_HIDE 隐藏窗口,可以结合 FindWindow,ShowWindow 控制.


    16. 如何使窗口始终在最前方?


    两种途径.


    BringWindowToTop(Handle);


    SetWindowPos函数,指定窗口的最顶风格,WS_EX_TOPMOST扩展窗口的风格


    Example:


    void ToggleTopMost( CWnd *pWnd)


    {


    ASSERT_VALID(pWnd);


    pWnd ->SetWindowPos(pWnd-> GetStyle( ) &WS_EX_TOPMOST)?


    &wndNoTopMOST: &wndTopMost,0,0,0,0,SSP_NOSIZE|WSP_NOMOVE);


    }


     


    17、如何创建一个字回绕的CEditView


    重载CWnd : : PreCreateWindow和修改CREATESTRUCT结构,关闭CEditView对象的ES_AUTOHSCROLLWS_HSCROLL风格位,由于CEditView : : PreCreateWindow显示设置cs. style,调用基类函数后要修改cs . style


    BOOL CSampleEDitView : : PreCreateWindow (CREATESTRUCT&cs)


    {


    //First call basse class function .


    BOOL bResutl =CEditView : : PreCreateWindow (cs) ;


    // Now specify the new window style .


    cs.style &= ~ (ES_AUTOHSCROLL WS_HSCROLL);


    return bResult ;


    }


    18、通用控件的显示窗口


    MFC提供了几个CView派生的视窗类,封装了通用控件的功能,但仍然使用工


    作框文档显示窗口体系结构:CEditView封装了编辑控件,CTreeView保持了树列表


    控件,CListView封装了列表显示窗口控件,CRichEditView可以处理多种编辑控件。


    19、移动窗口


    调用CWnd : : SetWindowPos并指定SWP_NOSIZE标志。目的位置与父窗口


    有关(顶层窗口与屏幕有关)。调用CWnd : : MoveWindow时必须要指定窗口


    的大小。


    //Move window to positoin 100 , 100 of its parent window .


    SetWindowPos (NULL, 100 , 100 , 0 , 0 , SWP_NOSIZE SWP_NOAORDER);


    20、重置窗口的大小


    调用CWnd: : SetWindowPos并指定SWP_NOMOVE标志, 也可调用


    CWnd : : MoveWindow 但必须指定窗口的位置。


    // Get the size of the window .


    Crect reWindow ;


    GetWindowRect (reWindow );


    //Make the window twice as wide and twice as tall .


    SetWindowPos (NULL , 0 , 0 , reWindow . Width ( ) *2,


    reWindow . Height () * 2,


    SWP_NOMOVE SWP_NOZORDER );


    21、如何单击除了窗口标题栏以外的区域使窗口移动


    当窗口需要确定鼠标位置时Windows向窗口发送WM_NCHITTEST信息,可以处理


    该信息使Windows认为鼠标在窗口标题上。对于对话框和基于对话的应用程序,可


    以使用ClassWizard处理该信息并调用基类函数,如果函数返回HTCLIENT 则表明


    鼠标在客房区域,返回HTCAPTION表明鼠标在Windows的标题栏中。


    UINT CSampleDialog : : OnNcHitTest (Cpoint point )


    {


    UINT nHitTest =Cdialog: : OnNcHitTest (point );


    return (nHitTest = =HTCLIENT)? HTCAPTION : nHitTest ;


    }


    上述技术有两点不利之处,其一是在窗口的客户区域双击时, 窗口将极大;


    其二,它不适合包含几个视窗的主框窗口。还有一种方法,当用户按下鼠标左键


    使主框窗口认为鼠标在其窗口标题上,使用ClassWizard在视窗中处理WM_LBUTTODOWN


    信息并向主框窗口发送一个WM_NCLBUTTONDOWN信息和一个单击测试HTCAPTION


    void CSampleView : : OnLButtonDown (UINT nFlags , Cpoint point )


    {


    CView : : OnLButtonDow (nFlags , pont );


    //Fool frame window into thinking somene clicked on


    its caption bar .


    GetParentFrame ( ) ?> PostMessage (


    WM_NCLBUTTONDOWN , HTCAPTION , MAKELPARAM (poitn .x , point .y) );


    }


    该技术也适用于对话框和基于对的应用程序,只是不必调用CWnd : : GetParentFrame


    void CSampleDialog : : OnLbuttonDown (UINT nFlags, Cpoint point )


    {


    Cdialog : : OnLButtonDow (nFlags, goint );


    //Fool dialog into thinking simeone clicked on its caption bar .


    PostMessage (WM_NCLBUTTONDOWN , HTCAPTION , MAKELPARM (point.x , point. y ) )


    }


    22、如何改变视窗的背景颜色


    Windows向窗口发送一个WM_ERASEBKGND消息通知该窗口擦除背景,可以使用


    ClassWizard重载该消息的缺省处理程序来擦除背景(实际是画),并返回TRUE


    防止Windows擦除窗口。


    //Paint area that needs to be erased.


    BOOL CSampleView : : OnEraseBkgnd (CDC* pDC)


    {


    // Create a pruple brush.


    CBrush Brush (RGB (128 , 0 , 128) );


    // Select the brush into the device context .


    CBrush* pOldBrush = pDC?>SelcetObject (&brush);


    // Get the area that needs to be erased .


    CRect reClip ;


    pDC?>GetCilpBox (&rcClip);


    //Paint the area.


    pDC?> PatBlt (rcClip.left , rcClip.top ,


    rcClip.Width ( ) , rcClip.Height ( ) , PATCOPY );


    //Unselect brush out of device context .


    pDC?>SelectObject (pOldBrush );


    // Return nonzero to half fruther processing .


    return TRUE;


    }


    23、如何改变窗口标题


    调用CWnd : : SetWindowText可以改变任何窗口(包括控件)的标题。


    //Set title for application's main frame window .


    AfxGetMainWnd ( ) ?> SetWindowText (_T("Application title") );


    //Set title for View's MDI child frame window .


    GetParentFrame ( ) ?> SetWindowText ("_T ("MDI Child Frame new title") );


    //Set title for dialog's push button control.


    GetDigitem (IDC_BUTTON) ?> SetWindowText (_T ("Button new title ") );


    如果需要经常修改窗口的标题(注:控件也是窗口),应该考虑使用半文档化


    的函数AfxSetWindowText。该函数在AFXPRIV.H中说明,在WINUTIL.CPP中实现,在


    联机帮助中找不到它,它在AFXPRIV.H中半文档化,在以后发行的MFC中将文档化。


    AfxSetWindowText的实现如下:


    voik AFXAPI AfxSetWindowText (HWND hWndCtrl , LPCTSTR IpszNew )


    {


    itn nNewLen= Istrlen (Ipaznew);


    TCHAR szOld [256];


    //fast check to see if text really changes (reduces flash in the controls )


    if (nNewLen >_contof (szOld) ||


    : : GetWindowText (hWndCrtl , szOld , _countof (szOld) !=nNewLen ||


    Istrcmp (szOld , IpszNew )! = 0


    {


    //change it


    : : SetWindowText (hWndCtrl , IpszNew );


    }


    }


    24、如何防止主框窗口在其说明中显示活动的文档名


    创建主框窗口和MDI子窗口进通常具有FWS_ADDTOTITLE风格位, 如果不希望在


    说明中自动添加文档名, 必须禁止该风格位,可以使用ClassWizard重置


    CWnd: : PreCreateWindow并关闭FWS_ADDTOTITLE风格。


    BOOL CMainFrame : : PreCreateWindow (CREATESTRUCT&cs)


    {


    //Turn off FWS_ADDTOTITLE in main frame .


    cs.styel & = ~FWS_ADDTOTITLE ; 


    return CMDIFrameWnd : : PreCreateWindow (cs );


    }


    关闭MDI子窗口的FWS _ADDTOTITLE风格将创建一个具有空标题的窗口,可以调


    CWnd: : SetWindowText来设置标题。记住自己设置标题时要遵循接口风格指南。


     

    相关文章
    本页查看次数: