Speech Library
Volume Number: | | 7
|
Issue Number: | | 3
|
Column Tag: | | Basic School
|
Adding Speech to QuickBasic
By William H. Ball, Braintree, MA
Adding Speech to QuickBASIC
After reading Dave Kellys review of QuickBASIC, I knew that the time was right to buy a Microsoft BASIC for my Mac. I wanted to create clickable apps, but I also wanted the convenience of an interpreter. Now I have both.
So far I am pleased with QuickBASIC. But I was surprised to find that Microsoft did not include Macintalk support in the compiler. Macintalk is a small resource on disk (about 30K) that can add a nice touch to your programs.
Clear Lake Research has an ad on a back page of the QB manual and offers libraries for speech, 3D graphics, and sorting, mathematical, and statistical functions. Im sure their libraries are professionally done, well-designed and extremely useful, but I didnt want to wait; after buying the compiler I wanted to add speech to my programs that afternoon!
Pure-Code Resource Libraries
Anyway, this article presents one way to write what Microsoft calls a pure-code resource (as opposed to an impure-code resource) for QB. Luckily, LightspeedC users can generate these resources through the Set Project menu item. The details on building your own libraries for QB are in the manual. Basically, you
create a project to build a code resource of type MBPC
add your libraries and QBs BasicLib.lib
write the code
compile to a code resource
use ResEdit to
set the file creator type to MSBB or MSBD
set the file type to MSBL
You can add more resources with ResEdit as long as you give each resource a different name and ID. Voila! Your own libraries for BASIC! You can then use the LIBRARY statement in QB to use your collection of routines. Microsoft provides 20 library support routines in BasicLib.lib, which comes with QB, to help you write your own libraries.
A Speech Library
The source for SAY() shows how you can make your Mac talk from QB. The QB program, test speech, tests the SAY() routine. Building additional routines for QB looks pretty easy, even for a non-programmer like me. Microsoft made the job a lot easier by providing oodles of technical information in the manual and over 50 example programs on a disk which comes with the compiler. MDS, MPWC, MPWP, and LightspeedC examples are included.
Note that the LightspeedC example on pp. 447-448 in the QB manual will not work if you try to compile it. Why? Because the function prototypes in BasicLSC.h as supplied on QBs examples disk were redefined after the manual went to print. Each argument type in the header now has a leading underscore. You will also have to change some of the casts.
I thought about writing a number of separate speech routines, but instead included the speech rate and pitch in SAY()s syntax. You may choose to write your own. I wanted to keep things simple, so right now my library Speech.lib only contains SAY(). How many routines make a library? Two?
BasicError() Problem?
I also wanted to include more error-checking in the program, but ran across a problem with QB. The QuickBASIC library support routine BasicError(), as documented on page 459 of the manual, does not seem to work correctly. The casual reader (me) would assume that calling BasicError(n), with n representing one of the error codes listed on pp 557-559, would return a QuickBASIC error dialog with the representative error message. This is not the case; instead, an Unprintable error dialog is shown on the screen when a syntax error is encountered by a pure-code resource. This happened even with an example LightspeedC program, AddStrings.c included on the Microsoft disk, which was intended to demonstrate a pure-code resource with error-trapping. Maybe Im doing something wrong? I tried trapping the error, but couldnt correctly identify it. Id welcome corrections, additions or enlightenments.
The code follows Microsofts suggestions, saves some registers for BASIC, and gets three arguments; two integer values for the speech rate and pitch, and a string to speak. Most of the work is done by GetNextLibArg(), a support routine that also tells you what type of argument was read. IntegerArg() converts the argument to an integer, and LoadStringDesc() returns a string with its length.
The general syntax for SAY() is
SAY(rate,pitch,string)
Where rate is an integer or integer variable from 0 to
425, and determines how fast string is spoken
pitch is an integer or integer variable from 0 to
500, and determines how high or low to speak
string is a string variable or string enclosed in
quotes
Make sure MacinTalk is somewhere on the volume, or SAY() will return a Syntax Error. If you pass the wrong number or type of arguments -- La Bomba! In fact, if you think that QB is the same, tame BASIC environment you grew up with on your first home computer, forget it. Because you now have full access to the Macs ROMs, there are a lot more opportunities to cause those sizzling, screen-splitting system crashes.
Factory Typo?
I was unable to generate a library icon of the speech library for the decimal version of QuickBASIC until I did some poking around with ResEdit. According to the manual on p. 451, you should use ResEdit to set the librarys file creator type to MSBB (for binary) or MSBD (for decimal). Well, everythings fine if you set the type for MSBB, but not for MSBD! All you get when you return to the desktop is a plain document icon, and not a library icon. If I called the library from a program, the decimal version wouldnt even show the filename on a GetFile dialog. Microsoft does not provide screenshots of the different icons generated by QB, so you cant tell the difference between a document, application or library on the desktop until you get familiar with the program.
ResEdit showed me the solution to the problem. If you Get-Info the decimal version, youll see that the creator type is MSBA, not MSBD! Is this a ResEdit typo from the factory? Anyway, I figured it was safer to change the librarys creator type to MSBA rather than change the decimal versions creator type. Once I did that, the decimal version was able to recognize the speech library, and the library had a library icon (which is a different icon from the binary version. Why?).
Drove me crazy for a while.
Printer Hangup
A further problem you might run across in QB is that printer errors are not trapped well by the QB environment. If your printer is off-line, QB will hang if you ask it to print the current file. Putting your printer back on-line gets you going again. Most Mac programs seem to catch this one.
Altogether, its great to have an interpreter and compiler in the same environment. One thing Id like to see in a future version is the ability to edit multiple files. Also, the compiler seems to be a bit untidy, as it can leave a bunch of temporary files on your disk if your compile is aborted. I also ran out of room on my hard disk after a file called QuickBasic Settings grew to over six megabytes in the System Folder. This happened after a crash or two, and after I trashed the file QuickBasic came up damaged but with a graceful exit.
Despite the fact that you can now go into La-La Land just as easily as in C or Pascal, if you look back at the capabilities of the early BASICs on mainframes and micros youll see that the present QB environment is very nice indeed -- and now talks to you!
Have a happy! And remember, practice safe computing! After Novembers virus scare, you never know what you can catch after connecting with another computer.
THINK C Project and Code Type
/* Speech.c -- QB code resource for access to MacinTalk routines
Copyright (C)1988 by William H. Ball for MacTutor magazine
Usage: SAY(rate,pitch,string)
where [rate] is an integer or integer variable
[pitch] is an integer or integer variable
[string] is a string variable or string enclosed by quotes.
for example,
hefty% = 100 : wimpy% = 300 : fast% = 200 : slow% = 100
m$ = Hello there
CALL SAY(slow%,hefty%,m$)
CALL SAY(fast%,wimpy%,m$)
or
CALL SAY(100,100,Hello there)
CALL SAY(200,300,Hello there)
Caveats: Can cause bombs of ID 2 or ID 10 if you fail to pass the correct
number or type of arguments.
*/
#include Macintalk.h /* LSCs header */
#include BasicLSC.h/* supplied with QB */
PUBLIC VOID main() /* see QB manual p. 448*/
{
SpeechErr err; /* error code*/
SpeechHandle theSpeech; /* pointer to speech*/
Handle spOut; /* speech handle */
INT16 TempFlag,len,type,rate,pitch;
/* INT16 == short */
StringPtr message;
LIBARGPTR valptr;
SAVEREGS();/* Save BASICs A4 and A5 registers*/
if ((err = SpeechOn(, &theSpeech)) != noErr) { return; }
if ((type = GetNextLibArg(&valptr,&TempFlag)) != _INTARG) rate = 150;
else rate = IntegerArg();
SpeechRate(theSpeech,(int)rate);
if ((type = GetNextLibArg(&valptr,&TempFlag)) != _INTARG) pitch = 95;
else pitch = IntegerArg();
SpeechPitch(theSpeech,(int)pitch,Natural);
type = GetNextLibArg(&valptr,&TempFlag);
if (type != _NULLARG) {
if (type == _STRGARG) {
if (TempFlag)
FreeTempDesc((SDRECPTR)valptr);
LoadStringDesc((SDRECPTR)valptr,&message,&len);
spOut = NewHandle(0L); /* allocate handle */
Reader(theSpeech,message,(long)len,spOut);
MacinTalk(theSpeech, spOut);
SpeechOff(theSpeech);
DisposHandle(spOut);
}
}
RESTREGS();/* Restore BASICs A4 and A5 registers*/
}
QuickBASIC program tests SAY() pure-code resource routine.
Copyright (C)1988 by William H. Ball for MacTutor magazine.
Usage: SAY(rate,pitch,string-expression)
Beware! Will bomb with wrong number or type of arguments!
Make sure MacinTalk is on same volume, or SAY() wont talk.
DEFINT a-z
LIBRARY Speechd.lib
ON ERROR GOTO 9
PRINT Lets test the SAY() routine:
GOSUB talk
9 LIBRARY CLOSE
END
talk:
rate = 0 declare integer and string variables
pitch = 0
message$ =
INPUT Enter something to say: ;message$
INPUT Enter a rate of speech between 0 and 425;rate
INPUT Enter a pitch of speech between 0 and 500;pitch
Just hit Return? Use a default value
IF rate = 0 THEN rate = 100
IF pitch = 0 THEN pitch = 100
CALL SAY(rate,pitch,message$)
PRINT another?
message$=INPUT$(1)
IF LEFT$(message$,1) = y THEN
GOSUB talk
END IF
RETURN