TweetFollow Us on Twitter

TCL OOPs Intro
Volume Number:6
Issue Number:9
Column Tag:C Forum

TCL OOPs Introduction

By Mark B. Kauffman, Tucson, AZ

Note: Source code files accompanying article are located on MacTech CD-ROM or source code disks.

An Introduction to Object-Oriented Programming With THINK C

[Mark Kauffman is currently employed at Burr-Brown Research Corp. in Tucson, Arizona, designing and programming test equipment. He has been programming micro-computers for 14 years. He has owned and programmed a Macintosh since its introduction in 1984.]

Introduction

This article is written for C programmers who are interested in learning the object-oriented programming concepts available in THINK C. It includes example code that defines a class of shapes and then draws some shapes in a window on the Macintosh screen. This example is designed to be used with the Starter project that comes with the THINK C 4.0 compiler. By reading this article and trying out the example code, you will gain a clear understanding of the object-oriented concepts of class, inheritance, objects, polymorphism and messages.

Create a New Project and Add the Example Code

Create a working copy of the Starter Folder included in the THINK C compiler as follows: Copy the Starter folder, change the new folder name from “Copy of Starter Folder” to “Object1 Folder”. Open the Object1 Folder and change the word “Starter” to “Object1” in all of the file names that contain the word “Starter”. Now open the Object1 project (Object1.Π) and use the THINK C Find... command in the Search menu to find and replace all occurrences of “Starter” with “Object1” in all of the files used by the project. This procedure allows you to preserve the original Starter Folder for later use in creating new projects.

Use the THINK C file editor to create and type in the three source file listings. Name the files CShapes.c, CShapes.h and DrawMyStuff.c. Add CShapes.c and DrawMyStuff.c to the Object1 project using the Add... command under the Source menu. Open the source code file CObject1Pane.c and find the Draw method...

void CObject1Pane::Draw(Rect *area)

