with FLTK.Widgets.Menus; with FLTK.Widgets.Groups.Windows; with FLTK.Text_Buffers; with FLTK.Dialogs; with Windows.Editor; with Windows.About; with Windows.Find; with Windows.Replace; with Ada.Strings.Unbounded; use Ada.Strings.Unbounded; package body Adapad is -- forward declarations of helper functions procedure Set_Title; function Safe_To_Discard return Boolean; procedure Do_Save; procedure Do_Save_As; procedure Load_File (Name : in String); procedure Save_File (Name : in String); procedure Centre (Win : in out FLTK.Widgets.Groups.Windows.Window'Class); -- global state of the text editor Editor : Windows.Editor.Editor_Window := Windows.Editor.Create (800, 500); Buffer : FLTK.Text_Buffers.Text_Buffer := FLTK.Text_Buffers.Create; About : Windows.About.About_Window := Windows.About.Create; Find : Windows.Find.Find_Window := Windows.Find.Create; Replace : Windows.Replace.Replace_Window := Windows.Replace.Create; Changed : Boolean := False; Filename : Unbounded_String := To_Unbounded_String (0); -- main program interface procedure Show is begin Editor.Show; end Show; procedure Hide is begin About.Hide; Find.Hide; Replace.Hide; Editor.Hide; end Hide; -- callbacks for the menu procedure New_CB (Item : in out FLTK.Widgets.Widget'Class) is begin if not Safe_To_Discard then return; end if; Filename := To_Unbounded_String (0); Buffer.Set_Selection (0, Buffer.Length); Buffer.Remove_Selected_Text; Changed := False; Buffer.Call_Modify_Callbacks; end New_CB; procedure Open_CB (Item : in out FLTK.Widgets.Widget'Class) is begin if not Safe_To_Discard then return; end if; declare New_Filename : String := FLTK.Dialogs.File_Chooser ("Open File?", "*", To_String (Filename)); begin if New_Filename /= "" then Load_File (New_Filename); end if; end; end Open_CB; procedure Save_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Do_Save; end Save_CB; procedure Save_As_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Do_Save_As; end Save_As_CB; procedure Quit_CB (Item : in out FLTK.Widgets.Widget'Class) is begin if not Safe_To_Discard then return; end if; Hide; end Quit_CB; procedure Undo_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Editor.Undo; end Undo_CB; procedure Cut_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Editor.Cut; end Cut_CB; procedure Copy_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Editor.Copy; end Copy_CB; procedure Paste_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Editor.Paste; end Paste_CB; procedure Delete_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Editor.Delete; end Delete_CB; procedure Select_All_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Buffer.Set_Selection (0, Buffer.Length); end Select_All_CB; procedure Find_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Centre (Find); Find.Show; end Find_CB; procedure Replace_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Centre (Replace); Replace.Show; end Replace_CB; procedure Jump_CB (Item : in out FLTK.Widgets.Widget'Class) is User_Input : String := FLTK.Dialogs.Text_Input ("Line number: "); Line_Number : Integer; begin Line_Number := Integer'Value (User_Input); if Line_Number > 0 then Editor.Set_Insert_Position (Buffer.Skip_Lines (0, Line_Number - 1)); Editor.Show_Insert_Position; end if; exception when Constraint_Error => null; -- user has entered nonsense input, do nothing end Jump_CB; procedure Count_CB (Item : in out FLTK.Widgets.Widget'Class) is Restore_Position : Natural := Editor.Get_Insert_Position; Current_Position, Result : Natural := 0; begin Editor.Set_Insert_Position (0); if Character'Pos (Buffer.Character_At (0)) > Character'Pos (' ') then Result := 1; end if; loop Editor.Next_Word; Current_Position := Editor.Get_Insert_Position; exit when Current_Position = Buffer.Length; Result := Result + 1; end loop; Editor.Set_Insert_Position (Restore_Position); FLTK.Dialogs.Message_Box ("There are " & Integer'Image (Result) & " words in the document."); end Count_CB; procedure Wrap_CB (Item : in out FLTK.Widgets.Widget'Class) is begin if FLTK.Widgets.Menus.Menu (Item).Chosen.Value then Editor.Set_Wrap_Mode (Windows.Editor.Wrap_At_Bounds); else Editor.Set_Wrap_Mode (Windows.Editor.Wrap_None); end if; end Wrap_CB; procedure Lines_CB (Item : in out FLTK.Widgets.Widget'Class) is begin if FLTK.Widgets.Menus.Menu (Item).Chosen.Value then -- 50 pixels should be enough for 5 digit line numbers Editor.Set_Linenumber_Width (50); else Editor.Set_Linenumber_Width (0); end if; end Lines_CB; procedure About_CB (Item : in out FLTK.Widgets.Widget'Class) is begin Centre (About); About.Show; end About_CB; -- callbacks for the text buffer procedure Mod_CB (Action : in FLTK.Text_Buffers.Modification; Place : in FLTK.Text_Buffers.Position; Length : in Natural; Deleted_Text : in String) is use type FLTK.Text_Buffers.Modification; begin if Action = FLTK.Text_Buffers.Insert or Action = FLTK.Text_Buffers.Delete then Changed := True; end if; Set_Title; end Mod_CB; -- callbacks for the find/replace windows procedure Do_Find_CB (Item : in String; Match_Case : in Boolean) is Current_Position, Found_At : Natural; begin Find.Hide; Current_Position := Editor.Get_Insert_Position; if Buffer.Search_Forward (Current_Position, Item, Found_At, Match_Case) then Buffer.Set_Selection (Found_At, Found_At + Item'Length); Editor.Set_Insert_Position (Found_At + Item'Length); Editor.Show_Insert_Position; else FLTK.Dialogs.Alert ("No occurrences of '" & Item & "' found!"); end if; end Do_Find_CB; procedure Do_Replace_CB (Item, Replace_With : in String; Match_Case, Replace_All : in Boolean) is Current_Position, Found_At : Natural; Times_Replaced : Natural := 0; begin Replace.Hide; if Replace_All then Editor.Set_Insert_Position (0); end if; loop Current_Position := Editor.Get_Insert_Position; exit when not Buffer.Search_Forward (Current_Position, Item, Found_At, Match_Case); Buffer.Set_Selection (Found_At, Found_At + Item'Length); Buffer.Remove_Selected_Text; Buffer.Insert_Text (Found_At, Replace_With); Editor.Set_Insert_Position (Found_At + Replace_With'Length); Editor.Show_Insert_Position; Times_Replaced := Times_Replaced + 1; exit when not Replace_All and Times_Replaced > 0; end loop; if Times_Replaced > 0 then FLTK.Dialogs.Message_Box ("Replaced " & Integer'Image (Times_Replaced) & " occurrences."); else FLTK.Dialogs.Alert ("No occurrences of '" & Item & "' found!"); end if; end Do_Replace_CB; -- helper functions procedure Set_Title is Title : Unbounded_String := To_Unbounded_String (0); begin if Changed then Append (Title, "*"); end if; if Filename = "" then Append (Title, "(Untitled)"); else Append (Title, Filename); end if; Editor.Set_Label (To_String (Title)); end Set_Title; function Safe_To_Discard return Boolean is User_Response : FLTK.Dialogs.Choice; begin if not Changed then return True; end if; User_Response := FLTK.Dialogs.Three_Way_Choice ("The current file has not been saved." & Character'Val (10) & "Would you like to save it now?", "Cancel", "Save", "Discard"); case User_Response is when FLTK.Dialogs.First => return False; when FLTK.Dialogs.Second => Do_Save; return not Changed; when FLTK.Dialogs.Third => return True; end case; end Safe_To_Discard; procedure Do_Save is begin if Filename = "" then Do_Save_As; else Save_File (To_String (Filename)); end if; end Do_Save; procedure Do_Save_As is New_Filename : String := FLTK.Dialogs.File_Chooser ("Save File As?", "*", To_String (Filename)); begin if New_Filename /= "" then Save_File (New_Filename); end if; end Do_Save_As; procedure Load_File (Name : in String) is begin Buffer.Load_File (Name); Filename := To_Unbounded_String (Name); Changed := False; Buffer.Call_Modify_Callbacks; exception when Storage_Error => FLTK.Dialogs.Alert ("Error reading from file " & Name); end Load_File; procedure Save_File (Name : in String) is begin Buffer.Save_File (Name); Filename := To_Unbounded_String (Name); Changed := False; Buffer.Call_Modify_Callbacks; exception when Storage_Error => FLTK.Dialogs.Alert ("Error writing to file " & Name); end Save_File; procedure Centre (Win : in out FLTK.Widgets.Groups.Windows.Window'Class) is Middle_X : Integer := Editor.Get_X + Editor.Get_W / 2; Middle_Y : Integer := Editor.Get_Y + Editor.Get_H / 2; begin Win.Reposition (Middle_X - Win.Get_W / 2, Middle_Y - Win.Get_H / 2); end Centre; begin declare use FLTK.Widgets.Menus; Bar : Menu_Cursor := Editor.Get_Menu; begin Bar.Add (Text => "&File", Flags => Flag_Submenu); Bar.Add ("File/&New", New_CB'Access, Mod_Ctrl + 'n'); Bar.Add ("File/&Open...", Open_CB'Access, Mod_Ctrl + 'o'); Bar.Add ("File/&Save", Save_CB'Access, Mod_Ctrl + 's'); Bar.Add ("File/Save &As...", Save_As_CB'Access, Mod_Shift + Mod_Ctrl + 's', Flag_Divider); Bar.Add ("File/&Quit", Quit_CB'Access, Mod_Ctrl + 'q'); Bar.Add (Text => "&Edit", Flags => Flag_Submenu); Bar.Add ("Edit/&Undo", Undo_CB'Access, Mod_Ctrl + 'z', Flag_Divider); Bar.Add ("Edit/Cu&t", Cut_CB'Access, Mod_Ctrl + 'x'); Bar.Add ("Edit/&Copy", Copy_CB'Access, Mod_Ctrl + 'c'); Bar.Add ("Edit/&Paste", Paste_CB'Access, Mod_Ctrl + 'v'); Bar.Add ("Edit/&Delete", Delete_CB'Access, No_Key, Flag_Divider); Bar.Add ("Edit/Select &All", Select_All_CB'Access, Mod_Ctrl + 'a'); Bar.Add (Text => "&Search", Flags => Flag_Submenu); Bar.Add ("Search/&Find...", Find_CB'Access, Mod_Ctrl + 'f'); Bar.Add ("Search/&Replace...", Replace_CB'Access, Mod_Ctrl + 'h', Flag_Divider); Bar.Add ("Search/Jump To...", Jump_CB'Access, Mod_Ctrl + 'j'); Bar.Add ("Search/Word Count", Count_CB'Access); Bar.Add (Text => "&Options", Flags => Flag_Submenu); Bar.Add ("Options/&Word Wrap", Wrap_CB'Access, No_Key, Flag_Toggle); Bar.Add ("Options/&Line Numbers", Lines_CB'Access, No_Key, Flag_Toggle); Bar.Add (Text => "&Help", Flags => Flag_Submenu); Bar.Add ("Help/&About", About_CB'Access); end; Find.Set_Find_Callback (Do_Find_CB'Access); Replace.Set_Replace_Callback (Do_Replace_CB'Access); Buffer.Add_Modify_Callback (Mod_CB'Access); Editor.Set_Callback (Quit_CB'Access); Editor.Set_Buffer (Buffer); end Adapad;