Pages

12.30.2013

Create a Digital Clock that Talks!

Digital Clock in Lazarus, that speaks!
We are going to make a digital clock that has LCD Screen looking digits. Not only that. It can also speak the time!


Clock is an essential part of our daily lives. Wherever the new technology taking us, we're rediscovering the clock but can't avoid it. We now have smartwatches that shows time, takes photos, receives messages and more.

Okey, all those new technology apart, today we'll build a small digital clock that sits on your screen and shows time in an LCD Display-like digits. Sounds difficult but it's not.

For the talking part, we will use TTS (Text to Speech), so no tension there either. If you are interested on how to make the computer speak, check out this tutorial.


So, how can we show LCD looking displays. The answer is there are many digit-looking fonts out there. We can use them. But using those fonts would mean that the font should be installed in the system that will run the program. That seems an extra burden. Especially if you consider cross-platform scenerio.

So I thought of a different approach. I would use an image to store digits. That way it wouldn't need to install the font and it is possible to use hundreds of images of different fonts without installing them in users' computer.

The Image

I have come up with this image:
digits.bmp for digital clock.

(You can right click and select "Save Image as..." to download and use the image in your project.)


I have downloaded a LCD digit-looking font, Let's Go Digital, from here. Opened photoshop and positioned the digits. Each digit has an imaginery area of 30px x 50px. It has 15 characters/digits/states of that dimension. Depending on the time we'll take certain parts of the image and draw it out in our form. That's our plan. Simple!

You can download thousands of fonts from the internet and place the digits like the above image.

I have saved the image as "digits.bmp" in an empty folder that I made. We'll use the folder as the project folder. There is the Photoshop psd file in the project download, just in case you want to play with it.

The project

Go ahead and start Lazarus.

Create a new Application Project (Project -> New Project -> Applicaiton -> OK).

Now, we would have to save the project. Save the project to the empty directory that you just made. Why are we saving our project that early? Because, we are using the digits.bmp from that folder. If we don't save the project it would run from the Temp directory and image would not be found.

Coding a bit

First, define the TBitmaps that will hold the digits.bmp (digitsBitmap) and the final clock image (clockBitmap).

var
  ...
  digitsBitmap: TBitmap;
  clockBitmap: TBitmap;

Now, add information about the single digits. Our every digit is 30px x 50px. So we store that value in variables:

var
  ...
  digitWidth: Integer = 30;
  digitHeight: Integer = 50;

Now, we'll define what's in our digits.bmp

var
  ...
  digitCount: Integer = 15;
  digitPositions: array[1..15] of String = (
      '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
      '-', ':', 'A', 'P', ' '
  );

digitCount contains the total number of digits in the bmp file. digitPositions array contains 1 to 15 string items containing each digits' real string. Now, we know all the information of the charaters in the bmp file.

Double click on the form and enter:
procedure TForm1.FormCreate(Sender: TObject);
begin
  clockBitmap := TBitmap.Create;
  clockBitmap.SetSize(Width, Height);

  digitsBitmap := TBitmap.Create;
  digitsBitmap.LoadFromFile('digits.bmp');

  drawDigits('--:-- ');
end;

We prepare both the TBitmaps. We create clockBitmap with the dimensions of our form. We will draw this exact TBitmap in the form on its OnPaint event.

We prepare digitsBitmap and load the digits.bmp on it.

We also use drawDigits, which we didn't declare yet. We use the procedure to draw an initial text "--:-- " on the form.

Now select the form and enter the following code in its OnPaint event (Object Inspector -> Events -> OnPaint -> [...]):
procedure TForm1.FormPaint(Sender: TObject);
begin
  if Assigned(clockBitmap) then
    Canvas.Draw(0,0,clockBitmap);
end;

We draw the clockBitmap everytime the form is painted. If you don't know, the form is "painted" everytime the form is resized, gets focus, releaved some part while moving, the form is painted. If we don't draw the bitmap, the revealing part would be blank!

We use Assigned() to see if the bitmap is Create-d (with TBitmap.Create). If the bitmap is not Create-d and we try to draw it then it will raise a SIGSEGV exception. So we check it everytime we use it.

Now enter the following code on the form's OnClose event:
procedure TForm1.FormClose(Sender: TObject; var CloseAction: TCloseAction);
begin
  if Assigned(digitsBitmap) then digitsBitmap.Free;
  if Assigned(clockBitmap) then clockBitmap.Free;
end;

