home *** CD-ROM | disk | FTP | other *** search
/ The World of Computer Software / World_Of_Computer_Software-02-385-Vol-1of3.iso / i / iv26_w_3.zip / EXAMPLES / STED / STED.C next >
C/C++ Source or Header  |  1992-03-27  |  14KB  |  467 lines

  1. /*
  2.  * Copyright (c) 1989 Stanford University
  3.  *
  4.  * Permission to use, copy, modify, distribute, and sell this software and its
  5.  * documentation for any purpose is hereby granted without fee, provided
  6.  * that the above copyright notice appear in all copies and that both that
  7.  * copyright notice and this permission notice appear in supporting
  8.  * documentation, and that the name of Stanford not be used in advertising or
  9.  * publicity pertaining to distribution of the software without specific,
  10.  * written prior permission.  Stanford makes no representations about
  11.  * the suitability of this software for any purpose.  It is provided "as is"
  12.  * without express or implied warranty.
  13.  *
  14.  * STANFORD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
  15.  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.
  16.  * IN NO EVENT SHALL STANFORD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
  17.  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
  18.  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
  19.  * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
  20.  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  21.  */
  22.  
  23. /*
  24.  * sted - a simple text editor
  25.  */
  26.  
  27. #include <InterViews/adjuster.h>
  28. #include <InterViews/border.h>
  29. #include <InterViews/box.h>
  30. #include <InterViews/button.h>
  31. #include <InterViews/glue.h>
  32. #include <InterViews/regexp.h>
  33. #include <InterViews/painter.h>
  34. #include <InterViews/scene.h>
  35. #include <InterViews/scroller.h>
  36. #include <InterViews/sensor.h>
  37. #include <InterViews/streditor.h>
  38. #include <InterViews/textbuffer.h>
  39. #include <InterViews/texteditor.h>
  40. #include <InterViews/world.h>
  41.  
  42. #include <ctype.h>
  43. #include <stdio.h>
  44. #include <stdlib.h>
  45. #include <string.h>
  46. #include <sys/types.h>
  47. #include <sys/stat.h>
  48.  
  49. static const int MINTEXTSIZE = 10000;
  50. static const int BUFFERSIZE = 100;
  51. static const int FILENAMESIZE = 100;
  52.  
  53. class StedWindow;
  54.  
  55. struct StedWindowInfo {
  56. public:
  57.     StedWindow* window;
  58.     boolean closed;
  59.     Coord x, y;
  60.     StedWindowInfo* next;
  61. };
  62.  
  63. class Sted {
  64. public:
  65.     Sted(World* world);
  66.     ~Sted();
  67.     void Open(const char*);
  68.     void Close(const char*);
  69.     void Quit();
  70.     void Run();
  71.     void Complain();
  72. private:
  73.     StedWindowInfo* Find(const char*);
  74.  
  75.     World* world;
  76.     StedWindowInfo* windows;
  77.     boolean done;
  78. };
  79.  
  80. class StedWindow : public MonoScene {
  81. public:
  82.     StedWindow(Sted*, const char* filename);
  83.     virtual ~StedWindow();
  84.     const char* Filename();
  85.  
  86.     virtual void Handle(Event&);
  87. protected:
  88.     virtual void Update();
  89. protected:
  90.     void Edit(const char* filename);
  91.     void Do(const char*);
  92.     void Ask(const char*, int, int);
  93.  
  94.     void HandleChar(char);
  95.     void LeftMouse(Event&);
  96.  
  97.     Sted* sted;
  98.     char lastchar;
  99.     char filename[FILENAMESIZE];
  100.     boolean modified;
  101.     boolean prefix1;
  102.     boolean prefix2;
  103.     char* buffer;
  104.     int size;
  105.     TextBuffer* text;
  106.     TextEditor* editor;
  107.     StringEditor* command;
  108.     ButtonState* state;
  109. };
  110.  
  111. Sted::Sted (World* w) {
  112.     world = w;
  113.     windows = nil;
  114.     done = false;
  115. }
  116.  
  117. Sted::~Sted () {
  118.     StedWindowInfo* w = windows;
  119.     while (w != nil) {
  120.         StedWindowInfo* prev = w;
  121.         w = w->next;
  122.         windows = w;                    // hinzugefügt: 10.01.92
  123.                                         // Ewald Salcher
  124.         if (!prev->closed) {
  125.             world->Remove(prev->window);
  126.         }
  127.         delete prev->window;
  128.         delete prev;
  129.     }
  130. }
  131.  
  132. void Sted::Open (const char* filename) {
  133.     StedWindowInfo* w = Find(filename);
  134.     if (w == nil) {
  135.         w = new StedWindowInfo();
  136.         w->window = new StedWindow(this, filename);
  137.         w->closed = false;
  138.         w->next = windows;
  139.         if (windows == nil) {
  140.             w->x = 0;
  141.             w->y = 0;
  142.             world->InsertApplication(w->window);
  143.             w->window->GetRelative(w->x, w->y);
  144.         } else {
  145.             w->x = windows->x - 15;
  146.             w->y = windows->y - 20;
  147.             world->InsertApplication(w->window, w->x, w->y);
  148.         }
  149.         windows = w;
  150.     } else {
  151.         if (w->closed) {
  152.             w->closed = false;
  153.             world->InsertApplication(w->window, w->x, w->y);
  154.         } else {
  155.             w->window->DeIconify();
  156.             world->Raise(w->window);
  157.         }
  158.     }
  159. }
  160.  
  161. void Sted::Close (const char* filename) {
  162.     StedWindowInfo* w = Find(filename);
  163.     if (w != nil) {
  164.         w->closed = true;
  165.     }
  166. }
  167.  
  168. StedWindowInfo* Sted::Find (const char* filename) {
  169.     StedWindowInfo* w;
  170.     for (w = windows; w != nil; w = w->next) {
  171.         if (strcmp(filename, w->window->Filename()) == 0) {
  172.             return w;
  173.         }
  174.     }
  175.     return nil;
  176. }
  177.  
  178. void Sted::Complain () {
  179.     world->RingBell(1);
  180. }
  181.  
  182. void Sted::Quit () {
  183.     done = true;
  184. }
  185.  
  186. void Sted::Run () {
  187.     Event e;
  188.     boolean alive;
  189.     do {
  190.         world->Read(e);
  191.         e.target->Handle(e);
  192.         alive = false;
  193.         for (StedWindowInfo* w = windows; w != nil; w = w->next) {
  194.             if (w->closed) {
  195.                 world->Remove(w->window);
  196.             } else {
  197.                 alive = true;
  198.             }
  199.         }
  200.     } while (alive && !done && e.target != nil);
  201. }
  202.  
  203. StedWindow::StedWindow (Sted* s, const char* name) {
  204.     sted = s;
  205.     input = new Sensor();
  206.     input->Catch(KeyEvent);
  207.     input->Catch(DownEvent);
  208.     state = new ButtonState((int)false);
  209.     state->Attach(this);
  210.     command = new StringEditor(state, "                    ");
  211.     buffer = nil;
  212.     size = 0;
  213.     text = nil;
  214.     editor = new TextEditor(24, 80, 8, Reversed);
  215.     Insert(
  216.         new VBox(
  217.             new HBox(
  218.                 new HGlue(5, 0, 0),
  219.                 new VBox(
  220.                     new VGlue(3, 0, 0),
  221.                     editor,
  222.                     new VGlue(3, 0, 0)
  223.                 ),
  224.                 new HGlue(5, 0, 0),
  225.                 new VBorder,
  226.                 new VBox(
  227.                     new UpMover(editor, 1),
  228.                     new HBorder(),
  229.                     new VScroller(editor),
  230.                     new HBorder(),
  231.                     new DownMover(editor, 1)
  232.                 )
  233.             ),
  234.             new HBorder,
  235.             new VGlue(2, 0, 0),
  236.             new HBox(
  237.                 new HGlue(5, 0, 0),
  238.                 command,
  239.                 new HGlue(5, 0, 0)
  240.             ),
  241.             new VGlue(2, 0, 0)
  242.         )
  243.     );
  244.     prefix1 = false;
  245.     prefix2 = false;
  246.     Edit(name);
  247. }
  248.  
  249. StedWindow::~StedWindow () {
  250.     sted->Close(filename);
  251.     delete text;
  252.     delete buffer;
  253.     state->Detach(this);
  254.     Unref(state);
  255. }
  256.  
  257. const char* StedWindow::Filename () {
  258.     return filename;
  259. }
  260.  
  261. void StedWindow::Edit (const char* name) {
  262.     delete buffer;
  263.     delete text;
  264.     sprintf(filename, "%s", name);
  265.     FILE* f = fopen(filename, "r");
  266.     if (f != nil) {
  267.         struct stat filestats;
  268.         stat(filename, &filestats);
  269.         size = max(round(filestats.st_size * 1.2), MINTEXTSIZE);
  270.         buffer = new char[size];
  271.         char* b = buffer;
  272.         int remaining = size;
  273.         while (remaining > 1 && fgets(b, remaining, f) != nil) {
  274.             int l = strlen(b);
  275.             remaining -= l;
  276.             b += l;
  277.         }
  278.         fclose(f);
  279.         text = new TextBuffer(buffer, b-buffer, size);
  280.         command->Message("");
  281.     } else {
  282.         size = MINTEXTSIZE;
  283.         buffer = new char[size];
  284.         text = new TextBuffer(buffer, 0, size);
  285.         command->Message("new file");
  286.     }
  287.     editor->Edit(text);
  288.     modified = false;
  289.     SetName(filename);
  290.     char* s = strrchr(filename, '/');
  291.     if (s != nil) {
  292.         SetIconName(s+1);
  293.     } else {
  294.         SetIconName(filename);
  295.     }
  296. }
  297.  
  298. void StedWindow::Update () {
  299.     int value;
  300.     state->GetValue(value);
  301.     if (value == '\r') {
  302.         Do(command->Text());
  303.     }
  304.     state->SetValue('\0');
  305. }
  306.  
  307. void StedWindow::Do (const char* s) {
  308.     static char operation[BUFFERSIZE];
  309.     static char parameter[BUFFERSIZE];
  310.     static char buffer[BUFFERSIZE];
  311.     strcpy(buffer, s);
  312.     sscanf(buffer, "%s %s", operation, parameter);
  313.     if (strcmp(operation, "backward-page") == 0) {
  314.         editor->BackwardPage(atoi(parameter));
  315.     } else if (strcmp(operation, "forward-page") == 0) {
  316.         editor->ForwardPage(atoi(parameter));
  317.     } else if (strcmp(operation, "backward-character") == 0) {
  318.         editor->BackwardCharacter(atoi(parameter));
  319.     } else if (strcmp(operation, "forward-character") == 0) {
  320.         editor->ForwardCharacter(atoi(parameter));
  321.     } else if (strcmp(operation, "backward-line") == 0) {
  322.         editor->BackwardLine(atoi(parameter));
  323.     } else if (strcmp(operation, "forward-line") == 0) {
  324.         editor->ForwardLine(atoi(parameter));
  325.     } else if (strcmp(operation, "beginning-of-text") == 0) {
  326.         editor->BeginningOfText();
  327.     } else if (strcmp(operation, "end-of-text") == 0) {
  328.         editor->EndOfText();
  329.     } else if (strcmp(operation, "beginning-of-line") == 0) {
  330.         editor->BeginningOfLine();
  331.     } else if (strcmp(operation, "end-of-line") == 0) {
  332.         editor->EndOfLine();
  333.     } else if (strcmp(operation, "delete-character") == 0) {
  334.         if (editor->Dot() != editor->Mark()) {
  335.             editor->DeleteSelection();
  336.         } else {
  337.             editor->DeleteText(atoi(parameter));
  338.         }
  339.         modified = true;
  340.     } else if (strcmp(operation, "insert-character") == 0) {
  341.         editor->DeleteSelection();
  342.         editor->InsertText(&lastchar, 1);
  343.         modified = true;
  344.     } else if (strcmp(operation, "quit") == 0) {
  345.         sted->Quit();
  346.     } else if (strcmp(operation, "close") == 0) {
  347.         sted->Close(filename);
  348.     } else if (strcmp(operation, "visit") == 0) {
  349.         sted->Open(parameter);
  350.     } else if (strcmp(operation, "file") == 0) {
  351.         Edit(parameter);
  352.     } else if (strcmp(operation, "search") == 0) {
  353.         Regexp re(parameter);
  354.         if (
  355.             text->ForwardSearch(&re, editor->Dot()) >= 0
  356.             || text->ForwardSearch(&re, text->BeginningOfText()) >= 0
  357.         ) {
  358.             editor->Select(re.EndOfMatch(), re.BeginningOfMatch());
  359.         } else {
  360.             sted->Complain();
  361.         }
  362.     } else if (strcmp(operation, "goto") == 0) {
  363.         editor->Select(text->LineIndex(atoi(parameter)));
  364.     } else {
  365.         sted->Complain();
  366.     }
  367.     editor->ScrollToSelection();
  368.     command->Message("");
  369. }
  370.  
  371. void StedWindow::Ask (const char* message, int begin, int end) {
  372.     command->Edit(message, begin, end);
  373. }
  374.  
  375. void StedWindow::Handle (Event& e) {
  376.     if (e.eventType == KeyEvent && e.len > 0) {
  377.         HandleChar(e.keystring[0]);
  378.  
  379.     } else if (e.eventType == DownEvent) {
  380.         switch (e.button) {
  381.             case LEFTMOUSE:     LeftMouse(e); break;
  382.             case MIDDLEMOUSE:   editor->GrabScroll(e); break;
  383.             case RIGHTMOUSE:    editor->RateScroll(e); break;
  384.         }
  385.     }
  386. }
  387.  
  388. void StedWindow::HandleChar (char c) {
  389.     if (prefix1) {
  390.         prefix1 = false;
  391.         switch (c) {
  392.           case 'v':     Do("backward-page 1"); break;
  393.           case 'x':     Ask("", 0, 0); break;
  394.           case '<':     Do("beginning-of-text"); break;
  395.           case '>':     Do("end-of-text"); break;
  396.           case '\r':    Do(""); break;
  397.           case '=':     Ask("goto ", 5, 5); break;
  398.           default:      Do("complain"); break;
  399.         }
  400.     } else if (prefix2) {
  401.         prefix2 = false;
  402.         switch (c) {
  403.           case '\006':  Ask("file ", 5, 5); break;
  404.           case '\026':  Ask("visit ", 6, 6); break;
  405.           case '\003':  Ask("quit", 0, 4); break;
  406.           case 'k':     Ask("close", 0, 5); break;
  407.           default:      Do("complain"); break;
  408.         }
  409.     } else {
  410.         switch (c) {
  411.           case '\033':  prefix1 = true; break;
  412.           case '\030':  prefix2 = true; break;
  413.           case '\026':  Do("forward-page 1"); break;
  414.           case '\001':  Do("beginning-of-line"); break;
  415.           case '\005':  Do("end-of-line"); break;
  416.           case '\006':  Do("forward-character 1"); break;
  417.           case '\002':  Do("backward-character 1"); break;
  418.           case '\016':  Do("forward-line 1"); break;
  419.           case '\020':  Do("backward-line 1"); break;
  420.           case '\023':  Ask("search ", 7, 7); break;
  421.           case '\004':  Do("delete-character 1"); break;
  422.           case '\010':  Do("delete-character -1"); break;
  423.           case '\177':  Do("delete-character -1"); break;
  424.           case '\011':  lastchar = '\t'; Do("insert-character"); break;
  425.           case '\015':  lastchar = '\n'; Do("insert-character"); break;
  426.           case '\007':  Do("complain"); break;
  427.             default:
  428.             if (!iscntrl(c)) {
  429.                 lastchar = c;
  430.                 Do("insert-character");
  431.             } else {
  432.                 Do("complain");
  433.             }
  434.             break;
  435.         }
  436.     }
  437. }
  438.  
  439. void StedWindow::LeftMouse (Event& e) {
  440.     GetRelative(e.x, e.y, editor);
  441.     editor->Select(editor->Locate(e.x, e.y));
  442.     do {
  443.         editor->ScrollToView(e.x, e.y);
  444.         editor->SelectMore(editor->Locate(e.x, e.y));
  445.         Poll(e);
  446.         GetRelative(e.x, e.y, editor);
  447.     } while (e.leftmouse);
  448. }
  449.  
  450. int IVMain (int argc, char** argv) {
  451.     World* world = new World("sted", argc, argv);
  452.     Sted* sted = new Sted(world);
  453.     if (argc > 1) {
  454.         for (int i = 1; i < argc; ++i) {
  455.             sted->Open(argv[i]);
  456.         }
  457.     } else {
  458.         sted->Open("d:\\borlandc\\iv\\examples\\sted\\sted.txt");
  459.     }
  460.  
  461.     sted->Run();
  462.  
  463.     delete sted;
  464.     delete world;
  465.     return 0;
  466. }
  467.