| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479 | {    $Id$    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by the Free Pascal development team    Graph unit implementation part    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}var  ExitSave: pointer;const  firstCallOfInitGraph: boolean = true;{$ifdef logging}var debuglog: text;function strf(l: longint): string;begin  str(l, strf)end;Procedure Log(Const s: String);Begin  Append(debuglog);  Write(debuglog, s);  Close(debuglog);End;Procedure LogLn(Const s: string);Begin  Append(debuglog);  Writeln(debuglog,s);  Close(debuglog);End;{$endif logging}const   StdBufferSize = 4096;   { Buffer size for FloodFill }type  tinttable = array[0..16383] of smallint;  pinttable = ^tinttable;  WordArray = Array [0..StdbufferSize] Of word;  PWordArray = ^WordArray;const   { Mask for each bit in byte used to determine pattern }   BitArray: Array[0..7] of byte =     ($01,$02,$04,$08,$10,$20,$40,$80);   RevbitArray: Array[0..7] of byte =     ($80,$40,$20,$10,$08,$04,$02,$01);    { pre expanded line patterns }    { 0 = LSB of byte pattern    }    { 15 = MSB of byte pattern   }    LinePatterns: Array[0..15] of BOOLEAN =     (TRUE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE,      TRUE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE,TRUE);const  BGIPath : string = '.';  { Default font 8x8 system from IBM PC }  {$i fontdata.inc}var  CurrentColor:     Word;  CurrentBkColor: Word;  CurrentX : smallint;   { viewport relative }  CurrentY : smallint;   { viewport relative }  ClipPixels: Boolean;  { Should cliiping be enabled }  CurrentWriteMode: smallint;  _GraphResult : smallint;  LineInfo : LineSettingsType;  FillSettings: FillSettingsType;  { information for Text Output routines }  CurrentTextInfo : TextSettingsType;  CurrentXRatio, CurrentYRatio: graph_float;  installedfonts: longint;  { Number of installed fonts }  StartXViewPort: smallint; { absolute }  StartYViewPort: smallint; { absolute }  ViewWidth : smallint;  ViewHeight: smallint;  IsGraphMode : Boolean; { Indicates if we are in graph mode or not }  ArcCall: ArcCoordsType;   { Information on the last call to Arc or Ellipse }var  { ******************** HARDWARE INFORMATION ********************* }  { Should be set in InitGraph once only.                           }  IntCurrentMode : smallint;  IntCurrentDriver : smallint;       { Currently loaded driver          }  IntCurrentNewDriver: smallint;  XAspect : word;  YAspect : word;  MaxX : smallint;       { Maximum resolution - ABSOLUTE }  MaxY : smallint;       { Maximum resolution - ABSOLUTE }  MaxColor : Longint;  PaletteSize : longint; { Maximum palette entry we can set, usually equal}                         { maxcolor.                                      }  HardwarePages : byte;  { maximum number of hardware visual pages        }  DriverName: String;  DirectColor : Boolean ; { Is it a direct color mode? }  ModeList : PModeInfo;{$ifndef nonewmodes}  newModeList: TNewModeInfo;{$endif nonewmodes}  DirectVideo : Boolean;  { Direct access to video memory? }{--------------------------------------------------------------------------}{                                                                          }{                    LINE AND LINE RELATED ROUTINES                        }{                                                                          }{--------------------------------------------------------------------------}  {$i clip.inc}  procedure HLineDefault(x,x2,y: smallint); {$ifndef fpc}far;{$endif fpc}   var    xtmp: smallint;   Begin    { must we swap the values? }    if x >= x2 then      Begin        xtmp := x2;        x2 := x;        x:= xtmp;      end;    { First convert to global coordinates }    X   := X + StartXViewPort;    X2  := X2 + StartXViewPort;    Y   := Y + StartYViewPort;    if ClipPixels then      Begin         if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,                StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then            exit;      end;    for x:= x to x2 do      DirectPutPixel(X,Y);   end;  procedure VLineDefault(x,y,y2: smallint); {$ifndef fpc}far;{$endif fpc}   var    ytmp: smallint;  Begin    { must we swap the values? }    if y >= y2 then     Begin       ytmp := y2;       y2 := y;       y:= ytmp;     end;    { First convert to global coordinates }    X   := X + StartXViewPort;    Y2  := Y2 + StartYViewPort;    Y   := Y + StartYViewPort;    if ClipPixels then      Begin         if LineClipped(x,y,x,y2,StartXViewPort,StartYViewPort,                StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then            exit;      end;    for y := y to y2 do Directputpixel(x,y)  End;  Procedure DirectPutPixelClip(x,y: smallint);  { for thickwidth lines, because they may call DirectPutPixel for coords }  { outside the current viewport (bug found by CEC)                       }  Begin    If (Not ClipPixels) Or       ((X >= StartXViewPort) And (X <= (StartXViewPort + ViewWidth)) And        (Y >= StartYViewPort) And (Y <= (StartYViewPort + ViewHeight))) then      Begin        DirectPutPixel(x,y)      End  End;  procedure LineDefault(X1, Y1, X2, Y2: smallint); {$ifndef fpc}far;{$endif fpc}  var X, Y :           smallint;      deltax, deltay : smallint;      d, dinc1, dinc2: smallint;      xinc1          : smallint;      xinc2          : smallint;      yinc1          : smallint;      yinc2          : smallint;      i              : smallint;      Flag           : Boolean; { determines pixel direction in thick lines }      NumPixels      : smallint;      PixelCount     : smallint;      OldCurrentColor: Word;      swtmp          : smallint;      TmpNumPixels   : smallint; begin{******************************************}{  SOLID LINES                             }{******************************************}  if lineinfo.LineStyle = SolidLn then    Begin       { we separate normal and thick width for speed }       { and because it would not be 100% compatible  }       { with the TP graph unit otherwise             }       if y1 = y2 then        Begin     {******************************************}     {  SOLID LINES HORIZONTAL                  }     {******************************************}          if lineinfo.Thickness=NormWidth then            hline(x1,x2,y2)          else            begin               { thick width }               hline(x1,x2,y2-1);               hline(x1,x2,y2);               hline(x2,x2,y2+1);            end;        end    else    if x1 = x2 then        Begin     {******************************************}     {  SOLID LINES VERTICAL                    }     {******************************************}          if lineinfo.Thickness=NormWidth then            vline(x1,y1,y2)          else            begin            { thick width }              vline(x1-1,y1,y2);              vline(x1,y1,y2);              vline(x1+1,y1,y2);            end;        end    else    begin     { Convert to global coordinates. }     x1 := x1 + StartXViewPort;     x2 := x2 + StartXViewPort;     y1 := y1 + StartYViewPort;     y2 := y2 + StartYViewPort;     { if fully clipped then exit... }     if ClipPixels then       begin       if LineClipped(x1,y1,x2,y2,StartXViewPort, StartYViewPort,           StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then              exit;       end;     {******************************************}     {  SLOPED SOLID LINES                      }     {******************************************}           oldCurrentColor :=           CurrentColor;           { Calculate deltax and deltay for initialisation }           deltax := abs(x2 - x1);           deltay := abs(y2 - y1);          { Initialize all vars based on which is the independent variable }          if deltax >= deltay then            begin             Flag := FALSE;             { x is independent variable }             numpixels := deltax + 1;             d := (2 * deltay) - deltax;             dinc1 := deltay Shl 1;             dinc2 := (deltay - deltax) shl 1;             xinc1 := 1;             xinc2 := 1;             yinc1 := 0;             yinc2 := 1;            end          else            begin             Flag := TRUE;             { y is independent variable }             numpixels := deltay + 1;             d := (2 * deltax) - deltay;             dinc1 := deltax Shl 1;             dinc2 := (deltax - deltay) shl 1;             xinc1 := 0;             xinc2 := 1;             yinc1 := 1;             yinc2 := 1;            end;         { Make sure x and y move in the right directions }         if x1 > x2 then           begin            xinc1 := - xinc1;            xinc2 := - xinc2;           end;         if y1 > y2 then          begin           yinc1 := - yinc1;           yinc2 := - yinc2;          end;         { Start drawing at <x1, y1> }         x := x1;         y := y1;         If LineInfo.Thickness=NormWidth then          Begin            { Draw the pixels }            for i := 1 to numpixels do              begin                DirectPutPixel(x, y);                if d < 0 then                  begin                   d := d + dinc1;                   x := x + xinc1;                   y := y + yinc1;                  end                else                  begin                   d := d + dinc2;                   x := x + xinc2;                   y := y + yinc2;                  end;                  CurrentColor := OldCurrentColor;             end;          end        else         { Thick width lines }          begin            { Draw the pixels }             for i := 1 to numpixels do               begin                { all depending on the slope, we can determine         }                { in what direction the extra width pixels will be put }                If Flag then                  Begin                    DirectPutPixelClip(x-1,y);                    DirectPutPixelClip(x,y);                    DirectPutPixelClip(x+1,y);                  end                else                  Begin                    DirectPutPixelClip(x, y-1);                    DirectPutPixelClip(x, y);                    DirectPutPixelClip(x, y+1);                  end;                if d < 0 then                  begin                    d := d + dinc1;                    x := x + xinc1;                    y := y + yinc1;                  end                else                  begin                    d := d + dinc2;                    x := x + xinc2;                    y := y + yinc2;                  end;                CurrentColor := OldCurrentColor;               end;          end;        end;  end   else{******************************************}{  begin patterned lines                   }{******************************************}    Begin      { Convert to global coordinates. }      x1 := x1 + StartXViewPort;      x2 := x2 + StartXViewPort;      y1 := y1 + StartYViewPort;      y2 := y2 + StartYViewPort;      { if fully clipped then exit... }      if ClipPixels then       begin       if LineClipped(x1,y1,x2,y2,StartXViewPort, StartYViewPort,           StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then              exit;       end;      OldCurrentColor := CurrentColor;      PixelCount:=0;      if y1 = y2 then            Begin             { Check if we must swap }         if x1 >= x2 then               Begin                 swtmp := x1;                 x1 := x2;                 x2 := swtmp;               end;         if LineInfo.Thickness = NormWidth then              Begin               for PixelCount:=x1 to x2 do                     { optimization: PixelCount mod 16 }                     if LinePatterns[PixelCount and 15] = TRUE then                      begin                        DirectPutPixel(PixelCount,y2);                      end;              end             else              Begin               for i:=-1 to 1 do                     Begin                       for PixelCount:=x1 to x2 do                         { Optimization from Thomas - mod 16 = and 15 }                         {this optimization has been performed by the compiler                          for while as well (JM)}                         if LinePatterns[PixelCount and 15] = TRUE then                           begin                                 DirectPutPixelClip(PixelCount,y2+i);                           end;                     end;              end;        end      else      if x1 = x2 then           Begin            { Check if we must swap }            if y1 >= y2 then              Begin                swtmp := y1;                y1 := y2;                y2 := swtmp;              end;            if LineInfo.Thickness = NormWidth then              Begin                for PixelCount:=y1 to y2 do                    { compare if we should plot a pixel here , compare }                    { with predefined line patterns...                 }                    if LinePatterns[PixelCount and 15] = TRUE then                      begin                    DirectPutPixel(x1,PixelCount);                      end;              end            else              Begin                for i:=-1 to 1 do                     Begin                       for PixelCount:=y1 to y2 do                       { compare if we should plot a pixel here , compare }                       { with predefined line patterns...                 }                         if LinePatterns[PixelCount and 15] = TRUE then                           begin                             DirectPutPixelClip(x1+i,PixelCount);                           end;                     end;              end;           end      else           Begin             oldCurrentColor := CurrentColor;             { Calculate deltax and deltay for initialisation }             deltax := abs(x2 - x1);             deltay := abs(y2 - y1);             { Initialize all vars based on which is the independent variable }             if deltax >= deltay then               begin                 Flag := FALSE;                 { x is independent variable }                 numpixels := deltax + 1;                 d := (2 * deltay) - deltax;                 dinc1 := deltay Shl 1;                 dinc2 := (deltay - deltax) shl 1;                 xinc1 := 1;                 xinc2 := 1;                 yinc1 := 0;                 yinc2 := 1;              end            else              begin                Flag := TRUE;                { y is independent variable }                numpixels := deltay + 1;                d := (2 * deltax) - deltay;                dinc1 := deltax Shl 1;                dinc2 := (deltax - deltay) shl 1;                xinc1 := 0;                xinc2 := 1;                yinc1 := 1;                yinc2 := 1;              end;            { Make sure x and y move in the right directions }            if x1 > x2 then              begin                xinc1 := - xinc1;                xinc2 := - xinc2;              end;            if y1 > y2 then              begin                yinc1 := - yinc1;                yinc2 := - yinc2;              end;            { Start drawing at <x1, y1> }            x := x1;            y := y1;            If LineInfo.Thickness=ThickWidth then             Begin               TmpNumPixels := NumPixels-1;               { Draw the pixels }               for i := 0 to TmpNumPixels do                 begin                     { all depending on the slope, we can determine         }                     { in what direction the extra width pixels will be put }                       If Flag then                          Begin                            { compare if we should plot a pixel here , compare }                            { with predefined line patterns...                 }                            if LinePatterns[i and 15] = TRUE then                              begin                                DirectPutPixelClip(x-1,y);                                DirectPutPixelClip(x,y);                                DirectPutPixelClip(x+1,y);                              end;                          end                       else                          Begin                            { compare if we should plot a pixel here , compare }                            { with predefined line patterns...                 }                            if LinePatterns[i and 15] = TRUE then                             begin                               DirectPutPixelClip(x,y-1);                               DirectPutPixelClip(x,y);                               DirectPutPixelClip(x,y+1);                             end;                          end;                   if d < 0 then                         begin                           d := d + dinc1;                           x := x + xinc1;                           y := y + yinc1;                         end                   else                         begin                   d := d + dinc2;                   x := x + xinc2;                   y := y + yinc2;                         end;                end;            end           else            Begin             { instead of putting in loop , substract by one now }             TmpNumPixels := NumPixels-1;            { NormWidth }             for i := 0 to TmpNumPixels do             begin                  if LinePatterns[i and 15] = TRUE then                    begin                          DirectPutPixel(x,y);                    end;             if d < 0 then                 begin                   d := d + dinc1;                   x := x + xinc1;                   y := y + yinc1;                 end             else                 begin                   d := d + dinc2;                   x := x + xinc2;                   y := y + yinc2;                 end;             end;            end        end;{******************************************}{  end patterned lines                     }{******************************************}       { restore color }       CurrentColor:=OldCurrentColor;   end; end;  { Line }  {********************************************************}  { Procedure DummyPatternLine()                           }  {--------------------------------------------------------}  { This is suimply an procedure that does nothing which   }  { can be passed as a patternlineproc for non-filled      }  { ellipses                                               }  {********************************************************}  Procedure DummyPatternLine(x1, x2, y: smallint); {$ifdef tp} far; {$endif tp}  begin  end;  {********************************************************}  { Procedure InternalEllipse()                            }  {--------------------------------------------------------}  { This routine first calculates all points required to   }  { draw a circle to the screen, and stores the points     }  { to display in a buffer before plotting them. The       }  { aspect ratio of the screen is taken into account when  }  { calculating the values.                                }  {--------------------------------------------------------}  { INPUTS: X,Y : Center coordinates of Ellipse.           }  {  XRadius - X-Axis radius of ellipse.                   }  {  YRadius - Y-Axis radius of ellipse.                   }  {  stAngle, EndAngle: Start angle and end angles of the  }  {  ellipse (used for partial ellipses and circles)       }  {  pl: procedure which either draws a patternline (for   }  {      FillEllipse) or does nothing (arc etc)            }  {--------------------------------------------------------}  { NOTE: -                                                }  {       -                                                }  {********************************************************}  Procedure InternalEllipseDefault(X,Y: smallint;XRadius: word;    YRadius:word; stAngle,EndAngle: word; pl: PatternLineProc); {$ifndef fpc}far;{$endif fpc}   Const ConvFac = Pi/180.0;   var    j, Delta, DeltaEnd: graph_float;    NumOfPixels: longint;    TempTerm: graph_float;    xtemp, ytemp, xp, yp, xm, ym, xnext, ynext,      plxpyp, plxmyp, plxpym, plxmym: smallint;    BackupColor, TmpAngle, OldLineWidth: word;  Begin   If LineInfo.ThickNess = ThickWidth Then    { first draw the two outer ellipses using normwidth and no filling (JM) }     Begin       OldLineWidth := LineInfo.Thickness;       LineInfo.Thickness := NormWidth;       InternalEllipseDefault(x,y,XRadius,YRadius,StAngle,EndAngle,                              {$ifdef fpc}@{$endif fpc}DummyPatternLine);       InternalEllipseDefault(x,y,XRadius+1,YRadius+1,StAngle,EndAngle,                              {$ifdef fpc}@{$endif fpc}DummyPatternLine);       If (XRadius > 0) and (YRadius > 0) Then         { draw the smallest ellipse last, since that one will use the }         { original pl, so it could possibly draw patternlines (JM)    }         Begin           Dec(XRadius);           Dec(YRadius);         End       Else Exit;       { restore line thickness }       LineInfo.Thickness := OldLineWidth;     End;   { Adjust for screen aspect ratio }   XRadius:=(longint(XRadius)*10000) div XAspect;   YRadius:=(longint(YRadius)*10000) div YAspect;   If xradius = 0 then inc(xradius);   if yradius = 0 then inc(yradius);   { check for an ellipse with negligable x and y radius }   If (xradius <= 1) and (yradius <= 1) then     begin       putpixel(x,y,CurrentColor);       ArcCall.X := X;       ArcCall.Y := Y;       ArcCall.XStart := X;       ArcCall.YStart := Y;       ArcCall.XEnd := X;       ArcCall.YEnd := Y;       exit;     end;   { check if valid angles }   stangle := stAngle mod 361;   EndAngle := EndAngle mod 361;   { if impossible angles then swap them! }   if Endangle < StAngle then     Begin       TmpAngle:=EndAngle;       EndAngle:=StAngle;       Stangle:=TmpAngle;     end;   { approximate the number of pixels required by using the circumference }   { equation of an ellipse.                                              }   { Changed this formula a it (trial and error), but the net result is that }   { less pixels have to be calculated now                                   }   NumOfPixels:=Round(Sqrt(3)*sqrt(sqr(XRadius)+sqr(YRadius)));   { Calculate the angle precision required }   Delta := 90.0 / NumOfPixels;   { for restoring after PatternLine }   BackupColor := CurrentColor;   { removed from inner loop to make faster }   { store some arccall info }   ArcCall.X := X;   ArcCall.Y := Y;   TempTerm := (StAngle)*ConvFac;   ArcCall.XStart := round(XRadius*Cos(TempTerm)) + X;   ArcCall.YStart := round(YRadius*Sin(TempTerm+Pi)) + Y;   TempTerm := (EndAngle)*ConvFac;   ArcCall.XEnd := round(XRadius*Cos(TempTerm)) + X;   ArcCall.YEnd := round(YRadius*Sin(TempTerm+Pi)) + Y;   { Always just go over the first 90 degrees. Could be optimized a   }   { bit if StAngle and EndAngle lie in the same quadrant, left as an }   { exercise for the reader :) (JM)                                  }   j := 0;   { calculate stop position, go 1 further than 90 because otherwise }   { 1 pixel is sometimes not drawn (JM)                             }   DeltaEnd := 91;   { Calculate points }   xnext := XRadius;   ynext := 0;   Repeat     xtemp := xnext;     ytemp := ynext;     { this is used by both sin and cos }     TempTerm := (j+Delta)*ConvFac;     { Calculate points }     xnext := round(XRadius*Cos(TempTerm));     ynext := round(YRadius*Sin(TempTerm+Pi));     xp := x + xtemp;     xm := x - xtemp;     yp := y + ytemp;     ym := y - ytemp;     plxpyp := maxsmallint;     plxmyp := -maxsmallint-1;     plxpym := maxsmallint;     plxmym := -maxsmallint-1;     If (j >= StAngle) and (j <= EndAngle) then       begin         plxpyp := xp;         PutPixel(xp,yp,CurrentColor);       end;     If ((180-j) >= StAngle) and ((180-j) <= EndAngle) then       begin         plxmyp := xm;         PutPixel(xm,yp,CurrentColor);       end;     If ((j+180) >= StAngle) and ((j+180) <= EndAngle) then       begin         plxmym := xm;         PutPixel(xm,ym,CurrentColor);       end;     If ((360-j) >= StAngle) and ((360-j) <= EndAngle) then       begin         plxpym := xp;         PutPixel(xp,ym,CurrentColor);       end;     If (ynext <> ytemp) and        (xp - xm >= 1) then       begin         CurrentColor := FillSettings.Color;         pl(plxmyp+1,plxpyp-1,yp);         pl(plxmym+1,plxpym-1,ym);         CurrentColor := BackupColor;       end;     j:=j+Delta;   Until j > (DeltaEnd);  end;  {********************************************************}  { Procedure InternalEllipse()                            }  {--------------------------------------------------------}  { This routine first calculates all points required to   }  { draw a circle to the screen, and stores the points     }  { to display in a buffer before plotting them. The       }  { aspect ratio of the screen is taken into account when  }  { calculating the values.                                }  {--------------------------------------------------------}  { INPUTS: X,Y : Center coordinates of Ellipse.           }  {  XRadius - X-Axis radius of ellipse.                   }  {  YRadius - Y-Axis radius of ellipse.                   }  {  stAngle, EndAngle: Start angle and end angles of the  }  {  ellipse (used for partial ellipses and circles)       }  {--------------------------------------------------------}  { NOTE: - uses the current write mode.                   }  {       - Angles must both be between 0 and 360          }  {********************************************************}(*Procedure InternalEllipseDefault (x, y : smallint;    xradius, yradius, stAngle, EndAngle : Word; pl: PatternLineProc); {$ifndef fpc} far; {$endif fpc}{ Draw an ellipse arc. Crude but it works (anyone have a better one?) }Var  aSqr, bSqr, twoaSqr, twobSqr, xa, ya, twoXbSqr, twoYaSqr, error : LongInt;  Alpha, TempTerm : graph_float;  BackupColor: Word;  plxpyp, plxmyp, plxpym, plxmym: smallint;const  RadToDeg = 180/Pi;Procedure PlotPoints;var i,j: smallint; xm, ym: smallint; xp, yp: smallint;Begin   ym := y-ya;   yp := y+ya;   xm := x-xa;   xp := x+xa;   plxpyp := maxsmallint;   plxmyp := -maxsmallint-1;   plxpym := maxsmallint;   plxmym := -maxsmallint-1;   if LineInfo.Thickness = Normwidth then     Begin       If (Alpha+270>=StAngle) And (Alpha+270<=EndAngle) then          Begin            plxmym := xm;            PutPixel (xm,ym, CurrentColor);          End;       If ((180+270)-Alpha>=StAngle) And ((180+270)-Alpha<=EndAngle) then          Begin            plxmyp := xm;            PutPixel (xm,yp, CurrentColor);          End;       If ((180+270)+Alpha>=StAngle) And ((180+270)+Alpha<=EndAngle) then          Begin            plxpyp := xp;            PutPixel (xp,yp, CurrentColor);          End;       If ((360+270)-Alpha>=StAngle) And ((360+270)-Alpha<=EndAngle) then          Begin            plxpym := xp;            PutPixel (xp,ym, CurrentColor);          End;     end   else     Begin       If (Alpha+270>=StAngle) And (Alpha+270<=EndAngle) then         Begin           plxmym := xm + 1;           for i:=-1 to 1 do             for j:=-1 to 1 do               PutPixel (xm+i,ym+j, CurrentColor);         End;       If ((180+270)-Alpha>=StAngle) And ((180+270)-Alpha<=EndAngle) then         Begin           plxmyp := xm + 1;           for i:=-1 to 1 do             for j:=-1 to 1 do               PutPixel (xm+i,yp+j, CurrentColor);         End;       If ((180+270)+Alpha>=StAngle) And ((180+270)+Alpha<=EndAngle) then         Begin           plxpyp := xp - 1;           for i:=-1 to 1 do             for j:=-1 to 1 do               PutPixel (xp+i,yp+j, CurrentColor);         End;       If ((360+270)-Alpha>=StAngle) And ((360+270)-Alpha<=EndAngle) then         Begin           plxpym := xp - 1;           for i:=-1 to 1 do             for j:=-1 to 1 do               PutPixel (xp+i,ym+j, CurrentColor);         End;     end;     If (xp <> xm) then       begin         CurrentColor := FillSettings.Color;         pl(plxmyp+1,plxpyp-1,yp);         pl(plxmym+1,plxpym-1,ym);         CurrentColor := BackupColor;       end;End;Begin  { check for an ellipse with negligable x and y radius }  If (xradius <= 1) and (yradius <= 1) then    begin      putpixel(x,y,CurrentColor);      ArcCall.X := X;      ArcCall.Y := Y;      ArcCall.XStart := X;      ArcCall.YStart := Y;      ArcCall.XEnd := X;      ArcCall.YEnd := Y;      exit;    end;  { for restoring after PatternLine }  BackupColor := CurrentColor;  If xradius = 0 then inc(xradius);  if yradius = 0 then inc(yradius);  { store arccall info }  ArcCall.x := x;  ArcCall.y := y;  TempTerm := StAngle*RadToDeg;  ArcCall.XStart := round(XRadius*Cos(TempTerm)) + X;  ArcCall.YStart := round(YRadius*Sin(TempTerm+Pi)) + Y;  TempTerm := EndAngle*RadToDeg;  ArcCall.XEnd := round(XRadius*Cos(TempTerm)) + X;  ArcCall.YEnd := round(YRadius*Sin(TempTerm+Pi)) + Y;  StAngle:=StAngle MOD 361;  EndAngle:=EndAngle MOD 361;  StAngle := StAngle + 270;  EndAngle := EndAngle + 270;  If StAngle>EndAngle then  Begin    StAngle:=StAngle Xor EndAngle; EndAngle:=EndAngle Xor StAngle; StAngle:=EndAngle Xor StAngle;  End;  { Adjust for screen aspect ratio }  XRadius:=(longint(XRadius)*10000) div XAspect;  YRadius:=(longint(YRadius)*10000) div YAspect;  aSqr:=LongInt (xradius)*LongInt (xradius);  bSqr:=LongInt (yradius)*LongInt (yradius);  twoaSqr:=2*aSqr;  twobSqr:=2*bSqr;  xa:=0;  ya:=yradius;  twoXbSqr:=0;  twoYaSqr:=ya*twoaSqr;  error:=-ya*aSqr;  While twoXbSqr<=twoYaSqr Do Begin    If ya=0 then Alpha:=90 Else Alpha:=RadToDeg*Arctan (xa/ya); { Crude but it works }    PlotPoints;    Inc (xa);    Inc (twoXbSqr,twobSqr);    Inc (error,twoXbSqr-bSqr);    If error>=0 then Begin      Dec (ya);      Dec (twoYaSqr,twoaSqr);      Dec (error,twoYaSqr);    End;  End;  xa:=xradius;  ya:=0;  twoXbSqr:=xa*twobSqr;  twoYaSqr:=0;  error:=-xa*bSqr;  While twoXbSqr>twoYaSqr Do Begin    If ya=0 then Alpha:=90 Else Alpha:=RadToDeg*Arctan (xa/ya);    PlotPoints;    Inc (ya);    Inc (twoYaSqr,twoaSqr);    Inc (error,twoYaSqr-aSqr);    If error>=0 then Begin      Dec (xa);      Dec (twoXbSqr,twobSqr);      Dec (error,twoXbSqr);    End;  End;End;*)  procedure PatternLineDefault(x1,x2,y: smallint); {$ifndef fpc}far;{$endif fpc}  {********************************************************}  { Draws a horizontal patterned line according to the     }  { current Fill Settings.                                 }  {********************************************************}  { Important notes:                                       }  {  - CurrentColor must be set correctly before entering  }  {    this routine.                                       }  {********************************************************}   var    NrIterations: smallint;    i           : smallint;    j           : smallint;    TmpFillPattern : byte;    OldWriteMode : word;    OldCurrentColor : word;   begin     { convert to global coordinates ... }     x1 := x1 + StartXViewPort;     x2 := x2 + StartXViewPort;     y  := y + StartYViewPort;     { if line was fully clipped then exit...}     if LineClipped(x1,y,x2,y,StartXViewPort,StartYViewPort,        StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then         exit;     OldWriteMode := CurrentWriteMode;     CurrentWriteMode := NormalPut;     { Get the current pattern }     TmpFillPattern := FillPatternTable       [FillSettings.Pattern][(y and $7)+1];     Case TmpFillPattern Of       0:         begin           OldCurrentColor := CurrentColor;           CurrentColor := CurrentBkColor;  { hline converts the coordinates to global ones, but that has been done }  { already here!!! Convert them back to local ones... (JM)                }           HLine(x1-StartXViewPort,x2-StartXViewPort,y-StartYViewPort);           CurrentColor := OldCurrentColor;         end;       $ff:         begin           HLine(x1-StartXViewPort,x2-StartXViewPort,y-StartYViewPort);         end;       else         begin           { number of times to go throuh the 8x8 pattern }           NrIterations := abs(x2 - x1+8) div 8;           For i:= 0 to NrIterations do             Begin               for j:=0 to 7 do                    Begin                            { x1 mod 8 }                    if RevBitArray[x1 and 7] and TmpFillPattern <> 0 then                       DirectPutpixel(x1,y)                    else                      begin                            { According to the TP graph manual, we overwrite everything }                            { which is filled up - checked against VGA and CGA drivers  }                            { of TP.                                                    }                            OldCurrentColor := CurrentColor;                            CurrentColor := CurrentBkColor;                            DirectPutPixel(x1,y);                            CurrentColor := OldCurrentColor;                      end;                    Inc(x1);                    if x1 > x2 then                     begin                           CurrentWriteMode := OldWriteMode;                           exit;                     end;                   end;             end;          end;     End;     CurrentWriteMode := OldWriteMode;   end;  procedure LineRel(Dx, Dy: smallint);   Begin     Line(CurrentX, CurrentY, CurrentX + Dx, CurrentY + Dy);     CurrentX := CurrentX + Dx;     CurrentY := CurrentY + Dy;   end;  procedure LineTo(x,y : smallint);   Begin     Line(CurrentX, CurrentY, X, Y);     CurrentX := X;     CurrentY := Y;   end;  procedure Rectangle(x1,y1,x2,y2:smallint);   begin     { Do not draw the end points }     Line(x1,y1,x2-1,y1);     Line(x1,y1+1,x1,y2);     Line(x2,y1,x2,y2-1);     Line(x1+1,y2,x2,y2);   end;  procedure GetLineSettings(var ActiveLineInfo : LineSettingsType);   begin    Activelineinfo:=Lineinfo;   end;  procedure SetLineStyle(LineStyle: word; Pattern: word; Thickness: word);   var    i: byte;    j: byte;   Begin    if (LineStyle > UserBitLn) or ((Thickness <> Normwidth) and (Thickness <> ThickWidth)) then      _GraphResult := grError    else      begin       LineInfo.Thickness := Thickness;       LineInfo.LineStyle := LineStyle;       case LineStyle of            UserBitLn: Lineinfo.Pattern := pattern;            SolidLn:   Lineinfo.Pattern  := $ffff;  { ------- }            DashedLn : Lineinfo.Pattern := $F8F8;   { -- -- --}            DottedLn:  LineInfo.Pattern := $CCCC;   { - - - - }            CenterLn: LineInfo.Pattern :=  $FC78;   { -- - -- }       end; { end case }       { setup pattern styles }       j:=16;       for i:=0 to 15 do        Begin         dec(j);         { bitwise mask for each bit in the word }         if (word($01 shl i) AND LineInfo.Pattern) <> 0 then               LinePatterns[j]:=TRUE             else               LinePatterns[j]:=FALSE;        end;      end;   end;{--------------------------------------------------------------------------}{                                                                          }{                    VIEWPORT RELATED ROUTINES                             }{                                                                          }{--------------------------------------------------------------------------}Procedure ClearViewPortDefault; {$ifndef fpc}far;{$endif fpc}var j: smallint; OldWriteMode, OldCurColor: word; LineSets : LineSettingsType;Begin  { CP is always RELATIVE coordinates }  CurrentX := 0;  CurrentY := 0;  { Save all old settings }  OldCurColor := CurrentColor;  CurrentColor:=CurrentBkColor;  OldWriteMode:=CurrentWriteMode;  CurrentWriteMode:=NormalPut;  GetLineSettings(LineSets);  { reset to normal line style...}  SetLineStyle(SolidLn, 0, NormWidth);  { routines are relative here...}  { ViewHeight is Height-1 !     }  for J:=0 to ViewHeight do       HLine(0, ViewWidth , J);  { restore old settings...}  SetLineStyle(LineSets.LineStyle, LineSets.Pattern, LineSets.Thickness);  CurrentColor := OldCurColor;  CurrentWriteMode := OldWriteMode;end;Procedure SetViewPort(X1, Y1, X2, Y2: smallint; Clip: Boolean);Begin  if (X1 > GetMaxX) or (X2 > GetMaxX) or (X1 > X2) or (X1 < 0) then  Begin{$ifdef logging}    logln('invalid setviewport parameters: ('      +strf(x1)+','+strf(y1)+'), ('+strf(x2)+','+strf(y2)+')');    logln('maxx = '+strf(getmaxx)+', maxy = '+strf(getmaxy));{$endif logging}    _GraphResult := grError;    exit;  end;  if (Y1 > GetMaxY) or (Y2 > GetMaxY) or (Y1 > Y2) or (Y1 < 0) then  Begin{$ifdef logging}    logln('invalid setviewport parameters: ('      +strf(x1)+','+strf(y1)+'), ('+strf(x2)+','+strf(y2)+')');    logln('maxx = '+strf(getmaxx)+', maxy = '+strf(getmaxy));{$endif logging}    _GraphResult := grError;    exit;  end;  { CP is always RELATIVE coordinates }  CurrentX := 0;  CurrentY := 0;  StartXViewPort := X1;  StartYViewPort := Y1;  ViewWidth :=  X2-X1;  ViewHeight:=  Y2-Y1;  ClipPixels := Clip;end;procedure GetViewSettings(var viewport : ViewPortType);begin  ViewPort.X1 := StartXViewPort;  ViewPort.Y1 := StartYViewPort;  ViewPort.X2 := ViewWidth + StartXViewPort;  ViewPort.Y2 := ViewHeight + StartYViewPort;  ViewPort.Clip := ClipPixels;end;procedure ClearDevice;var  ViewPort: ViewPortType;begin  { Reset the CP }  CurrentX := 0;  CurrentY := 0;  { save viewport }  ViewPort.X1 :=  StartXviewPort;  ViewPort.X2 :=  ViewWidth - StartXViewPort;  ViewPort.Y1 :=  StartYViewPort;  ViewPort.Y2 :=  ViewHeight - StartYViewPort;  ViewPort.Clip := ClipPixels;  { put viewport to full screen }  StartXViewPort := 0;  ViewHeight := MaxY;  StartYViewPort := 0;  ViewWidth := MaxX;  ClipPixels := TRUE;  ClearViewPort;  { restore old viewport }  StartXViewPort := ViewPort.X1;  ViewWidth := ViewPort.X2-ViewPort.X1;  StartYViewPort := ViewPort.Y1;  ViewHeight := ViewPort.Y2-ViewPort.Y1;  ClipPixels := ViewPort.Clip;end;{--------------------------------------------------------------------------}{                                                                          }{                      BITMAP PUT/GET ROUTINES                             }{                                                                          }{--------------------------------------------------------------------------}  Procedure GetScanlineDefault (X1, X2, Y : smallint; Var Data); {$ifndef fpc}far;{$endif fpc}  {**********************************************************}  { Procedure GetScanLine()                                  }  {----------------------------------------------------------}  { Returns the full scanline of the video line of the Y     }  { coordinate. The values are returned in a WORD array      }  { each WORD representing a pixel of the specified scanline }  { note: we only need the pixels inside the ViewPort! (JM)  }  { note2: extended so you can specify start and end X coord }  {   so it is usable for GetImage too (JM)                  }  {**********************************************************}  Var    x : smallint;  Begin     For x:=X1 to X2 Do       WordArray(Data)[x-x1]:=GetPixel(x, y);  End;Function DefaultImageSize(X1,Y1,X2,Y2: smallint): longint; {$ifndef fpc}far;{$endif fpc}Begin  { each pixel uses two bytes, to enable modes with colors up to 64K }  { to work.                                                         }  DefaultImageSize := 12 + (((X2-X1+1)*(Y2-Y1+1))*2);end;Procedure DefaultPutImage(X,Y: smallint; var Bitmap; BitBlt: Word); {$ifndef fpc}far;{$endif fpc}type  pt = array[0..$fffffff] of word;  ptw = array[0..2] of longint;var  k: longint;  oldCurrentColor: word;  oldCurrentWriteMode, i, j, y1, x1, deltaX, deltaX1, deltaY: smallint;Begin{$ifdef logging}  LogLn('putImage at ('+strf(x)+','+strf(y)+') with width '+strf(ptw(Bitmap)[0])+    ' and height '+strf(ptw(Bitmap)[1]));  deltaY := 0;{$endif logging}  inc(x,startXViewPort);  inc(y,startYViewPort);  { width/height are 1-based, coordinates are zero based }  x1 := ptw(Bitmap)[0]+x-1; { get width and adjust end coordinate accordingly }  y1 := ptw(Bitmap)[1]+y-1; { get height and adjust end coordinate accordingly }  deltaX := 0;  deltaX1 := 0;  k := 3 * sizeOf(Longint) div sizeOf(Word); { Three reserved longs at start of bitmap } { check which part of the image is in the viewport }  if clipPixels then    begin      if y < startYViewPort then        begin          deltaY := startYViewPort - y;          inc(k,(x1-x+1)*deltaY);          y := startYViewPort;         end;      if y1 > startYViewPort+viewHeight then        y1 := startYViewPort+viewHeight;      if x < startXViewPort then        begin          deltaX := startXViewPort-x;          x := startXViewPort;        end;      if x1 > startXViewPort + viewWidth then        begin          deltaX1 := x1 - (startXViewPort + viewWidth);          x1 := startXViewPort + viewWidth;        end;    end;{$ifdef logging}  LogLn('deltax: '+strf(deltax)+', deltax1: '+strf(deltax1)+',deltay: '+strf(deltay));{$endif logging}  oldCurrentColor := currentColor;  oldCurrentWriteMode := currentWriteMode;  currentWriteMode := bitBlt;  for j:=Y to Y1 do   Begin     inc(k,deltaX);     for i:=X to X1 do      begin        currentColor := pt(bitmap)[k];        directPutPixel(i,j);        inc(k);     end;     inc(k,deltaX1);   end;  currentWriteMode := oldCurrentWriteMode;  currentColor := oldCurrentColor;end;Procedure DefaultGetImage(X1,Y1,X2,Y2: smallint; Var Bitmap); {$ifndef fpc}far;{$endif fpc}type  pt = array[0..$fffffff] of word;  ptw = array[0..2] of longint;var  i,j: smallint;  k: longint;Begin  k:= 3 * Sizeof(longint) div sizeof(word); { Three reserved longs at start of bitmap }  i := x2 - x1 + 1;  for j:=Y1 to Y2 do   Begin     GetScanLine(x1,x2,j,pt(Bitmap)[k]);     inc(k,i);   end;   ptw(Bitmap)[0] := X2-X1+1;   { First longint  is width  }   ptw(Bitmap)[1] := Y2-Y1+1;   { Second longint is height }   ptw(bitmap)[2] := 0;       { Third longint is reserved}end;  Procedure GetArcCoords(var ArcCoords: ArcCoordsType);   Begin     ArcCoords.X := ArcCall.X;     ArcCoords.Y := ArcCall.Y;     ArcCoords.XStart := ArcCall.XStart;     ArcCoords.YStart := ArcCall.YStart;     ArcCoords.XEnd := ArcCall.XEnd;     ArcCoords.YEnd := ArcCall.YEnd;   end;  procedure SetVisualPageDefault(page : word); {$ifndef fpc}far;{$endif fpc}   begin   end;  procedure SetActivePageDefault(page : word); {$ifndef fpc}far;{$endif fpc}   begin   end;  procedure DirectPutPixelDefault(X,Y: smallint);   begin     Writeln(stderr,'Error: Not in graphics mode (use InitGraph and test GraphResult afterwards)');     Halt(1);   end;  function GetPixelDefault(X,Y: smallint): word;   begin     Writeln(stderr,'Error: Not in graphics mode (use InitGraph and test GraphResult afterwards)');     Halt(1);     exit(0); { avoid warning }   end;  procedure PutPixelDefault(X,Y: smallint; Color: Word);   begin     Writeln(stderr,'Error: Not in graphics mode (use InitGraph and test GraphResult afterwards)');     Halt(1);   end;  procedure SetRGBPaletteDefault(ColorNum, RedValue, GreenValue, BlueValue: smallint);   begin     Writeln(stderr,'Error: Not in graphics mode (use InitGraph and test GraphResult afterwards)');     Halt(1);   end;  procedure GetRGBPaletteDefault(ColorNum: smallint; var            RedValue, GreenValue, BlueValue: smallint);   begin     Writeln(stderr,'Error: Not in graphics mode (use InitGraph and test GraphResult afterwards)');     Halt(1);   end;  procedure OutTextXYDefault(x,y : smallint;const TextString : string);forward;  procedure CircleDefault(X, Y: smallint; Radius:Word);forward;{$i palette.inc}  Procedure DefaultHooks;  {********************************************************}  { Procedure DefaultHooks()                               }  {--------------------------------------------------------}  { Resets all hookable routine either to nil for those    }  { which need overrides, and others to defaults.          }  { This is called each time SetGraphMode() is called.     }  {********************************************************}  Begin    { All default hooks procedures }    { required...}    DirectPutPixel := {$ifdef fpc}@{$endif}DirectPutPixelDefault;    PutPixel := {$ifdef fpc}@{$endif}PutPixelDefault;    GetPixel := {$ifdef fpc}@{$endif}GetPixelDefault;    SetRGBPalette := {$ifdef fpc}@{$endif}SetRGBPaletteDefault;    GetRGBPalette := {$ifdef fpc}@{$endif}GetRGBPaletteDefault;    { optional...}    SetAllPalette := {$ifdef fpc}@{$endif}SetAllPaletteDefault;    SetActivePage := {$ifdef fpc}@{$endif}SetActivePageDefault;    SetVisualPage := {$ifdef fpc}@{$endif}SetVisualPageDefault;    ClearViewPort := {$ifdef fpc}@{$endif}ClearViewportDefault;    PutImage := {$ifdef fpc}@{$endif}DefaultPutImage;    GetImage := {$ifdef fpc}@{$endif}DefaultGetImage;    ImageSize := {$ifdef fpc}@{$endif}DefaultImageSize;    GraphFreeMemPtr := nil;    GraphGetMemPtr := nil;    GetScanLine := {$ifdef fpc}@{$endif}GetScanLineDefault;    Line := {$ifdef fpc}@{$endif}LineDefault;    InternalEllipse := {$ifdef fpc}@{$endif}InternalEllipseDefault;    PatternLine := {$ifdef fpc}@{$endif}PatternLineDefault;    HLine := {$ifdef fpc}@{$endif}HLineDefault;    VLine := {$ifdef fpc}@{$endif}VLineDefault;    OuttextXY := {$ifdef fpc}@{$endif}OuttextXYDefault;    Circle := {$ifdef fpc}@{$endif}CircleDefault;  end;  Procedure InitVars;  {********************************************************}  { Procedure InitVars()                                   }  {--------------------------------------------------------}  { Resets all internal variables, and resets all          }  { overridable routines.                                  }  {********************************************************}   Begin    DirectVideo := TRUE;  { By default use fastest access possible }    ArcCall.X := 0;    ArcCall.Y := 0;    ArcCall.XStart := 0;    ArcCall.YStart := 0;    ArcCall.XEnd := 0;    ArcCall.YEnd := 0;    { Reset to default values }    IntCurrentMode := 0;    IntCurrentDriver := 0;    IntCurrentNewDriver := 0;    XAspect := 0;    YAspect := 0;    MaxX := 0;    MaxY := 0;    MaxColor := 0;    PaletteSize := 0;    DirectColor := FALSE;    HardwarePages := 0;    if hardwarepages=0 then; { remove note }    DefaultHooks;  end;{$i modes.inc}  function InstallUserDriver(Name: string; AutoDetectPtr: Pointer): smallint;   begin     _graphResult := grError;     InstallUserDriver:=grError;   end;  function RegisterBGIDriver(driver: pointer): smallint;   begin     _graphResult := grError;     RegisterBGIDriver:=grError;   end;{ ----------------------------------------------------------------- }  Procedure Arc(X,Y : smallint; StAngle,EndAngle,Radius: word);{   var    OldWriteMode: word;}   Begin     { Only if we are using thickwidths lines do we accept }     { XORput write modes.                                 }{     OldWriteMode := CurrentWriteMode;     if (LineInfo.Thickness = NormWidth) then       CurrentWriteMode := NormalPut;}     InternalEllipse(X,Y,Radius,Radius,StAngle,Endangle,{$ifdef fpc}@{$endif}DummyPatternLine);{     CurrentWriteMode := OldWriteMode;}   end; procedure Ellipse(X,Y : smallint; stAngle, EndAngle: word; XRadius,YRadius: word);  Begin    InternalEllipse(X,Y,XRadius,YRadius,StAngle,Endangle,{$ifdef fpc}@{$endif}DummyPatternLine);  end; procedure FillEllipse(X, Y: smallint; XRadius, YRadius: Word);  {********************************************************}  { Procedure FillEllipse()                                }  {--------------------------------------------------------}  { Draws a filled ellipse using (X,Y) as a center point   }  { and XRadius and YRadius as the horizontal and vertical }  { axes. The ellipse is filled with the current fill color}  { and fill style, and is bordered with the current color.}  {********************************************************}  begin    InternalEllipse(X,Y,XRadius,YRadius,0,360,PatternLine)  end; procedure CircleDefault(X, Y: smallint; Radius:Word);  {********************************************************}  { Draws a circle centered at X,Y with the given Radius.  }  {********************************************************}  { Important notes:                                       }  {  - Thickwidth circles use the current write mode, while}  {    normal width circles ALWAYS use CopyPut/NormalPut   }  {    mode. (Tested against VGA BGI driver -CEC 13/Aug/99 }  {********************************************************}  var OriginalArcInfo: ArcCoordsType;      OldWriteMode: word;  begin     if (Radius = 0) then          Exit;     if (Radius = 1) then     begin      { only normal put mode is supported by a call to PutPixel }          PutPixel(X, Y, CurrentColor);          Exit;     end;     { save state of arc information }     { because it is not needed for  }     { a circle call.                }     move(ArcCall,OriginalArcInfo, sizeof(ArcCall));     if LineInfo.Thickness = Normwidth then       begin             OldWriteMode := CurrentWriteMode;             CurrentWriteMode := CopyPut;       end;     InternalEllipse(X,Y,Radius,Radius,0,360,{$ifdef fpc}@{$endif}DummyPatternLine);     if LineInfo.Thickness = Normwidth then         CurrentWriteMode := OldWriteMode;     { restore arc information }     move(OriginalArcInfo, ArcCall,sizeof(ArcCall)); end; procedure SectorPL(x1,x2,y: smallint); {$ifndef fpc}far;{$endif fpc} var plx1, plx2: smallint;{$ifdef sectorpldebug}     t : text;{$endif sectorpldebug} begin{$ifdef sectorpldebug}   assign(t,'sector.log');   append(t);   writeln(t,'Got here for line ',y);   close(t);{$endif sectorpldebug}   If (x1 = -maxsmallint) Then     If (x2 = maxsmallint-1) Then       { no ellipse points drawn on this line }       If (((Y < ArcCall.Y) and (Y > ArcCall.YStart)) or          ((Y > ArcCall.Y) and (Y < ArcCall.YStart))) Then         { there is a part of the sector at this y coordinate, but no    }         { ellips points are plotted on this line, so draw a patternline }         { between the lines connecting (arccall.x,arccall.y) with       }         { the start and the end of the arc (JM)                         }         { use: y-y1=(y2-y1)/(x2-x1)*(x-x1) =>                           }         { x = (y-y1)/(y2-y1)*(x2-x1)+x1                                 }         Begin{$ifdef sectorpldebug}           If (ArcCall.YStart-ArcCall.Y) = 0 then             begin               append(t);               writeln(t,'bug1');               close(t);               runerror(202);             end;{$endif sectorpldebug}           plx1 := (y-ArcCall.Y)*(ArcCall.XStart-ArcCall.X)                   div (ArcCall.YStart-ArcCall.Y)+ArcCall.X;{$ifdef sectorpldebug}           If (ArcCall.YEnd-ArcCall.Y) = 0 then             begin               append(t);               writeln(t,'bug2');               close(t);               runerror(202);             end;{$endif sectorpldebug}           plx2 := (y-ArcCall.Y)*(ArcCall.XEnd-ArcCall.X)                   div (ArcCall.YEnd-ArcCall.Y)+ArcCall.X;           If plx1 > plx2 then             begin               plx1 := plx1 xor plx2;               plx2 := plx1 xor plx2;               plx1 := plx1 xor plx2;             end;{$ifdef sectorpldebug}           append(t);           writeln(t,'lines: ',plx1,' - ',plx2);           close(t);{$endif sectorpldebug}         End       { otherwise two points which have nothing to do with the sector }       Else exit     Else       { the arc is plotted at the right side, but not at the left side, }       { fill till the line between (ArcCall.X,ArcCall.Y) and            }       { (ArcCall.XStart,ArcCall.YStart)                                 }       Begin         If (y < ArcCall.Y) then           begin{$ifdef sectorpldebug}             If (ArcCall.YEnd-ArcCall.Y) = 0 then               begin                 append(t);                 writeln(t,'bug3');                 close(t);                 runerror(202);               end;{$endif sectorpldebug}             plx1 := (y-ArcCall.Y)*(ArcCall.XEnd-ArcCall.X)                     div (ArcCall.YEnd-ArcCall.Y)+ArcCall.X           end         else if (y > ArcCall.Y) then           begin{$ifdef sectorpldebug}             If (ArcCall.YStart-ArcCall.Y) = 0 then               begin                 append(t);                 writeln(t,'bug4');                 close(t);                 runerror(202);               end;{$endif sectorpldebug}             plx1 := (y-ArcCall.Y)*(ArcCall.XStart-ArcCall.X)                     div (ArcCall.YStart-ArcCall.Y)+ArcCall.X             end         else plx1 := ArcCall.X;         plx2 := x2;{$ifdef sectorpldebug}         append(t);         writeln(t,'right: ',plx1,' - ',plx2);         close(t);{$endif sectorpldebug}       End   Else     If (x2 = maxsmallint-1) Then       { the arc is plotted at the left side, but not at the rigth side.   }       { the right limit can be either the first or second line. Just take }       { the closest one, but watch out for division by zero!              }       Begin         If (y < ArcCall.Y) then           begin{$ifdef sectorpldebug}             If (ArcCall.YStart-ArcCall.Y) = 0 then               begin                 append(t);                 writeln(t,'bug5');                 close(t);                 runerror(202);               end;{$endif sectorpldebug}             plx2 := (y-ArcCall.Y)*(ArcCall.XStart-ArcCall.X)                     div (ArcCall.YStart-ArcCall.Y)+ArcCall.X           end         else if (y > ArcCall.Y) then           begin{$ifdef sectorpldebug}             If (ArcCall.YEnd-ArcCall.Y) = 0 then               begin                 append(t);                 writeln(t,'bug6');                 close(t);                 runerror(202);               end;{$endif sectorpldebug}             plx2 := (y-ArcCall.Y)*(ArcCall.XEnd-ArcCall.X)                     div (ArcCall.YEnd-ArcCall.Y)+ArcCall.X           end         else plx2 := ArcCall.X;         plx1 := x1;{$ifdef sectorpldebug}         append(t);         writeln(t,'left: ',plx1,' - ',plx2);         close(t);{$endif sectorpldebug}       End     Else       { the arc is plotted at both sides }       Begin         plx1 := x1;         plx2 := x2;{$ifdef sectorpldebug}         append(t);         writeln(t,'normal: ',plx1,' - ',plx2);         close(t);{$endif sectorpldebug}       End;   If plx2 > plx1 then     Begin{$ifdef sectorpldebug}       append(t);       Writeln(t,'drawing...');       close(t);{$endif sectorpldebug}       PatternLine(plx1,plx2,y);     end; end; procedure Sector(x, y: smallint; StAngle,EndAngle, XRadius, YRadius: Word);  begin     internalellipse(x,y,XRadius, YRadius, StAngle, EndAngle, {$ifdef fpc}@{$endif}SectorPL);     Line(ArcCall.XStart, ArcCall.YStart, x,y);     Line(x,y,ArcCall.Xend,ArcCall.YEnd);  end;   procedure SetFillStyle(Pattern : word; Color: word);   begin     { on invalid input, the current fill setting will be }     { unchanged.                                         }     if (Pattern > UserFill) or (Color > GetMaxColor) then      begin{$ifdef logging}           logln('invalid fillstyle parameters');{$endif logging}           _GraphResult := grError;           exit;      end;     FillSettings.Color := Color;     FillSettings.Pattern := Pattern;   end;  procedure SetFillPattern(Pattern: FillPatternType; Color: word);  {********************************************************}  { Changes the Current FillPattern to a user defined      }  { pattern and changes also the current fill color.       }  { The FillPattern is saved in the FillPattern array so   }  { it can still be used with SetFillStyle(UserFill,Color) }  {********************************************************}   var    i: smallint;   begin     if Color > GetMaxColor then       begin{$ifdef logging}            logln('invalid fillpattern parameters');{$endif logging}            _GraphResult := grError;            exit;       end;     FillSettings.Color := Color;     FillSettings.Pattern := UserFill;     { Save the pattern in the buffer }     For i:=1 to 8 do       FillPatternTable[UserFill][i] := Pattern[i];   end;  procedure Bar(x1,y1,x2,y2:smallint);  {********************************************************}  { Important notes for compatibility with BP:             }  {     - WriteMode is always CopyPut                      }  {     - No contour is drawn for the lines                }  {********************************************************}  var y               : smallint;      origcolor       : longint;      origlinesettings: Linesettingstype;      origwritemode   : smallint;   begin     origlinesettings:=lineinfo;     origcolor:=CurrentColor;     if y1>y2 then       begin          y:=y1;          y1:=y2;          y2:=y;       end;     { Always copy mode for Bars }     origwritemode := CurrentWriteMode;     CurrentWriteMode := CopyPut;     { All lines used are of this style }     Lineinfo.linestyle:=solidln;     Lineinfo.thickness:=normwidth;     case Fillsettings.pattern of     EmptyFill :       begin         Currentcolor:=CurrentBkColor;         for y:=y1 to y2 do           Hline(x1,x2,y);       end;     SolidFill :       begin         CurrentColor:=FillSettings.color;           for y:=y1 to y2 do              Hline(x1,x2,y);       end;     else      Begin        CurrentColor:=FillSettings.color;        for y:=y1 to y2 do          patternline(x1,x2,y);      end;    end;    CurrentColor:= Origcolor;    LineInfo := OrigLineSettings;    CurrentWriteMode := OrigWritemode;   end;procedure bar3D(x1, y1, x2, y2 : smallint;depth : word;top : boolean);var origwritemode : smallint; OldX, OldY : smallint;begin  origwritemode := CurrentWriteMode;  CurrentWriteMode := CopyPut;  Bar(x1,y1,x2,y2);  Rectangle(x1,y1,x2,y2);  { Current CP should not be updated in Bar3D }  { therefore save it and then restore it on  }  { exit.                                     }  OldX := CurrentX;  OldY := CurrentY;  if top then begin    Moveto(x1,y1);    Lineto(x1+depth,y1-depth);    Lineto(x2+depth,y1-depth);    Lineto(x2,y1);  end;  if Depth <> 0 then    Begin      Moveto(x2+depth,y1-depth);      Lineto(x2+depth,y2-depth);      Lineto(x2,y2);    end;  { restore CP }  CurrentX := OldX;  CurrentY := OldY;  CurrentWriteMode := origwritemode;end;{--------------------------------------------------------------------------}{                                                                          }{                       COLOR AND PALETTE ROUTINES                         }{                                                                          }{--------------------------------------------------------------------------}  procedure SetColor(Color: Word);   Begin     CurrentColor := Color;   end;  function GetColor: Word;   Begin     GetColor := CurrentColor;   end;  function GetBkColor: Word;   Begin     GetBkColor := CurrentBkColor;   end;  procedure SetBkColor(ColorNum: Word);  { Background color means background screen color in this case, and it is  }  { INDEPENDANT of the viewport settings, so we must clear the whole screen }  { with the color.                                                         }   var     ViewPort: ViewportType;   Begin     GetViewSettings(Viewport);{$ifdef logging}      logln('calling setviewport from setbkcolor');{$endif logging}     SetViewPort(0,0,MaxX,MaxY,FALSE);{$ifdef logging}      logln('calling setviewport from setbkcolor done');{$endif logging}     CurrentBkColor := ColorNum;     {ClearViewPort;}     if not DirectColor and (ColorNum<256) then      SetRGBPalette(0,          DefaultColors[ColorNum].Red,          DefaultColors[ColorNum].Green,          DefaultColors[ColorNum].Blue);     SetViewport(ViewPort.X1,Viewport.Y1,Viewport.X2,Viewport.Y2,Viewport.Clip);   end;  function GetMaxColor: word;  { Checked against TP VGA driver - CEC }   begin      GetMaxColor:=MaxColor-1; { based on an index of zero so subtract one }   end;   Procedure MoveRel(Dx, Dy: smallint);    Begin     CurrentX := CurrentX + Dx;     CurrentY := CurrentY + Dy;   end;   Procedure MoveTo(X,Y: smallint);  {********************************************************}  { Procedure MoveTo()                                     }  {--------------------------------------------------------}  { Moves the current pointer in VIEWPORT relative         }  { coordinates to the specified X,Y coordinate.           }  {********************************************************}    Begin     CurrentX := X;     CurrentY := Y;    end;function GraphErrorMsg(ErrorCode: smallint): string;Begin GraphErrorMsg:=''; case ErrorCode of  grOk,grFileNotFound,grInvalidDriver: exit;  grNoInitGraph: GraphErrorMsg:='Graphics driver not installed';  grNotDetected: GraphErrorMsg:='Graphics hardware not detected';  grNoLoadMem,grNoScanMem,grNoFloodMem: GraphErrorMsg := 'Not enough memory for graphics';  grNoFontMem: GraphErrorMsg := 'Not enough memory to load font';  grFontNotFound: GraphErrorMsg:= 'Font file not found';  grInvalidMode: GraphErrorMsg := 'Invalid graphics mode';  grError: GraphErrorMsg:='Graphics error';  grIoError: GraphErrorMsg:='Graphics I/O error';  grInvalidFont,grInvalidFontNum: GraphErrorMsg := 'Invalid font';  grInvalidVersion: GraphErrorMsg:='Invalid driver version'; end;end;  Function GetMaxX: smallint;  { Routine checked against VGA driver - CEC }   Begin     GetMaxX := MaxX;   end;  Function GetMaxY: smallint;  { Routine checked against VGA driver - CEC }   Begin    GetMaxY := MaxY;   end;Function GraphResult: smallint;Begin  GraphResult := _GraphResult;  _GraphResult := grOk;end;  Function GetX: smallint;   Begin     GetX := CurrentX;   end;  Function GetY: smallint;   Begin     GetY := CurrentY;   end;   Function GetDriverName: string;    begin      GetDriverName:=DriverName;    end;   procedure graphdefaults;   { PS: GraphDefaults does not ZERO the ArcCall structure }   { so a call to GetArcCoords will not change even the    }   { returned values even if GraphDefaults is called in    }   { between.                                              }    var     i: smallint;   begin     lineinfo.linestyle:=solidln;     lineinfo.thickness:=normwidth;     { reset line style pattern }     for i:=0 to 15 do       LinePatterns[i] := TRUE;     { By default, according to the TP prog's reference }     { the default pattern is solid, and the default    }     { color is the maximum color in the palette.       }     fillsettings.color:=GetMaxColor;     fillsettings.pattern:=solidfill;     { GraphDefaults resets the User Fill pattern to $ff }     { checked with VGA BGI driver - CEC                 }     for i:=1 to 8 do       FillPatternTable[UserFill][i] := $ff;     CurrentColor:=white;     ClipPixels := TRUE;     { Reset the viewport }     StartXViewPort := 0;     StartYViewPort := 0;     ViewWidth := MaxX;     ViewHeight := MaxY;     { Reset CP }     CurrentX := 0;     CurrentY := 0;     SetBkColor(Black);     { normal write mode }     CurrentWriteMode := CopyPut;     { Schriftart einstellen }     CurrentTextInfo.font := DefaultFont;     CurrentTextInfo.direction:=HorizDir;     CurrentTextInfo.charsize:=1;     CurrentTextInfo.horiz:=LeftText;     CurrentTextInfo.vert:=TopText;     XAspect:=10000; YAspect:=10000;   end;  procedure GetAspectRatio(var Xasp,Yasp : word);  begin    XAsp:=XAspect;    YAsp:=YAspect;  end;  procedure SetAspectRatio(Xasp, Yasp : word);  begin    Xaspect:= XAsp;    YAspect:= YAsp;  end;  procedure SetWriteMode(WriteMode : smallint);  { TP sets the writemodes according to the following scheme (JM) }   begin     Case writemode of       xorput, andput: CurrentWriteMode := XorPut;       notput, orput, copyput: CurrentWriteMode := CopyPut;     End;   end;  procedure GetFillSettings(var Fillinfo:Fillsettingstype);   begin     Fillinfo:=Fillsettings;   end;  procedure GetFillPattern(var FillPattern:FillPatternType);   begin     FillPattern:=FillpatternTable[UserFill];   end;  procedure DrawPoly(numpoints : word;var polypoints);      type            ppointtype = ^pointtype;        pt = array[0..16000] of pointtype;      var            i : longint;    begin         if numpoints < 2 then           begin             _GraphResult := grError;             exit;           end;         for i:=0 to numpoints-2 do           line(pt(polypoints)[i].x,                pt(polypoints)[i].y,                pt(polypoints)[i+1].x,                pt(polypoints)[i+1].y);    end;  procedure PieSlice(X,Y,stangle,endAngle:smallint;Radius: Word);  begin    Sector(x,y,stangle,endangle,radius,radius);  end;{$i fills.inc}{$i gtext.inc}  procedure internDetectGraph(var GraphDriver, GraphMode:smallint;    calledFromInitGraph: boolean);  var LoMode, HiMode: smallint;      CpyMode: smallint;      CpyDriver: smallint;  begin    HiMode := -1;    LoMode := -1;{$ifndef nonewmodes}    if not calledFromInitGraph or       (graphDriver < lowNewDriver) or       (graphDriver > highNewDriver) then      begin        { Search lowest supported bitDepth }        graphdriver := D1bit;        while (graphDriver <= highNewDriver) and              (hiMode = -1) do          begin            getModeRange(graphDriver,loMode,hiMode);            inc(graphDriver);          end;        dec(graphdriver);        if hiMode = -1 then          begin            _GraphResult := grNotDetected;            exit;          end;        CpyMode := 0;        repeat           GetModeRange(GraphDriver,LoMode,HiMode);           { save the highest mode possible...}           {$ifdef logging}           logln('Found driver '+strf(graphdriver)+' with modes '+                  strf(lomode)+' - '+strf(himode));           {$endif logging}           if HiMode <> -1 then             begin               CpyMode:=HiMode;               CpyDriver:=GraphDriver;             end;           { go to next driver if it exists...}           Inc(graphDriver);        until (graphDriver > highNewDriver);      end    else      begin        cpyMode := 0;        getModeRange(graphDriver,loMode,hiMode);        if hiMode <> -1 then          begin            cpyDriver := graphDriver;            cpyMode := hiMode;          end;      end;    if cpyMode = 0 then      begin        _GraphResult := grNotDetected;        exit;      end;{$else nonewmodes}    { We start at VGA }    GraphDriver := VGA;    CpyMode := 0;    { search all possible graphic drivers in ascending order...}    { usually the new driver numbers indicate newest hardware...}    { Internal driver numbers start at VGA=9 }    repeat       GetModeRange(GraphDriver,LoMode,HiMode);       { save the highest mode possible...}       {$ifdef logging}       logln('Found driver '+strf(graphdriver)+' with modes '+              strf(lomode)+' - '+strf(himode));       {$endif logging}       if HiMode = -1 then break;       CpyMode:=HiMode;       CpyDriver:=GraphDriver;       { go to next driver if it exists...}       Inc(GraphDriver);    until (CpyMode=-1);    { If this is equal to -1 then no graph mode possible...}    if CpyMode = -1 then      begin        _GraphResult := grNotDetected;        exit;      end;{$endif nonewmodes}    _GraphResult := grOK;    GraphDriver := CpyDriver;    GraphMode := CpyMode;  end;  procedure detectGraph(var GraphDriver: smallint; var GraphMode:smallint);  begin    internDetectGraph(graphDriver,graphMode,false);  end;  procedure InitGraph(var GraphDriver:smallint;var GraphMode:smallint;    const PathToDriver:String);  const    {$IFDEF Linux}    dirchar = '/';    {$ELSE}    dirchar = '\';    {$ENDIF}  begin    InitVars;    { path to the fonts (where they will be searched)...}    bgipath:=PathToDriver;    if (Length(bgipath) > 0) and (bgipath[length(bgipath)]<>dirchar) then      bgipath:=bgipath+dirchar;    if not assigned(SaveVideoState) then      RunError(216);    DriverName:=InternalDriverName;   { DOS Graphics driver }    if (Graphdriver=Detect){$ifndef nonewmodes}       or (GraphMode = detectMode){$endif}       then      begin        internDetectGraph(GraphDriver,GraphMode,true);        If _GraphResult = grNotDetected then Exit;        { _GraphResult is now already set to grOK by DetectGraph }        IntCurrentDriver := GraphDriver;        if (graphDriver >= lowNewDriver) and           (graphDriver <= highNewDriver) then          IntCurrentNewDriver := GraphDriver        else IntCurrentNewDriver := -1;        { Actually set the graph mode...}        if firstCallOfInitgraph then          begin        SaveVideoState;            firstCallOfInitgraph := false;          end;        SetGraphMode(GraphMode);      end    else      begin        { Search if that graphics modec actually exists...}        if SearchMode(GraphDriver,GraphMode) = nil then          begin            _GraphResult := grInvalidMode;            exit;         end        else         begin           _GraphResult := grOK;           IntCurrentDriver := GraphDriver;           if (graphDriver >= lowNewDriver) and              (graphDriver <= highNewDriver) then             IntCurrentNewDriver := GraphDriver           else IntCurrentNewDriver := -1;           if firstCallOfInitgraph then             begin           SaveVideoState;               firstCallOfInitgraph := false;             end;           SetGraphMode(GraphMode);         end;      end;  end; procedure SetDirectVideo(DirectAccess: boolean);  begin    DirectVideo := DirectAccess;  end; function GetDirectVideo: boolean;  begin    GetDirectVideo := DirectVideo;  end; procedure GraphExitProc; {$ifndef fpc} far; {$endif fpc} { deallocates all memory allocated by the graph unit }  var    list: PModeInfo;    tmp : PModeInfo;    c: graph_int;  begin   { restore old exitproc! }   exitproc := exitsave;   if IsGraphMode and ((errorcode<>0) or (erroraddr<>nil)) then     CloseGraph;   { release memory allocated for fonts }   for c := 1 to installedfonts do     with fonts[c] Do     If assigned(instr) Then       Freemem(instr,instrlength);   { release memory allocated for modelist }   list := ModeList;   while assigned(list) do     begin       tmp := list;       list:=list^.next;       dispose(tmp);     end;{$ifndef nonewmodes}   for c := lowNewDriver to highNewDriver do     begin       list := newModeList.modeinfo[c];       while assigned(list) do         begin           tmp := list;           list:=list^.next;           dispose(tmp);         end;     end;{$endif nonewmodes}{$IFDEF DPMI}  { We had copied the buffer of mode information }  { and allocated it dynamically... now free it  }  { Warning: if GetVESAInfo returned false, this buffer is not allocated! (JM)}   If hasVesa then     Dispose(VESAInfo.ModeList);{$ENDIF}  end;procedure InitializeGraph;begin{$ifdef logging} assign(debuglog,'grlog.txt'); rewrite(debuglog); close(debuglog);{$endif logging} isgraphmode := false; ModeList := nil;{$ifndef nonewmodes} fillChar(newModeList.modeinfo,sizeof(newModeList.modeinfo),#0); { lo and hi modenumber are -1 currently (no modes supported) } fillChar(newModeList.loHiModeNr,sizeof(newModeList.loHiModeNr),#255);{$endif nonewmodes} SaveVideoState := nil; RestoreVideoState := nil;{$ifdef oldfont}{$ifdef go32v2} LoadFont8x8;{$endif go32v2}{$endif oldfont} { This must be called at startup... because GetGraphMode may } { be called even when not in graph mode.                     }{$ifdef logging} LogLn('Calling QueryAdapterInfo...');{$endif logging} QueryAdapterInfo; { Install standard fonts } { This is done BEFORE startup... } InstalledFonts := 0; InstallUserFont('TRIP'); InstallUserFont('LITT'); InstallUserFont('SANS'); InstallUserFont('GOTH'); InstallUserFont('SCRI'); InstallUserFont('SIMP'); InstallUserFont('TSCR'); InstallUserFont('LCOM'); InstallUserFont('EURO'); InstallUserFont('BOLD'); { This installs an exit procedure which cleans up the mode list...} ExitSave := ExitProc; ExitProc := @GraphExitProc;{$ifdef win32} charmessagehandler:=nil;{$endif win32}end;{  $Log$  Revision 1.6  2001-04-14 16:06:03  jonas    * fixed problem with new modes system (reported by Vladimir Ravodin) (merged)  Revision 1.5  2000/12/07 17:19:47  jonas    * new constant handling: from now on, hex constants >$7fffffff are      parsed as unsigned constants (otherwise, $80000000 got sign extended      and became $ffffffff80000000), all constants in the longint range      become longints, all constants >$7fffffff and <=cardinal($ffffffff)      are cardinals and the rest are int64's.    * added lots of longint typecast to prevent range check errors in the      compiler and rtl    * type casts of symbolic ordinal constants are now preserved    * fixed bug where the original resulttype wasn't restored correctly      after doing a 64bit rangecheck  Revision 1.4  2000/08/12 12:27:13  jonas    + setallpalette hook    + setallpalette implemented for standard vga and VESA 2.0+  Revision 1.3  2000/08/05 18:34:47  peter    * merged setvideostate patch  Revision 1.2  2000/07/13 11:33:46  michael  + removed logs}
 |