Anybody who has dabbled with Windows even for a few hours has knowledge about the existence of a program called Screen Saver. A Screen Saver can be installed by right clicking on the screen saver’s executable file and selecting the ‘Install’ option from the menu that appears. As soon as we do this, our screen saver file is entered in the list of screen savers that the Windows OS maintains. This list is displayed in a combo box in a dialog along with its preview. This dialog box, the combo box and the preview window are shown in the Figure 1. Note that our screen saver ‘mysaver’ also appears in the list.
Figure 1
The screen savers are different than other executables in their extension. All screen savers have an extension of ‘.scr’ rather than the traditional ‘.exe’. To ensure that it gets this extension, follow these steps:
- Select ‘Project | Settings | Debug’ property page from the ‘Developer Studio’. Change the extension from ‘.exe’ file to ‘.scr’ in the edit box with a title ‘Executable for debug session’.
- Select ‘Project | Settings | Link’ property page from the ‘Developer Studio’. Change the extension from ‘.exe’ file to ‘.scr’ in the edit box that has a title ‘Output file name’. Press OK.
Figure 2
If we click on the ‘Preview’ button in the dialog box in Figure 1 the screen saver would be put into action. If we click on another button called ‘Settings’ one more dialog box would appear which would help the user to control the various elements of the screen saver. Naturally, the appearance and contents of this dialog box would vary from one screen saver to another. The dialog box for our screen saver is shown in the following figure.
Figure 3
As soon as we change any of the settings in the dialog box the change is immediately noticed in the preview window in the dialog box.
Working
The screen saver developed here randomly displays colorful squares/circles on the screen. It uses an array of structures s[ ] to display the squares/circles. We have defined this array to hold 10 elements. The information stored in this array controls the size, position and color of each square/circle. At any point of time, a total of 10 squares/circles are displayed. When the 11th square/circle is drawn the 1st one is erased. This continues till either some key is depressed or the mouse is moved or any button on the mouse is clicked.
The screen saver program runs in three different modes:
- Full Screen mode: The Screen Saver runs in this mode in 4 cases:
- When the PC is idle for a time greater than one that has been set up through the spin button control in the ‘Start | Settings | Control Panel | Display | Screen Saver’ property page.
- When the ‘Preview’ button in ‘Start | Settings | Control Panel | Display | Screen Saver’ property page is clicked.
- When we right click on the screen saver file and select ‘Test’ from the menu that appears.
- When we run the screen saver by double clicking the screen saver from the ‘Explorer’ or from ‘Start | Run’.
- Small Preview Window mode: The Screen Saver runs in this mode in 2 cases:
- When ‘Start | Settings | Control Panel | Display | Screen Saver’ property page is displayed.
- When we right click on ‘.scr’ file and select ‘Install’.
- Settings Dialog Preview mode: The Screen Saver runs in this mode in 3 cases:
- When we run the screen saver from the ‘Developer Studio’.
- When we right click on the screen saver file and select ‘Configure’ from the menu that appears as shown in the Figure 4.
- When we click on the ‘Settings’ button from ‘Start | Settings | Control Panel | Display | Screen Saver’ property page.
Figure 4
Figure 1 shows the screen saver active in the ‘Small Window Preview’ mode, whereas Figure 3 shows it active in the ‘Settings Dialog Preview’ mode. The following figure shows it active in the ‘Full Screen’ mode.
Figure 5
Depending upon which of the above three modes is used to invoke the screen saver it is passed an argument ‘s’, ‘p’ and ‘c’ respectively. However there are two exceptions here. In cases 3(a) and 3(b) above no argument is passed to the program. However, in these two cases too we want the screen saver to run in ‘Settings Dialog preview mode’. It is necessary to identify which of these three cases has occurred, because the size of the window where the random squares/circles are to appear is different in the three cases.
When the ‘Settings’ dialog box is popped up, we have made a provision to let the user select the shape of the objects (rectangle or circle) that are going to get displayed on the screen when the screen saver goes into action. Additionally, the user can also select the fill style for the rectangles/circles. We have provided three styles; solid, hatch and pattern to choose from. For permitting the user to select the shape and the style we have provided two combo boxes in the ‘Settings’ dialog box.
Irrespective of the mode in which our Screen Saver is running, the control would ultimately land in the myapp::InitInstance( ) function. Here it is first determined which command line argument has been passed (if at all) to our program by calling the myapp::checkoption( ) function. It then appropriately calls the function myapp::doconfig( ), myapp::dofullscreen( ), or myapp:: doperview( ).
The myapp::doconfig( ) Function
If it is determined that the settings dialog box should be popped up, an object of the settings dialog class is created through the statement,
settingdialog d ;
This calls the constructor of the settings dialog. Here the settings dialog is created in memory. This dialog is then displayed by calling CDialog::DoModal( ).
The myapp::dopreview( ) Function
The code of this function given below:
void myapp::dopreview( )
{
CWnd* parent = CWnd::FromHandle ( ( HWND ) atol ( __argv[2] ) ) ;
CRect r ;
parent -> GetClientRect ( &r ) ;
drawwnd* p = new drawwnd ( TRUE ) ;
p -> create ( NULL, WS_VISIBLE | WS_CHILD, r, parent, NULL ) ;
m_pMainWnd = p ;
}
When the function is called the drawing activity should take place in the preview window. The drawing activity would be managed by the drawwnd class object. When this object is created the window associated with it should be the child of the preview window present in the property page. Hence the address of this parent window needs to be determined. This has been achieved by calling the function CWnd::FromHandle( ). The argument passed to this function is the handle to the preview window of the property page. This handle is passed to our program as a command line argument (argv[2]). Once the pointer to the parent window has been obtained the drawwnd::create( ) function is called to create the window.
The myapp::dofullscreen( ) Function
Lastly, if it is determined that the actual screen saver should be put into action then a full screen window is created. The code of dofullscreen( ) is shown below:
void myapp::dofullscreen( )
{
saverwindow* p = new saverwindow ( TRUE ) ;
p -> create( ) ;
m_pMainWnd = p ;
}
The Settings Dialog
When the DoModal( ) function is called to display the dialog box control first reaches settingdialog::OnInitDialog( ). The code of this function is given below.
BOOL mydialog::OnInitDialog( )
{
m_shape = AfxGetApp( ) -> GetProfileInt ( "Config", "Shape", 0 ) ;
m_fillstyle = AfxGetApp( ) -> GetProfileInt ( "Config", "FillStyle", 0 ) ;
CDialog::OnInitDialog( ) ;
CRect r ;
CStatic *s = ( CStatic * ) GetDlgItem ( IDC_PREVIEW ) ;
s -> GetWindowRect ( &r ) ;
ScreenToClient ( &r ) ;
m_preview.create ( NULL, WS_VISIBLE | WS_CHILD, r, this, NULL ) ;
CenterWindow( ) ;
return TRUE ;
}
In this function, to begin with, the values of shape and fill style are read from the registry by calling the CWinApp::GetProfileInt( ) function. When you run the program for the first time there won’t be any entry in the registry. Hence default values of 0 and 0 (last parameter passed to GetProfileInt( )) would be assumed for the shape and fill style. The values read from the registry (or the assumed values if there are no entries in the registry) are used to show the default selections for shape and fill style when the dialog is popped up.
Next, the base class implementation of OnInitDialog( ) is called, which in turn calls settingdialog::DoDataExchange( ). In this function the values of m_shape and m_fillstyle are used to set up the default selections of the two combo boxes. Next, in the OnInitDialog( ) function the size of the preview window is obtained and a preview window is created. Note that the object m_preview (object of drawwnd class) which is a private data member is used to create the preview window. As the window gets created, the drawwnd::OnCreate( ) handler gets called. Here, to set the speed at which the squares/circles should be displayed in the window, the CWnd::SetTimer( ) function is called. This function simply sets a timer. Later on we would see how this timer gets serviced.
As we make selections in the dialog box its results are immediately shown in the preview window. To ensure this a function settingdialog::newshapestyle( ) is called as soon as the selections from the combo boxes are changed. This function updates the drawing tools by calling the drawwnd::setdrawingtool( ) function.
Finally, when the user dismisses the dialog box by clicking OK the selections made by him are written to the registry by calling CWinApp::WriteProfileInt( ) function. The logic of drawing in the preview window has been managed in the drawwnd class.
Drawing in The Window
To manage drawing in the small preview window or the full screen window we have developed a class called drawwnd. Whenever an object of this class is created its zero argument constructor gets called. This constructor function is shown below:
drawwnd::drawwnd ( BOOL deleteflag )
{
m_deleteflag = deleteflag ;
m_shape = AfxGetApp( ) -> GetProfileInt ( "Config", "Shape", 0 ) ;
m_fillstyle = AfxGetApp( ) -> GetProfileInt ( "Config", "FillStyle", 0 ) ;
for ( int i = 0 ; i < 10 ; i++ )
s[i].x = s[i].y = -1 ;
setdrawingtool ( m_shape, m_fillstyle ) ;
currentnum = 0 ;
}
Here, to begin with, the values of shape and fill style are read from the registry. Next, the array s[ ] is initialised such that the x, y coordinates of all 10 squares/circles are set to -1. These values are later on used to keep track of how many square/circles have been drawn so far. Lastly, the values read from the registry are used to create 10 brushes by calling the drawwnd::setdrawingtool( ) function.
Whenever the full screen window or the small preview window is created, control reaches drawwnd::OnCreate( ). Here we have set up a timer by calling CWnd::SetTimer( ).
When the time interval set in CWnd::SetTimer( ) is over the function drawwnd::OnTimer( ) gets called. The code of this function is shown below:
void drawwnd::OnTimer ( UINT id )
{
CClientDC *dc ;
dc = new CClientDC ( this ) ;
CRect rect ;
GetClientRect ( &rect ) ;
draw ( dc, rect ) ;
delete dc ;
}
In this function we have obtained the device context, determined the client area size and then called drawwnd::draw( ) to actually carry out the drawing.
Full Screen Mode
When the screen saver runs in the full screen mode the myapp::dofullscreen( ) function gets called. In this function we have created an object of saverwindow class which has been derived from the drawwnd class. On creating this object the constructor of saverwindow gets called. In the constructor we have set up the saverwindow::lastpoint variable with a value (-1,-1). This value is later on used in the saverwindow::OnMouseMove( ) function, as we would soon see. Once the object has been created, using it, saverwindow::create( ) function is called. This function in turn calls drawwnd::create( ) with WS_EX_TOPMOST as the first argument and WS_POPOUP as one of the values in the second argument. The first argument ensures that the full screen window created becomes the topmost window. The value WS_POP ensures that the window created doesn’t have a caption bar and the border.
Once the window is created the drawnd::OnCreate( ) gets called which once again sets up a timer in response to which the OnTimer( ) function calls drawwnd::draw( ) to do the drawing in the full screen window.
Disabling The Screen Saver
When the screen saver runs in the full screen mode it should get disabled whenever the user presses any key or performs any mouse operation. It means in the saverwindow class we must handle all keyboard and mouse related messages. Hence we have written the handlers OnKeyDown( ), OnSysKeyDown( ), OnLButtonDown( ), etc. in the saverwindow class. In each of these handlers all that we have done is, sent a WM_CLOSE message to the message queue and then called the base class implementation of the handler. When the WM_CLOSE message would get picked up from the message queue, the application would close itself. The only exception is the OnMouseMove( ) handler. The code of this handler is as shown below:
void saverwindow::OnMouseMove ( UINT flags, CPoint pt )
{
if ( lastpoint == CPoint ( -1, -1 ) )
lastpoint = pt ;
else if ( lastpoint != pt )
PostMessage ( WM_CLOSE ) ;
drawwnd::OnMouseMove ( flags, pt ) ;
}
If you remember, in the constructor of the saverwindow class we have set up the variable saverwindow::lastpoint to a value (-1, -1). In the OnMouseMove( ) handler we have checked whether the coordinates of lastpoint are still (-1, -1). If they are then we do not post the WM_CLOSE message into the message queue. It would be necessary to do so in the following situation. Suppose a WM_MOUSEMOVE message is present in the message queue and the screen saver gets active before the WM_MOUSEMOVE message could get processed. In such an event, as soon as the screen saver becomes active it would get deactivated when the pending WM_MOUSEMOVE message gets processed. Our code given above prevents this by not posting the WM_CLOSE into the message queue in this situation. We also update the value of lastpoint such that next time a WM_MOUSEMOVE message arrives the screen saver is shut down by posting a WM_CLOSE into the message queue.
Destroying The Window
Once the screen saver becomes active we can deactivate it in a number of ways, as discussed above. Even though it has been deactivated from the screen it continues to run in memory. If once again a considerable time elapses without you hitting a key or clicking a mouse yet again it would become active. If it gets activated again then it would yet again reach the myapp::dofullscreen( ) function to create a window. Here the code,
p = new saverwindow;
would get executed again.
But what has happened to the object that p was pointing last time around. It is still surviving, and now if p starts pointing to another object then a memory leak would occur since we would have no way to access the earlier object. Note that the earlier object would not get destroyed through the destructor of myapp. This is because the destructor doesn’t get called since the application is still running and the myapp a object has not gone out of scope.
To avoid this situation we have used a drawwnd::deleteflag variable. While creating the preview window of ‘Start | Control Panel | Display | Screen Saver’, or the full screen window we have passed a value TRUE to the drawwnd constructor which in turn sets it up in drawwnd::m_deleteflag. As against this, in case of the preview window of the settings dialog we have passed a value FALSE to the drawwnd constructor. This is because on dismissing the dialog box the application would come to an end resulting in call to the myapp class’s destructor function. This would destroy the object associated with the preview window using its address which is stored in myapp::m_pMainWnd.
In case of the preview window of ‘Start | Control Panel | Display | Screen Saver’ and the full screen window the window object would be destroyed in the drawwnd::PostNcDestroy( ) function using the m_deleteflag variable.