Subversion Repositories Kolibri OS

Rev

Rev 9187 | Rev 9208 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | Download | RSS feed

  1. (*
  2.     Copyright 2021 Anton Krotov
  3.  
  4.     This file is part of CEdit.
  5.  
  6.     CEdit is free software: you can redistribute it and/or modify
  7.     it under the terms of the GNU General Public License as published by
  8.     the Free Software Foundation, either version 3 of the License, or
  9.     (at your option) any later version.
  10.  
  11.     CEdit is distributed in the hope that it will be useful,
  12.     but WITHOUT ANY WARRANTY; without even the implied warranty of
  13.     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14.     GNU General Public License for more details.
  15.  
  16.     You should have received a copy of the GNU General Public License
  17.     along with CEdit. If not, see <http://www.gnu.org/licenses/>.
  18. *)
  19.  
  20. MODULE Menu;
  21.  
  22. IMPORT
  23.     SYSTEM, G := Graph, List, K := KolibriOS, KOSAPI;
  24.  
  25. CONST
  26.     fontHeight = 22;
  27.     fontWidth = 8;
  28.  
  29.     MainMenuHeight* = K.fontHeight + 7;
  30.  
  31.     RIGHT = 16;
  32.     LEFT = 16;
  33.     TOP = 1;
  34.  
  35.     maxLEVEL = 1;
  36.  
  37.     backColor = 0F0F0F0H;
  38.     foreColor = 0;
  39.     selBackColor = 091C9F7H;
  40.     selForeColor = 0;
  41.     disBackColor = backColor;
  42.     disForeColor = 808080H;
  43.     disSelBackColor = 0E4E4E4H;
  44.     disSelForeColor = disForeColor;
  45.  
  46.  
  47. TYPE
  48.  
  49.         tMainItem* = POINTER TO descMainItem;
  50.  
  51.     tMain* = POINTER TO RECORD (List.tList)
  52.         id: INTEGER
  53.     END;
  54.  
  55.     tMenu* = POINTER TO RECORD
  56.         tid*: INTEGER;
  57.         active*, keyboard: BOOLEAN;
  58.         parent*, child: tMenu;
  59.         mainID: INTEGER;
  60.         winX, winY, width*, height*: INTEGER;
  61.         selItem, cliItem: INTEGER;
  62.  
  63.         font: G.tFont;
  64.         canvas: G.tCanvas;
  65.  
  66.         items: List.tList;
  67.         click: PROCEDURE (menu: tMenu; id: INTEGER);
  68.         key: PROCEDURE (menu: tMenu; key: INTEGER): BOOLEAN
  69.     END;
  70.  
  71.     tItem* = POINTER TO RECORD (List.tItem)
  72.         id*, check: INTEGER;
  73.         text: ARRAY 32 OF WCHAR;
  74.         enabled, delim: BOOLEAN;
  75.         child: tMenu
  76.     END;
  77.  
  78.     descMainItem = RECORD (List.tItem)
  79.         id*, x: INTEGER;
  80.         text: ARRAY 32 OF WCHAR;
  81.         menu*: tMenu;
  82.         main: tMain
  83.     END;
  84.  
  85.     tClick = PROCEDURE (menu: tMenu; id: INTEGER);
  86.     tKey = PROCEDURE (menu: tMenu; key: INTEGER): BOOLEAN;
  87.     tProc = PROCEDURE;
  88.  
  89. VAR
  90.     stack: ARRAY maxLEVEL + 1, 250000 OF INTEGER;
  91.     TIDs: ARRAY maxLEVEL + 1 OF INTEGER;
  92.     resetTimer: tProc;
  93.     _open: PROCEDURE (m: tMenu; x, y: INTEGER);
  94. (*
  95.     backColor, foreColor, selBackColor, selForeColor,
  96.     disBackColor, disForeColor, disSelBackColor, disSelForeColor: INTEGER;
  97. *)
  98.  
  99. PROCEDURE AddMainItem* (main: tMain; text: ARRAY OF WCHAR; menu: tMenu);
  100. VAR
  101.         item, prev: tMainItem;
  102. BEGIN
  103.         NEW(item);
  104.         item.id := main.id + main.count;
  105.         COPY(text, item.text);
  106.         item.menu := menu;
  107.         item.main := main;
  108.         menu.mainID := item.id;
  109.         List.append(main, item);
  110.         prev := item.prev(tMainItem);
  111.         IF prev # NIL THEN
  112.                 item.x := prev.x + LENGTH(prev.text)*fontWidth + 9
  113.         ELSE
  114.                 item.x := 0
  115.         END
  116. END AddMainItem;
  117.  
  118.  
  119. PROCEDURE CreateMain* (id: INTEGER): tMain;
  120. VAR
  121.         res: tMain;
  122.         list: List.tList;
  123. BEGIN
  124.         NEW(res);
  125.         res.id := id;
  126.         list := List.create(res)
  127.         RETURN res
  128. END CreateMain;
  129.  
  130.  
  131. PROCEDURE drawMainItem (item: tMainItem);
  132. VAR
  133.     menuColor, textColor, n: INTEGER;
  134. BEGIN
  135.     IF item.menu.tid # 0 THEN
  136.         menuColor := K.textColor;
  137.         textColor := K.winColor
  138.     ELSE
  139.         menuColor := K.winColor;
  140.         textColor := K.textColor
  141.     END;
  142.     n := LENGTH(item.text);
  143.     K.DrawRect(item.x, 0, n*fontWidth + 2, MainMenuHeight, menuColor);
  144.     K.CreateButton(item.id + ORD({30}), item.x, 0, n*fontWidth + 2, MainMenuHeight, K.btnColor, "");
  145.     K.DrawText(item.x + 1, (MainMenuHeight - K.fontHeight) DIV 2 + 1, textColor, item.text)
  146. END drawMainItem;
  147.  
  148.  
  149. PROCEDURE DrawMain* (main: tMain);
  150. VAR
  151.         item: List.tItem;
  152. BEGIN
  153.         item := main.first;
  154.         WHILE item # NIL DO
  155.                 drawMainItem(item(tMainItem));
  156.                 item := item.next
  157.         END
  158. END DrawMain;
  159.  
  160.  
  161. PROCEDURE getMainID (m: tMenu): INTEGER;
  162. BEGIN
  163.         WHILE m.parent # NIL DO
  164.                 m := m.parent
  165.         END
  166.         RETURN m.mainID
  167. END getMainID;
  168.  
  169.  
  170. PROCEDURE ClickMain* (main: tMain; btn: INTEGER): tMenu;
  171. VAR
  172.         item: List.tItem;
  173.         res: tMenu;
  174. BEGIN
  175.         item := List.getItem(main, btn - main.id);
  176.         IF item # NIL THEN
  177.                 res := item(tMainItem).menu
  178.         ELSE
  179.                 res := NIL
  180.         END
  181.         RETURN res
  182. END ClickMain;
  183.  
  184.  
  185. PROCEDURE isSender* (tid: INTEGER): BOOLEAN;
  186. VAR
  187.         i: INTEGER;
  188. BEGIN
  189.         i := 0;
  190.         WHILE (i <= maxLEVEL) & (TIDs[i] # tid) DO
  191.                 INC(i)
  192.         END
  193.         RETURN i <= maxLEVEL
  194. END isSender;
  195.  
  196.  
  197. PROCEDURE exit (m: tMenu);
  198. BEGIN
  199.     m.active := FALSE;
  200.         resetTimer;
  201.     m.tid := 0;
  202.     K.Exit
  203. END exit;
  204.  
  205.  
  206. PROCEDURE escape (m: tMenu);
  207. BEGIN
  208.     m.active := FALSE;
  209.     IF m.parent = NIL THEN
  210.         resetTimer
  211.     END;
  212.     m.tid := 0;
  213.     K.Exit
  214. END escape;
  215.  
  216.  
  217. PROCEDURE repaint (m: tMenu);
  218. VAR
  219.     y, i, X, Y1, Y2: INTEGER;
  220.     item: tItem;
  221.     BkColor, TextColor: INTEGER;
  222.     canvas: G.tCanvas;
  223.  
  224. BEGIN
  225. (*
  226.     backColor := K.winColor;
  227.     foreColor := K.textColor;
  228.     selBackColor := K.btnColor;
  229.     selForeColor := K.btnTextColor;
  230.  
  231.     disBackColor := backColor;
  232.     disForeColor := K.darkColor;
  233.     disSelBackColor := K.lightColor;
  234.     disSelForeColor := disForeColor;
  235. *)
  236.     canvas := m.canvas;
  237.     G.SetColor(canvas, backColor);
  238.     G.clear(canvas);
  239.     G.SetColor(canvas, foreColor);
  240.     G.Rect(canvas, 0, 0, m.width, m.height);
  241.     y := TOP;
  242.     i := 0;
  243.     item := m.items.first(tItem);
  244.     WHILE item # NIL DO
  245.         IF item.enabled THEN
  246.             IF i # m.selItem THEN
  247.                 BkColor := backColor;
  248.                 TextColor := foreColor
  249.             ELSE
  250.                 BkColor := selBackColor;
  251.                 TextColor := selForeColor
  252.             END
  253.         ELSE
  254.             IF i # m.selItem THEN
  255.                 BkColor := disBackColor;
  256.                 TextColor := disForeColor
  257.             ELSE
  258.                 BkColor := disSelBackColor;
  259.                 TextColor := disSelForeColor
  260.             END
  261.         END;
  262.         G.SetColor(canvas, BkColor);
  263.         G.FillRect(canvas, 1, y, m.width - 1, y + fontHeight - 4);
  264.         G.SetTextColor(canvas, TextColor);
  265.         G.SetBkColor(canvas, BkColor);
  266.         G.TextOut2(canvas, LEFT, y + (fontHeight - 16) DIV 2 - 2, item.text, LENGTH(item.text));
  267.  
  268.         G.SetColor(canvas, TextColor);
  269.         IF item.check = 1 THEN
  270.             G.DLine(canvas, 4, 7, y + (fontHeight - 16) DIV 2 + 5, -1);
  271.             G.DLine(canvas, 4, 7, y + (fontHeight - 16) DIV 2 + 6, -1);
  272.             G.DLine(canvas, 7, 12, y + (fontHeight - 16) DIV 2 + 8, 1);
  273.             G.DLine(canvas, 7, 12, y + (fontHeight - 16) DIV 2 + 9, 1);
  274.         ELSIF item.check = 2 THEN
  275.             G.FillRect(canvas, 6, y + fontHeight DIV 2 - 4, 10, y + fontHeight DIV 2)
  276.         END;
  277.  
  278.         IF item.child # NIL THEN
  279.             X := m.width - 9;
  280.             Y1 := y + (fontHeight - 16) DIV 2 + 2;
  281.             Y2 := Y1 + 8;
  282.                 G.Triangle(canvas, X, Y1, X, Y2, G.triRight)
  283.         END;
  284.  
  285.         INC(y, fontHeight);
  286.         IF item.delim THEN
  287.             G.SetColor(canvas, foreColor);
  288.             G.HLine(canvas, y - 2, 1, m.width - 1)
  289.         END;
  290.         INC(i);
  291.         item := item.next(tItem)
  292.     END;
  293.     G.DrawCanvas(canvas, 0, 0)
  294. END repaint;
  295.  
  296.  
  297. PROCEDURE draw_window (m: tMenu);
  298. BEGIN
  299.     K.BeginDraw;
  300.     K.CreateWindow(m.winX, m.winY, m.width, m.height, 0, 61H, 0, 1, "");
  301.     repaint(m);
  302.     K.EndDraw
  303. END draw_window;
  304.  
  305.  
  306. PROCEDURE mouse (m: tMenu; VAR x, y: INTEGER);
  307. VAR
  308.     mouseX, mouseY: INTEGER;
  309. BEGIN
  310.     K.MousePos(mouseX, mouseY);
  311.     x := mouseX - m.winX;
  312.     y := mouseY - m.winY;
  313. END mouse;
  314.  
  315.  
  316. PROCEDURE close* (m: tMenu);
  317. VAR
  318.         temp: INTEGER;
  319. BEGIN
  320.     IF (m # NIL) & (m.tid # 0) THEN
  321.         IF m.child # NIL THEN
  322.                 close(m.child);
  323.                 m.child := NIL
  324.         END;
  325.         temp := m.tid;
  326.         m.tid := 0;
  327.         K.ExitID(temp);
  328.         m.active := FALSE
  329.     END
  330. END close;
  331.  
  332.  
  333. PROCEDURE click (m: tMenu; i: INTEGER);
  334. VAR
  335.     item: List.tItem;
  336.     p: tMenu;
  337.     id: INTEGER;
  338. BEGIN
  339.         id := -1;
  340.         IF i < 0 THEN
  341.                 id := i
  342.         ELSE
  343.             item := List.getItem(m.items, i);
  344.             IF (item # NIL) & item(tItem).enabled & (item(tItem).child = NIL) THEN
  345.                 id := item(tItem).id
  346.             END
  347.     END;
  348.     IF id # -1 THEN
  349.             m.click(m, id);
  350.             p := m.parent;
  351.                 WHILE p # NIL DO
  352.                         p.child := NIL;
  353.                         close(p);
  354.                         p := p.parent
  355.                 END;
  356.                 exit(m)
  357.         END
  358. END click;
  359.  
  360.  
  361. PROCEDURE opened* (m: tMenu): BOOLEAN;
  362.     RETURN m.tid # 0
  363. END opened;
  364.  
  365.  
  366. PROCEDURE isActive (m: tMenu): BOOLEAN;
  367.         RETURN (m # NIL) & ((m.tid # 0) & m.active OR isActive(m.child))
  368. END isActive;
  369.  
  370.  
  371. PROCEDURE closeChild (m: tMenu);
  372. BEGIN
  373.         IF m.child # NIL THEN
  374.                 close(m.child);
  375.                 m.child := NIL
  376.         END
  377. END closeChild;
  378.  
  379.  
  380. PROCEDURE submenu (m: tMenu; keyboard: BOOLEAN): BOOLEAN;
  381. VAR
  382.         item: List.tItem;
  383.         res: BOOLEAN;
  384. BEGIN
  385.         res := FALSE;
  386.     item := List.getItem(m.items, m.selItem);
  387.     IF (item # NIL) & item(tItem).enabled & (item(tItem).child # NIL) THEN
  388.         res := TRUE;
  389.         IF ~opened(item(tItem).child) THEN
  390.                 closeChild(m);
  391.                 item(tItem).child.keyboard := keyboard;
  392.                 _open(item(tItem).child, m.winX + m.width - 2, m.winY + m.selItem*fontHeight);
  393.                 m.child := item(tItem).child;
  394.         END
  395.     ELSE
  396.         closeChild(m)
  397.     END
  398.     RETURN res
  399. END submenu;
  400.  
  401.  
  402. PROCEDURE [stdcall] window (m: tMenu);
  403. VAR
  404.     x, y: INTEGER;
  405.     key, temp: INTEGER;
  406.     msState: SET;
  407.     shift, ctrl: BOOLEAN;
  408. BEGIN
  409.     m.selItem := ORD(m.keyboard) - 1;
  410.     m.cliItem := -1;
  411.     m.keyboard := FALSE;
  412.     K.SetEventsMask({0, 1, 5});
  413.     WHILE TRUE DO
  414.         CASE K.WaitForEvent() OF
  415.         |1:
  416.             draw_window(m)
  417.         |2:
  418.                 K.getKBState(shift, ctrl);
  419.             key := K.GetKey();
  420.             IF ~shift & ~ ctrl THEN
  421.                     IF key DIV 65536 = 72 THEN
  422.                         DEC(m.selItem);
  423.                         IF m.selItem < 0 THEN
  424.                             m.selItem := m.items.count - 1
  425.                         END
  426.                     ELSIF key DIV 65536 = 80 THEN
  427.                         INC(m.selItem);
  428.                         IF m.selItem >= m.items.count THEN
  429.                             m.selItem := 0
  430.                         END
  431.                     ELSIF key DIV 65536 = 28 THEN
  432.                         IF m.selItem >= 0 THEN
  433.                             click(m, m.selItem)
  434.                         END;
  435.                         m.cliItem := -1
  436.                     ELSIF key DIV 65536 = 77 THEN
  437.                         IF ~submenu(m, TRUE) THEN
  438.                                 click(m, -(getMainID(m) + 1))
  439.                         END;
  440.                         m.cliItem := -1
  441.                     ELSIF key DIV 65536 = 75 THEN
  442.                         IF m.parent # NIL THEN
  443.                                 escape(m)
  444.                         ELSE
  445.                                 click(m, -(getMainID(m) - 1))
  446.                         END;
  447.                         m.cliItem := -1
  448.                     ELSIF key DIV 65536 = 1 THEN
  449.                         escape(m)
  450.                     ELSE
  451.                         IF m.key(m, key) THEN
  452.                                 IF m.parent # NIL THEN
  453.                                         temp := m.parent.tid;
  454.                                         m.parent.tid := 0;
  455.                                         K.ExitID(temp)
  456.                                 END;
  457.                             exit(m)
  458.                         END
  459.                 END
  460.             ELSE
  461.                 IF m.key(m, key) THEN
  462.                         IF m.parent # NIL THEN
  463.                                 temp := m.parent.tid;
  464.                                 m.parent.tid := 0;
  465.                                 K.ExitID(temp)
  466.                         END;
  467.                     exit(m)
  468.                 END
  469.             END;
  470.             repaint(m)
  471.         |6:
  472.             msState := K.MouseState();
  473.             mouse(m, x, y);
  474.             IF (0 <= x) & (x < m.width) & (0 <= y) & (y < m.height) THEN
  475.                 m.active := TRUE;
  476.                 m.selItem := (y - TOP) DIV fontHeight;
  477.                 IF 8 IN msState THEN
  478.                     m.cliItem := (y - TOP) DIV fontHeight
  479.                 END;
  480.                 IF 16 IN msState THEN
  481.                     IF m.cliItem = m.selItem THEN
  482.                         click(m, m.cliItem)
  483.                     END;
  484.                     m.cliItem := -1
  485.                 END
  486.             ELSE
  487.                 m.active := FALSE;
  488.                 m.cliItem := -1;
  489.                 IF ({8, 9, 10, 16} * msState # {}) & ~isActive(m.child) THEN
  490.                     exit(m)
  491.                 END
  492.             END;
  493.             repaint(m);
  494.             IF submenu(m, FALSE) THEN END
  495.         END
  496.     END
  497. END window;
  498.  
  499.  
  500. PROCEDURE level (m: tMenu): INTEGER;
  501. VAR
  502.         res: INTEGER;
  503. BEGIN
  504.         res := 0;
  505.         WHILE m.parent # NIL DO
  506.                 INC(res);
  507.                 m := m.parent
  508.         END
  509.         RETURN res
  510. END level;
  511.  
  512.  
  513. PROCEDURE open* (m: tMenu; x, y: INTEGER);
  514. VAR
  515.         L: INTEGER;
  516. BEGIN
  517.     IF m.tid = 0 THEN
  518.         L := level(m);
  519.         IF KOSAPI.sysfunc3(18, 21, TIDs[L]) = 0 THEN
  520.                 m.winX := x;
  521.                 m.winY := y;
  522.                 SYSTEM.PUT(SYSTEM.ADR(stack[L][LEN(stack[0]) - 1]), m);
  523.             m.tid := K.CreateThread(SYSTEM.ADR(window), stack[L]);
  524.                 TIDs[L] := m.tid
  525.         END
  526.     END
  527. END open;
  528.  
  529.  
  530. PROCEDURE AddMenuItem* (items: List.tList; id: INTEGER; s: ARRAY OF WCHAR);
  531. VAR
  532.     item: tItem;
  533. BEGIN
  534.     NEW(item);
  535.     item.id := id;
  536.     item.text := s;
  537.     item.enabled := TRUE;
  538.     item.delim := FALSE;
  539.     item.child := NIL;
  540.     List.append(items, item);
  541. END AddMenuItem;
  542.  
  543.  
  544. PROCEDURE delimiter* (items: List.tList);
  545. BEGIN
  546.     items.last(tItem).delim := TRUE
  547. END delimiter;
  548.  
  549.  
  550. PROCEDURE child* (items: List.tList; menu: tMenu);
  551. BEGIN
  552.     items.last(tItem).child := menu
  553. END child;
  554.  
  555.  
  556. PROCEDURE getItem (m: tMenu; id: INTEGER): tItem;
  557. VAR
  558.     item: tItem;
  559. BEGIN
  560.     item := m.items.first(tItem);
  561.     WHILE (item # NIL) & (item.id # id) DO
  562.         item := item.next(tItem)
  563.     END
  564.     RETURN item
  565. END getItem;
  566.  
  567.  
  568. PROCEDURE setEnabled* (m: tMenu; id: INTEGER; value: BOOLEAN);
  569. VAR
  570.     item: tItem;
  571. BEGIN
  572.     item := getItem(m, id);
  573.     IF item # NIL THEN
  574.         item.enabled := value
  575.     END
  576. END setEnabled;
  577.  
  578.  
  579. PROCEDURE check* (m: tMenu; id: INTEGER; value: BOOLEAN);
  580. VAR
  581.     item: tItem;
  582. BEGIN
  583.     item := getItem(m, id);
  584.     IF item # NIL THEN
  585.         item.check := ORD(value)
  586.     END
  587. END check;
  588.  
  589.  
  590. PROCEDURE option* (m: tMenu; id: INTEGER; value: BOOLEAN);
  591. VAR
  592.     item: tItem;
  593. BEGIN
  594.     item := getItem(m, id);
  595.     IF item # NIL THEN
  596.         item.check := ORD(value)*2
  597.     END
  598. END option;
  599.  
  600.  
  601. PROCEDURE isEnabled* (m: tMenu; id: INTEGER): BOOLEAN;
  602. VAR
  603.     item: tItem;
  604. BEGIN
  605.     item := getItem(m, id)
  606.     RETURN (item # NIL) & item.enabled
  607. END isEnabled;
  608.  
  609.  
  610. PROCEDURE create* (items: List.tList; click: tClick; key: tKey): tMenu;
  611. VAR
  612.     m: tMenu;
  613.     maxLength: INTEGER;
  614.     item: tItem;
  615. BEGIN
  616.     NEW(m);
  617.     m.tid := 0;
  618.     m.active := FALSE;
  619.     m.parent := NIL;
  620.     m.child := NIL;
  621.     m.mainID := 0;
  622.     m.items  := items;
  623.     m.click := click;
  624.     m.key := key;
  625.     maxLength := 0;
  626.     item := items.first(tItem);
  627.     WHILE item # NIL DO
  628.         maxLength := MAX(maxLength, LENGTH(item.text));
  629.         item := item.next(tItem)
  630.     END;
  631.     m.width  := maxLength*fontWidth + LEFT + RIGHT;
  632.     m.height := items.count*fontHeight - 2;
  633.     m.font := G.CreateFont(1, "", {});
  634.     m.canvas := G.CreateCanvas(m.width + 1, m.height + 1);
  635.     G.SetFont(m.canvas, m.font);
  636.     RETURN m
  637. END create;
  638.  
  639.  
  640. PROCEDURE init* (_resetTimer: tProc);
  641. VAR
  642.         i: INTEGER;
  643. BEGIN
  644.         resetTimer := _resetTimer;
  645.         _open := open;
  646.         FOR i := 0 TO maxLEVEL DO
  647.                 TIDs[i] := 0
  648.         END
  649. END init;
  650.  
  651.  
  652. END Menu.