| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274 | program tclock;{$MODE OBJFPC}uses  ncurses, sysutils;{$linklib c}procedure setlocale(cat : integer; p : pchar); cdecl; external 'c';const  LC_ALL = 6;const  ASPECT = 2.2;  _2PI = 2.0 * PI;function sign(_x: Integer): Integer;begin  if _x < 0 then    sign := -1  else    sign := 1end;function A2X(angle,radius: Double): Integer; inline;begin  A2X := round(ASPECT * radius * sin(angle))end;function A2Y(angle,radius: Double): Integer; inline;begin  A2Y := round(radius * cos(angle))end;type  PRchar = ^TRchar;  TRchar = record    ry,rx: Smallint;    rch: chtype;  end;procedure restore( rest: PRchar );var  i: Longint = 0;begin  while rest[i].rch <> 0 do  begin    with rest[i] do      mvaddch(ry, rx, rch);    Inc(i);  end;  freemem(rest)end;(* Draw a diagonal(arbitrary) line using Bresenham's alogrithm. *)procedure dline(from_y, from_x, end_y,  end_x: Smallint; ch: chtype; var rest: PRchar);var  dx, dy: Smallint;  ax, ay: Smallint;  sx, sy: Smallint;  x, y, d, i: Smallint;begin  dx := end_x - from_x;  dy := end_y - from_y;  ax := abs(dx * 2);  ay := abs(dy * 2);  sx := sign(dx);  sy := sign(dy);  x := from_x;  y := from_y;  i := 0;  if (ax > ay) then  begin    getmem(rest, sizeof(TRchar)*(abs(dx)+3));    d := ay - (ax DIV 2);    while true do    begin      move(y, x);      with rest[i] do      begin        rch := inch;        ry := y;        rx := x;        Inc(i)      end;      addch(ch);      if (x = end_x) then      begin        rest[i].rch := 0;        exit;      end;      if (d >= 0) then      begin        y += sy;        d -= ax;      end;      x += sx;      d += ay;    end  end  else  begin    getmem(rest, sizeof(TRchar)*(abs(dy)+3));    d := ax - (ay DIV 2);    while true do    begin      move(y, x);      with rest[i] do      begin        rch := inch;        ry := y;        rx := x;        Inc(i)      end;      addch(ch);      if (y = end_y) then      begin        rest[i].rch := 0;        exit;      end;      if (d >= 0) then      begin        x += sx;        d -= ay;      end;      y += sy;      d += ax;    end  endend;var  cx, cy: Integer;  cr, sradius, mradius, hradius: Double;procedure clockinit;const  title1 = 'Free pascal';  title2 = 'ncurses clock';  title3 = 'Press F10 or q to exit';var  i: Integer;  vstr, tstr: AnsiString;  angle: Double;begin  cx := (COLS - 1) DIV 2;  cy := LINES DIV 2;  if (cx / ASPECT < cy) then  cr := cx / ASPECT    else  cr := cy;  sradius := (8 * cr) / 9;  mradius := (3 * cr) / 4;  hradius := cr / 2;  for i := 1 to 24 do  begin    angle := i * _2PI / 24.0;    if (i MOD 2) = 0 then    begin      Str (i DIV 2, tstr);      attron(A_BOLD OR COLOR_PAIR(5));      mvaddstr(cy - A2Y(angle, sradius), cx + A2X(angle, sradius), @tstr[1]);      attroff(A_BOLD OR COLOR_PAIR(5));    end    else    begin      attron(COLOR_PAIR(1));      mvaddch(cy - A2Y(angle, sradius), cx + A2X(angle, sradius), chtype('.'));      attroff(COLOR_PAIR(1));    end  end;  vstr := curses_version;  attron(A_DIM OR COLOR_PAIR(2));  mvhline(cy , cx - round(sradius * ASPECT) + 1, ACS_HLINE,  round(sradius * ASPECT) * 2 - 1);  mvvline(cy - round(sradius) + 1, cx , ACS_VLINE,  round(sradius) * 2 - 1);  attroff(A_DIM OR COLOR_PAIR(1));  attron(COLOR_PAIR(3));  mvaddstr(cy - 5, cx - Length(title1) DIV 2, title1);  mvaddstr(cy - 4, cx - Length(title2) DIV 2, title2);  mvaddstr(cy - 3, cx - Length(vstr) DIV 2, PChar(vstr));  attroff(COLOR_PAIR(3));  attron(A_UNDERLINE);  mvaddstr(cy + 2, cx - Length(title3) DIV 2, title3);  attroff(A_UNDERLINE);end;var  angle: Double;  ch: chtype = 0;  Hour, Min, Sec, Msec: Word;  Hrest, Mrest, Srest: PRchar;  timestr: AnsiString;  my_bg: Smallint = COLOR_BLACK;begin  setlocale(LC_ALL, '');  try    initscr();    noecho();    cbreak();    halfdelay(10);    keypad(stdscr, TRUE);    curs_set(0);    if (has_colors()) then    begin      start_color();      if (use_default_colors() = OK) then        my_bg := -1;      init_pair(1, COLOR_YELLOW, my_bg);      init_pair(2, COLOR_RED, my_bg);      init_pair(3, COLOR_GREEN, my_bg);      init_pair(4, COLOR_CYAN, my_bg);      init_pair(5, COLOR_YELLOW, COLOR_BLACK) ;    end;    clockinit;    repeat      if (ch = KEY_RESIZE) then      begin        flash();        erase();        wrefresh(curscr);        clockinit;      end;      decodeTime(Time, Hour, Min, Sec, Msec);      Hour := Hour MOD 12;      timestr := DateTimeToStr(Now);      mvaddstr(cy + round(sradius) - 4, cx - Length(timestr) DIV 2, PChar(timestr));      angle := Hour * _2PI / 12;      dline(cy, cx, cy - A2Y(angle, hradius), cx + A2X(angle, hradius), chtype('*'),Hrest);      angle := Min * _2PI / 60;      dline(cy, cx, cy - A2Y(angle, mradius), cx + A2X(angle, mradius), chtype('*'),Mrest);      angle := Sec * _2PI / 60;      dline(cy, cx, cy - A2Y(angle, sradius), cx + A2X(angle, sradius), chtype('.'),Srest);      ch := getch();      restore(Srest);      restore(Mrest);      restore(Hrest);    until (ch = chtype('q')) OR (ch = KEY_F(10));  finally    curs_set(1);    endwin();  end;end.
 |