Command Pattern


Command objects manage the processing that occurs when a user
manipulates the application in some way.

Objectifying a command admits all kinds of interesting possibilities.
You can put the command on a stack to implement undo and redo, or write
the command to disk to implement record/playback.

You don't even have to execute the command right away.  In a simulation
each command has a scenario time.  You put the command on a queue and
execute the command later, when the simulation reaches that point in the
scenario.

In the example here, a menu-based application manages a group of open
documents.  Each menu item contains a command object, bound at creation
time to a document object, which when executed calls a document
operation.


Implementation

In C++, you can bind a command to a document by passing the document as
a parameter in the constructor for the command.

For binding one object to another in Ada95 we use access discriminants,
a language feature created specifically for this purpose.  The advantage
of this approach over C++ constructors is that the language guarantees
that a dangling reference cannot occur.

In order to have an access discriminant, the type must be limited.  This
was already the case for the command class, because we used the
limited-and-indefinite idiom to control instance creation:

  package Commands is

    type Root_Command_Type (<>) is abstract tagged limited private;

That the type is also indefinite means we can hide the access
discriminant, by declaring it only in the full view of the type:

  package Commands.Open is

    type Open_Command_Type is
      new Root_Command_Type with private;
    ...
  private

   type Open_Command_Type (App : access Application_Type'Class) is
     new Root_Command_Type with null record;
   ...
  end Commands.Open;


We bind an open command to an application instance by passing the
reference in the constructor for the type (very similar, actually, to
the C++ mechanism):

   function New_Open_Command (App : access Application_Type'Class)
     return Open_Command_Access is
   begin
      return new Open_Command_Type (App);
   end;


When an open command is executed, the command creates a new document and
opens it, and adds it to the application:

   procedure Execute (Command : access Open_Command) is
   ...
      Add (Doc, To => Command.App.all);


The paste command is implemented similarly, except that it is bound to a
document instance:

     type Paste_Command_Type (Doc : access Document_Type'Class) is
       new Root_Command_Type with private;


When you execute a paste command, it calls the paste operation of the
document to which it is bound:

   procedure Execute (Command : access Paste_Command_Type) is
   begin
      Documents.Paste (Command.Doc);
   end;


You can generalize this idea by creating a generic package, allowing a
command to be bound to any kind of type:

   generic

      type Receiver_Type (<>) is limited private;

      with procedure Action (Receiver : in out Receiver_Type);

   package Commands.Simple is

      type Simple_Command_Type (Receiver : access Receiver_Type) is
        new Root_Command_Type with private;


The generic formal procedure Action specifies what you do to the object
when you execute the command:

   procedure Execute (Command : access Simple_Command_Type) is
   begin
      Action (Command.Receiver.all);
   end;


In the sample code, I use the simple command generic to create a command
that counts the number of documents in an application.

We want to be able to count the number of documents for any type in the
class, so we will instantiate the generic on the class-wide type.
Actual types that are indefinite (like Application_Type'Class) are
allowed because the formal type (Receiver_Type) was declared with
unknown discriminants.

There was no existing class-wide operation for applications that counts
documents, so we just extend package Applications with a public child
subprogram:

  procedure Applications.Count_Docs
    (App : in out Application_Type'Class);

Because it is a child, Count_Docs has access to the representation of
Application_Type, and it can therefore be implemented by iterating over
the documents in the collection maintained by the application.

A macro command maintains a list of commands, and so doesn't need an
access discriminant of its own.  When you execute a macro command, it
calls the execute operation for all the commands on the list:

  procedure Execute (Command : access Macro_Command_Type) is
     Index : Root_Command_Access := Command.Next;
  begin
     while Index /= null loop
        Execute (Index);
        Index := Index.Next;
     end loop;
  end Execute;

This is an example of the Composite pattern.

Note that if you have a well-known object (singleton or otherwise) that
is the recipient of command activity, then you don't need to bind the
command to anything.  In the implementation of Execute, you just call
the well-known object directly.


The code below is in a format suitable for use with gnatchop.

--STX
with Ada.Text_IO; use Ada.Text_IO;

procedure Applications.Count_Docs
  (App : in out Application_Type'Class) is

   Count : Natural := 0;

   procedure Inc_Count (Doc : access Document_Type'Class) is
   begin
      Count := Count + 1;
   end;

   procedure Do_Count_Docs is
     new For_Every_Document (Inc_Count);

begin

   Do_Count_Docs (App.Container);

   case Count is
      when 0 =>
         Put_Line ("There are no docs in the app.");

      when 1 =>
         Put_Line ("There is 1 doc in the app.");

      when others =>
         Put_Line
           ("There are" &
            Integer'Image (Count) &
            " docs in the app.");

   end case;

end Applications.Count_Docs;
procedure Applications.Count_Docs
  (App : in out Application_Type'Class);
package body Applications is

   procedure Add
     (Document : access Document_Type'Class;
      To       : in out Application_Type) is

      App : Application_Type renames To;
   begin
      Add (Document, To => App.Container);
   end;

end Applications;


with Documents;            use Documents;
with Documents.Containers;

package Applications is

   type Application_Type is tagged limited private;

   procedure Add
     (Document : access Document_Type'Class;
      To       : in out Application_Type);

private

   use Documents.Containers;

   type Application_Type is
      tagged limited record
         Container : Document_Container;
      end record;

end Applications;


with Applications.Count_Docs; use Applications;
with Commands.Simple;

package Commands.App is
  new Simple (Application_Type'Class, Count_Docs);

with Ada.Unchecked_Deallocation;

package body Commands.Macro is

   procedure Deallocate is
     new Ada.Unchecked_Deallocation
     (Macro_Command_Type,
      Macro_Command_Access);


   function New_Macro_Command return Macro_Command_Access is
   begin
      return new Macro_Command_Type;
   end;


   procedure Do_Free (Command : access Macro_Command_Type) is

      Index : Root_Command_Access := Command.Next;
      Item  : Root_Command_Access;
   begin
      while Index /= null loop
         Item := Index;
         Index := Index.Next;
         Free (Item);
      end loop;

      declare
         CA : Macro_Command_Access :=
           Macro_Command_Access (Command);
      begin
         Deallocate (CA);
      end;
   end Do_Free;



   procedure Execute (Command : access Macro_Command_Type) is
      Index : Root_Command_Access := Command.Next;
   begin
      while Index /= null loop
         Execute (Index);
         Index := Index.Next;
      end loop;
   end Execute;


   procedure Add (Command : access Root_Command_Type'Class;
                  To      : access Macro_Command_Type) is
   begin
      Command.Next := To.Next;
      To.Next := Root_Command_Access (Command);
   end;


end Commands.Macro;




package Commands.Macro is

   type Macro_Command_Type is new Root_Command_Type with private;

   type Macro_Command_Access is access all Macro_Command_Type;

   function New_Macro_Command return Macro_Command_Access;

   procedure Execute (Command : access Macro_Command_Type);

   procedure Add (Command : access Root_Command_Type'Class;
                  To      : access Macro_Command_Type);

private

   type Macro_Command_Type is new Root_Command_Type with null record;
   --
   -- We've already got a Next pointer that we (can) use for storage
   -- management of commands, so let's just use that as the head of our
   -- command list.
   --

   procedure Do_Free (Command : access Macro_Command_Type);

end Commands.Macro;




with Documents; use Documents;

with Ada.Text_IO;  use Ada.Text_IO;
with Ada.Unchecked_Deallocation;

package body Commands.Open is

   procedure Deallocate is
     new Ada.Unchecked_Deallocation
     (Open_Command_Type,
      Open_Command_Access);


   function New_Open_Command (App : access Application_Type'Class)
     return Open_Command_Access is
   begin
      return new Open_Command_Type (App);
   end;


   procedure Do_Free (Command : access Open_Command_Type) is

      CA : Open_Command_Access :=
        Open_Command_Access (Command);
   begin
      Deallocate (CA);
   end;


   function Ask_User_For_Name return String is

      Line : String (1 .. 80);
      Last : Natural;
   begin
      Put ("Name: ");
      Get_Line (Line, Last);
      return Line (1 .. Last);
   end;


   procedure Execute (Command : access Open_Command_Type) is

      Doc : Document_Access;

      Name : constant String := Ask_User_For_Name;

   begin

      if Name /= "" then

         Doc := New_Doc (Name);

         Add (Doc, To => Command.App.all);

         Documents.Open (Doc);

      end if;

   end Execute;

end Commands.Open;





with Applications; use Applications;

package Commands.Open is

   type Open_Command_Type is
     new Root_Command_Type with private;

   type Open_Command_Access is access all Open_Command_Type;

   function New_Open_Command (App : access Application_Type'Class)
     return Open_Command_Access;


   procedure Execute (Command : access Open_Command_Type);

private

   type Open_Command_Type (App : access Application_Type'Class) is
     new Root_Command_Type with null record;

   function Ask_User_For_Name return String;

   procedure Do_Free (Command : access Open_Command_Type);

end Commands.Open;





with Ada.Unchecked_Deallocation;

package body Commands.Paste is

   procedure Deallocate is
     new Ada.Unchecked_Deallocation
     (Paste_Command_Type,
      Paste_Command_Access);


   function New_Paste_Command (Doc : access Document_Type'Class)
     return Paste_Command_Access is
   begin
     return new Paste_Command_Type (Doc);
   end;


   procedure Do_Free (Command : access Paste_Command_Type) is

      CA : Paste_Command_Access :=
        Paste_Command_Access (Command);
   begin
      Deallocate (CA);
   end;


   procedure Execute (Command : access Paste_Command_Type) is
   begin
      Documents.Paste (Command.Doc);
   end;

end Commands.Paste;



with Documents; use Documents;

package Commands.Paste is

   type Paste_Command_Type is
     new Root_Command_Type with private;

   type Paste_Command_Access is access all Paste_Command_Type;

   function New_Paste_Command (Doc : access Document_Type'Class)
     return Paste_Command_Access;

   procedure Execute (Command : access Paste_Command_Type);

private

   type Paste_Command_Type (Doc : access Document_Type'Class) is
     new Root_Command_Type with null record;

   procedure Do_Free (Command : access Paste_Command_Type);

end Commands.Paste;



with Ada.Unchecked_Deallocation;

package body Commands.Simple is

   procedure Deallocate is
     new Ada.Unchecked_Deallocation
     (Simple_Command_Type,
      Simple_Command_Access);


   function New_Simple_Command (Receiver : access Receiver_Type)
     return Simple_Command_Access is
   begin
      return new Simple_Command_Type (Receiver);
   end;


   procedure Do_Free (Command : access Simple_Command_Type) is

      CA : Simple_Command_Access :=
        Simple_Command_Access (Command);
   begin
      Deallocate (CA);
   end;


   procedure Execute (Command : access Simple_Command_Type) is
   begin
      Action (Command.Receiver.all);
   end;

end Commands.Simple;




generic
   type Receiver_Type (<>) is limited private;
   with procedure Action (Receiver : in out Receiver_Type);
package Commands.Simple is

   type Simple_Command_Type (Receiver : access Receiver_Type) is
     new Root_Command_Type with private;

   type Simple_Command_Access is access all Simple_Command_Type;

   function New_Simple_Command (Receiver : access Receiver_Type)
     return Simple_Command_Access;

   procedure Execute (Command : access Simple_Command_Type);

private

   type Simple_Command_Type (Receiver : access Receiver_Type) is
     new Root_Command_Type with null record;

   procedure Do_Free (Command : access Simple_Command_Type);

end Commands.Simple;




package body Commands is

   procedure Free (Command : in out Root_Command_Access) is
   begin
      Do_Free (Command);
      Command := null;
   end;

   procedure Do_Free (Command : access Root_Command_Type) is
   begin
      null;
   end;

end Commands;






package Commands is

   type Root_Command_Type (<>) is abstract tagged limited private;

   type Root_Command_Access is access all Root_Command_Type'Class;

   procedure Execute
     (Command : access Root_Command_Type) is abstract;

   procedure Free (Command : in out Root_Command_Access);

private

   type Root_Command_Type is
     abstract tagged limited record
        Next : Root_Command_Access;
     end record;

   procedure Do_Free (Command : access Root_Command_Type);

end Commands;






package body Documents.Containers is

   procedure Add (Document : access Document_Type'Class;
                  To       : in out Document_Container) is

      Container : Document_Container renames To;
   begin
      Document.Next := Container.Head;
      Container.Head := Document_Class_Access (Document);
   end;


   procedure For_Every_Document (Container : in Document_Container) is

      Index : Document_Class_Access := Container.Head;
   begin
      while Index /= null loop
         Process (Index);
         Index := Index.Next;
      end loop;
   end For_Every_Document;


end Documents.Containers;


package Documents.Containers is

   type Document_Container is limited private;

   procedure Add (Document : access Document_Type'Class;
                  To       : in out Document_Container);

   generic
      with procedure Process (Doc : access Document_Type'Class);
   procedure For_Every_Document (Container : in Document_Container);

private

   type Document_Container is
      limited record
         Head : Document_Class_Access;
      end record;

end Documents.Containers;


with Ada.Unchecked_Deallocation;
with Ada.Text_IO;                 use Ada.Text_IO;

package body Documents is

   procedure Deallocate is
     new Ada.Unchecked_Deallocation
     (Document_Type, Document_Access);


   function New_Doc (Name : String) return Document_Access is

      Doc : constant Document_Access :=
        Document_Access'(new Document_Type);
   begin
      Doc.Name := To_Unbounded_String (Name);
      return Doc;
   end;


   procedure Do_Free (Doc : access Document_Type) is

      DA : Document_Access := Document_Access (Doc);
   begin
      Deallocate (DA);
   end;


   procedure Open
     (Document : access Document_Type) is
   begin
      Put_Line
        ("opening document named '" &
         To_String (Document.Name) &
         "'");
   end;

   procedure Close
     (Document : access Document_Type) is
   begin
      Put_Line ("closing document");
   end;

   procedure Cut
     (Document : access Document_Type) is
   begin
      Put_Line
        ("cut text from doc named '" &
         To_String (Document.Name) &
         "'");
   end;

   procedure Copy
     (Document : access Document_Type) is
   begin
      Put_Line ("copy text from doc");
   end;

   procedure Paste
     (Document : access Document_Type) is
   begin
      Put_Line
        ("paste text into doc named '" &
         To_String (Document.Name) &
         "'");
   end;


   procedure Free (Document : in out Document_Class_Access) is
   begin
      Do_Free (Document);
      Document := null;
   end;

end Documents;


with Ada.Strings.Unbounded;

package Documents is

   type Document_Type (<>) is tagged limited private;

   type Document_Class_Access is access all Document_Type'Class;


   type Document_Access is access all Document_Type;

   function New_Doc (Name : String) return Document_Access;


   procedure Open
     (Document : access Document_Type);

   procedure Close
     (Document : access Document_Type);

   procedure Cut
     (Document : access Document_Type);

   procedure Copy
     (Document : access Document_Type);

   procedure Paste
     (Document : access Document_Type);


   procedure Free (Document : in out Document_Class_Access);

private

   use Ada.Strings.Unbounded;

   type Document_Type is
      tagged limited record
         Next : Document_Class_Access;
         Name : Unbounded_String;
      end record;

   procedure Do_Free (Doc : access Document_Type);

end Documents;


with Documents;    use Documents;
with Applications; use Applications;

with Commands.Paste;
with Commands.Open;
with Commands.App;  use Commands.App;

procedure Test_Commands is

   App : aliased Application_Type;

   App_Command : constant Commands.App.Simple_Command_Access :=
     New_Simple_Command (App'Access);

begin

   Execute (App_Command);

   declare
      use Commands.Open;

      Command : Open_Command_Access :=
        New_Open_Command (App'Access);
   begin
      Execute (Command);
   end;

   Execute (App_Command);

   declare
      use Commands.Paste;

      Doc : constant Document_Access :=
        New_Doc ("pastedoc.dat");

      Command : constant Paste_Command_Access :=
        New_Paste_Command (Doc);
   begin
      Execute (Command);
      Add (Doc, To => App);
   end;

   Execute (App_Command);

end Test_Commands;

Contributed by: Matthew Heaney
Contributed on: May 24, 1999
License: Public Domain

Back