If you want to display a control to the user when the parent window comes up, at design time, set the Visible property to True. If you are programmatically creating the control, add the WS_VISIBLE property. Here is an example: void CSecondDlg::OnSecondControl() { // TODO: Add your control notification handler code here CWnd *Second = new CWnd; Second->Create(NULL, NULL, WS_CHILD | WS_VISIBLE, ); } If you do not set the Visible property to True or do not add the WS_VISIBLE property, the control would be hidden (but possibly available) when its parent window comes up. Later on, you can display the control by calling the CWnd::ShowWindow() method. Remember that this method is used to either hide or to reveal a control by passing the appropriate constant, SW_HIDE to hide and SW_SHOW to display it. Here is an example that displays a control that missed the WS_VISIBLE property when it was created: void CSecondDlg::OnFirstControl() { // TODO: Add your control notification handler code here CWnd *First = new CWnd; CString StrClsName = AfxRegisterWndClass(CS_VREDRAW | CS_HREDRAW, LoadCursor(NULL, IDC_CROSS), (HBRUSH)GetStockObject(BLACK_BRUSH), LoadIcon(NULL, IDI_WARNING)); First->Create(StrClsName, NULL, WS_CHILD); First->ShowWindow(SW_SHOW); } When the ShowWindow() method is called with the SW_SHOW value, if the control was hidden, it would become visible; if the control was already visible, nothing would happen. In the same way, when this method is called with the SW_HIDE argument, the control would be hidden, whether it was already hidden or not. If you want to check the visibility of a control before calling this method, you can call the CWnd::IsWindowVisible() method. Its syntax is: BOOL IsWindowVisible() const; This method returns TRUE if the control that called it is already visible. If the control is hidden, the method returns FALSE. By default, after a control has been created, it is available to the user. If it is a non-static control, the user can possibly change select or change its value. If you do not want the control to be immediately usable but do not want to hide it, you can disable it. If you are designing the control, set its Disable property to True or checked. If you are programmatically creating the control, add the WS_DISABLED style as follows: void CSecondDlg::OnSecondControl() { // TODO: Add your control notification handler code here CWnd *Second = new CWnd; Second->Create(NULL, NULL, WS_CHILD | WS_VISIBLE | WS_DISABLED, ); } When a control is disabled, the user can see it but cannot change its value. If for any reason a control is disabled, to enable it, you can call the CWnd::EnableWindow() method. In fact, the EnableWindow() method is used either to enable or to disable a window. Its syntax is: BOOL EnableWindow(BOOL bEnable = TRUE); Here is an example that disables a control called Memo: void CSecondDlg::OnDisableMemo() { // TODO: Add your control notification handler code here Memo->EnableWindow(FALSE); } When calling the the EnableWindow() method, if you pass the TRUE value, the control is disabled, whether it was already disabled or not. If you pass the FALSE constant, it gets enabled, even it was already enabled. Sometimes you may want to check first whether the control is already enabled or disabled. This can be accomplished by calling the CWnd::IsWindowEnabled(). Its syntax is: BOOL IsWindowEnabled( ) const; This method checks the control that called it. If the control is enabled, the member function returns TRUE. If the control is disabled, this method returns FALSE. Here is an example: void CSecondDlg::OnDisableMemo() { // TODO: Add your control notification handler code here if( Memo->IsWindowEnabled() == TRUE ) Memo->EnableWindow(FALSE); else // if( !Memo->IsWindowEnabled() ) Memo->EnableWindow(); } Here is a simplified version of the above code: void CSecondDlg::OnDisableMemo() { // TODO: Add your control notification handler code here Memo->EnableWindow(!Memo->IsWindowEnabled()); } One of the visible features you can give a control is to draw its borders. Fortunately, most Microsoft Windows controls have a 3-D look by default. Some other controls must explicitly be given a border. To add a border to a control, at design time, set its Border property to True. If you are programmatically creating the control, add the WS_BORDER style to it: void CSecondDlg::OnSecondControl() { // TODO: Add your control notification handler code here CWnd *Second = new CWnd; Second->Create(NULL, NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, ); } | | To raised the borders of such a control, add the WS_THICKFRAME to its styles: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CtrlBorders->Create(NULL, NULL, WS_CHILD | WS_VISIBLE | WS_THICKFRAME, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | When using the controls of a dialog, the user may choose to press Tab to navigate from one control to another. This process follows a sequence of controls that can receive input in an incremental order. To include a control in this sequence, at design time, set its Tab Stop property to True. If you are creating your control with code, to make sure that it can fit in the tab sequence, add the WS_TABTOP. Here is an example: void CSecondDlg::OnThirdControl() { // TODO: Add your control notification handler code here Memo->Create("EDIT", "Voice Recorder", WS_CHILD | WS_VISIBLE | WS_BORDER | WS_TABSTOP, ); CWnd *CanTab = new CWnd; CanTab->Create("BUTTON", "&Apply", WS_CHILD | WS_VISIBLE | WS_BORDER | WS_TABSTOP, ); } Introduction to Extended Styles |
| Besides the above regular styles and properties used on controls, if you to add a more features to a window, you can create it using the CWnd::CreateEx() method. It comes in two versions as follows: BOOL CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle, int x, int y, int nWidth, int nHeight, HWND hwndParent, HMENU nIDorHMenu, LPVOID lpParam = NULL ); | BOOL CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, LPVOID lpParam = NULL); | In Visual C++ 6, some of the extended styles are on the Styles tab of the Properties window and some others are in the Extended Styles tab: In Visual C++ 7, all styles are listed in the Properties window's vertical list: Text-based controls (such as the static label, the edit box, or the rich edit control) align their text to the left by default. This means that the control displays, its text starts on the left side of its area. To align text to the left on a control that allows it, at design time, select the Left value in the Align Text combo box. The default text alignment of a text-based control is to the left. This is because the WS_EX_LEFT extended style is applied to it. If you want to reinforce this, you can add that style as the dwExStyle argument of the CreateEx() method. This can be done as follows: void CSecondDlg::OnThirdControl() { // TODO: Add your control notification handler code here Memo->CreateEx(WS_EX_LEFT, "EDIT", "Voice Recorder", WS_CHILD | WS_VISIBLE | WS_BORDER | WS_TABSTOP, ); } Many text-based controls, including the button, allow you to set their text close to the right border of their confined rectangle. To do this at design time, select the Right value in the Align Text combo box. If you are programmatically creating the control, to align its text to the right, set or add the WS_EX_RIGHT extended style. Here is an example: void CSecondDlg::OnThirdControl() { // TODO: Add your control notification handler code here CWnd *CanTab = new CWnd; CanTab->CreateEx(WS_EX_RIGHT, "BUTTON", "&Apply", WS_CHILD | WS_VISIBLE | WS_BORDER | WS_TABSTOP, ); } | | Besides the border features of the dwStyle argument of the Create() member function, the CreateEx() method provides other border drawings to apply to a control. Static Borders: To give a 3-D appearance to a control, especially one that does not receive input from the user (such as a static control), you can apply a static edge to it. To do this, at design time, set the Static Edge property to True. At run time, you can set or add the WS_EX_STATICEDGE extended style. Here is an example: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_STATICEDGE, "STATIC", NULL, WS_CHILD | WS_VISIBLE, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | Sunken Borders: You can further sink the borders of a control, give them an advanced 3-D appearance. To apply this effect at design time, set the Client Edge property to True. If you are programmatically creating a control, add the WS_EX_CLIENTEDGE extended style. Here is an example: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_CLIENTEDGE, "STATIC", NULL, WS_CHILD | WS_VISIBLE, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | Raised Borders: A control appears raised when it borders come out of the client area. This creates the effect of light left and top borders while the right and bottom borders appear darker. To apply this property, at design time, set the Modal Frame property to True or, at run time, add the WS_EX_DLGMODALFRAME extended style. Here is an example: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_DLGMODALFRAME, "STATIC", NULL, WS_CHILD | WS_VISIBLE, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | To raise only the borders of a control without including the body of the control, you can combine WS_EX_CLIENTEDGE and the WS_EX_DLGMODALFRAME extended styles. Here is an example: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_CLIENTEDGE | WS_EX_DLGMODALFRAME, "STATIC", NULL, WS_CHILD | WS_VISIBLE, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | Ownership and Identification: Parent Windows Logistics |
| One of the main roles of a parent window is its ability to host other controls. Such controls are called its children because the parent carries them when it is moved and it controls their availability. These controls are also called its clients because they request the parental service from it. As mentioned already, at design time, you can pick up a control from the Controls toolbox and position it on the parent window. The child controls are added in an area referred to as the body of the parent window. The body of a window is a rectangular shape also called the client area: A control can be added only in the client area and cannot exceed that area. To find out how much room a parent window is making available to its children, that is, to get the dimensions (and screen location) of the client area, you can call the CWnd::GetClientRect() function. Its syntax is: void GetClientRect(LPRECT lpRect) const; This function takes as argument a RECT or CRect variable and stores the location and dimension of the client rectangular area in it. In the following example, the GetClientRect() function is called to get the dimensions of the client area of a dialog box and use the resulting rectangular to paint that area: void CSecondDlg::OnPaint() { CPaintDC dc(this); // device context for painting // TODO: Add your message handler code here CRect Recto; CBrush BlueBrush(RGB(0, 128, 255)); CPen BluePen(PS_SOLID, 1, RGB(0, 128, 255)); GetClientRect(&Recto); CPen *OldPen = dc.SelectObject(&BluePen); CBrush *OldBrush = dc.SelectObject(&BlueBrush); dc.Rectangle(Recto); dc.SelectObject(OldBrush); dc.SelectObject(OldPen); // Do not call CDialog::OnPaint() for painting messages } To manage the area available to the control you are adding at design time, Visual C++ presents two options. If you display the grids, which is done by clicking the Toggle Grid button , the control can be positioned and resized anywhere in the area inside the borders of the parent window, as seen on the above red rectangle. No control can be positioned outside of the client area. If you hide the grids, which is done by clicking the Toggle Guides button , a (blue dotted) rectangle appears in the client area: This (blue dotted) rectangle allows you to effectively control the area available to your controls. To set the location and dimensions of the available area, click one of the (blue dotted) rectangle borders or corners and drag in the desired direction. If a control is positioned on a border that is moving, the control would be repositioned accordingly: After setting the (blue dotted) rectangular location and area, you can add but cannot move a control outside of that (blue dotted) rectangle. The idea is to allow you to design a control or a group of controls in a (temporary) confined area. The (blue dotted) rectangle provides an effective means of aligning controls. After using it, if you want to add and manipulate controls outside of it, you should display the grids. As mentioned already, the controls are confined to the client area offered by the parent window. After visually adding a control to a parent window, it assumes a location and takes some dimensions in the client area. The origin of the rectangular client area is on the upper-left corner of the parent window. The horizontal measurements move from the origin to the right. The vertical measurements move from the origin to the bottom: At design time, to set the location and dimension of a control, after placing it on the parent, click and hold your mouse on the control, then drag left, top, right, and down. While doing that, refer to the right section of the status bar for the current location and dimensions of the control: While moving and resizing the control, if you need help with this exercise, click the Toggle Grid button . When the grids are displaying, the control can only be located on grid indicators and when resizing the control, it can only fit on grid indicators. If you want to be more precise and ignore the grids, you have various options: - If you want to keep the grids but modify their distances, open the Guide Settings dialog box available from the Layout (MSVC 6) or Format (MSVC 7) menu then modify the Width and/or Height values in the Grid Spacing section:
- If you want to keep the grids, press and hold Alt while you are dragging. While alt is down, you can position and dimension the control anyhow regardless of the grids settings, as long as you stay confined to the client area
- You can hide the grids by clicking the Toggle Guides button . When this button is down, a control can moved and dimensioned anyhow as long as it stays within the confinements of the (blue dotted) rectangle
- To use neither the grids nor the rectangular guide, click the button that is down to remove it.
Whether the grids or the rectangular guide are displaying, to position a control with more precision, click it to select, then press one of the arrow keys to move the control by one pixel If you are programmatically creating the control, to set its location, if you are using the first version of the CreateEx() method, specify the x value for the distance from the left border of the client area to the left border of the control, and pass the desired y value for the upper distance from the top border of the client area to the top border of the control. Here is an example: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* Panel = new CStatic; Panel->CreateEx(NULL, "STATIC", NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, 32, 15, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | If you specify negative values for the left and top distances, either the left or the top borders, respectively, will be hidden. To set the dimensions of the control, if you are using the first version of the CreateEx() member function, specify the nWidth and the width and the nHeight for the height of the control. If you specify measures that are higher than the width of the client area - x or the height of the client area - y, the right border or the bottom border, respectively, of the control will be hidden. To specify the location and the dimensions of the control at the same time, pass a RECT or a CRect variable as the rect argument of the Create() or the second version of the CreateEx() methods. Here are examples: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* btnApply = new CWnd; RECT Recto = {12, 16, 128, 64}; btnApply->Create("BUTTON", "&Apply", WS_CHILD | WS_VISIBLE, Recto, ); CWnd *btnDismiss = new CWnd; btnDismiss->Create("BUTTON", "&Dismiss", WS_CHILD | WS_VISIBLE, CRect(12, 68, 128, 120), ); CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_DLGMODALFRAME, "STATIC", NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, 132, 16, 220, 120, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | | It is important to understand that there is a tremendous difference on the way the dimensions of a control are calculated. Although we have not mentioned it, and we will not spend time on it because we are not going to treat controls differently, Common Controls are objects that started to be released with Windows 95. They calculate their width from the left border of the control to the right and the height of the control is calculated from the top border of the control down. Other (earlier) controls calculate their width by subtracting the specified nWidth value from the x value. This means that the nWidth starts on the left border of the client area. Based on this, for example if you are creating a button (class name = "BUTTON"), if you provide a value that <= x, the control would not be displayed because its width would be either nil or negative. Observe the following code: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* btnApply = new CWnd; RECT Recto = {12, 16, 12, 64}; btnApply->Create("BUTTON", "&Apply", WS_CHILD | WS_VISIBLE, Recto, ); CWnd* Panel = new CStatic; Panel->CreateEx(WS_EX_DLGMODALFRAME, "STATIC", NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, 132, 16, 132, 120, ); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | The x and nWidth values of the button have the same value. In the same way, the x and the nWidth values of the panel object have the same value. When this program is executed, the button would not display because its width = 0. The panel object would display with a width of 132. Once the control is already positioned on the client area, to get its location and dimensions, you can call the CWnd::GetWindowRect() method. Here is an example: void CClientAreaDlg::OnBtnCalculate() { // TODO: Add your control notification handler code here CRect Recto; char LocDim[80]; m_Rectangle.GetWindowRect(&Recto); sprintf(LocDim, "Left: %d, Top: %d, Width: %d, Height: %d", Recto.left, Recto.top, Recto.Width(), Recto.Height()); SetWindowText(LocDim); } | The Control Location and its Origin |
| When calling either the GetClientRect() or the GetWindowRect() methods to get the location and the dimensions of a control or another object, it is important to know the origin of the produced rectangle. By default, the rectangle returned by the GetWindowRect() method called by a control has its origin on the top left corner of the monitor and not on the top-left corner of the parent window. Consider the following button event. It gets the location and dimensions of a static control and stores them in a CRect variable. Then it paints a rectangle (it is supposed to paint the static control) located on, and equal to the dimensions, of the static control. Finally, it displays the properties of the rectangle that holds a static control: void CClientAreaDlg::OnBtnCalculate() { // TODO: Add your control notification handler code here CRect Recto; char LocDim[80]; m_Rectangle.GetWindowRect(&Recto); CClientDC dc(this); CBrush BlueBrush(RGB(0, 128, 192)); dc.SelectObject(&BlueBrush); dc.Rectangle(Recto); sprintf(LocDim, "Left: %d, Top: %d, Width: %d, Height: %d", Recto.left, Recto.top, Recto.Width(), Recto.Height()); SetWindowText(LocDim); } After executing the program and moving the dialog box somewhere to the middle center of the screen and clicking the button, the result is as follows: After moving the dialog box close to the top-left section of the screen and clicking the button again, the result is the following: This clearly demonstrates that, although the static control is a child of the dialog box, the rectangle returned by the GetWindowRect() method is based on the screen and not the client area of the parent window. This is not an anomaly. It is purposely done so you can specify what origin you want to consider. As seen in previous lessons, the origin of the screen is positioned on the top-left corner of the monitor. The origin of a client area is placed on its top-left corner. For example, the origin used by the above GetWindowRect() method is based on the screen. If you want the rectangle resulting from a call to either the GetClientRect() or the GetWindowRect() methods to be based on the client area of the control that called it, you can transfer the origin from the screen to the client. This is conveniently done with a simple call to the CWnd::ClientToScreen() method. It is overloaded as follows: void ClientToScreen(LPPOINT lpPoint) const; void ClientToScreen(LPRECT lpRect) const; If the location you had requested is a point, pass its POINT or CPoint variable to the ClientToScreen() method. If the value you requested is a rectangle, pass its RECT or CRect variable. Here is an example: void CClientAreaDlg::OnBtnCalculate() { // TODO: Add your control notification handler code here CRect Recto; char LocDim[80]; m_Rectangle.GetWindowRect(&Recto); CClientDC dc(this); CBrush BlueBrush(RGB(0, 128, 192)); ScreenToClient(Recto); dc.SelectObject(&BlueBrush); dc.Rectangle(Recto); sprintf(LocDim, "Left: %d, Top: %d, Width: %d, Height: %d", Recto.left, Recto.top, Recto.Width(), Recto.Height()); SetWindowText(LocDim); } | | This time, even if the dialog box moves, the GetWindowRect() method returns the same rectangle. If the location and/or dimension are given in client coordinates, to convert them to screen coordinates, call the ScreenToClient() method. It is overloaded as follows: void ScreenToClient(LPPOINT lpPoint) const; void ScreenToClient(LPRECT lpRect) const; After specifying that the control you are creating is a child, which is done by placing the control on a form or a dialog box or by using or adding the WS_CHILD style, you must specify what window is the parent of your control. The parent of a control is automatically set when the control is placed on a host: the host becomes the parent. If you are programmatically creating the control, you can specify its parent by passing a window handle as the pParentWnd argument of the Create() or or the second version of the CreateEx() member functions. If you are creating the control in a member function or an event of the parent window that is a descendent of CWnd, you can pass this argument as the this pointer. Here is an example: void CSecondDlg::OnFirstControl() { // TODO: Add your control notification handler code here CWnd *First = new CWnd; CString StrClsName = AfxRegisterWndClass(CS_VREDRAW | CS_HREDRAW, LoadCursor(NULL, IDC_CROSS), (HBRUSH)GetStockObject(BLACK_BRUSH) LoadIcon(NULL, IDI_WARNING)); First->Create(StrClsName, NULL,WS_CHILD | WS_VISIBLE | WS_BORDER, CRect(20, 20, 120, 60), this, ); } Specifying the parent as the this pointer indicates that when the parent is destroyed, it will also make sure the child control is destroyed. If you want the application to be the parent and owner of the control, use the first version of the CreateEx() method and pass the handle of your application as the hwndParent argument. Here is an example: BOOL CClientAreaDlg::OnInitDialog() { CDialog::OnInitDialog(); SetIcon(m_hIcon, TRUE); // Set big icon SetIcon(m_hIcon, FALSE); // Set small icon // TODO: Add extra initialization here CWnd* stcLogo = new CWnd; stcLogo->CreateEx(WS_EX_DLGMODALFRAME, "STATIC", NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, 240, 90, 90, 40, this->m_hWnd, ); return TRUE; // return TRUE unless you set the focus to a control } If, during the lifetime of your program you want to change a control's parent, you can call the CWnd::SetParent() method. Its syntax is: CWnd* SetParent(CWnd* pWndNewParent); Here is a rough example: class CSecondDlg : public CDialog { // Construction public: CSecondDlg(CWnd* pParent = NULL); // standard constructor . . . private: CWnd *lblFirstName; CWnd *AnotherParent; }; | CSecondDlg::CSecondDlg(CWnd* pParent /*=NULL*/) : CDialog(CSecondDlg::IDD, pParent) { //{{AFX_DATA_INIT(CSecondDlg) // NOTE: the ClassWizard will add member initialization here //}}AFX_DATA_INIT lblFirstName = new CWnd; AnotherParent = new CWnd; } | void CSecondDlg::OnThirdControl() { // TODO: Add your control notification handler code here CWnd *Memo = new CWnd; Memo->Create("EDIT", "Voice Recorder", WS_CHILD | WS_VISIBLE | WS_VSCROLL | WS_BORDER | ES_MULTILINE | ES_WANTRETURN, CRect(20, 80, 185, 224), this, ); Memo->SetParent(AnotherParent); } | To find out what window is the parent of a certain control, call the CWnd::GetParent() method. Its syntax is: CWnd* GetParent() const; This method returns a pointer to the parent of the control. After adding a control to your your application, you must identify it. An identifier is not a string, it is a constant integer that is used to identify a control. Therefore, it is not the name of the control. If you have just added the control, Visual C++ would assign a default identifier. If the control is static, it would have the IDC_STATIC identifier. Every other control gets an identifier that mimics its class name followed by a number. An example would be IDC_EDIT1. If you add another control of the same kind, it would receive an identifier with an incremental number. To have a better idea of each control, you should change their IDs to make them more intuitive. To do this, if you are visually adding the control, first display the Properties window: The identifier can be a word or a hexadecimal number. Here are the rules you must follow when assigning identifiers: It you decide to use a word | If you decide to use a hexadecimal number | - The identifier must start with an underscore or a letter
- The identifier must be in one word
- After the first character, the identifier can have letters, underscores, and digits in any combination
- The identifier must not have non-alphanumeric characters
- The identifier cannot have space(s)
| - The identifier must start with a digit or one of the following letters: a, b, c, d, e, f, A, B, C, D, E, or F
- The identifier can have only digits and the above letters except this: if the identifier starts with 0, the second character can be x or X followed by the above letters and digits so that, when converted to decimal, it must have a value between -32768 and 65535
| | Here are suggestions you should follow: - The identifier should be in all uppercase
- The identifier should have a maximum of 30 characters
- If the object is a form or dialog box, its identifier should start with IDD_ followed by a valid name. Examples: IDD_EMPLOYEES or IDD_EMPL_RECORS
- If you are creating a control, the identifier should start with IDC_ followed by a name. Examples: IDC_ADDRESS or IDC_FIRST_NAME
- If you are creating a static control but plans to use it in your code, (you must) change its identifier to something more meaningful. If the static control will hold text and you plan to change that text, you can identify it with IDC_LABEL. For a static control used to display a picture of employees, you can identify it as IDC_EMPL_PICTURE
The identifiers of the controls used in your application must be listed in a header file called Resource.h If you visually add controls to your application, their identifiers are automatically added to this file. Even if you change the identifier of the control, Visual C++ updates this file. You can also manually add identifiers to this file, provided you know why you are doing that. If you are programmatically creating a control, you can locally set an identifier by specifying a (randomly selected) decimal or hexadecimal number as the nID argument of the Create() or the second version of the CreateEx() methods. Here are examples: BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd* btnApply = new CWnd; RECT Recto = {12, 16, 128, 64}; btnApply->Create("BUTTON", "&Apply", WS_CHILD | WS_VISIBLE, Recto, this, 0x10); CWnd *btnDismiss = new CWnd; btnDismiss->Create("BUTTON", "&Dismiss", WS_CHILD | WS_VISIBLE, CRect(12, 68, 128, 120), this, 258); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | Alternatively, you can first create an identifier in the String Table. We saw how to do this in Lesson 3. You can also create an identifier using the Resource Symbols dialog box. Then use that identifier for your control. Here is an example: | BOOL CBordersDlg::OnInitDialog() { CDialog::OnInitDialog(); // TODO: Add extra initialization here CWnd *btnDismiss = new CWnd; btnDismiss->Create("BUTTON", "&Dismiss", WS_CHILD | WS_VISIBLE, CRect(12, 68, 128, 120), this, IDN_DISMISS); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } | Control Functionality: Variables |
| There are two main ways you can refer to a control in your code. The first technique consists of using its identifier. This can help you find out what type of control the identifier refers to. In this case, you usually must cast the result to the right control's class. There are two prerequisites. First, you must know the ID or nID of the control you want to refer to. Then, you can call the CWnd::GetGldItem() method. It comes in two versions as follows: CWnd* GetDlgItem(int nID) const; void CWnd::GetDlgItem(int nID, HWND* phWnd) const; To refer to a control in your code, you can also declare a variable for it. There are two types of variable you can use to refer to a control. If you are more interested in the value held by a control, then you can declare a value variable for it. The variable must be a valid type of value that the control can hold. For example, the text in a text-based control (such as an edit box or a combo box) is a CString. Therefore, you can declare a CString variable for the control. If you want to refer to the object as a control and get access to its properties, you should declare the variable as Control. To help with making that decision, you should use either ClassWizard of MSVC 6 or the Add Member Variable Wizard of MSVC 7. Because the wizard also inserts the appropriate code in the header and source files, you should rely on it instead of manually declaring the variable(s). If you want to refer to the object as a control and separately refer to its value, you can declare both the Value and the Control variables. |