home *** CD-ROM | disk | FTP | other *** search
/ Amiga Times / AmigaTimes.iso / internet / Term / Source / term-4.7a / TextBufferWindow.c < prev    next >
Encoding:
C/C++ Source or Header  |  1998-10-06  |  67.4 KB  |  2,930 lines

  1. /*
  2. **    TextBufferWindow.c
  3. **
  4. **    Support routines for the text buffer window and screen.
  5. **
  6. **    Copyright © 1990-1996 by Olaf `Olsen' Barthel
  7. **        All Rights Reserved
  8. **
  9. **    :ts=4
  10. */
  11.  
  12. #ifndef _GLOBAL_H
  13. #include "Global.h"
  14. #endif
  15.  
  16. /****************************************************************************/
  17.  
  18. STATIC VOID Buffer_AskPosition(struct MarkerContext *Context, LONG *Column, LONG *Line, WORD Mode);
  19. STATIC VOID Buffer_Scroll(struct MarkerContext *Context, LONG DeltaX, LONG DeltaY);
  20. STATIC VOID Buffer_Highlight(struct MarkerContext *Context, LONG Line, LONG Left, LONG Right);
  21. STATIC BOOL Buffer_Put(APTR UserData, STRPTR Buffer, LONG Length);
  22. STATIC BOOL Buffer_PutLine(APTR UserData, STRPTR Buffer, LONG Length);
  23. STATIC BOOL Buffer_Transfer(struct MarkerContext *Context, LONG Line, LONG Left, LONG Right, MARKER_Put Put, APTR UserData);
  24. STATIC BOOL Buffer_PickWord(struct MarkerContext *Context, LONG Left, LONG Top, LONG *WordLeft, LONG *WordRight);
  25. STATIC APTR Buffer_TransferStartStop(struct MarkerContext *Context, APTR UserData, ULONG Qualifier);
  26. STATIC VOID BufferMarkerStop(TextBufferInfo *BufferInfo);
  27. STATIC VOID BufferMarkWord(TextBufferInfo *BufferInfo);
  28. STATIC VOID BufferMarkerStart(TextBufferInfo *BufferInfo, UWORD MsgQualifier);
  29. STATIC VOID BufferMarkerInterrupt(TextBufferInfo *BufferInfo);
  30. STATIC VOID BufferMarkerMoveMouse(TextBufferInfo *BufferInfo);
  31. STATIC VOID BufferMarkerTransfer(TextBufferInfo *BufferInfo, UWORD MsgQualifier);
  32. STATIC VOID FlushMsg(struct Window *Window);
  33. STATIC VOID BufferClear(TextBufferInfo *BufferInfo, LONG Left, LONG Top, LONG Right, LONG Bottom);
  34. STATIC VOID BufferScroll(TextBufferInfo *BufferInfo, LONG Lines);
  35. STATIC VOID BufferComplement(TextBufferInfo *BufferInfo, LONG SrcX, LONG SrcY, LONG Width, LONG Height);
  36. STATIC VOID DeleteScroller(TextBufferInfo *BufferInfo);
  37. STATIC BOOL CreateScroller(TextBufferInfo *BufferInfo, LONG Height);
  38. STATIC VOID PrintLine(TextBufferInfo *BufferInfo, STRPTR Buffer, LONG LineNumber);
  39. STATIC VOID PrintLineWithMarker(TextBufferInfo *BufferInfo, STRPTR Buffer, LONG LineNumber, LONG AbsoluteLine);
  40. STATIC LONG RedrawScreen(TextBufferInfo *BufferInfo, LONG FirstLine);
  41. STATIC VOID BufferDestructor(struct DataMsg *Item);
  42. STATIC VOID BufferSerWrite(TextBufferInfo *BufferInfo, APTR Data, LONG Size);
  43. STATIC VOID StartSearch(TextBufferInfo *BufferInfo, struct SearchInfo *SearchInfo, STRPTR SearchBuffer);
  44. STATIC BOOL HandleBuffer(struct SignalSemaphore *Access, TextBufferInfo **Data);
  45. STATIC VOID DeleteBufferInfo(TextBufferInfo *BufferInfo);
  46. STATIC TextBufferInfo *CreateBufferInfo(struct Screen *Parent, BOOL *pSearchForward, BOOL *pIgnoreCase, BOOL *pWholeWords, LONG *pTopLine);
  47. STATIC VOID SAVE_DS BufferServer(VOID);
  48. STATIC VOID SAVE_DS ReviewServer(VOID);
  49.  
  50. /****************************************************************************/
  51.  
  52.     /* Gadget ID codes. */
  53.  
  54. enum    {    GAD_SCROLLER,
  55.             GAD_UP,
  56.             GAD_DOWN
  57.         };
  58.  
  59.     /* Menu ID codes. */
  60.  
  61. enum    {    MEN_SEARCH,
  62.             MEN_REPEAT,
  63.             MEN_GOTO,
  64.             MEN_CLEARBUF_CONTENTS,
  65.             MEN_QUITBUF,
  66.             MEN_COPYCLIP,
  67.             MEN_PASTECLIP,
  68.             MEN_CLEARCLIP,
  69.             MEN_SELECT_ALL_CLIP
  70.         };
  71.  
  72. /****************************************************************************/
  73.  
  74. STATIC VOID
  75. Buffer_AskPosition(struct MarkerContext *Context,LONG *Column,LONG *Line,WORD Mode)
  76. {
  77.     TextBufferInfo *BufferInfo;
  78.     struct TextExtent Extent;
  79.     LONG Left,Top;
  80.  
  81.     BufferInfo = Context->UserData;
  82.  
  83.     Left    = BufferInfo->Window->MouseX - BufferInfo->Left;
  84.     Top        = BufferInfo->Window->MouseY - BufferInfo->Top;
  85.  
  86.     SafeObtainSemaphoreShared(&BufferSemaphore);
  87.  
  88.     if(Mode == MARKERASK_Scroll)
  89.     {
  90.         if(Top < 0)
  91.         {
  92.             *Column    = 0;
  93.             *Line    = -1;
  94.         }
  95.         else if(Top >= BufferInfo->Height)
  96.         {
  97.             *Column    = Context->NumVisibleColumns - 1;
  98.             *Line    = Context->NumVisibleLines;
  99.         }
  100.         else
  101.         {
  102.             LONG ThisLine;
  103.  
  104.             ThisLine = Top / BufferInfo->LocalTextFontHeight;
  105.  
  106.             if(BufferInfo->TopLine + ThisLine >= Lines)
  107.             {
  108.                 *Column    = Context->NumVisibleColumns - 1;
  109.                 *Line    = Context->NumVisibleLines;
  110.             }
  111.             else
  112.             {
  113.                 if(Left < 0)
  114.                     *Column = -1;
  115.                 else if (Left >= BufferInfo->Width)
  116.                     *Column = Context->NumVisibleColumns;
  117.                 else
  118.                 {
  119.                     STRPTR TheLine = BufferLines[BufferInfo->TopLine + ThisLine];
  120.  
  121.                     if(TheLine[-1])
  122.                         *Column = TextFit(BufferInfo->RPort,TheLine,TheLine[-1],&Extent,NULL,1,Left,32767);
  123.                     else
  124.                         *Column = 0;
  125.                 }
  126.  
  127.                 *Line = ThisLine;
  128.             }
  129.         }
  130.     }
  131.     else
  132.     {
  133.         LONG ThisLine;
  134.  
  135.         if(Top < 0)
  136.             Top = 0;
  137.         else if(Top >= BufferInfo->Height)
  138.             Top = BufferInfo->Height - 1;
  139.  
  140.         if(Left < 0)
  141.             Left = 0;
  142.         else if (Left >= BufferInfo->Width)
  143.             Left = BufferInfo->Width - 1;
  144.  
  145.         ThisLine = Top / BufferInfo->LocalTextFontHeight;
  146.  
  147.         if(BufferInfo->TopLine + ThisLine >= Lines)
  148.         {
  149.             STRPTR TheLine;
  150.  
  151.             ThisLine = Lines - BufferInfo->TopLine - 1;
  152.  
  153.             TheLine = BufferLines[BufferInfo->TopLine + ThisLine];
  154.  
  155.             if(TheLine[-1] > 0)
  156.                 *Column = TextFit(BufferInfo->RPort,TheLine,TheLine[-1],&Extent,NULL,1,Left,32767);
  157.             else
  158.                 *Column = 0;
  159.  
  160.             *Line = ThisLine;
  161.         }
  162.         else
  163.         {
  164.             if(Left < 0)
  165.                 *Column = 0;
  166.             else
  167.             {
  168.                 STRPTR TheLine;
  169.  
  170.                 TheLine = BufferLines[BufferInfo->TopLine + ThisLine];
  171.  
  172.                 if(TheLine[-1] > 0)
  173.                     *Column = TextFit(BufferInfo->RPort,TheLine,TheLine[-1],&Extent,NULL,1,Left,32767);
  174.                 else
  175.                     *Column = 0;
  176.             }
  177.  
  178.             *Line = ThisLine;
  179.         }
  180.     }
  181.  
  182.     ReleaseSemaphore(&BufferSemaphore);
  183. }
  184.  
  185.     /* Scroll the visible region. */
  186.  
  187. STATIC VOID
  188. Buffer_Scroll(struct MarkerContext *Context,LONG UnusedDeltaX,LONG DeltaY)
  189. {
  190.     TextBufferInfo *BufferInfo;
  191.     LONG NewTopLine;
  192.  
  193.     BufferInfo = Context->UserData;
  194.  
  195.     NewTopLine = BufferInfo->TopLine + DeltaY;
  196.  
  197.     if(NewTopLine < 0)
  198.         NewTopLine = 0;
  199.     else if (NewTopLine + BufferInfo->NumBufferLines >= Lines)
  200.         NewTopLine = Lines - BufferInfo->NumBufferLines;
  201.  
  202.     if(NewTopLine >= 0 && NewTopLine != BufferInfo->TopLine)
  203.     {
  204.         BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,NewTopLine);
  205.  
  206.         BufferInfo->TopLine = NewTopLine;
  207.  
  208.         SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  209.             PGA_Top,BufferInfo->TopLine,
  210.         TAG_DONE);
  211.     }
  212. }
  213.  
  214.     /* Highlight a line of text between the "Left" and "Right" column (inclusive).
  215.      */
  216.  
  217. STATIC VOID
  218. Buffer_Highlight(struct MarkerContext *Context,LONG Line,LONG Left,LONG Right)
  219. {
  220.     TextBufferInfo *BufferInfo;
  221.  
  222.     BufferInfo = Context->UserData;
  223.  
  224.     if(Line >= BufferInfo->TopLine && Line < BufferInfo->TopLine + BufferInfo->NumBufferLines && Line < Lines)
  225.     {
  226.         if(Left >= BufferInfo->NumBufferColumns)
  227.             return;
  228.  
  229.         if(Right >= BufferInfo->NumBufferColumns)
  230.             Right = BufferInfo->NumBufferColumns - 1;
  231.  
  232.         if(Left <= Right)
  233.             BufferComplement(BufferInfo,BufferInfo->BufferColumnOffsets[Left],BufferInfo->BufferLineOffsets[Line - BufferInfo->TopLine],BufferInfo->BufferColumnOffsets[Right - Left + 1],BufferInfo->BufferLineOffsets[1]);
  234.     }
  235. }
  236.  
  237.     /* Transfer the buffer contents line by line. */
  238.  
  239. STATIC BOOL
  240. Buffer_Put(APTR UserData,STRPTR Buffer,LONG Length)
  241. {
  242.     if(Length <= 0)
  243.         return(TRUE);
  244.     else
  245.     {
  246.         BOOL Result;
  247.  
  248.         if(Config->TerminalConfig->FontMode == FONT_STANDARD)
  249.             Result = (BOOL)(WriteChunkBytes(UserData,Buffer,Length) == Length);
  250.         else
  251.             Result = WriteTranslatedToClip(UserData,Buffer,Length);
  252.  
  253.         return(Result);
  254.     }
  255. }
  256.  
  257. STATIC BOOL
  258. Buffer_PutLine(APTR UserData,STRPTR Buffer,LONG Length)
  259. {
  260.     BOOL Result;
  261.  
  262.     if(Length <= 0)
  263.         Result = TRUE;
  264.     else
  265.     {
  266.         if(Config->TerminalConfig->FontMode == FONT_STANDARD)
  267.             Result = (BOOL)(WriteChunkBytes(UserData,Buffer,Length) == Length);
  268.         else
  269.             Result = WriteTranslatedToClip(UserData,Buffer,Length);
  270.     }
  271.  
  272.     if(Result)
  273.         Result = (BOOL)(WriteChunkBytes(UserData,"\n",1) == 1);
  274.  
  275.     return(Result);
  276. }
  277.  
  278. STATIC BOOL
  279. Buffer_Transfer(struct MarkerContext *Context,LONG Line,LONG Left,LONG Right,MARKER_Put Put,APTR UserData)
  280. {
  281.     BOOL Result;
  282.  
  283.     Result = TRUE;
  284.  
  285.     SafeObtainSemaphoreShared(&RasterSemaphore);
  286.  
  287.     if(Line >= 0 && Line < Lines)
  288.     {
  289.         STRPTR TheLine;
  290.  
  291.         TheLine = BufferLines[Line];
  292.  
  293.         if(Left < 0)
  294.             Left = 0;
  295.  
  296.         if(Right >= TheLine[-1])
  297.             Right = TheLine[-1] - 1;
  298.  
  299.         if(Left <= Right)
  300.             Result = Put(UserData,&TheLine[Left],Right - Left + 1);
  301.         else if (TheLine[-1] == 0)
  302.             Result = Put(UserData,&TheLine[Left],0);
  303.     }
  304.  
  305.     ReleaseSemaphore(&RasterSemaphore);
  306.  
  307.     return(Result);
  308. }
  309.  
  310. STATIC BOOL
  311. Buffer_PickWord(struct MarkerContext *UnusedContext,LONG Left,LONG Top,LONG *WordLeft,LONG *WordRight)
  312. {
  313.     BOOL GotIt;
  314.  
  315.     GotIt = FALSE;
  316.  
  317.     SafeObtainSemaphoreShared(&BufferSemaphore);
  318.  
  319.     if(Top >= 0 && Top < Lines)
  320.     {
  321.         STRPTR TheLine;
  322.  
  323.         TheLine = BufferLines[Top];
  324.  
  325.         if(Left >= 0 && Left < TheLine[-1] && TheLine[Left] != ' ')
  326.         {
  327.             LONG From,To;
  328.  
  329.             From = To = Left;
  330.  
  331.             while(From > 0 && TheLine[From - 1] != ' ')
  332.                 From--;
  333.  
  334.             while(To < TheLine[-1] - 1 && TheLine[To + 1] != ' ')
  335.                 To++;
  336.  
  337.             GotIt = TRUE;
  338.  
  339.             *WordLeft    = From;
  340.             *WordRight    = To;
  341.         }
  342.     }
  343.  
  344.     ReleaseSemaphore(&BufferSemaphore);
  345.  
  346.     return(GotIt);
  347. }
  348.  
  349. STATIC APTR
  350. Buffer_TransferStartStop(struct MarkerContext *Context,APTR UserData,ULONG Qualifier)
  351. {
  352.     struct IFFHandle *Handle;
  353.     TextBufferInfo *BufferInfo;
  354.  
  355.     BufferInfo = Context->UserData;
  356.  
  357.     if(Handle = UserData)
  358.     {
  359.         if(!PopChunk(Handle))
  360.             PopChunk(Handle);
  361.     }
  362.     else
  363.     {
  364.         APTR Buffer;
  365.         LONG Size;
  366.  
  367.         SafeObtainSemaphoreShared(&BufferSemaphore);
  368.  
  369.         SetWait(BufferInfo->Window);
  370.  
  371.         Buffer    = NULL;
  372.         Size    = 0;
  373.  
  374.         if(Qualifier & SHIFT_KEY)
  375.             GetClipContents(Config->ClipConfig->ClipboardUnit,&Buffer,&Size);
  376.  
  377.         if(Handle = OpenIFFClip(Config->ClipConfig->ClipboardUnit,MODE_NEWFILE))
  378.         {
  379.             if(!PushChunk(Handle,ID_FTXT,ID_FORM,IFFSIZE_UNKNOWN))
  380.             {
  381.                 if(!PushChunk(Handle,0,ID_CHRS,IFFSIZE_UNKNOWN))
  382.                 {
  383.                     if(Size > 0)
  384.                     {
  385.                         BOOL Ok;
  386.  
  387.                         Ok = (BOOL)(WriteChunkBytes(Handle,Buffer,Size) == Size);
  388.  
  389.                         FreeVecPooled(Buffer);
  390.  
  391.                         if(Ok)
  392.                             return(Handle);
  393.                     }
  394.                     else
  395.                         return(Handle);
  396.                 }
  397.             }
  398.         }
  399.     }
  400.  
  401.     CloseIFFClip(Handle);
  402.  
  403.     ReleaseSemaphore(&BufferSemaphore);
  404.  
  405.     ClrWait(BufferInfo->Window);
  406.  
  407.     return(NULL);
  408. }
  409.  
  410. STATIC VOID
  411. BufferMarkerStop(TextBufferInfo *BufferInfo)
  412. {
  413.     if(BufferInfo->Marker)
  414.     {
  415.         UpMarker(BufferInfo->Marker);
  416.  
  417.         FreeVecPooled(BufferInfo->Marker);
  418.         BufferInfo->Marker = NULL;
  419.  
  420.         ReportMouse(FALSE,BufferInfo->Window);
  421.  
  422.         ModifyIDCMP(BufferInfo->Window,BufferInfo->Window->IDCMPFlags & ~IDCMP_INTUITICKS);
  423.     }
  424.  
  425.     OffMenu(BufferInfo->Window,FULLMENUNUM(1,0,NOSUB));
  426.     OffMenu(BufferInfo->Window,FULLMENUNUM(1,2,NOSUB));
  427. }
  428.  
  429. STATIC VOID
  430. BufferSetMarker(TextBufferInfo *BufferInfo,LONG Left,LONG Top,LONG Width)
  431. {
  432.     BufferMarkerStop(BufferInfo);
  433.  
  434.     if(BufferInfo->Marker = CreateMarkerContext(
  435.         Buffer_AskPosition,
  436.         Buffer_Scroll,
  437.         Buffer_Highlight,
  438.         Buffer_Highlight,
  439.         Buffer_TransferStartStop,
  440.         Buffer_Transfer,
  441.         Buffer_Put,
  442.         Buffer_PutLine,
  443.         Buffer_PickWord
  444.     ))
  445.     {
  446.         BufferInfo->Marker->UserData = BufferInfo;
  447.  
  448.         SetMarker(BufferInfo->Marker,BufferInfo->TopLine,BufferInfo->DisplayedLines,Lines,0,BufferInfo->NumBufferColumns,BufferInfo->NumBufferColumns,Top,Left,Left + Width - 1);
  449.  
  450.         BufferMarkerInterrupt(BufferInfo);
  451.     }
  452. }
  453.  
  454. STATIC VOID
  455. BufferMarkerSelectAll(TextBufferInfo *BufferInfo)
  456. {
  457.     SafeObtainSemaphoreShared(&BufferSemaphore);
  458.  
  459.     if(Lines > 0)
  460.     {
  461.         BufferMarkerStop(BufferInfo);
  462.  
  463.         if(BufferInfo->Marker = CreateMarkerContext(
  464.             Buffer_AskPosition,
  465.             Buffer_Scroll,
  466.             Buffer_Highlight,
  467.             Buffer_Highlight,
  468.             Buffer_TransferStartStop,
  469.             Buffer_Transfer,
  470.             Buffer_Put,
  471.             Buffer_PutLine,
  472.             Buffer_PickWord
  473.         ))
  474.         {
  475.             BufferInfo->Marker->UserData = BufferInfo;
  476.  
  477.             SelectAllMarker(BufferInfo->Marker,BufferInfo->TopLine,BufferInfo->DisplayedLines,Lines,0,BufferInfo->NumBufferColumns,BufferInfo->NumBufferColumns,
  478.                 0,0,
  479.                 BufferInfo->NumBufferColumns - 1,Lines - 1
  480.             );
  481.  
  482.             BufferMarkerInterrupt(BufferInfo);
  483.         }
  484.     }
  485.  
  486.     ReleaseSemaphore(&BufferSemaphore);
  487. }
  488.  
  489. STATIC VOID
  490. BufferMarkWord(TextBufferInfo *BufferInfo)
  491. {
  492.     BufferMarkerStop(BufferInfo);
  493.  
  494.     if(BufferInfo->Marker = CreateMarkerContext(
  495.         Buffer_AskPosition,
  496.         Buffer_Scroll,
  497.         Buffer_Highlight,
  498.         Buffer_Highlight,
  499.         Buffer_TransferStartStop,
  500.         Buffer_Transfer,
  501.         Buffer_Put,
  502.         Buffer_PutLine,
  503.         Buffer_PickWord
  504.     ))
  505.     {
  506.         BufferInfo->Marker->UserData = BufferInfo;
  507.  
  508.         SafeObtainSemaphoreShared(&BufferSemaphore);
  509.  
  510.         if(!SetWordMarker(BufferInfo->Marker,BufferInfo->TopLine,BufferInfo->DisplayedLines,Lines,0,BufferInfo->NumBufferColumns,BufferInfo->NumBufferColumns))
  511.             BufferMarkerStop(BufferInfo);
  512.         else
  513.             BufferMarkerInterrupt(BufferInfo);
  514.  
  515.         ReleaseSemaphore(&BufferSemaphore);
  516.     }
  517. }
  518.  
  519. STATIC VOID
  520. BufferMarkerStart(TextBufferInfo *BufferInfo,UWORD MsgQualifier)
  521. {
  522.     if(BufferInfo->Marker && !(MsgQualifier & SHIFT_KEY))
  523.         BufferMarkerStop(BufferInfo);
  524.  
  525.     BufferInfo->Interrupted = FALSE;
  526.  
  527.     if(BufferInfo->Marker)
  528.         MoveMouseMarker(BufferInfo->Marker);
  529.     else
  530.     {
  531.         if(BufferInfo->Marker = CreateMarkerContext(
  532.             Buffer_AskPosition,
  533.             Buffer_Scroll,
  534.             Buffer_Highlight,
  535.             Buffer_Highlight,
  536.             Buffer_TransferStartStop,
  537.             Buffer_Transfer,
  538.             Buffer_Put,
  539.             Buffer_PutLine,
  540.             Buffer_PickWord
  541.         ))
  542.         {
  543.             BufferInfo->Marker->UserData = BufferInfo;
  544.  
  545.             SafeObtainSemaphoreShared(&BufferSemaphore);
  546.  
  547.             DownMarker(BufferInfo->Marker,BufferInfo->TopLine,BufferInfo->DisplayedLines,Lines,0,BufferInfo->NumBufferColumns,BufferInfo->NumBufferColumns);
  548.  
  549.             ReleaseSemaphore(&BufferSemaphore);
  550.         }
  551.     }
  552.  
  553.     if(BufferInfo->Marker)
  554.     {
  555.         ReportMouse(TRUE,BufferInfo->Window);
  556.         ModifyIDCMP(BufferInfo->Window,BufferInfo->Window->IDCMPFlags | IDCMP_INTUITICKS);
  557.     }
  558. }
  559.  
  560. STATIC VOID
  561. BufferMarkerInterrupt(TextBufferInfo *BufferInfo)
  562. {
  563.     if(CheckMarker(BufferInfo->Marker))
  564.     {
  565.         OnMenu(BufferInfo->Window,FULLMENUNUM(1,0,NOSUB));
  566.         OnMenu(BufferInfo->Window,FULLMENUNUM(1,2,NOSUB));
  567.     }
  568.     else
  569.     {
  570.         OffMenu(BufferInfo->Window,FULLMENUNUM(1,0,NOSUB));
  571.         OffMenu(BufferInfo->Window,FULLMENUNUM(1,2,NOSUB));
  572.     }
  573.  
  574.     BufferInfo->Interrupted = TRUE;
  575.  
  576.     ReportMouse(FALSE,BufferInfo->Window);
  577.     ModifyIDCMP(BufferInfo->Window,BufferInfo->Window->IDCMPFlags & ~IDCMP_INTUITICKS);
  578. }
  579.  
  580. STATIC VOID
  581. BufferMarkerMoveMouse(TextBufferInfo *BufferInfo)
  582. {
  583.     if(BufferInfo->Marker && !BufferInfo->Interrupted)
  584.         MoveMouseMarker(BufferInfo->Marker);
  585. }
  586.  
  587. STATIC VOID
  588. BufferMarkerTransfer(TextBufferInfo *BufferInfo,UWORD MsgQualifier)
  589. {
  590.     if(BufferInfo->Marker)
  591.     {
  592.         TransferMarker(BufferInfo->Marker,MsgQualifier);
  593.  
  594.         BufferMarkerStop(BufferInfo);
  595.     }
  596. }
  597.  
  598. /****************************************************************************/
  599.  
  600.     /* FlushMsg(struct Window *Window):
  601.      *
  602.      *    Cancel all pending messages of a window.
  603.      */
  604.  
  605. STATIC VOID
  606. FlushMsg(struct Window *Window)
  607. {
  608.     struct Message *Message;
  609.  
  610.     while(Message = GetMsg(Window->UserPort))
  611.         ReplyMsg(Message);
  612. }
  613.  
  614. STATIC VOID
  615. BufferClear(TextBufferInfo *BufferInfo,LONG Left,LONG Top,LONG Right,LONG Bottom)
  616. {
  617.     struct RastPort    *RPort = BufferInfo->RPort;
  618.  
  619.     Left    += BufferInfo->Left;
  620.     Top        += BufferInfo->Top;
  621.     Right    += BufferInfo->Left;
  622.     Bottom    += BufferInfo->Top;
  623.  
  624.     if(Right >= BufferInfo->Left + BufferInfo->Width)
  625.         Right = BufferInfo->Left + BufferInfo->Width - 1;
  626.  
  627.     if(Bottom >= BufferInfo->Top + BufferInfo->Height)
  628.         Bottom = BufferInfo->Top + BufferInfo->Height - 1;
  629.  
  630.     if(Left <= Right && Top <= Bottom)
  631.         EraseRect(RPort,Left,Top,Right,Bottom);
  632. }
  633.  
  634. STATIC VOID
  635. BufferScroll(TextBufferInfo *BufferInfo,LONG Lines)
  636. {
  637.     struct RastPort *RPort;
  638.  
  639.     if(Lines < 0)
  640.         Lines = -BufferInfo->BufferLineOffsets[-Lines];
  641.     else
  642.         Lines = BufferInfo->BufferLineOffsets[Lines];
  643.  
  644.     RPort = BufferInfo->RPort;
  645.  
  646.     ScrollRaster(RPort,0,Lines,BufferInfo->Left,BufferInfo->Top,BufferInfo->Left + BufferInfo->Width - 1,BufferInfo->Top + BufferInfo->Height - 1);
  647.  
  648.     BeginUpdate(RPort->Layer);
  649.     EndUpdate(RPort->Layer,TRUE);
  650. }
  651.  
  652. STATIC VOID
  653. BufferComplement(TextBufferInfo *BufferInfo,LONG SrcX,LONG SrcY,LONG Width,LONG Height)
  654. {
  655.     struct RastPort *RPort = BufferInfo->RPort;
  656.  
  657.     SetPens(RPort,(ULONG)~0,0,JAM1 | COMPLEMENT);
  658.  
  659.     SrcX += BufferInfo->Left;
  660.     SrcY += BufferInfo->Top;
  661.  
  662.     FillBox(RPort,SrcX,SrcY,Width,Height);
  663.  
  664.     SetPens(RPort,BufferInfo->TextFrontPen,BufferInfo->TextBackPen,JAM2);
  665. }
  666.  
  667.     /* DeleteScroller():
  668.      *
  669.      *    Delete scroller and arrow objects.
  670.      */
  671.  
  672. STATIC VOID
  673. DeleteScroller(TextBufferInfo *BufferInfo)
  674. {
  675.     DisposeObject(BufferInfo->Scroller);
  676.     DisposeObject(BufferInfo->UpArrow);
  677.     DisposeObject(BufferInfo->DownArrow);
  678.     DisposeObject(BufferInfo->UpImage);
  679.     DisposeObject(BufferInfo->DownImage);
  680. }
  681.  
  682.     /* CreateScroller(LONG Height):
  683.      *
  684.      *    Create scroller and arrow objects.
  685.      */
  686.  
  687. STATIC BOOL
  688. CreateScroller(TextBufferInfo *BufferInfo,LONG Height)
  689. {
  690.     STATIC struct TagItem ArrowMappings[] =
  691.     {
  692.         GA_ID,    GA_ID,
  693.  
  694.         TAG_END
  695.     };
  696.  
  697.     ULONG ArrowWidth,ArrowHeight;
  698.     struct Screen *Screen;
  699.     LONG SizeType;
  700.     BOOL Result;
  701.  
  702.     Result = FALSE;
  703.  
  704.     if(BufferInfo->Parent)
  705.         Screen = BufferInfo->Parent;
  706.     else
  707.         Screen = BufferInfo->Screen;
  708.  
  709.     if(Screen->Flags & SCREENHIRES)
  710.         SizeType = SYSISIZE_MEDRES;
  711.     else
  712.         SizeType = SYSISIZE_LOWRES;
  713.  
  714.     if(!BufferInfo->Parent)
  715.     {
  716.         if(BufferInfo->UpImage = NewObject(NULL,SYSICLASS,
  717.             SYSIA_Size,        SizeType,
  718.             SYSIA_Which,    UPIMAGE,
  719.             SYSIA_DrawInfo,    BufferInfo->BufferDrawInfo,
  720.         TAG_DONE))
  721.         {
  722.             if(BufferInfo->DownImage = NewObject(NULL,SYSICLASS,
  723.                 SYSIA_Size,        SizeType,
  724.                 SYSIA_Which,    DOWNIMAGE,
  725.                 SYSIA_DrawInfo,    BufferInfo->BufferDrawInfo,
  726.             TAG_DONE))
  727.             {
  728.                 LONG ScrollerHeight,LeftEdge;
  729.  
  730.                 GetAttr(IA_Height,    BufferInfo->UpImage,&ArrowHeight);
  731.                 GetAttr(IA_Width,    BufferInfo->UpImage,&ArrowWidth);
  732.  
  733.                 ScrollerHeight = Height - 2 * ArrowHeight;
  734.  
  735.                 LeftEdge = BufferInfo->Screen->Width - ArrowWidth;
  736.  
  737.                 if(BufferInfo->Scroller = NewObject(NULL,PROPGCLASS,
  738.                     GA_ID,            GAD_SCROLLER,
  739.  
  740.                     GA_Top,            0,
  741.                     GA_Left,        LeftEdge,
  742.                     GA_Width,        ArrowWidth,
  743.                     GA_Height,        ScrollerHeight,
  744.                     GA_Immediate,    TRUE,
  745.                     GA_FollowMouse,    TRUE,
  746.                     GA_RelVerify,    TRUE,
  747.  
  748.                     PGA_Freedom,    FREEVERT,
  749.                     PGA_NewLook,    TRUE,
  750.  
  751.                     PGA_Visible,    1,
  752.                     PGA_Total,        1,
  753.                 TAG_DONE))
  754.                 {
  755.                     if(BufferInfo->UpArrow = NewObject(NULL,BUTTONGCLASS,
  756.                         GA_ID,            GAD_UP,
  757.                         GA_Image,        BufferInfo->UpImage,
  758.                         GA_Left,        LeftEdge,
  759.                         GA_Top,            ScrollerHeight,
  760.                         GA_Height,        ArrowHeight,
  761.                         GA_Width,        ArrowWidth,
  762.                         GA_Previous,    BufferInfo->Scroller,
  763.  
  764.                         ICA_TARGET,        ICTARGET_IDCMP,
  765.                         ICA_MAP,        ArrowMappings,
  766.                     TAG_DONE))
  767.                     {
  768.                         if(BufferInfo->DownArrow = NewObject(NULL,BUTTONGCLASS,
  769.                             GA_ID,            GAD_DOWN,
  770.                             GA_Image,        BufferInfo->DownImage,
  771.                             GA_Left,        LeftEdge,
  772.                             GA_Top,            ScrollerHeight + ArrowHeight,
  773.                             GA_Height,        ArrowHeight,
  774.                             GA_Width,        ArrowWidth,
  775.                             GA_Previous,    BufferInfo->UpArrow,
  776.  
  777.                             ICA_TARGET,        ICTARGET_IDCMP,
  778.                             ICA_MAP,        ArrowMappings,
  779.                         TAG_DONE))
  780.                             Result = TRUE;
  781.                     }
  782.                 }
  783.             }
  784.         }
  785.     }
  786.     else
  787.     {
  788.         Object *SizeImage;
  789.  
  790.         if(SizeImage = NewObject(NULL,SYSICLASS,
  791.             SYSIA_Size,        SizeType,
  792.             SYSIA_Which,    SIZEIMAGE,
  793.             SYSIA_DrawInfo,    BufferInfo->BufferDrawInfo,
  794.         TAG_DONE))
  795.         {
  796.             ULONG SizeWidth,SizeHeight;
  797.  
  798.             GetAttr(IA_Width,    SizeImage,&SizeWidth);
  799.             GetAttr(IA_Height,    SizeImage,&SizeHeight);
  800.  
  801.             DisposeObject(SizeImage);
  802.  
  803.             BufferInfo->RightBorderWidth = SizeWidth;
  804.  
  805.             if(BufferInfo->UpImage = NewObject(NULL,SYSICLASS,
  806.                 SYSIA_Size,        SizeType,
  807.                 SYSIA_Which,    UPIMAGE,
  808.                 SYSIA_DrawInfo,    BufferInfo->BufferDrawInfo,
  809.             TAG_DONE))
  810.             {
  811.                 GetAttr(IA_Height,BufferInfo->UpImage,&ArrowHeight);
  812.  
  813.                 if(BufferInfo->DownImage = NewObject(NULL,SYSICLASS,
  814.                     SYSIA_Size,        SizeType,
  815.                     SYSIA_Which,    DOWNIMAGE,
  816.                     SYSIA_DrawInfo,    BufferInfo->BufferDrawInfo,
  817.                 TAG_DONE))
  818.                 {
  819.                     if(BufferInfo->Scroller = NewObject(NULL,PROPGCLASS,
  820.                         GA_ID,            GAD_SCROLLER,
  821.  
  822.                         GA_Top,            Screen->WBorTop + Screen->Font->ta_YSize + 2,
  823.                         GA_RelHeight,    -(Screen->WBorTop + Screen->Font->ta_YSize + 2 + SizeHeight + 1 + 2 * ArrowHeight),
  824.                         GA_Width,        SizeWidth - 8,
  825.                         GA_RelRight,    -(SizeWidth - 5),
  826.  
  827.                         GA_Immediate,    TRUE,
  828.                         GA_FollowMouse,    TRUE,
  829.                         GA_RelVerify,    TRUE,
  830.                         GA_RightBorder,    TRUE,
  831.  
  832.                         PGA_Freedom,    FREEVERT,
  833.                         PGA_NewLook,    TRUE,
  834.                         PGA_Borderless,    TRUE,
  835.  
  836.                         PGA_Visible,    1,
  837.                         PGA_Total,        1,
  838.                     TAG_DONE))
  839.                     {
  840.                         if(BufferInfo->UpArrow = NewObject(NULL,BUTTONGCLASS,
  841.                             GA_ID,            GAD_UP,
  842.  
  843.                             GA_Image,        BufferInfo->UpImage,
  844.                             GA_RelRight,    -(SizeWidth - 1),
  845.                             GA_RelBottom,    -(SizeHeight - 1 + 2 * ArrowHeight),
  846.                             GA_Height,        ArrowHeight,
  847.                             GA_Width,        SizeWidth,
  848.                             GA_Previous,    BufferInfo->Scroller,
  849.                             GA_RightBorder,    TRUE,
  850.  
  851.                             ICA_TARGET,        ICTARGET_IDCMP,
  852.                             ICA_MAP,        ArrowMappings,
  853.                         TAG_DONE))
  854.                         {
  855.                             if(BufferInfo->DownArrow = NewObject(NULL,BUTTONGCLASS,
  856.                                 GA_ID,            GAD_DOWN,
  857.  
  858.                                 GA_Image,        BufferInfo->DownImage,
  859.                                 GA_RelRight,    -(SizeWidth - 1),
  860.                                 GA_RelBottom,    -(SizeHeight - 1 + ArrowHeight),
  861.                                 GA_Height,        ArrowHeight,
  862.                                 GA_Width,        SizeWidth,
  863.                                 GA_Previous,    BufferInfo->UpArrow,
  864.                                 GA_RightBorder,    TRUE,
  865.  
  866.                                 ICA_TARGET,        ICTARGET_IDCMP,
  867.                                 ICA_MAP,        ArrowMappings,
  868.                             TAG_DONE))
  869.                                 Result = TRUE;
  870.                         }
  871.                     }
  872.                 }
  873.             }
  874.         }
  875.     }
  876.  
  877.     if(!Result)
  878.         DeleteScroller(BufferInfo);
  879.     else
  880.         BufferInfo->ArrowWidth = ArrowWidth;
  881.  
  882.     return(Result);
  883. }
  884.  
  885.     /* PrintLine(STRPTR Buffer,LONG LineNumber):
  886.      *
  887.      *    Print a line at a given line number in the displayed area.
  888.      */
  889.  
  890. STATIC VOID
  891. PrintLine(TextBufferInfo *BufferInfo,STRPTR Buffer,LONG LineNumber)
  892. {
  893.     LONG Length;
  894.  
  895.     Length = Buffer[-1];
  896.  
  897.         /* Print the text. */
  898.  
  899.     if(Length > 0)
  900.     {
  901.         struct TextExtent Extent;
  902.  
  903.         Length = TextFit(BufferInfo->RPort,Buffer,Length,&Extent,NULL,1,BufferInfo->Width,32767);
  904.  
  905.         if(Length > 0)
  906.             PlaceText(BufferInfo->RPort,BufferInfo->Left,BufferInfo->Top + BufferInfo->BufferLineOffsets[LineNumber],Buffer,Length);
  907.     }
  908.  
  909.         /* The line doesn't exactly fill the displayed line,
  910.          * so erase the remaining columns.
  911.          */
  912.  
  913.     if(Length < BufferInfo->NumBufferColumns)
  914.         BufferClear(BufferInfo,BufferInfo->BufferColumnOffsets[Length],BufferInfo->BufferLineOffsets[LineNumber],BufferInfo->BufferColumnOffsets[BufferInfo->NumBufferColumns] - 1,BufferInfo->BufferLineOffsets[LineNumber + 1] - 1);
  915. }
  916.  
  917. STATIC VOID
  918. PrintLineWithMarker(TextBufferInfo *BufferInfo,STRPTR Buffer,LONG LineNumber,LONG AbsoluteLine)
  919. {
  920.     LONG Length,Left,Right;
  921.  
  922.     Length = Buffer[-1];
  923.  
  924.         /* Print the text. */
  925.  
  926.     if(Length > 0)
  927.     {
  928.         struct TextExtent Extent;
  929.  
  930.         Length = TextFit(BufferInfo->RPort,Buffer,Length,&Extent,NULL,1,BufferInfo->Width,32767);
  931.  
  932.         if(Length > 0)
  933.             PlaceText(BufferInfo->RPort,BufferInfo->Left,BufferInfo->Top + BufferInfo->BufferLineOffsets[LineNumber],Buffer,Length);
  934.     }
  935.  
  936.         /* The line doesn't exactly fill the displayed line,
  937.          * so erase the remaining columns.
  938.          */
  939.  
  940.     if(Length < BufferInfo->NumBufferColumns)
  941.         BufferClear(BufferInfo,BufferInfo->BufferColumnOffsets[Length],BufferInfo->BufferLineOffsets[LineNumber],BufferInfo->BufferColumnOffsets[BufferInfo->NumBufferColumns] - 1,BufferInfo->BufferLineOffsets[LineNumber + 1] - 1);
  942.  
  943.     if(CheckMarkerHighlighting(BufferInfo->Marker,AbsoluteLine,&Left,&Right))
  944.     {
  945.         if(Left < BufferInfo->NumBufferColumns)
  946.         {
  947.             if(Right >= BufferInfo->NumBufferColumns)
  948.                 Right = BufferInfo->NumBufferColumns - 1;
  949.  
  950.             if(Left <= Right)
  951.                 BufferComplement(BufferInfo,BufferInfo->BufferColumnOffsets[Left],BufferInfo->BufferLineOffsets[LineNumber],BufferInfo->BufferColumnOffsets[Right - Left + 1],BufferInfo->BufferLineOffsets[1]);
  952.         }
  953.     }
  954. }
  955.  
  956.     /* RedrawScreen(LONG FirstLine):
  957.      *
  958.      *    Redraw the contents of the entire screen and return the
  959.      *    number of lines actually drawn.
  960.      */
  961.  
  962. STATIC LONG
  963. RedrawScreen(TextBufferInfo *BufferInfo,LONG FirstLine)
  964. {
  965.     LONG i,Last,Line = 0,Result;
  966.  
  967.     SafeObtainSemaphoreShared(&BufferSemaphore);
  968.  
  969.         /* Determine last line to display. */
  970.  
  971.     if((Last = FirstLine + BufferInfo->NumBufferLines) > Lines)
  972.     {
  973.         Last = Lines;
  974.  
  975.         if((FirstLine = Last - BufferInfo->NumBufferLines) < 0)
  976.             FirstLine = 0;
  977.  
  978.         BufferInfo->TopLine = FirstLine;
  979.     }
  980.  
  981.     if(Last > FirstLine + BufferInfo->NumBufferLines)
  982.         Last = FirstLine + BufferInfo->NumBufferLines;
  983.  
  984.     Result = Last - FirstLine;
  985.  
  986.     if(Lines)
  987.     {
  988.         if(BufferInfo->LastTopLine == -1)
  989.             BufferInfo->LastTopLine = FirstLine;
  990.         else
  991.         {
  992.             LONG Delta = FirstLine - BufferInfo->LastTopLine;
  993.  
  994.             if(ABS(Delta) >= BufferInfo->NumBufferLines)
  995.                 BufferInfo->LastTopLine = FirstLine;
  996.             else
  997.             {
  998.                     /* No change? */
  999.  
  1000.                 if(!Delta)
  1001.                 {
  1002.                     ReleaseSemaphore(&BufferSemaphore);
  1003.  
  1004.                     return(Result);
  1005.                 }
  1006.                 else
  1007.                 {
  1008.                     BufferInfo->LastTopLine = FirstLine;
  1009.  
  1010.                         /* Scrolled up? */
  1011.  
  1012.                     if(Delta < 0)
  1013.                     {
  1014.                         BufferScroll(BufferInfo,Delta);
  1015.  
  1016.                         Last = FirstLine - Delta;
  1017.                     }
  1018.                     else
  1019.                     {
  1020.                             /* Scrolled down. */
  1021.  
  1022.                         BufferScroll(BufferInfo,Delta);
  1023.  
  1024.                         FirstLine += BufferInfo->NumBufferLines - Delta;
  1025.  
  1026.                         Line = BufferInfo->NumBufferLines - Delta;
  1027.                     }
  1028.                 }
  1029.             }
  1030.         }
  1031.  
  1032.         if(BufferLines)
  1033.         {
  1034.             if(BufferInfo->Marker)
  1035.             {
  1036.                 for(i = FirstLine ; i < Last ; i++)
  1037.                     PrintLineWithMarker(BufferInfo,BufferLines[i],Line++,i);
  1038.             }
  1039.             else
  1040.             {
  1041.                 for(i = FirstLine ; i < Last ; i++)
  1042.                     PrintLine(BufferInfo,BufferLines[i],Line++);
  1043.             }
  1044.         }
  1045.     }
  1046.  
  1047.         /* We didn't fill the whole screen, so clear the rest. */
  1048.  
  1049.     if(Result < BufferInfo->NumBufferLines)
  1050.         BufferClear(BufferInfo,0,BufferInfo->BufferLineOffsets[Result],BufferInfo->BufferColumnOffsets[BufferInfo->NumBufferColumns] - 1,BufferInfo->BufferLineOffsets[BufferInfo->NumBufferLines] - 1);
  1051.  
  1052.     ReleaseSemaphore(&BufferSemaphore);
  1053.  
  1054.     return(Result);
  1055. }
  1056.  
  1057. STATIC VOID
  1058. BufferDestructor(struct DataMsg *Item)
  1059. {
  1060.     Signal(Item->Client,Item->Mask);
  1061. }
  1062.  
  1063. STATIC VOID
  1064. BufferSerWrite(TextBufferInfo *BufferInfo,APTR Data,LONG Size)
  1065. {
  1066.     struct DataMsg Msg;
  1067.  
  1068.     InitMsgItem(&Msg,(DESTRUCTOR)BufferDestructor);
  1069.  
  1070.     Msg.Type    = DATAMSGTYPE_WRITE;
  1071.     Msg.Data    = Data;
  1072.     Msg.Size    = Size;
  1073.     Msg.Client    = FindTask(NULL);
  1074.     Msg.Mask    = 1UL << BufferInfo->BufferSignal;
  1075.  
  1076.     Forbid();
  1077.  
  1078.     ClrSignal(Msg.Mask);
  1079.  
  1080.     PutMsgItem(SpecialQueue,(struct MsgItem *)&Msg);
  1081.  
  1082.     Wait(Msg.Mask);
  1083.  
  1084.     Permit();
  1085. }
  1086.  
  1087. STATIC VOID
  1088. StartSearch(TextBufferInfo *BufferInfo,struct SearchInfo *SearchInfo,STRPTR SearchBuffer)
  1089. {
  1090.     SafeObtainSemaphoreShared(&BufferSemaphore);
  1091.  
  1092.     LT_LockWindow(BufferInfo->Window);
  1093.  
  1094.     if(Lines)
  1095.     {
  1096.         LONG LineNumber;
  1097.  
  1098.         LineNumber = SearchTextBuffer(SearchInfo);
  1099.  
  1100.         BufferMarkerStop(BufferInfo);
  1101.  
  1102.         if(LineNumber == -1)
  1103.         {
  1104.             ShowRequest(BufferInfo->Window,LocaleString(MSG_TERMBUFFER_DID_NOT_FIND_TXT),LocaleString(MSG_GLOBAL_CONTINUE_TXT),SearchBuffer);
  1105.  
  1106.             FlushMsg(BufferInfo->Window);
  1107.  
  1108.             SearchInfo->FoundY = -1;
  1109.         }
  1110.         else
  1111.         {
  1112.             if(LineNumber < BufferInfo->TopLine)
  1113.             {
  1114.                 BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine = LineNumber);
  1115.  
  1116.                 SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1117.                     PGA_Top,BufferInfo->TopLine,
  1118.                 TAG_DONE);
  1119.             }
  1120.             else
  1121.             {
  1122.                 if(LineNumber > BufferInfo->TopLine + BufferInfo->DisplayedLines - 1)
  1123.                 {
  1124.                     if(LineNumber >= Lines - BufferInfo->NumBufferLines)
  1125.                     {
  1126.                         LONG NewCurrentLine;
  1127.  
  1128.                         if((NewCurrentLine = Lines - BufferInfo->NumBufferLines) < 0)
  1129.                             NewCurrentLine = 0;
  1130.  
  1131.                         if(BufferInfo->TopLine != NewCurrentLine)
  1132.                             BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine = NewCurrentLine);
  1133.                     }
  1134.                     else
  1135.                         BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine = LineNumber);
  1136.  
  1137.                     SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1138.                         PGA_Top,BufferInfo->TopLine,
  1139.                     TAG_DONE);
  1140.                 }
  1141.             }
  1142.  
  1143.             BufferSetMarker(BufferInfo,SearchInfo->FoundX,LineNumber,SearchInfo->PatternWidth);
  1144.         }
  1145.     }
  1146.     else
  1147.         ShowRequest(BufferInfo->Window,LocaleString(MSG_GLOBAL_NOTHING_IN_THE_BUFFER_TXT),LocaleString(MSG_GLOBAL_CONTINUE_TXT));
  1148.  
  1149.     LT_UnlockWindow(BufferInfo->Window);
  1150.  
  1151.     ReleaseSemaphore(&BufferSemaphore);
  1152. }
  1153.  
  1154. STATIC BOOL
  1155. HandleBuffer(struct SignalSemaphore *Access,TextBufferInfo **Data)
  1156. {
  1157.     ULONG LastSeconds,LastMicros,Seconds,Micros;
  1158.     BOOL WindowJustActivated,UpdatePercent,RingBack;
  1159.     ULONG SignalSet;
  1160.     struct IntuiMessage *Message;
  1161.     ULONG MsgClass;
  1162.     UWORD MsgCode,MsgQualifier,MsgGadgetID;
  1163.     UBYTE Char,FullChar;
  1164.     UBYTE PercentBuffer[80],SearchBuffer[256];
  1165.     STRPTR PercentTemplate;
  1166.     struct SearchContext *Context;
  1167.     struct SearchInfo *SearchInfo;
  1168.     struct Hook HistoryHook;
  1169.     TextBufferInfo *BufferInfo;
  1170.     BOOL DisplayDirty,DoNotUpdateScroller;
  1171.     LONG NewTopLine;
  1172.     ULONG ClickSeconds,ClickMicros;
  1173.     BOOL TestingDoubleClick;
  1174.  
  1175.     TestingDoubleClick    = FALSE;
  1176.     ClickSeconds        = 0;
  1177.     ClickMicros            = 0;
  1178.  
  1179.     LastSeconds            = 0;
  1180.     LastMicros            = 0;
  1181.     UpdatePercent        = TRUE;
  1182.     RingBack             = FALSE;
  1183.  
  1184.     Context                = NULL;
  1185.     SearchInfo            = NULL;
  1186.  
  1187.     DoNotUpdateScroller    = FALSE;
  1188.     DisplayDirty        = FALSE;
  1189.     NewTopLine            = 0;
  1190.  
  1191.     BufferInfo = *Data;
  1192.  
  1193.     if(LocaleBase)
  1194.         PercentTemplate = "%lD/%lD (%ld%%)";
  1195.     else
  1196.         PercentTemplate = "%ld/%ld (%ld%%)";
  1197.  
  1198.     while(TRUE)
  1199.     {
  1200.         if(DisplayDirty)
  1201.         {
  1202.             DisplayDirty = FALSE;
  1203.  
  1204.             if(NewTopLine < 0)
  1205.                 NewTopLine = 0;
  1206.             else if (NewTopLine + BufferInfo->NumBufferLines >= Lines)
  1207.                 NewTopLine = Lines - BufferInfo->NumBufferLines;
  1208.  
  1209.             if(NewTopLine >= 0 && NewTopLine != BufferInfo->TopLine)
  1210.             {
  1211.                 BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,NewTopLine);
  1212.  
  1213.                 BufferInfo->TopLine = NewTopLine;
  1214.  
  1215.                 if(DoNotUpdateScroller)
  1216.                     DoNotUpdateScroller    = FALSE;
  1217.                 else
  1218.                 {
  1219.                     SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1220.                         PGA_Top,BufferInfo->TopLine,
  1221.                     TAG_DONE);
  1222.                 }
  1223.  
  1224.                 UpdatePercent = TRUE;
  1225.             }
  1226.         }
  1227.  
  1228.         if(BufferInfo->TopLine < 0)
  1229.             BufferInfo->TopLine = 0;
  1230.  
  1231.             /* Show where we are. */
  1232.  
  1233.         if(UpdatePercent)
  1234.         {
  1235.             STRPTR ScreenTitle,WindowTitle;
  1236.  
  1237.             SafeObtainSemaphoreShared(&BufferSemaphore);
  1238.  
  1239.             if(Lines > 0)
  1240.                 LimitedSPrintf(sizeof(PercentBuffer),PercentBuffer,PercentTemplate,BufferInfo->TopLine,MAX(0,Lines - BufferInfo->NumBufferLines),(100 * (BufferInfo->TopLine + BufferInfo->DisplayedLines)) / Lines);
  1241.             else
  1242.                 PercentBuffer[0] = 0;
  1243.  
  1244.             LimitedSPrintf(sizeof(BufferInfo->TitleBuffer),BufferInfo->TitleBuffer,"%s  %s",LocaleString(MSG_TERMBUFFER_TERM_BUFFER_TXT),PercentBuffer);
  1245.  
  1246.             if(BufferInfo->Parent)
  1247.             {
  1248.                 WindowTitle    = BufferInfo->TitleBuffer;
  1249.                 ScreenTitle = (STRPTR)~0;
  1250.             }
  1251.             else
  1252.                 ScreenTitle = WindowTitle = BufferInfo->TitleBuffer;
  1253.  
  1254.             SetWindowTitles(BufferInfo->Window,WindowTitle,ScreenTitle);
  1255.  
  1256.             UpdatePercent = FALSE;
  1257.  
  1258.             ReleaseSemaphore(&BufferSemaphore);
  1259.         }
  1260.  
  1261.         if(!BufferInfo->BufferTerminated)
  1262.             SignalSet = Wait(SIG_KILL | SIG_TOFRONT | SIG_UPDATE | SIG_MOVEUP | PORTMASK(BufferInfo->Window->UserPort) | BufferInfo->QueueMask);
  1263.         else
  1264.         {
  1265.             ObtainSemaphore(Access);
  1266.  
  1267.             SignalSet = SIG_KILL | SIG_TOFRONT | SIG_UPDATE | SIG_MOVEUP | PORTMASK(BufferInfo->Window->UserPort) | BufferInfo->QueueMask;
  1268.             SignalSet = SetSignal(0,SignalSet) & SignalSet;
  1269.  
  1270.                 /* Disconnect */
  1271.  
  1272.             if(!SignalSet)
  1273.             {
  1274.                 BufferInfo->Buddy = NULL;
  1275.  
  1276.                 *Data = NULL;
  1277.  
  1278.                 ReleaseSemaphore(Access);
  1279.  
  1280.                 break;
  1281.             }
  1282.             else
  1283.                 ReleaseSemaphore(Access);
  1284.         }
  1285.  
  1286.             /* Leave this town? */
  1287.  
  1288.         if(SignalSet & SIG_KILL)
  1289.             BufferInfo->BufferTerminated = RingBack = TRUE;
  1290.  
  1291.             /* Bring our window to the front. */
  1292.  
  1293.         if(SignalSet & SIG_TOFRONT)
  1294.         {
  1295.             if(Context)
  1296.                 LT_ShowWindow(Context->SearchHandle,TRUE);
  1297.  
  1298.             BumpWindow(BufferInfo->Window);
  1299.         }
  1300.  
  1301.             /* Redraw the entire screen. */
  1302.  
  1303.         if(SignalSet & SIG_MOVEUP)
  1304.         {
  1305.             BufferMarkerStop(BufferInfo);
  1306.  
  1307.             BufferInfo->LastTopLine = -1;
  1308.  
  1309.             SafeObtainSemaphoreShared(&BufferSemaphore);
  1310.  
  1311.             BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine);
  1312.  
  1313.             SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1314.                 PGA_Total,Lines,
  1315.             TAG_DONE);
  1316.  
  1317.             ReleaseSemaphore(&BufferSemaphore);
  1318.  
  1319.             UpdatePercent = TRUE;
  1320.  
  1321.             Signal((struct Task *)ThisProcess,SIG_HANDSHAKE);
  1322.         }
  1323.  
  1324.             /* Redraw the entire screen unless it is already drawn. */
  1325.  
  1326.         if(SignalSet & SIG_UPDATE)
  1327.         {
  1328.             SafeObtainSemaphoreShared(&BufferSemaphore);
  1329.  
  1330.             if(BufferInfo->DisplayedLines < BufferInfo->NumBufferLines || !Lines)
  1331.             {
  1332.                 BufferMarkerStop(BufferInfo);
  1333.  
  1334.                 BufferInfo->LastTopLine = -1;
  1335.  
  1336.                 BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine);
  1337.             }
  1338.  
  1339.             SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1340.                 PGA_Total,Lines,
  1341.             TAG_DONE);
  1342.  
  1343.             ReleaseSemaphore(&BufferSemaphore);
  1344.  
  1345.             UpdatePercent = TRUE;
  1346.  
  1347.             Signal((struct Task *)ThisProcess,SIG_HANDSHAKE);
  1348.         }
  1349.  
  1350.             /* A request from the queue? */
  1351.  
  1352.         if(SignalSet & BufferInfo->QueueMask)
  1353.         {
  1354.             struct DataMsg *Msg;
  1355.  
  1356.             while(Msg = (struct DataMsg *)GetMsgItem(BufferInfo->Queue))
  1357.             {
  1358.                 switch(Msg->Size)
  1359.                 {
  1360.                     case REVIEW_MOVE_TOP:
  1361.  
  1362.                         DisplayDirty    = TRUE;
  1363.                         NewTopLine        = 0;
  1364.  
  1365.                         break;
  1366.  
  1367.                     case REVIEW_MOVE_BOTTOM:
  1368.  
  1369.                         DisplayDirty    = TRUE;
  1370.                         NewTopLine        = Lines - BufferInfo->NumBufferLines;
  1371.  
  1372.                         break;
  1373.  
  1374.                     case REVIEW_MOVE_UP:
  1375.  
  1376.                         DisplayDirty    = TRUE;
  1377.                         NewTopLine        = BufferInfo->TopLine - (BufferInfo->NumBufferLines - 1);
  1378.  
  1379.                         break;
  1380.  
  1381.                     case REVIEW_MOVE_DOWN:
  1382.  
  1383.                         DisplayDirty    = TRUE;
  1384.                         NewTopLine        = BufferInfo->TopLine + (BufferInfo->NumBufferLines - 1);
  1385.  
  1386.                         break;
  1387.                 }
  1388.  
  1389.                 DeleteMsgItem((struct MsgItem *)Msg);
  1390.             }
  1391.         }
  1392.  
  1393.             /* Process the incoming window
  1394.              * input.
  1395.              */
  1396.  
  1397.         while(Message = (struct IntuiMessage *)GetMsg(BufferInfo->Window->UserPort))
  1398.         {
  1399.             if(Context && Context->SearchWindow == Message->IDCMPWindow)
  1400.             {
  1401.                 MsgClass = NULL;
  1402.  
  1403.                 if(HandleSearchMessage(Context,&Message))
  1404.                 {
  1405.                     BOOL Ok = Context->Ok;
  1406.  
  1407.                     DeleteSearchContext(Context);
  1408.  
  1409.                     Context = NULL;
  1410.  
  1411.                     if(Ok)
  1412.                     {
  1413.                         if(SearchInfo)
  1414.                             DeleteSearchInfo(SearchInfo);
  1415.  
  1416.                         if(SearchInfo = CreateSearchInfo(SearchBuffer,BufferInfo->SearchForward,BufferInfo->IgnoreCase,BufferInfo->WholeWords))
  1417.                             StartSearch(BufferInfo,SearchInfo,SearchBuffer);
  1418.                     }
  1419.                     else
  1420.                     {
  1421.                         if(SearchInfo)
  1422.                             DeleteSearchInfo(SearchInfo);
  1423.  
  1424.                         SearchInfo = NULL;
  1425.                     }
  1426.                 }
  1427.             }
  1428.             else
  1429.             {
  1430.                 MsgClass        = Message->Class;
  1431.                 MsgCode            = Message->Code;
  1432.                 MsgQualifier    = Message->Qualifier;
  1433.                 Seconds            = Message->Seconds;
  1434.                 Micros            = Message->Micros;
  1435.  
  1436.                 WindowJustActivated    = FALSE;
  1437.                 MsgGadgetID            = 0;
  1438.  
  1439.                 switch(MsgClass)
  1440.                 {
  1441.                     case IDCMP_SIZEVERIFY:
  1442.  
  1443.                         BufferMarkerStop(BufferInfo);
  1444.                         break;
  1445.  
  1446.                     case IDCMP_IDCMPUPDATE:
  1447.  
  1448.                         MsgGadgetID = (UWORD)GetTagData(GA_ID,0,(struct TagItem *)Message->IAddress);
  1449.                         break;
  1450.  
  1451.                     case IDCMP_GADGETUP:
  1452.                     case IDCMP_GADGETDOWN:
  1453.  
  1454.                         MsgGadgetID = ((struct Gadget *)Message->IAddress)->GadgetID;
  1455.                         break;
  1456.  
  1457.                     case IDCMP_RAWKEY:
  1458.  
  1459.                         FullChar = ConvertTheKey(NULL,0,MsgCode,MsgQualifier,*(ULONG *)Message->IAddress);
  1460.  
  1461.                         if(!(MsgQualifier & IEQUALIFIER_NUMERICPAD))
  1462.                             Char = FullChar;
  1463.                         else
  1464.                             Char = ConvertTheKey(NULL,0,MsgCode,MsgQualifier & ~(CONTROL_KEY | ALT_KEY | SHIFT_KEY),*(ULONG *)Message->IAddress);
  1465.  
  1466.                         break;
  1467.  
  1468.                     case IDCMP_ACTIVEWINDOW:
  1469.                     case IDCMP_MOUSEBUTTONS:
  1470.  
  1471.                         WindowJustActivated = (BOOL)(Seconds == LastSeconds && Micros == LastMicros);
  1472.                         break;
  1473.                 }
  1474.  
  1475.                 LastSeconds    = Seconds;
  1476.                 LastMicros    = Micros;
  1477.  
  1478.                 ReplyMsg((struct Message *)Message);
  1479.             }
  1480.  
  1481.             if(MsgClass == IDCMP_INACTIVEWINDOW)
  1482.                 BufferMarkerInterrupt(BufferInfo);
  1483.  
  1484.             if(MsgClass == IDCMP_IDCMPUPDATE)
  1485.             {
  1486.                 switch(MsgGadgetID)
  1487.                 {
  1488.                     case GAD_UP:
  1489.  
  1490.                         DisplayDirty    = TRUE;
  1491.                         NewTopLine        = BufferInfo->TopLine - 1;
  1492.  
  1493.                         break;
  1494.  
  1495.                     case GAD_DOWN:
  1496.  
  1497.                         DisplayDirty    = TRUE;
  1498.                         NewTopLine        = BufferInfo->TopLine + 1;
  1499.  
  1500.                         break;
  1501.                 }
  1502.             }
  1503.  
  1504.             if(MsgClass == IDCMP_MOUSEMOVE || MsgClass == IDCMP_INTUITICKS)
  1505.                 BufferMarkerMoveMouse(BufferInfo);
  1506.  
  1507.             if(MsgClass == IDCMP_MOUSEMOVE || ((MsgClass == IDCMP_GADGETDOWN || MsgClass == IDCMP_GADGETUP) && MsgGadgetID == GAD_SCROLLER))
  1508.             {
  1509.                 GetAttr(PGA_Top,BufferInfo->Scroller,(ULONG *)&NewTopLine);
  1510.  
  1511.                 DisplayDirty        = TRUE;
  1512.                 DoNotUpdateScroller    = TRUE;
  1513.             }
  1514.  
  1515.             if(MsgClass == IDCMP_MOUSEBUTTONS)
  1516.             {
  1517.                 if(!WindowJustActivated || MsgCode != SELECTDOWN)
  1518.                 {
  1519.                     switch(MsgCode)
  1520.                     {
  1521.                         case SELECTUP:
  1522.  
  1523.                             BufferMarkerInterrupt(BufferInfo);
  1524.                             break;
  1525.  
  1526.                         case SELECTDOWN:
  1527.  
  1528.                             if(TestingDoubleClick)
  1529.                             {
  1530.                                 ULONG CurrentSeconds,CurrentMicros;
  1531.  
  1532.                                 TestingDoubleClick = FALSE;
  1533.  
  1534.                                 CurrentTime(&CurrentSeconds,&CurrentMicros);
  1535.  
  1536.                                 if(DoubleClick(ClickSeconds,ClickMicros,CurrentSeconds,CurrentMicros))
  1537.                                 {
  1538.                                     BufferMarkWord(BufferInfo);
  1539.  
  1540.                                     break;
  1541.                                 }
  1542.                             }
  1543.                             else
  1544.                             {
  1545.                                 TestingDoubleClick = TRUE;
  1546.  
  1547.                                 CurrentTime(&ClickSeconds,&ClickMicros);
  1548.                             }
  1549.  
  1550.                             BufferMarkerStart(BufferInfo,MsgQualifier);
  1551.                             break;
  1552.                     }
  1553.                 }
  1554.             }
  1555.  
  1556.             if(MsgClass == IDCMP_RAWKEY)
  1557.             {
  1558.                 switch(MsgCode)
  1559.                 {
  1560.                     case HELP_CODE:
  1561.  
  1562.                         GuideDisplay(CONTEXT_TEXTBUFFER);
  1563.                         break;
  1564.  
  1565.                         /* Scroll the buffer up. */
  1566.  
  1567.                     case CURSOR_UP_CODE:
  1568.  
  1569.                         DisplayDirty = TRUE;
  1570.  
  1571.                         if(MsgQualifier & (CONTROL_KEY | ALT_KEY))
  1572.                         {
  1573.                             NewTopLine = 0;
  1574.  
  1575.                             break;
  1576.                         }
  1577.  
  1578.                         if(MsgQualifier & SHIFT_KEY)
  1579.                         {
  1580.                             NewTopLine = BufferInfo->TopLine - (BufferInfo->NumBufferLines - 1);
  1581.  
  1582.                             break;
  1583.                         }
  1584.  
  1585.                         NewTopLine = BufferInfo->TopLine - 1;
  1586.                         break;
  1587.  
  1588.                         /* Scroll the buffer down. */
  1589.  
  1590.                     case CURSOR_DOWN_CODE:
  1591.  
  1592.                         DisplayDirty = TRUE;
  1593.  
  1594.                         if(MsgQualifier & (CONTROL_KEY | ALT_KEY))
  1595.                         {
  1596.                             NewTopLine = Lines - BufferInfo->NumBufferLines;
  1597.  
  1598.                             break;
  1599.                         }
  1600.  
  1601.                         if(MsgQualifier & SHIFT_KEY)
  1602.                         {
  1603.                             NewTopLine = BufferInfo->TopLine + (BufferInfo->NumBufferLines - 1);
  1604.  
  1605.                             break;
  1606.                         }
  1607.  
  1608.                         NewTopLine = BufferInfo->TopLine + 1;
  1609.                         break;
  1610.  
  1611.                     default:
  1612.  
  1613.                             /* Use the numeric keypad keys to
  1614.                              * move through the buffer.
  1615.                              */
  1616.  
  1617.                         if(!(MsgQualifier & IEQUALIFIER_NUMERICPAD))
  1618.                             Char = FullChar;
  1619.                         else
  1620.                         {
  1621.                             switch(Char - '0')
  1622.                             {
  1623.                                     /* Jump to bottom. */
  1624.  
  1625.                                 case 1:
  1626.  
  1627.                                     DisplayDirty    = TRUE;
  1628.                                     NewTopLine        = Lines - BufferInfo->NumBufferLines;
  1629.  
  1630.                                     break;
  1631.  
  1632.                                     /* Move one line down. */
  1633.  
  1634.                                 case 2:
  1635.  
  1636.                                     DisplayDirty    = TRUE;
  1637.                                     NewTopLine        = BufferInfo->TopLine + 1;
  1638.  
  1639.                                     break;
  1640.  
  1641.                                     /* Move one page down. */
  1642.  
  1643.                                 case 3:
  1644.  
  1645.                                     DisplayDirty    = TRUE;
  1646.                                     NewTopLine        = BufferInfo->TopLine + (BufferInfo->NumBufferLines - 1);
  1647.  
  1648.                                     break;
  1649.  
  1650.                                     /* Jump to top. */
  1651.  
  1652.                                 case 7:
  1653.  
  1654.                                     DisplayDirty    = TRUE;
  1655.                                     NewTopLine        = 0;
  1656.  
  1657.                                     break;
  1658.  
  1659.                                     /* Move one line up. */
  1660.  
  1661.                                 case 8:
  1662.  
  1663.                                     DisplayDirty    = TRUE;
  1664.                                     NewTopLine        = BufferInfo->TopLine - 1;
  1665.  
  1666.                                     break;
  1667.  
  1668.                                     /* Move one page up. */
  1669.  
  1670.                                 case 9:
  1671.  
  1672.                                     DisplayDirty    = TRUE;
  1673.                                     NewTopLine        = BufferInfo->TopLine - (BufferInfo->NumBufferLines - 1);
  1674.  
  1675.                                     break;
  1676.                             }
  1677.  
  1678.                             if(Char >= '0' && Char <= '9')
  1679.                                 Char = 0;
  1680.                             else
  1681.                                 Char = FullChar;
  1682.                         }
  1683.  
  1684.                         if(Char > 0 && BufferInfo->Parent)
  1685.                         {
  1686.                             if(Config->SerialConfig->StripBit8)
  1687.                                 Char &= 0x7F;
  1688.  
  1689.                             if(Get_xOFF())
  1690.                             {
  1691.                                 if(Char == XON)
  1692.                                     Clear_xOFF();
  1693.                             }
  1694.                             else
  1695.                             {
  1696.                                 STRPTR String;
  1697.                                 LONG Length;
  1698.  
  1699.                                 String = &Char;
  1700.                                 Length = 1;
  1701.  
  1702.                                     /* Convert chars as appropriate */
  1703.  
  1704.                                 switch(Char)
  1705.                                 {
  1706.                                     case '\n':
  1707.  
  1708.                                         switch(Config->TerminalConfig->SendLF)
  1709.                                         {
  1710.                                             case EOL_LF:
  1711.  
  1712.                                                 break;
  1713.  
  1714.                                             case EOL_LFCR:
  1715.  
  1716.                                                 String = "\n\r";
  1717.                                                 Length = 2;
  1718.                                                 break;
  1719.  
  1720.                                             case EOL_CRLF:
  1721.  
  1722.                                                 String = "\r\n";
  1723.                                                 Length = 2;
  1724.                                                 break;
  1725.  
  1726.                                             case EOL_CR:
  1727.  
  1728.                                                 String = "\r";
  1729.                                                 break;
  1730.                                         }
  1731.  
  1732.                                         break;
  1733.  
  1734.                                     case '\r':
  1735.  
  1736.                                         switch(Config->TerminalConfig->SendCR)
  1737.                                         {
  1738.                                             case EOL_CR:
  1739.  
  1740.                                                 break;
  1741.  
  1742.                                             case EOL_LFCR:
  1743.  
  1744.                                                 String = "\n\r";
  1745.                                                 Length = 2;
  1746.                                                 break;
  1747.  
  1748.                                             case EOL_CRLF:
  1749.  
  1750.                                                 String = "\r\n";
  1751.                                                 Length = 2;
  1752.                                                 break;
  1753.  
  1754.                                             case EOL_LF:
  1755.  
  1756.                                                 String = "\n";
  1757.                                                 break;
  1758.                                         }
  1759.  
  1760.                                         break;
  1761.  
  1762.                                         /* Restart in/output. */
  1763.  
  1764.                                     case XON:
  1765.  
  1766.                                         if(Config->SerialConfig->xONxOFF)
  1767.                                             Clear_xOFF();
  1768.  
  1769.                                         if(!Config->SerialConfig->PassThrough)
  1770.                                             Length = 0;
  1771.  
  1772.                                         break;
  1773.  
  1774.                                         /* Stop in/output. */
  1775.  
  1776.                                     case XOF:
  1777.  
  1778.                                         if(Config->SerialConfig->xONxOFF)
  1779.                                             Set_xOFF();
  1780.  
  1781.                                         if(!Config->SerialConfig->PassThrough)
  1782.                                             Length = 0;
  1783.  
  1784.                                         break;
  1785.  
  1786.                                         /* Convert special Amiga characters into
  1787.                                          * alien IBM dialect.
  1788.                                          */
  1789.  
  1790.                                     default:
  1791.  
  1792.                                         if(Config->TerminalConfig->FontMode == FONT_IBM)
  1793.                                         {
  1794.                                             if(IBMConversion[Char])
  1795.                                                 Char = IBMConversion[Char];
  1796.                                             else
  1797.                                                 Length = 0;
  1798.                                         }
  1799.  
  1800.                                         break;
  1801.                                 }
  1802.  
  1803.                                 if(Length > 0)
  1804.                                     BufferSerWrite(BufferInfo,String,Length);
  1805.                             }
  1806.                         }
  1807.  
  1808.                         break;
  1809.                 }
  1810.             }
  1811.  
  1812.             if(MsgClass == IDCMP_MENUHELP)
  1813.                 GuideDisplay(CONTEXT_BUFFER_MENU);
  1814.  
  1815.             if(MsgClass == IDCMP_CLOSEWINDOW)
  1816.                 BufferInfo->BufferTerminated = TRUE;
  1817.  
  1818.             if(MsgClass == IDCMP_NEWSIZE)
  1819.             {
  1820.                 LONG Width,Height;
  1821.  
  1822.                 BufferInfo->Width                = BufferInfo->Window->Width        - (BufferInfo->Window->BorderLeft    + BufferInfo->Window->BorderRight);
  1823.                 BufferInfo->Height                = BufferInfo->Window->Height    - (BufferInfo->Window->BorderTop    + BufferInfo->Window->BorderBottom);
  1824.  
  1825.                 Width                            = BufferInfo->Width        - (BufferInfo->Width    % BufferInfo->LocalTextFontWidth);
  1826.                 Height                            = BufferInfo->Height    - (BufferInfo->Height    % BufferInfo->LocalTextFontHeight);
  1827.  
  1828.                 BufferInfo->NumBufferColumns    = Width / BufferInfo->LocalTextFontWidth;
  1829.                 BufferInfo->NumBufferLines        = Height / BufferInfo->LocalTextFontHeight;
  1830.  
  1831.                 if(Width < BufferInfo->Width)
  1832.                     BufferClear(BufferInfo,Width,0,BufferInfo->Width - 1,BufferInfo->Height - 1);
  1833.  
  1834.                 if(Height < BufferInfo->Height)
  1835.                     BufferClear(BufferInfo,0,Height,BufferInfo->Width - 1,BufferInfo->Height - 1);
  1836.  
  1837.                 BufferInfo->Width    = Width;
  1838.                 BufferInfo->Height    = Height;
  1839.  
  1840.                 BufferInfo->LastTopLine = -1;
  1841.                 BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine);
  1842.  
  1843.                 SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  1844.                     PGA_Total,        Lines,
  1845.                     PGA_Top,        BufferInfo->TopLine,
  1846.                     PGA_Visible,    BufferInfo->DisplayedLines,
  1847.                 TAG_DONE);
  1848.  
  1849.                 UpdatePercent = TRUE;
  1850.             }
  1851.  
  1852.             if(MsgClass == IDCMP_MENUPICK)
  1853.             {
  1854.                 struct MenuItem    *MenuItem;
  1855.                 struct DataMsg     Msg;
  1856.  
  1857.                 while(MsgCode != MENUNULL)
  1858.                 {
  1859.                     if(!(MenuItem = ItemAddress(BufferInfo->BufferMenuStrip,MsgCode)))
  1860.                         break;
  1861.                     else
  1862.                     {
  1863.                         switch((ULONG)GTMENUITEM_USERDATA(MenuItem))
  1864.                         {
  1865.                             case MEN_COPYCLIP:
  1866.  
  1867.                                 BufferMarkerTransfer(BufferInfo,MsgQualifier);
  1868.                                 break;
  1869.  
  1870.                             case MEN_CLEARCLIP:
  1871.  
  1872.                                 BufferMarkerStop(BufferInfo);
  1873.                                 break;
  1874.  
  1875.                             case MEN_SELECT_ALL_CLIP:
  1876.  
  1877.                                 BufferMarkerSelectAll(BufferInfo);
  1878.                                 break;
  1879.  
  1880.                             case MEN_PASTECLIP:
  1881.  
  1882.                                 InitMsgItem(&Msg,(DESTRUCTOR)BufferDestructor);
  1883.  
  1884.                                 Msg.Type    = DATAMSGTYPE_WRITECLIP;
  1885.                                 Msg.Size    = Config->ClipConfig->ClipboardUnit;
  1886.                                 Msg.Client    = FindTask(NULL);
  1887.                                 Msg.Mask    = 1UL << BufferInfo->BufferSignal;
  1888.  
  1889.                                 Forbid();
  1890.  
  1891.                                 ClrSignal(Msg.Mask);
  1892.  
  1893.                                 PutMsgItem(SpecialQueue,(struct MsgItem *)&Msg);
  1894.  
  1895.                                 Wait(Msg.Mask);
  1896.  
  1897.                                 Permit();
  1898.                                 break;
  1899.  
  1900.                             case MEN_SEARCH:
  1901.  
  1902.                                 if(Context)
  1903.                                     LT_ShowWindow(Context->SearchHandle,TRUE);
  1904.                                 else
  1905.                                     Context = CreateSearchContext(BufferInfo->Window,SearchBuffer,&HistoryHook,(struct List *)&TextBufferHistory,&BufferInfo->SearchForward,&BufferInfo->IgnoreCase,&BufferInfo->WholeWords);
  1906.  
  1907.                                 UpdatePercent = TRUE;
  1908.  
  1909.                                 break;
  1910.  
  1911.                             case MEN_REPEAT:
  1912.  
  1913.                                 if(Context)
  1914.                                     LT_ShowWindow(Context->SearchHandle,TRUE);
  1915.                                 else
  1916.                                 {
  1917.                                     if(SearchInfo)
  1918.                                         StartSearch(BufferInfo,SearchInfo,SearchBuffer);
  1919.                                     else
  1920.                                         Context = CreateSearchContext(BufferInfo->Window,SearchBuffer,&HistoryHook,(struct List *)&TextBufferHistory,&BufferInfo->SearchForward,&BufferInfo->IgnoreCase,&BufferInfo->WholeWords);
  1921.                                 }
  1922.  
  1923.                                 UpdatePercent = TRUE;
  1924.  
  1925.                                 break;
  1926.  
  1927.                             case MEN_GOTO:
  1928.  
  1929.                                 if(Window)
  1930.                                     BumpWindow(Window);
  1931.  
  1932.                                 break;
  1933.  
  1934.                             case MEN_QUITBUF:
  1935.  
  1936.                                 BufferInfo->BufferTerminated = TRUE;
  1937.                                 break;
  1938.  
  1939.                             case MEN_CLEARBUF_CONTENTS:
  1940.  
  1941.                                 if(Lines)
  1942.                                 {
  1943.                                     struct DataMsg *Msg;
  1944.  
  1945.                                     if(Msg = (struct DataMsg *)CreateMsgItem(sizeof(struct DataMsg)))
  1946.                                     {
  1947.                                         Msg->Type = DATAMSGTYPE_CLEARBUFFER;
  1948.  
  1949.                                         UpdatePercent = TRUE;
  1950.  
  1951.                                         LT_LockWindow(BufferInfo->Window);
  1952.  
  1953.                                         if(MsgQualifier & SHIFT_KEY)
  1954.                                         {
  1955.                                             PutMsgItem(SpecialQueue,(struct MsgItem *)Msg);
  1956.                                             Msg = NULL;
  1957.                                         }
  1958.                                         else
  1959.                                         {
  1960.                                             if(Config->MiscConfig->ProtectiveMode)
  1961.                                             {
  1962.                                                 if(!ShowRequest(BufferInfo->Window,LocaleString(MSG_TERMBUFFER_BUFFER_STILL_HOLDS_LINES_TXT),LocaleString(MSG_GLOBAL_YES_NO_TXT),Lines))
  1963.                                                 {
  1964.                                                     LT_UnlockWindow(BufferInfo->Window);
  1965.  
  1966.                                                     break;
  1967.                                                 }
  1968.                                             }
  1969.  
  1970.                                             PutMsgItem(SpecialQueue,(struct MsgItem *)Msg);
  1971.                                             Msg = NULL;
  1972.                                         }
  1973.  
  1974.                                         DeleteMsgItem((struct MsgItem *)Msg);
  1975.                                     }
  1976.  
  1977.                                     LT_UnlockWindow(BufferInfo->Window);
  1978.  
  1979.                                     FlushMsg(BufferInfo->Window);
  1980.                                 }
  1981.  
  1982.                                 break;
  1983.                         }
  1984.  
  1985.                         MsgCode = MenuItem->NextSelect;
  1986.                     }
  1987.                 }
  1988.             }
  1989.  
  1990.             if(DisplayDirty)
  1991.             {
  1992.                 DisplayDirty = FALSE;
  1993.  
  1994.                 if(NewTopLine < 0)
  1995.                     NewTopLine = 0;
  1996.                 else if (NewTopLine + BufferInfo->NumBufferLines >= Lines)
  1997.                     NewTopLine = Lines - BufferInfo->NumBufferLines;
  1998.  
  1999.                 if(NewTopLine >= 0 && NewTopLine != BufferInfo->TopLine)
  2000.                 {
  2001.                     BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,NewTopLine);
  2002.  
  2003.                     BufferInfo->TopLine = NewTopLine;
  2004.  
  2005.                     if(DoNotUpdateScroller)
  2006.                         DoNotUpdateScroller = FALSE;
  2007.                     else
  2008.                     {
  2009.                         SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  2010.                             PGA_Top,BufferInfo->TopLine,
  2011.                         TAG_DONE);
  2012.                     }
  2013.  
  2014.                     UpdatePercent = TRUE;
  2015.                 }
  2016.             }
  2017.         }
  2018.     }
  2019.  
  2020.     if(Context)
  2021.         DeleteSearchContext(Context);
  2022.  
  2023.     if(SearchInfo)
  2024.         DeleteSearchInfo(SearchInfo);
  2025.  
  2026.         /* Put the stuff back */
  2027.  
  2028.     *BufferInfo->pTopLine        = BufferInfo->TopLine;
  2029.     *BufferInfo->pSearchForward    = BufferInfo->SearchForward;
  2030.     *BufferInfo->pIgnoreCase    = BufferInfo->IgnoreCase;
  2031.     *BufferInfo->pWholeWords    = BufferInfo->WholeWords;
  2032.  
  2033.     return(RingBack);
  2034. }
  2035.  
  2036. STATIC VOID
  2037. DeleteBufferInfo(TextBufferInfo *BufferInfo)
  2038. {
  2039.     if(BufferInfo)
  2040.     {
  2041.         FreeVecPooled(BufferInfo->BufferLineOffsets);
  2042.  
  2043.         if(BufferInfo->Window)
  2044.         {
  2045.             ClearMenuStrip(BufferInfo->Window);
  2046.  
  2047.             LT_DeleteWindowLock(BufferInfo->Window);
  2048.  
  2049.             CloseWindow(BufferInfo->Window);
  2050.         }
  2051.  
  2052.         FreeVecPooled(BufferInfo->Marker);
  2053.  
  2054.         DeleteScroller(BufferInfo);
  2055.  
  2056.         DisposeObject(BufferInfo->BufferAmigaGlyph);
  2057.         DisposeObject(BufferInfo->BufferCheckGlyph);
  2058.  
  2059.         FreeScreenDrawInfo(BufferInfo->Screen,BufferInfo->BufferDrawInfo);
  2060.  
  2061.         LT_DisposeMenu(BufferInfo->BufferMenuStrip);
  2062.  
  2063.         if(BufferInfo->Screen && !BufferInfo->Parent)
  2064.             CloseScreen(BufferInfo->Screen);
  2065.  
  2066.         if(BufferInfo->LocalFont)
  2067.             CloseFont(BufferInfo->LocalFont);
  2068.  
  2069.         if(BufferInfo->BufferSignal != -1)
  2070.             FreeSignal(BufferInfo->BufferSignal);
  2071.  
  2072.         DeleteMsgQueue(BufferInfo->Queue);
  2073.  
  2074.         FreeVecPooled(BufferInfo);
  2075.     }
  2076. }
  2077.  
  2078. STATIC TextBufferInfo *
  2079. CreateBufferInfo(struct Screen *Parent,BOOL *pSearchForward,BOOL *pIgnoreCase,BOOL *pWholeWords,LONG *pTopLine)
  2080. {
  2081.     TextBufferInfo *BufferInfo;
  2082.  
  2083.     if(BufferInfo = (TextBufferInfo *)AllocVecPooled(sizeof(TextBufferInfo),MEMF_ANY | MEMF_CLEAR))
  2084.     {
  2085.         LONG                         Width,Height;
  2086.         ULONG                         DisplayMode;
  2087.  
  2088.         struct Rectangle             DisplayClip;
  2089.         struct DimensionInfo         DimensionInfo;
  2090.  
  2091.         BufferInfo->pSearchForward    = pSearchForward;
  2092.         BufferInfo->pIgnoreCase        = pIgnoreCase;
  2093.         BufferInfo->pWholeWords        = pWholeWords;
  2094.         BufferInfo->pTopLine        = pTopLine;
  2095.  
  2096.         BufferInfo->TopLine            = *pTopLine;
  2097.         BufferInfo->LastTopLine        = -1;
  2098.  
  2099.         BufferInfo->SearchForward    = *pSearchForward;
  2100.         BufferInfo->IgnoreCase        = *pIgnoreCase;
  2101.         BufferInfo->WholeWords        = *pWholeWords;
  2102.  
  2103.         BufferInfo->OldColumn        = -1;
  2104.         BufferInfo->OldLine            = -1;
  2105.         BufferInfo->OldLength        = -1;
  2106.  
  2107.         BufferInfo->Buddy            = (struct Process *)FindTask(NULL);
  2108.  
  2109.         CopyMem(&TextFont,&BufferInfo->LocalTextFont,sizeof(struct TTextAttr));
  2110.  
  2111.         BufferInfo->LocalTextFont.tta_Name    = BufferInfo->LocalTextFontName;
  2112.         BufferInfo->LocalTextFont.tta_YSize    = Config->TerminalConfig->TextFontHeight;
  2113.  
  2114.         strcpy(BufferInfo->LocalTextFontName,Config->TerminalConfig->TextFontName);
  2115.  
  2116.         CopyMem(&UserFont,&BufferInfo->LocalUserFont,sizeof(struct TTextAttr));
  2117.  
  2118.         BufferInfo->LocalUserFont.tta_Name = BufferInfo->LocalUserFontName;
  2119.  
  2120.         strcpy(BufferInfo->LocalUserFontName,UserFont.tta_Name);
  2121.  
  2122.         if(!Config->CaptureConfig->ConvertChars && Config->TerminalConfig->FontMode != FONT_STANDARD)
  2123.         {
  2124.             strcpy(BufferInfo->LocalTextFontName,Config->TerminalConfig->IBMFontName);
  2125.  
  2126.             BufferInfo->LocalTextFont.tta_YSize = Config->TerminalConfig->IBMFontHeight;
  2127.  
  2128.             BufferInfo->NeedClipConversion = TRUE;
  2129.         }
  2130.         else
  2131.             BufferInfo->NeedClipConversion = FALSE;
  2132.  
  2133.         if(BufferInfo->LocalFont = SmartOpenDiskFont((struct TextAttr *)&BufferInfo->LocalTextFont))
  2134.         {
  2135.             BufferInfo->LocalTextFontWidth    = BufferInfo->LocalFont->tf_XSize;
  2136.             BufferInfo->LocalTextFontHeight    = BufferInfo->LocalFont->tf_YSize;
  2137.  
  2138.             DisplayMode = Config->CaptureConfig->BufferScreenMode;
  2139.  
  2140.             if(DisplayMode == INVALID_ID && Window)
  2141.                 DisplayMode = GetVPModeID(&Window->WScreen->ViewPort);
  2142.  
  2143.             if(ModeNotAvailable(DisplayMode))
  2144.                 DisplayMode = Config->ScreenConfig->DisplayMode;
  2145.  
  2146.             if(ModeNotAvailable(DisplayMode))
  2147.             {
  2148.                 struct Screen *PubScreen = LockPubScreen(NULL);
  2149.  
  2150.                 if(PubScreen)
  2151.                 {
  2152.                     DisplayMode = GetVPModeID(&PubScreen->ViewPort);
  2153.  
  2154.                     UnlockPubScreen(NULL,PubScreen);
  2155.                 }
  2156.                 else
  2157.                     DisplayMode = DEFAULT_MONITOR_ID | HIRES_KEY;
  2158.             }
  2159.  
  2160.                 /* Set up the actual width of the screen we want. */
  2161.  
  2162.             Width = Config->CaptureConfig->BufferWidth * BufferInfo->LocalTextFontWidth + BufferInfo->ArrowWidth + 1;
  2163.  
  2164.             if((BufferInfo->BufferSignal = AllocSignal(-1)) != -1)
  2165.             {
  2166.                     /* Get the mode dimension info. */
  2167.  
  2168.                 if(GetDisplayInfoData(NULL,(APTR)&DimensionInfo,sizeof(struct DimensionInfo),DTAG_DIMS,DisplayMode))
  2169.                 {
  2170.                         /* Determine maximum text overscan width. */
  2171.  
  2172.                     LONG TextWidth = DimensionInfo.TxtOScan.MaxX - DimensionInfo.TxtOScan.MinX + 1;
  2173.  
  2174.                         /* Too small? */
  2175.  
  2176.                     if(Width < DimensionInfo.MinRasterWidth)
  2177.                         Width = DimensionInfo.MinRasterWidth;
  2178.  
  2179.                         /* Far too large? */
  2180.  
  2181.                     if(Width > DimensionInfo.MaxRasterWidth)
  2182.                         Width = DimensionInfo.MaxRasterWidth;
  2183.  
  2184.                         /* A bit too large? */
  2185.  
  2186.                     if(Width > TextWidth)
  2187.                         Width = TextWidth;
  2188.  
  2189.                         /* Inquire the text overscan dimensions. */
  2190.  
  2191.                     if(QueryOverscan(DisplayMode,&DisplayClip,OSCAN_TEXT))
  2192.                     {
  2193.                             /* Centre the buffer screen. */
  2194.  
  2195.                         if(DisplayClip.MaxX - DisplayClip.MinX + 1 > Width)
  2196.                         {
  2197.                             LONG Differ = (DisplayClip.MaxX - DisplayClip.MinX + 1 - Width) / 2;
  2198.  
  2199.                             switch(Config->CaptureConfig->BufferScreenPosition)
  2200.                             {
  2201.                                 case SCREEN_LEFT:
  2202.  
  2203.                                     DisplayClip.MaxX = DisplayClip.MinX + Width;
  2204.                                     break;
  2205.  
  2206.                                 case SCREEN_RIGHT:
  2207.  
  2208.                                     DisplayClip.MinX = DisplayClip.MaxX - Width;
  2209.                                     break;
  2210.  
  2211.                                 case SCREEN_CENTRE:
  2212.  
  2213.                                     DisplayClip.MinX += Differ;
  2214.                                     DisplayClip.MaxX -= Differ;
  2215.  
  2216.                                     break;
  2217.                             }
  2218.                         }
  2219.  
  2220.                             /* Open a two bitplane clone of the main screen. */
  2221.  
  2222.                         if(Parent)
  2223.                         {
  2224.                             BufferInfo->Parent = Parent;
  2225.                             BufferInfo->Screen = Parent;
  2226.  
  2227.                             if(BufferInfo->Queue = CreateMsgQueue(NULL,0))
  2228.                                 BufferInfo->QueueMask = BufferInfo->Queue->SigMask;
  2229.                             else
  2230.                                 BufferInfo->Screen = NULL;
  2231.                         }
  2232.                         else
  2233.                         {
  2234.                             UWORD Pens = (UWORD)~0;
  2235.  
  2236.                             BufferInfo->Screen = OpenScreenTags(NULL,
  2237.                                 SA_Title,        LocaleString(MSG_TERMBUFFER_TERM_BUFFER_TXT),
  2238.                                 SA_Depth,        2,
  2239.                                 SA_Left,        DisplayClip.MinX,
  2240.                                 SA_DClip,        &DisplayClip,
  2241.                                 SA_DisplayID,    DisplayMode,
  2242.                                 SA_Font,        &BufferInfo->LocalUserFont,
  2243.                                 SA_AutoScroll,    TRUE,
  2244.                                 SA_Pens,        &Pens,
  2245.                             TAG_END);
  2246.                         }
  2247.  
  2248.                         if(BufferInfo->Screen)
  2249.                         {
  2250.                             if(BufferInfo->BufferDrawInfo = GetScreenDrawInfo(BufferInfo->Screen))
  2251.                             {
  2252.                                 ULONG *MenuTags;
  2253.  
  2254.                                 CreateMenuGlyphs(BufferInfo->Screen,BufferInfo->BufferDrawInfo,&BufferInfo->BufferAmigaGlyph,&BufferInfo->BufferCheckGlyph);
  2255.  
  2256.                                 if(Parent)
  2257.                                 {
  2258.                                     STATIC ULONG ReviewTags[] =
  2259.                                     {
  2260.                                         LAMN_TitleID,            MSG_TERMREVIEW_PROJECT_MEN,
  2261.                                             LAMN_ItemID,        MSG_TERMREVIEW_SEARCH_MEN,
  2262.                                                 LAMN_UserData,    MEN_SEARCH,
  2263.                                             LAMN_ItemID,        MSG_TERMREVIEW_REPEAT_SEARCH_MEN,
  2264.                                                 LAMN_UserData,    MEN_REPEAT,
  2265.  
  2266.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2267.  
  2268.                                             LAMN_ItemID,        MSG_TERMREVIEW_CLEAR_BUFFER_MEN,
  2269.                                                 LAMN_UserData,    MEN_CLEARBUF_CONTENTS,
  2270.  
  2271.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2272.  
  2273.                                             LAMN_ItemID,        MSG_TERMREVIEW_CLOSE_BUFFER_MEN,
  2274.                                                 LAMN_UserData,    MEN_QUITBUF,
  2275.  
  2276.                                             LAMN_ItemID,        MSG_CLOSE_WINDOW_TXT,
  2277.                                                 LAMN_KeyText,    (ULONG)":",
  2278.                                                 LAMN_UserData,    MEN_QUITBUF,
  2279.  
  2280.                                         LAMN_TitleID,            MSG_TERMBUFFER_EDIT_MEN,
  2281.                                             LAMN_ItemID,        MSG_TERMDATA_COPY_MEN,
  2282.                                                 LAMN_UserData,    MEN_COPYCLIP,
  2283.                                             LAMN_ItemID,        MSG_TERMDATA_PASTE_MEN,
  2284.                                                 LAMN_UserData,    MEN_PASTECLIP,
  2285.                                             LAMN_ItemID,        MSG_TERMDATA_CLEAR_MEN,
  2286.                                                 LAMN_UserData,    MEN_CLEARCLIP,
  2287.  
  2288.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2289.  
  2290.                                             LAMN_ItemID,        MSG_SELECT_ALL_MEN,
  2291.                                                 LAMN_UserData,    MEN_SELECT_ALL_CLIP,
  2292.  
  2293.                                         TAG_DONE
  2294.                                     };
  2295.  
  2296.                                     MenuTags = ReviewTags;
  2297.                                 }
  2298.                                 else
  2299.                                 {
  2300.                                     STATIC ULONG BufferTags[] =
  2301.                                     {
  2302.                                         LAMN_TitleID,            MSG_TERMBUFFER_PROJECT_MEN,
  2303.                                             LAMN_ItemID,        MSG_TERMBUFFER_SEARCH_MEN,
  2304.                                                 LAMN_UserData,    MEN_SEARCH,
  2305.                                             LAMN_ItemID,        MSG_TERMBUFFER_REPEAT_SEARCH_MEN,
  2306.                                                 LAMN_UserData,    MEN_REPEAT,
  2307.  
  2308.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2309.  
  2310.                                             LAMN_ItemID,        MSG_TERMBUFFER_GO_TO_MAIN_SCREEN_MEN,
  2311.                                                 LAMN_UserData,    MEN_GOTO,
  2312.  
  2313.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2314.  
  2315.                                             LAMN_ItemID,        MSG_TERMBUFFER_CLEAR_BUFFER_MEN,
  2316.                                                 LAMN_UserData,    MEN_CLEARBUF_CONTENTS,
  2317.  
  2318.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2319.  
  2320.                                             LAMN_ItemID,        MSG_TERMBUFFER_CLOSE_BUFFER_MEN,
  2321.                                                 LAMN_UserData,    MEN_QUITBUF,
  2322.  
  2323.                                         LAMN_TitleID,            MSG_TERMBUFFER_EDIT_MEN,
  2324.                                             LAMN_ItemID,        MSG_TERMDATA_COPY_MEN,
  2325.                                                 LAMN_UserData,    MEN_COPYCLIP,
  2326.                                             LAMN_ItemID,        MSG_TERMDATA_PASTE_MEN,
  2327.                                                 LAMN_UserData,    MEN_PASTECLIP,
  2328.                                             LAMN_ItemID,        MSG_TERMDATA_CLEAR_MEN,
  2329.                                                 LAMN_UserData,    MEN_CLEARCLIP,
  2330.  
  2331.                                             LAMN_ItemText,        (ULONG)NM_BARLABEL,
  2332.  
  2333.                                             LAMN_ItemID,        MSG_SELECT_ALL_MEN,
  2334.                                                 LAMN_UserData,    MEN_SELECT_ALL_CLIP,
  2335.  
  2336.                                         TAG_DONE
  2337.                                     };
  2338.  
  2339.                                     MenuTags = BufferTags;
  2340.                                 }
  2341.  
  2342.                                 if(BufferInfo->BufferMenuStrip = LT_NewMenuTags(
  2343.                                     LAHN_LocaleHook,        &LocaleHook,
  2344.                                     LAMN_Screen,            BufferInfo->Screen,
  2345.                                     LAMN_TextAttr,            &BufferInfo->LocalUserFont,
  2346.                                     LAMN_AmigaGlyph,        BufferInfo->BufferAmigaGlyph,
  2347.                                     LAMN_CheckmarkGlyph,    BufferInfo->BufferCheckGlyph,
  2348.                                 TAG_MORE,MenuTags))
  2349.                                 {
  2350.                                     UWORD *Pens = BufferInfo->BufferDrawInfo->dri_Pens;
  2351.  
  2352.                                     BufferInfo->TextFrontPen    = Pens[TEXTPEN];
  2353.                                     BufferInfo->TextBackPen        = Pens[BACKGROUNDPEN];
  2354.  
  2355.                                     BufferInfo->MaxPen = MAX(BufferInfo->TextFrontPen,BufferInfo->TextBackPen);
  2356.  
  2357.                                     Height = BufferInfo->Screen->Height - (BufferInfo->Screen->BarHeight + 2);
  2358.  
  2359.                                     if(CreateScroller(BufferInfo,Height))
  2360.                                     {
  2361.                                         LONG Left,Top;
  2362.  
  2363.                                         if(Parent)
  2364.                                         {
  2365.                                             Left    = Window->LeftEdge;
  2366.                                             Top        = Window->TopEdge + Window->BorderTop;
  2367.                                             Width    = Window->Width;
  2368.                                             Height    = Window->Height - Window->BorderTop;
  2369.  
  2370.                                             GetWindowInfo(WINDOW_REVIEW,&Left,&Top,&Width,&Height,Width,Height);
  2371.                                         }
  2372.                                         else
  2373.                                         {
  2374.                                             Left    = 0;
  2375.                                             Top        = BufferInfo->Screen->BarHeight + 2;
  2376.                                             Width    = BufferInfo->Screen->Width;
  2377.                                             Height    = BufferInfo->Screen->Height - Top;
  2378.                                         }
  2379.  
  2380.                                         if(BufferInfo->Window = OpenWindowTags(NULL,
  2381.                                             WA_Left,            Left,
  2382.                                             WA_Top,                Top,
  2383.                                             WA_Width,            Width,
  2384.                                             WA_Height,            Height,
  2385.                                             WA_Backdrop,        Parent == NULL,
  2386.                                             WA_Borderless,        Parent == NULL,
  2387.                                             WA_DepthGadget,        Parent != NULL,
  2388.                                             WA_DragBar,            Parent != NULL,
  2389.                                             WA_CloseGadget,        Parent != NULL,
  2390.                                             WA_SizeGadget,        Parent != NULL,
  2391.                                             WA_NoCareRefresh,    TRUE,
  2392.                                             WA_NewLookMenus,    TRUE,
  2393.                                             WA_IDCMP,            IDCMP_IDCMPUPDATE | IDCMP_RAWKEY | IDCMP_INACTIVEWINDOW | IDCMP_ACTIVEWINDOW | IDCMP_MOUSEBUTTONS | IDCMP_MENUPICK | IDCMP_GADGETUP | IDCMP_GADGETDOWN | IDCMP_MOUSEMOVE | IDCMP_MENUHELP | IDCMP_SIZEVERIFY | IDCMP_NEWSIZE | IDCMP_CLOSEWINDOW,
  2394.                                             WA_MenuHelp,        TRUE,
  2395.                                             WA_Gadgets,            BufferInfo->Scroller,
  2396.                                             WA_CustomScreen,    BufferInfo->Screen,
  2397.                                             WA_Activate,        TRUE,
  2398.                                             BackfillTag,        Parent ? &BackfillHook : NULL,
  2399.  
  2400.                                             BufferInfo->BufferAmigaGlyph ? WA_AmigaKey  : TAG_IGNORE, BufferInfo->BufferAmigaGlyph,
  2401.                                             BufferInfo->BufferCheckGlyph ? WA_Checkmark : TAG_IGNORE, BufferInfo->BufferCheckGlyph,
  2402.                                         TAG_DONE))
  2403.                                         {
  2404.                                             LONG MaxLines,MaxColumns;
  2405.  
  2406.                                             SetMenuStrip(BufferInfo->Window,BufferInfo->BufferMenuStrip);
  2407.  
  2408.                                             OffMenu(BufferInfo->Window,FULLMENUNUM(1,0,NOSUB));
  2409.                                             OffMenu(BufferInfo->Window,FULLMENUNUM(1,2,NOSUB));
  2410.  
  2411.                                             if(!Parent)
  2412.                                                 OffMenu(BufferInfo->Window,FULLMENUNUM(1,1,NOSUB));
  2413.  
  2414.                                             BufferInfo->Left    = BufferInfo->Window->BorderLeft;
  2415.                                             BufferInfo->Top        = BufferInfo->Window->BorderTop;
  2416.                                             BufferInfo->Width    = BufferInfo->Window->Width        - (BufferInfo->Window->BorderLeft    + BufferInfo->Window->BorderRight);
  2417.                                             BufferInfo->Height    = BufferInfo->Window->Height    - (BufferInfo->Window->BorderTop    + BufferInfo->Window->BorderBottom);
  2418.  
  2419.                                             if(!Parent)
  2420.                                                 BufferInfo->Width -= BufferInfo->ArrowWidth;
  2421.  
  2422.                                             BufferInfo->Width    -= BufferInfo->Width    % BufferInfo->LocalTextFontWidth;
  2423.                                             BufferInfo->Height    -= BufferInfo->Height    % BufferInfo->LocalTextFontHeight;
  2424.  
  2425.                                             if(Parent)
  2426.                                                 WindowLimits(BufferInfo->Window,BufferInfo->Window->BorderLeft + 20 * BufferInfo->LocalTextFontWidth + BufferInfo->Window->BorderRight,BufferInfo->Window->BorderTop + BufferInfo->LocalTextFontHeight + BufferInfo->Window->BorderBottom,BufferInfo->Screen->Width,BufferInfo->Screen->Height);
  2427.  
  2428.                                             MaxColumns    = BufferInfo->Window->WScreen->Width / BufferInfo->LocalTextFontWidth + 1;
  2429.                                             MaxLines    = BufferInfo->Window->WScreen->Height / BufferInfo->LocalTextFontHeight + 1;
  2430.  
  2431.                                             if(BufferInfo->BufferLineOffsets = (UWORD *)AllocVecPooled(sizeof(UWORD) * (MaxLines + MaxColumns),MEMF_ANY | MEMF_CLEAR))
  2432.                                             {
  2433.                                                 LONG Index;
  2434.                                                 LONG i;
  2435.  
  2436.                                                 BufferInfo->BufferColumnOffsets    = &BufferInfo->BufferLineOffsets[MaxLines];
  2437.  
  2438.                                                 for(i = Index = 0 ; i < MaxLines ; i++)
  2439.                                                 {
  2440.                                                     BufferInfo->BufferLineOffsets[i] = Index;
  2441.  
  2442.                                                     Index += BufferInfo->LocalTextFontHeight;
  2443.                                                 }
  2444.  
  2445.                                                 for(i = Index = 0 ; i < MaxColumns ; i++)
  2446.                                                 {
  2447.                                                     BufferInfo->BufferColumnOffsets[i] = Index;
  2448.  
  2449.                                                     Index += BufferInfo->LocalTextFontWidth;
  2450.                                                 }
  2451.  
  2452.                                                     /* Determine maximum dimensions of
  2453.                                                      * the buffer screen (in rows and
  2454.                                                      * columns).
  2455.                                                      */
  2456.  
  2457.                                                 if(Parent)
  2458.                                                     BufferInfo->NumBufferColumns = BufferInfo->Width / BufferInfo->LocalTextFontWidth;
  2459.                                                 else
  2460.                                                     BufferInfo->NumBufferColumns = (BufferInfo->Window->Width - (BufferInfo->ArrowWidth + 1)) / BufferInfo->LocalTextFontWidth;
  2461.  
  2462.                                                 BufferInfo->NumBufferLines = BufferInfo->Height / BufferInfo->LocalTextFontHeight;
  2463.  
  2464.                                                 if(BufferInfo->TopLine == -1 || !Config->CaptureConfig->RememberBufferScreen)
  2465.                                                 {
  2466.                                                     switch(Config->CaptureConfig->OpenBufferScreen)
  2467.                                                     {
  2468.                                                         case BUFFER_TOP:
  2469.  
  2470.                                                             BufferInfo->TopLine = 0;
  2471.                                                             break;
  2472.  
  2473.                                                         case BUFFER_END:
  2474.  
  2475.                                                             if((BufferInfo->TopLine = Lines - BufferInfo->NumBufferLines) < 0)
  2476.                                                                 BufferInfo->TopLine = 0;
  2477.  
  2478.                                                             break;
  2479.  
  2480.                                                         default:
  2481.  
  2482.                                                             BufferInfo->TopLine = 0;
  2483.                                                             break;
  2484.                                                     }
  2485.                                                 }
  2486.  
  2487.                                                 if(BufferInfo->TopLine > Lines - BufferInfo->NumBufferLines || BufferInfo->TopLine < 0)
  2488.                                                     BufferInfo->TopLine = 0;
  2489.  
  2490.                                                 BufferInfo->RPort = BufferInfo->Window->RPort;
  2491.  
  2492.                                                 SetFont(BufferInfo->RPort,BufferInfo->LocalFont);
  2493.  
  2494.                                                     /* Bring the screen to the front. */
  2495.  
  2496.                                                 BumpWindow(BufferInfo->Window);
  2497.  
  2498.                                                     /* Set the drawing pens for the window. */
  2499.  
  2500.                                                 if(Kick30)
  2501.                                                 {
  2502.                                                         /* Don't slow down display updates when using an
  2503.                                                          * interleaved screen.
  2504.                                                          */
  2505.  
  2506.                                                     if(!(GetBitMapAttr(BufferInfo->RPort->BitMap,BMA_FLAGS) & BMF_INTERLEAVED))
  2507.                                                         SetMaxPen(BufferInfo->RPort,BufferInfo->MaxPen);
  2508.                                                 }
  2509.  
  2510.                                                 SetPens(BufferInfo->RPort,BufferInfo->TextFrontPen,BufferInfo->TextBackPen,JAM2);
  2511.  
  2512.                                                     /* Initial creation of the buffer display. */
  2513.  
  2514.                                                 BufferInfo->DisplayedLines = RedrawScreen(BufferInfo,BufferInfo->TopLine);
  2515.  
  2516.                                                 SetGadgetAttrs((struct Gadget *)BufferInfo->Scroller,BufferInfo->Window,NULL,
  2517.                                                     PGA_Top,        BufferInfo->TopLine,
  2518.                                                     PGA_Total,        Lines,
  2519.                                                     PGA_Visible,    BufferInfo->NumBufferLines,
  2520.                                                 TAG_DONE);
  2521.  
  2522.                                                 if(!Parent)
  2523.                                                 {
  2524.                                                     struct RastPort *RPort = BufferInfo->Screen->BarLayer->rp;
  2525.  
  2526.                                                     BufferInfo->TitleOffset = TextLength(RPort,LocaleString(MSG_TERMBUFFER_TERM_BUFFER_TXT),strlen(LocaleString(MSG_TERMBUFFER_TERM_BUFFER_TXT))) + TextLength(RPort,"  ",1) + 4;
  2527.                                                 }
  2528.  
  2529.                                                 return(BufferInfo);
  2530.                                             }
  2531.                                         }
  2532.                                     }
  2533.                                 }
  2534.                             }
  2535.                         }
  2536.                     }
  2537.                 }
  2538.             }
  2539.         }
  2540.     }
  2541.  
  2542.     DeleteBufferInfo(BufferInfo);
  2543.  
  2544.     return(NULL);
  2545. }
  2546.  
  2547.     /* BufferServer():
  2548.      *
  2549.      *    Asynchronous task to display the data stored in the
  2550.      *    scrollback display buffer.
  2551.      */
  2552.  
  2553. STATIC VOID SAVE_DS
  2554. BufferServer()
  2555. {
  2556.     STATIC BOOL        SearchForward    = TRUE,
  2557.                     IgnoreCase        = TRUE,
  2558.                     WholeWords        = FALSE;
  2559.     STATIC LONG        TopLine            = -1;
  2560.  
  2561.     TextBufferInfo    *BufferInfo;
  2562.     struct Task        *Father,*Me;
  2563.     BOOL             RingBack = TRUE;
  2564.  
  2565.         /* Wake this guy up */
  2566.  
  2567.     Me = FindTask(NULL);
  2568.  
  2569.     Father = (struct Task *)Me->tc_UserData;
  2570.  
  2571.         /* Snap our fingers... */
  2572.  
  2573.     if(BufferInfo = CreateBufferInfo(NULL,&SearchForward,&IgnoreCase,&WholeWords,&TopLine))
  2574.     {
  2575.             /* Open wide, here I come! */
  2576.  
  2577.         ObtainSemaphore(&BufferTaskSemaphore);
  2578.  
  2579.         BufferInfoData = BufferInfo;
  2580.  
  2581.         ReleaseSemaphore(&BufferTaskSemaphore);
  2582.  
  2583.             /* Up and running */
  2584.  
  2585.         Signal(Father,SIG_HANDSHAKE);
  2586.  
  2587.         Father = NULL;
  2588.  
  2589.             /* Swish the tinsel */
  2590.  
  2591.         RingBack = HandleBuffer(&BufferTaskSemaphore,&BufferInfoData);
  2592.  
  2593.             /* Activate the main window again */
  2594.  
  2595.         if(Window)
  2596.             BumpWindow(Window);
  2597.  
  2598.             /* The wrecking crew */
  2599.  
  2600.         DeleteBufferInfo(BufferInfo);
  2601.     }
  2602.  
  2603.         /* Shutdown in an orderly fashion */
  2604.  
  2605.     Forbid();
  2606.  
  2607.     if(RingBack)
  2608.     {
  2609.         if(Father)
  2610.             Signal(Father,SIG_HANDSHAKE);
  2611.         else
  2612.             Signal((struct Task *)ThisProcess,SIG_HANDSHAKE);
  2613.     }
  2614. }
  2615.  
  2616.     /* LaunchBuffer():
  2617.      *
  2618.      *    Launch the buffer process.
  2619.      */
  2620.  
  2621. BOOL
  2622. LaunchBuffer()
  2623. {
  2624.     ObtainSemaphore(&BufferTaskSemaphore);
  2625.  
  2626.         /* Is the buffer process already running? */
  2627.  
  2628.     if(BufferInfoData)
  2629.     {
  2630.             /* Tell it to bring its screen to the front. */
  2631.  
  2632.         Signal((struct Task *)BufferInfoData->Buddy,SIG_TOFRONT);
  2633.  
  2634.         ReleaseSemaphore(&BufferTaskSemaphore);
  2635.  
  2636.             /* Return success. */
  2637.  
  2638.         return(TRUE);
  2639.     }
  2640.     else
  2641.     {
  2642.         struct Task    *Child;
  2643.         BOOL         Result = FALSE;
  2644.  
  2645.         ReleaseSemaphore(&BufferTaskSemaphore);
  2646.  
  2647.         Forbid();
  2648.  
  2649.             /* Launch the buffer process. */
  2650.  
  2651.         if(Child = (struct Task *)CreateNewProcTags(
  2652.             NP_Entry,        BufferServer,
  2653.             NP_Name,        "term Buffer Process",
  2654.             NP_WindowPtr,    -1,
  2655.             NP_StackSize,    8192,
  2656.         TAG_END))
  2657.         {
  2658.             Child->tc_UserData = FindTask(NULL);
  2659.  
  2660.             WaitForHandshake();
  2661.  
  2662.             ObtainSemaphore(&BufferTaskSemaphore);
  2663.  
  2664.             if(BufferInfoData)
  2665.                 Result = TRUE;
  2666.  
  2667.             ReleaseSemaphore(&BufferTaskSemaphore);
  2668.         }
  2669.  
  2670.         Permit();
  2671.  
  2672.             /* Return the result. */
  2673.  
  2674.         return(Result);
  2675.     }
  2676. }
  2677.  
  2678.     /* TerminateBuffer():
  2679.      *
  2680.      *    Terminate the buffer process.
  2681.      */
  2682.  
  2683. VOID
  2684. TerminateBuffer()
  2685. {
  2686.     ObtainSemaphore(&BufferTaskSemaphore);
  2687.  
  2688.     if(BufferInfoData)
  2689.     {
  2690.         Forbid();
  2691.  
  2692.         ReleaseSemaphore(&BufferTaskSemaphore);
  2693.         ShakeHands((struct Task *)BufferInfoData->Buddy,SIG_KILL);
  2694.  
  2695.         Permit();
  2696.     }
  2697.     else
  2698.         ReleaseSemaphore(&BufferTaskSemaphore);
  2699. }
  2700.  
  2701.     /* MoveBuffer(TextBufferInfo *BufferInfo,BYTE Mode):
  2702.      *
  2703.      *    Move the currently displayed buffer area somewhere.
  2704.      */
  2705.  
  2706. VOID
  2707. MoveBuffer(struct SignalSemaphore *Access,struct TextBufferInfo **Data,LONG Mode)
  2708. {
  2709.     struct TextBufferInfo *BufferInfo;
  2710.  
  2711.     ObtainSemaphore(Access);
  2712.  
  2713.     BufferInfo = (TextBufferInfo *)*Data;
  2714.  
  2715.     if(BufferInfo && BufferInfo->Queue)
  2716.     {
  2717.         LONG SigBit;
  2718.  
  2719.         if((SigBit = AllocSignal(-1)) != -1)
  2720.         {
  2721.             struct DataMsg Msg;
  2722.  
  2723.             InitMsgItem(&Msg,(DESTRUCTOR)BufferDestructor);
  2724.  
  2725.             Msg.Type    = DATAMSGTYPE_MOVEREVIEW;
  2726.             Msg.Size    = Mode;
  2727.             Msg.Client    = FindTask(NULL);
  2728.             Msg.Mask    = 1L << SigBit;
  2729.  
  2730.             ClrSignal(Msg.Mask);
  2731.  
  2732.             PutMsgItem(BufferInfo->Queue,(struct MsgItem *)&Msg);
  2733.  
  2734.             ReleaseSemaphore(Access);
  2735.  
  2736.             Wait(Msg.Mask);
  2737.  
  2738.             FreeSignal(SigBit);
  2739.  
  2740.             return;
  2741.         }
  2742.     }
  2743.  
  2744.     ReleaseSemaphore(Access);
  2745. }
  2746.  
  2747.     /* NotifyBuffer(struct SignalSemaphore *Access,struct TextBufferInfo **Data,ULONG Signals):
  2748.      *
  2749.      *    Send a signal to the buffer process.
  2750.      */
  2751.  
  2752. VOID
  2753. NotifyBuffer(struct SignalSemaphore *Access,struct TextBufferInfo **Data,ULONG Signals)
  2754. {
  2755.     struct TextBufferInfo *BufferInfo;
  2756.  
  2757.     ObtainSemaphore(Access);
  2758.  
  2759.     BufferInfo = (TextBufferInfo *)*Data;
  2760.  
  2761.     if(BufferInfo && BufferInfo->Buddy)
  2762.     {
  2763.         Forbid();
  2764.  
  2765.         ReleaseSemaphore(Access);
  2766.         ShakeHands((struct Task *)BufferInfo->Buddy,Signals);
  2767.  
  2768.         Permit();
  2769.     }
  2770.     else
  2771.         ReleaseSemaphore(Access);
  2772. }
  2773.  
  2774.  
  2775. /*****************************************************************************/
  2776.  
  2777.  
  2778. STATIC VOID SAVE_DS
  2779. ReviewServer()
  2780. {
  2781.     STATIC BOOL        SearchForward    = TRUE,
  2782.                     IgnoreCase        = TRUE,
  2783.                     WholeWords        = FALSE;
  2784.     STATIC LONG        TopLine            = -1;
  2785.  
  2786.     TextBufferInfo    *BufferInfo;
  2787.     struct Task        *Father,*Me;
  2788.     BOOL             RingBack = TRUE;
  2789.  
  2790.         /* Wake this guy up */
  2791.  
  2792.     Me = FindTask(NULL);
  2793.  
  2794.     Father = (struct Task *)Me->tc_UserData;
  2795.  
  2796.         /* Snap our fingers... */
  2797.  
  2798.     if(BufferInfo = CreateBufferInfo(Window->WScreen,&SearchForward,&IgnoreCase,&WholeWords,&TopLine))
  2799.     {
  2800.             /* Open wide, here I come! */
  2801.  
  2802.         ObtainSemaphore(&ReviewTaskSemaphore);
  2803.  
  2804.         ReviewInfoData = BufferInfo;
  2805.  
  2806.         ReleaseSemaphore(&ReviewTaskSemaphore);
  2807.  
  2808.             /* Up and running */
  2809.  
  2810.         Signal(Father,SIG_HANDSHAKE);
  2811.  
  2812.         Father = NULL;
  2813.  
  2814.             /* Swish the tinsel */
  2815.  
  2816.         RingBack = HandleBuffer(&ReviewTaskSemaphore,&ReviewInfoData);
  2817.  
  2818.         if(Config->CaptureConfig->RememberBufferWindow)
  2819.             PutWindowInfo(WINDOW_REVIEW,BufferInfo->Window->LeftEdge,BufferInfo->Window->TopEdge,BufferInfo->Window->Width,BufferInfo->Window->Height);
  2820.  
  2821.             /* Activate the main window again */
  2822.  
  2823.         if(Window)
  2824.             BumpWindow(Window);
  2825.  
  2826.             /* The wrecking crew */
  2827.  
  2828.         DeleteBufferInfo(BufferInfo);
  2829.  
  2830.         CheckItem(MEN_REVIEW_WINDOW,FALSE);
  2831.     }
  2832.  
  2833.         /* Shutdown in an orderly fashion */
  2834.  
  2835.     Forbid();
  2836.  
  2837.     if(RingBack)
  2838.     {
  2839.         if(Father)
  2840.             Signal(Father,SIG_HANDSHAKE);
  2841.         else
  2842.             Signal((struct Task *)ThisProcess,SIG_HANDSHAKE);
  2843.     }
  2844. }
  2845.  
  2846. VOID
  2847. MoveReview(LONG Mode)
  2848. {
  2849.     MoveBuffer(&ReviewTaskSemaphore,&ReviewInfoData,Mode);
  2850. }
  2851.  
  2852. VOID
  2853. DeleteReview()
  2854. {
  2855.     CheckItem(MEN_REVIEW_WINDOW,FALSE);
  2856.  
  2857.     ObtainSemaphore(&ReviewTaskSemaphore);
  2858.  
  2859.     if(ReviewInfoData)
  2860.     {
  2861.         Forbid();
  2862.  
  2863.         ReleaseSemaphore(&ReviewTaskSemaphore);
  2864.         ShakeHands((struct Task *)ReviewInfoData->Buddy,SIG_KILL);
  2865.  
  2866.         Permit();
  2867.     }
  2868.     else
  2869.         ReleaseSemaphore(&ReviewTaskSemaphore);
  2870. }
  2871.  
  2872. BOOL
  2873. CreateReview()
  2874. {
  2875.     BOOL Result = FALSE;
  2876.  
  2877.     ObtainSemaphore(&ReviewTaskSemaphore);
  2878.  
  2879.         /* Is the Review process already running? */
  2880.  
  2881.     if(ReviewInfoData)
  2882.     {
  2883.             /* Tell it to bring its screen to the front. */
  2884.  
  2885.         Signal((struct Task *)ReviewInfoData->Buddy,SIG_TOFRONT);
  2886.  
  2887.         ReleaseSemaphore(&ReviewTaskSemaphore);
  2888.  
  2889.             /* Return success. */
  2890.  
  2891.         Result = TRUE;
  2892.     }
  2893.     else
  2894.     {
  2895.         struct Task    *Child;
  2896.  
  2897.         ReleaseSemaphore(&ReviewTaskSemaphore);
  2898.  
  2899.         Forbid();
  2900.  
  2901.             /* Launch the Review process. */
  2902.  
  2903.         if(Child = (struct Task *)CreateNewProcTags(
  2904.             NP_Entry,        ReviewServer,
  2905.             NP_Name,        "term Review Process",
  2906.             NP_WindowPtr,    -1,
  2907.             NP_StackSize,    8192,
  2908.         TAG_END))
  2909.         {
  2910.             Child->tc_UserData = FindTask(NULL);
  2911.  
  2912.             WaitForHandshake();
  2913.  
  2914.             ObtainSemaphore(&ReviewTaskSemaphore);
  2915.  
  2916.             if(ReviewInfoData)
  2917.                 Result = TRUE;
  2918.  
  2919.             ReleaseSemaphore(&ReviewTaskSemaphore);
  2920.         }
  2921.  
  2922.         Permit();
  2923.     }
  2924.  
  2925.     if(Result)
  2926.         CheckItem(MEN_REVIEW_WINDOW,TRUE);
  2927.  
  2928.     return(Result);
  2929. }
  2930.