Underneath the line,

 {  /* draw your stuff */

add the following line...

   DrawMyStuff();

When you run the project you should see some circles, rectangles and a line drawn in a window. One of the rectangles should be missing a line across the top and one of the “circles” will be oval shaped. See figure 1.

Figure 1.

OOP Concepts

CShapes.c and CShapes.h are the files that define the shape classes that DrawMyStuff.c uses to create shapes. I’ve followed Symantec’s convention of separating class definitions into two files, a header file (the dot-h file) and a code file (the dot-c file); the header file is always included at the beginning of the code file. CShapes.h, like all dot-h files used to define a class, contains class declarations. A class declaration declares the existence of a class of objects by naming the class (by convention all of the class names begin with a C), by identifying the parent class (In Think C, unlike in some other object-oriented languages, classes undergo asexual reproduction and therefore each class has only one parent.), by stating the method prototypes (i.e. stating the names and parameters of all methods available to the class), and by declaring the instance variables of the class. CShapes.c, like all dot-c files used to define a class, consists of all the methods whose prototypes are given in its dot-h file.

Each method is an algorithm that describes an action. Messages are requests for the action that are sent to an object from the user of the program or from other objects. For example, in response to the message “Draw”, an object of class “CLine” uses the Macintosh QuickDraw routines “MoveTo” and “LineTo” to draw a line at the location given in its instance variable “Location”. Each instance of a class (i.e. each individual object in the class) may respond to the same message differently, depending on the values of its instance variables.

Instance variables are the attributes of objects. Different instances (objects) of the same class can have different values of their instance variables. For example, in the class of objects CPeople, individual objects in the class can have different values for the instance variable “leg strength”. As a result, each object of the class may respond to the same message differently. You may be able to jump (a method) higher than I can because your leg strength (an instance variable) is larger (the value of the instance variable) than mine.

The Sample Program’s Classes

In the example program there are four classes. Class CShape is the root or parent class of the other three classes. CShape is also referred to as an abstract class because there will never be any objects created of class CShape. All of the objects used in the example are of classes that are descendants of CShape. CShapes purpose is to define the methods and instance variables that its descendants will inherit. The inherited methods are: SetShapeLoc, Draw, and Erase. The inherited instance variable is Location.

The descendants of CShape are COval, CRectangle and CLine. These descendants do not need to define the method SetShapeLoc or the instance variable Location because that method and instance variable are inherited from the parent class CShape. One of the advantages of object-oriented programming is this property of new classes inheriting the methods and instance variables of their parent class.

I’ve defined three methods for the CShape class: SetShapeLoc, Draw and Erase. Objects that belong to a class that is a descendant of CShape will all be able to respond to the messages SetShapeLoc, Draw and Erase. Objects in a descendant class of CShape will respond to SetShapeLoc by setting their Location instance variable to the values passed with the SetShapeLoc message. The response inherited by classes that are descendants of CShape to the Draw or Erase message is to do nothing. However, descendants of class CShape can respond to Draw and Erase messages with actions other than the inherited response by overriding the Draw and Erase methods. A descendant class overrides an inherited method by declaring the method prototype in its class declaration, using the same method name, and redefining the method. This ability of objects of different classes to respond differently to the same message is called polymorphism.

Draw!

The sample program’s pane object gets a Draw message whenever it needs to be updated due to some action by the user such as starting the application or resizing the window. In the example the pane’s Draw method has been modified to call DrawMyStuff, a function that draws various lines, rectangles and ovals on the screen. The code in DrawMyStuff could have been in the CObject1Pane.c file as part of its Draw method but by placing the code as a function in a file separate from CObject1Pane.c, compile times are cut tremendously.

DrawMyStuff draws by creating shape objects using the Think C new() function, and then sending SetShapeLoc and Draw messages to the objects. Rectangle and oval objects are created in arrays of their respective class type. Each shape in the arrays is then sent a SetShapeLoc message and a Draw message.

There is only one instance of a line object. The line object is used twice. First, it is sent a SetShapeLoc message with parameters that are the end points of the top of one of the rectangles, and an Erase message so that the top of the rectangle is erased. Second, it is sent another SetShapeLoc message with end points that are above the rectangles and ovals, and a Draw message so that a line is drawn above the rectangles and ovals.

What Next?

This shape library has many possibilities. Obviously, one could add more shapes. There could be an instance variable, “pattern”, so that shapes could be drawn filled with a pattern. It should be possible to animate the shapes using the Dawdle message sent by the application object to the pane by adding a Dawdle method to the pane.

The concepts presented in this article are the basics of object-oriented programming with THINK C. Once you understand these concepts, you will be able to use the THINK Class Library to create complete Macintosh applications with less time and effort than was possible with previous development systems. Have fun!

Listing:  CShape.h

/*
 * Source - CShape.h
 * Author - Mark Bykerk Kauffman
 * Purpose- To define a small class of shapes
 * using THINK C, and to illustrate several
 * key concepts of object oriented programming.
 * All of these classes use the convention of
 * having a C as the first letter their class
 * name.
 */
      
struct CShape : indirect  
{
 Rect Location;
 
 void SetShapeLoc(int A, int B,int C,int D);
 void Draw(void);
 void Erase(void);
};

/*
 * All of the following classes are descendants
 * of CShape.  They inherit all of CShapes 
 * methods and instance variables.
 */ 
    
struct COval : CShape
{
 void Draw(void);
 void Erase(void);
};

struct CRectangle : CShape
{
 void Draw(void);
 void Erase(void);
};

struct CLine : CShape
{
 void Draw(void);
 void Erase(void);
};
Listing:  CShape.c

/*
 * Source - CShape.c
 * Author - Mark Bykerk Kauffman
 * Purpose- This file contains the
 * implementaion of the methods
 * prototyped by shape classes in CShape.h
 */
#include “CShape.h”
#include “Quickdraw.h”

void CShape :: Draw(void)
{
}

void CShape :: Erase(void)
{
}

void CShape :: SetShapeLoc(Left,Top,Right,Bottom)
int Left;
int Top;
int Right;
intBottom;
{
 Rect Location;  
/* LOCAL location variable */
 
 SetRect(&Location,Left,Top,Right,Bottom);
 this->Location = Location;

/* 
 * Set the instance variable Location to
 * the local Location variable value. The 
 * reason for doing this is that
 * Semantec states in the THINK C manual
 * “Your program should not rely on the
 * addresses of instance variables...”.
 * Semantec describes the above technique as
 * “shadowing”.  
 */
}

void COval :: Draw(void)
{
 Rect Location;
 
 Location = this->Location; 
 FrameOval(&Location);    
}

void COval :: Erase(void)
{
 Rect Location;
 
 Location = this->Location;
 EraseOval(&Location);  
}

void CRectangle :: Draw(void)
{
 Rect Location;
 
 Location = this->Location;
 FrameRect(&Location);  
}

void CRectangle :: Erase(void)
{
 Rect Location;
 
 Location = this->Location;
 EraseRect(&Location);  
}

void CLine :: Draw(void)
{
 MoveTo(Location.left,Location.top);
 LineTo(Location.right,Location.bottom);
}

void CLine :: Erase(void)
{
 int    oldPenMode;

 oldPenMode = thePort->pnMode;
 PenMode(notPatCopy);
 MoveTo(Location.left,Location.top);
 LineTo(Location.right,Location.bottom);
 PenMode(oldPenMode);
}
Listing:  DrawMyStuff.c

/*
 * Source - DrawMyStuff.c
 * Author - Mark Bykerk Kauffman
 * Purpose- To demonstrate how to create
 * instances of objects and send them
 * messages using Think C 4.0.  Call this 
 * function from the Draw method of the Pane 
 * object for your application. 
 */
 
#include “oops.h”
#include “CShape.h”

/* 
 * oops.h is included so the compiler knows how 
 * to deal with the object-oriented programming 
 * in this section of code.  CShape.h is
 * included so that all of the shapes defined 
 * in CShape.h are available. 
 */
 
void DrawMyStuff()
{
 int    i;
 CLine  *ALine;
 COval  *AnOval[4];
 CRectangle *ARectangle[4];
 
 for (i=0;i<4;i++) 
 ARectangle[i] = new(CRectangle);
 ARectangle[0]->SetShapeLoc(200,100,300,200);
 ARectangle[1]->SetShapeLoc(210,110,290,190);
 ARectangle[2]->SetShapeLoc(220,120,280,180);
 ARectangle[3]->SetShapeLoc(230,130,270,170);
 for (i=0;i<4;i++) 
 ARectangle[i]->Draw();
 
 ALine = new(CLine); 
 ALine->SetShapeLoc(210,110,290,110);
 ALine->Erase();
 ALine->SetShapeLoc(100,50,300,50);
 ALine->Draw();
 
 for (i=0;i<4;i++) 
 AnOval[i] = new(COval);
 AnOval[0]->SetShapeLoc(100,100,200,200);
 AnOval[1]->SetShapeLoc(110,110,190,190);
 AnOval[2]->SetShapeLoc(120,120,180,185); 
 AnOval[3]->SetShapeLoc(130,130,170,170);
 for (i=0;i<4;i++) 
 AnOval[i]->Draw();
 
 for (i=0;i<4;i++)
 { 
 delete(ARectangle[i]);   
 delete(AnOval[i]);
 }
 delete(ALine);
}

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Latest Forum Discussions

See All

Combo Quest (Games)
Combo Quest 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: Combo Quest is an epic, time tap role-playing adventure. In this unique masterpiece, you are a knight on a heroic quest to retrieve... | Read more »
Hero Emblems (Games)
Hero Emblems 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: ** 25% OFF for a limited time to celebrate the release ** ** Note for iPhone 6 user: If it doesn't run fullscreen on your device... | Read more »
Puzzle Blitz (Games)
Puzzle Blitz 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: Puzzle Blitz is a frantic puzzle solving race against the clock! Solve as many puzzles as you can, before time runs out! You have... | Read more »
Sky Patrol (Games)
Sky Patrol 1.0.1 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0.1 (iTunes) Description: 'Strategic Twist On The Classic Shooter Genre' - Indie Game Mag... | Read more »
The Princess Bride - The Official Game...
The Princess Bride - The Official Game 1.1 Device: iOS Universal Category: Games Price: $3.99, Version: 1.1 (iTunes) Description: An epic game based on the beloved classic movie? Inconceivable! Play the world of The Princess Bride... | Read more »
Frozen Synapse (Games)
Frozen Synapse 1.0 Device: iOS iPhone Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: Frozen Synapse is a multi-award-winning tactical game. (Full cross-play with desktop and tablet versions) 9/10 Edge 9/10 Eurogamer... | Read more »
Space Marshals (Games)
Space Marshals 1.0.1 Device: iOS Universal Category: Games Price: $4.99, Version: 1.0.1 (iTunes) Description: ### IMPORTANT ### Please note that iPhone 4 is not supported. Space Marshals is a Sci-fi Wild West adventure taking place... | Read more »
Battle Slimes (Games)
Battle Slimes 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: BATTLE SLIMES is a fun local multiplayer game. Control speedy & bouncy slime blobs as you compete with friends and family.... | Read more »
Spectrum - 3D Avenue (Games)
Spectrum - 3D Avenue 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: "Spectrum is a pretty cool take on twitchy/reaction-based gameplay with enough complexity and style to stand out from the... | Read more »
Drop Wizard (Games)
Drop Wizard 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: Bring back the joy of arcade games! Drop Wizard is an action arcade game where you play as Teo, a wizard on a quest to save his... | Read more »

Price Scanner via MacPrices.net

Apple’s M4 Mac minis on sale for record-low p...
B&H Photo has M4 and M4 Pro Mac minis in stock and on sale right now for up to $150 off Apple’s MSRP, each including free 1-2 day shipping to most US addresses. Prices start at only $469: – M4... Read more
Deal Alert! Mac Studio with M4 Max CPU on sal...
B&H Photo has the standard-configuration Mac Studio model with Apple’s M4 Max CPU in stock today and on sale for $300 off MSRP, now $1699 (10-Core CPU and 32GB RAM/512GB SSD). B&H also... Read more

Jobs Board

All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.