Again, we check with Assigned() then we use .Free to free the TBitmaps we craeted. (It's important that we .Free everything that we .Create-d. If we fail to do that a memory leak might be created.)

Add the following procedure (before the "end." line, under the implementation clause):
procedure TForm1.drawDigit(digit:string; X, Y: Integer);
var
  rectDest, rectSrc: TRect;
  digitNumber: Integer;
  i: Integer;
begin
  // if the bitmaps have not been initialized (Create'd)
  // then we can't work with them!!
  if (Assigned(digitsBitmap)=false)
  or (Assigned(clockBitmap)=false) then
       exit;

  digitNumber:=15; // the default digit position
  for i := 1 to digitCount do begin
    if digitPositions[i] = UpperCase(digit) then begin
       digitNumber:=i;
       Break;
    end;
  end;

  with rectDest do begin
    Left:=X;
    Top:=Y;
    Right:=Left+30;
    Bottom:=Top+digitHeight;
  end;

  with rectSrc do begin
    Left:=digitWidth * (digitNumber-1);
    Top:=0;
    Right:=Left+digitWidth;
    Bottom:=Top+digitHeight;
  end;

  clockBitmap.Canvas.CopyRect(rectDest, digitsBitmap.Canvas, rectSrc);
  FormPaint(Form1);
end;

Now, place your cursor inside the procedure and press Ctrl+Shift+C. That will create a forward declaration under the TForm1 class.

This procedure draws a single digit, given its position, which can be 1 through 15, at a given position (with X and Y). This procedure is only half of the drawing we would do. We would use it in our drawDigits procedure which will use this procedure to draw each single digit.

  digitNumber:=15; // the default digit position
  for i := 1 to digitCount do begin
    if digitPositions[i] = UpperCase(digit) then begin
       digitNumber:=i;
       Break;
    end;
  end;

First we loop through every digit position to find the character in the digits.bmp.

Then we prepare 2 TRects for CopyRect. We copy pixels from the appropriate position from the bmp and draw it in our clockBitmap, to be eventually drawn on the form.

Now add the following procedure:
procedure TForm1.drawDigits(digits:String);
var
  c: Char;
  x: Integer;
begin
  x := 0;
  for c in digits do begin
    drawDigit(c, x, 0);
    Inc(x, digitWidth);
  end;
end;

Like the above procedure, take your cursor inside the procedure and press Ctrl+Shift+C.

Now draw a TTimer (from System tab) in the form. The Interval property is set to 1000. 1000 miliseconds = 1 second. Now double click it and enter:
procedure TForm1.Timer1Timer(Sender: TObject);
var
  TimeString: String;
begin
  TimeString := FormatDateTime('hh:nna/p', Now);
  drawDigits(TimeString);
end;

Now Run the project (F9 ot Run->Run) to see how it looks.

You will see that the form is bigger than the clock. Also, the form is resizable. Set the properties like the following:
Width = 180
Height = 50
BorderIcons->biMaximize = False : to disable maximize button
BorderStyle = bsSingle : to disable resizing

Now Run the project again.

This time it looks more perfect.

Our Digital Clock project source code Lazarus, cross platform


Let it talk!

It would be cool if our little digital clock could talk the time! It is possible, thanks to TTS (Text to Speech). Here is a good wiki about TTS in Lazarus/FPC. We can either use Microsoft Speech API (SAPI) or eSpeak. SAPI is Windows-only. We are going to use SAPI here for a shortcut way. You can implement eSpeak with the code very easily.

Add comobj in your uses clause:
uses
  ...xyz
  {$IFDEF MSWINDOWS}
      ,comobj
  {$ENDIF};

We have used compiler directive {$IFDEF MSWINDOWS} because comobj is only supported in Windows. So if the program is compiled in a platform other than Windows, it will not be used.

On the Form's OnClick event enter:
procedure TForm1.FormClick(Sender: TObject);
var
  SavedCW: Word;
  SpVoice: Variant;
  TimeString: String;
  TextToBeSpoken:Variant;
begin
  {$IFDEF MSWINDOWS}
    TimeString := FormatDateTime('h:n am/pm', Now);
    TextToBeSpoken := 'The time is: '+TimeString;

    SpVoice := CreateOleObject('SAPI.SpVoice');
    // Change FPU interrupt mask to avoid SIGFPE exceptions
    SavedCW := Get8087CW;
    try
      Set8087CW(SavedCW or $4);
      SpVoice.Speak(TextToBeSpoken, 0);
    finally
      // Restore FPU mask
      Set8087CW(SavedCW);
      SpVoice:=Unassigned;
    end;
  {$ENDIF}
end;

Now run your project.

Click on the form to listen to the time being spoken.

Optionally, add the following 2 lines to make the clock appear at the bottom right part of the screen:
procedure TForm1.FormCreate(Sender: TObject);
begin
  ...
  Left := Screen.Width - Width - 30;
  Top := Screen.Height - Height - 90;
end;

You can also set the FormStyle of the form to fsStaysOnTop to set it as a "Always on Top" mode (it won't disappear even if you focus on other windows).


Now run it and enjoy!


Further experiments

Enhancing a source code is a great way of learning, the fun way. Try these:
- Make the clock borderless and make it movable. Create a right click menu to minimize, always on top etc.
- Implement an alarm feature. That'd be cool.
- Give the user the option to choose "skin" image (a.k.a. digits.bmp). So that they can change the appearance of the clock as they wish. You can also change the values of digitWidth and digitHeight according to the new image, so that users can use skins of different sizes.
- Implement eSpeak for cross-platform compatibility

Download Sample Code ZIP

You can download the above example tutorial project's source code from here
Size: 680 KB
The package contains compiled executable EXE file.

4 comments:

  1. Nice project. However, it is not compiling for me on Linux. Debian 'Testing' 64-bit, Lazarus 1.0.14 and RC1.
    Error: http://pastebin.com/zG5q1sNz

    ReplyDelete
  2. Hello hpp3,

    Sometimes 64 bit version of Lazarus has problems with external components. (For example, Gecko also has problem in 64 bit Lazarus, but 32 bit is ok.)

    So my guess is, if you try with 32 bit Lazarus, it might be ok.

    Regards
    Adnan

    ReplyDelete
  3. Hello, great site.
    can you reupload files.?

    regards, Luis

    ReplyDelete
  4. @Luis Esteban Vilchez
    Hello and welcome to LazPlanet.

    Thanks for pointing out the problem. Dropbox had some changes in policy and the file link was discontinued. I have changed the download link to a newer one. It should now work.

    Regards

    ReplyDelete