2019-07-11 13:52:33 -05:00
# include "TextEditorWidget.h"
2019-07-13 19:58:04 -05:00
# include <AK/Optional.h>
2019-07-11 13:52:33 -05:00
# include <AK/StringBuilder.h>
# include <LibCore/CFile.h>
2019-09-02 19:48:47 +02:00
# include <LibDraw/PNGLoader.h>
# include <LibGUI/GAboutDialog.h>
2019-07-11 13:52:33 -05:00
# include <LibGUI/GAction.h>
# include <LibGUI/GBoxLayout.h>
2019-08-21 21:30:20 +02:00
# include <LibGUI/GButton.h>
2019-07-11 13:52:33 -05:00
# include <LibGUI/GFilePicker.h>
# include <LibGUI/GFontDatabase.h>
# include <LibGUI/GMenuBar.h>
# include <LibGUI/GMessageBox.h>
# include <LibGUI/GStatusBar.h>
2019-08-21 21:30:20 +02:00
# include <LibGUI/GTextBox.h>
2019-07-11 13:52:33 -05:00
# include <LibGUI/GTextEditor.h>
# include <LibGUI/GToolBar.h>
TextEditorWidget : : TextEditorWidget ( )
{
set_layout ( make < GBoxLayout > ( Orientation : : Vertical ) ) ;
layout ( ) - > set_spacing ( 0 ) ;
auto * toolbar = new GToolBar ( this ) ;
m_editor = new GTextEditor ( GTextEditor : : MultiLine , this ) ;
m_editor - > set_ruler_visible ( true ) ;
m_editor - > set_automatic_indentation_enabled ( true ) ;
2019-08-27 17:05:01 +02:00
m_editor - > set_line_wrapping_enabled ( true ) ;
2019-08-21 21:30:20 +02:00
2019-08-27 20:18:19 +02:00
m_editor - > on_change = [ this ] {
bool was_dirty = m_document_dirty ;
m_document_dirty = true ;
if ( ! was_dirty )
update_title ( ) ;
} ;
2019-08-22 11:09:25 +02:00
m_find_widget = new GWidget ( this ) ;
m_find_widget - > set_fill_with_background_color ( true ) ;
m_find_widget - > set_size_policy ( SizePolicy : : Fill , SizePolicy : : Fixed ) ;
m_find_widget - > set_preferred_size ( 0 , 22 ) ;
m_find_widget - > set_layout ( make < GBoxLayout > ( Orientation : : Horizontal ) ) ;
m_find_widget - > layout ( ) - > set_margins ( { 2 , 2 , 2 , 2 } ) ;
m_find_widget - > set_visible ( false ) ;
2019-08-21 21:30:20 +02:00
2019-08-22 11:09:25 +02:00
m_find_textbox = new GTextBox ( m_find_widget ) ;
2019-08-21 21:30:20 +02:00
2019-08-25 21:35:58 +02:00
m_find_next_action = GAction : : create ( " Find next " , { Mod_Ctrl , Key_G } , [ & ] ( auto & ) {
2019-08-21 21:30:20 +02:00
auto needle = m_find_textbox - > text ( ) ;
2019-08-24 12:09:35 -06:00
auto found_range = m_editor - > find_next ( needle , m_editor - > normalized_selection ( ) . end ( ) ) ;
dbg ( ) < < " find_next( \" " < < needle < < " \" ) returned " < < found_range ;
if ( found_range . is_valid ( ) ) {
m_editor - > set_selection ( found_range ) ;
} else {
GMessageBox : : show (
String : : format ( " Not found: \" %s \" " , needle . characters ( ) ) ,
" Not found " ,
GMessageBox : : Type : : Information ,
GMessageBox : : InputType : : OK , window ( ) ) ;
}
2019-08-25 21:35:58 +02:00
} ) ;
m_find_previous_action = GAction : : create ( " Find previous " , { Mod_Ctrl | Mod_Shift , Key_G } , [ & ] ( auto & ) {
2019-08-24 12:09:35 -06:00
auto needle = m_find_textbox - > text ( ) ;
auto selection_start = m_editor - > normalized_selection ( ) . start ( ) ;
if ( ! selection_start . is_valid ( ) )
selection_start = m_editor - > normalized_selection ( ) . end ( ) ;
auto found_range = m_editor - > find_prev ( needle , selection_start ) ;
2019-08-25 12:23:34 +02:00
2019-08-24 12:09:35 -06:00
dbg ( ) < < " find_prev( \" " < < needle < < " \" ) returned " < < found_range ;
2019-08-21 21:30:20 +02:00
if ( found_range . is_valid ( ) ) {
m_editor - > set_selection ( found_range ) ;
} else {
GMessageBox : : show (
String : : format ( " Not found: \" %s \" " , needle . characters ( ) ) ,
" Not found " ,
GMessageBox : : Type : : Information ,
GMessageBox : : InputType : : OK , window ( ) ) ;
}
2019-08-25 21:35:58 +02:00
} ) ;
m_find_previous_button = new GButton ( " Previous " , m_find_widget ) ;
m_find_previous_button - > set_size_policy ( SizePolicy : : Fixed , SizePolicy : : Fill ) ;
m_find_previous_button - > set_preferred_size ( 64 , 0 ) ;
m_find_previous_button - > set_action ( * m_find_previous_action ) ;
m_find_next_button = new GButton ( " Next " , m_find_widget ) ;
m_find_next_button - > set_size_policy ( SizePolicy : : Fixed , SizePolicy : : Fill ) ;
m_find_next_button - > set_preferred_size ( 64 , 0 ) ;
m_find_next_button - > set_action ( * m_find_next_action ) ;
2019-08-21 21:30:20 +02:00
2019-08-22 11:09:25 +02:00
m_find_textbox - > on_return_pressed = [ this ] {
2019-08-24 12:09:35 -06:00
m_find_next_button - > click ( ) ;
2019-08-22 11:09:25 +02:00
} ;
m_find_textbox - > on_escape_pressed = [ this ] {
m_find_widget - > set_visible ( false ) ;
m_editor - > set_focus ( true ) ;
} ;
2019-09-13 23:56:37 +02:00
m_find_action = GAction : : create ( " Find... " , { Mod_Ctrl , Key_F } , load_png ( " /res/icons/16x16/find.png " ) , [ this ] ( auto & ) {
2019-08-22 11:09:25 +02:00
m_find_widget - > set_visible ( true ) ;
2019-08-22 11:02:03 +02:00
m_find_textbox - > set_focus ( true ) ;
2019-08-25 21:44:59 +02:00
m_find_textbox - > select_all ( ) ;
2019-08-22 11:02:03 +02:00
} ) ;
2019-08-25 21:38:13 +02:00
m_editor - > add_custom_context_menu_action ( * m_find_action ) ;
m_editor - > add_custom_context_menu_action ( * m_find_next_action ) ;
m_editor - > add_custom_context_menu_action ( * m_find_previous_action ) ;
2019-07-11 13:52:33 -05:00
auto * statusbar = new GStatusBar ( this ) ;
m_editor - > on_cursor_change = [ statusbar , this ] {
StringBuilder builder ;
2019-07-27 21:20:38 +02:00
builder . appendf ( " Line: %d, Column: %d " , m_editor - > cursor ( ) . line ( ) + 1 , m_editor - > cursor ( ) . column ( ) ) ;
2019-07-11 13:52:33 -05:00
statusbar - > set_text ( builder . to_string ( ) ) ;
} ;
2019-09-05 23:05:28 -06:00
m_new_action = GAction : : create ( " New " , { Mod_Ctrl , Key_N } , GraphicsBitmap : : load_from_file ( " /res/icons/16x16/new.png " ) , [ this ] ( const GAction & ) {
if ( m_document_dirty ) {
2019-09-14 22:10:44 +02:00
GMessageBox save_document_first_box ( " Save Document First? " , " Warning " , GMessageBox : : Type : : Warning , GMessageBox : : InputType : : OKCancel , window ( ) ) ;
2019-09-05 23:05:28 -06:00
auto save_document_first_result = save_document_first_box . exec ( ) ;
if ( save_document_first_result ! = GDialog : : ExecResult : : ExecOK )
2019-09-14 22:10:44 +02:00
return ;
2019-09-05 23:05:28 -06:00
m_save_action - > activate ( ) ;
}
2019-09-14 22:10:44 +02:00
2019-09-05 23:05:28 -06:00
m_document_dirty = false ;
m_editor - > set_text ( StringView ( ) ) ;
set_path ( FileSystemPath ( ) ) ;
update_title ( ) ;
2019-07-11 13:52:33 -05:00
} ) ;
2019-07-25 23:48:39 -05:00
m_open_action = GAction : : create ( " Open... " , { Mod_Ctrl , Key_O } , GraphicsBitmap : : load_from_file ( " /res/icons/16x16/open.png " ) , [ this ] ( const GAction & ) {
2019-07-28 23:45:50 -05:00
Optional < String > open_path = GFilePicker : : get_open_filepath ( ) ;
2019-07-11 13:52:33 -05:00
2019-07-28 23:45:50 -05:00
if ( ! open_path . has_value ( ) )
2019-07-13 19:58:04 -05:00
return ;
2019-07-28 23:45:50 -05:00
open_sesame ( open_path . value ( ) ) ;
2019-07-11 13:52:33 -05:00
} ) ;
2019-07-25 23:48:39 -05:00
m_save_as_action = GAction : : create ( " Save as... " , { Mod_None , Key_F12 } , GraphicsBitmap : : load_from_file ( " /res/icons/16x16/save.png " ) , [ this ] ( const GAction & ) {
2019-07-28 23:45:50 -05:00
Optional < String > save_path = GFilePicker : : get_save_filepath ( m_name . is_null ( ) ? " Untitled " : m_name , m_extension . is_null ( ) ? " txt " : m_extension ) ;
if ( ! save_path . has_value ( ) )
2019-07-13 19:58:04 -05:00
return ;
2019-07-28 23:45:50 -05:00
if ( ! m_editor - > write_to_file ( save_path . value ( ) ) ) {
2019-07-16 21:32:10 +02:00
GMessageBox : : show ( " Unable to save file. \n " , " Error " , GMessageBox : : Type : : Error , GMessageBox : : InputType : : OK , window ( ) ) ;
2019-07-13 19:58:04 -05:00
return ;
}
2019-08-27 20:18:19 +02:00
m_document_dirty = false ;
2019-07-28 23:45:50 -05:00
set_path ( FileSystemPath ( save_path . value ( ) ) ) ;
dbg ( ) < < " Wrote document to " < < save_path . value ( ) ;
2019-07-24 06:32:30 +02:00
} ) ;
2019-07-25 23:48:39 -05:00
m_save_action = GAction : : create ( " Save " , { Mod_Ctrl , Key_S } , GraphicsBitmap : : load_from_file ( " /res/icons/16x16/save.png " ) , [ & ] ( const GAction & ) {
2019-07-24 06:32:30 +02:00
if ( ! m_path . is_empty ( ) ) {
2019-08-27 20:18:19 +02:00
if ( ! m_editor - > write_to_file ( m_path ) ) {
2019-07-24 06:32:30 +02:00
GMessageBox : : show ( " Unable to save file. \n " , " Error " , GMessageBox : : Type : : Error , GMessageBox : : InputType : : OK , window ( ) ) ;
2019-08-27 20:18:19 +02:00
} else {
m_document_dirty = false ;
update_title ( ) ;
}
2019-07-24 06:32:30 +02:00
return ;
}
2019-07-25 23:48:39 -05:00
m_save_as_action - > activate ( ) ;
2019-07-11 13:52:33 -05:00
} ) ;
2019-08-25 12:23:34 +02:00
m_line_wrapping_setting_action = GAction : : create ( " Line wrapping " , [ & ] ( GAction & action ) {
action . set_checked ( ! action . is_checked ( ) ) ;
m_editor - > set_line_wrapping_enabled ( action . is_checked ( ) ) ;
} ) ;
m_line_wrapping_setting_action - > set_checkable ( true ) ;
m_line_wrapping_setting_action - > set_checked ( m_editor - > is_line_wrapping_enabled ( ) ) ;
2019-07-11 13:52:33 -05:00
auto menubar = make < GMenuBar > ( ) ;
auto app_menu = make < GMenu > ( " Text Editor " ) ;
2019-08-26 17:59:05 +02:00
app_menu - > add_action ( * m_new_action ) ;
app_menu - > add_action ( * m_open_action ) ;
app_menu - > add_action ( * m_save_action ) ;
app_menu - > add_action ( * m_save_as_action ) ;
app_menu - > add_separator ( ) ;
2019-09-14 22:10:44 +02:00
app_menu - > add_action ( GCommonActions : : make_quit_action ( [ this ] ( auto & ) {
2019-08-27 20:37:41 +02:00
if ( ! request_close ( ) )
return ;
2019-07-11 13:52:33 -05:00
GApplication : : the ( ) . quit ( 0 ) ;
} ) ) ;
menubar - > add_menu ( move ( app_menu ) ) ;
auto edit_menu = make < GMenu > ( " Edit " ) ;
edit_menu - > add_action ( m_editor - > undo_action ( ) ) ;
edit_menu - > add_action ( m_editor - > redo_action ( ) ) ;
edit_menu - > add_separator ( ) ;
edit_menu - > add_action ( m_editor - > cut_action ( ) ) ;
edit_menu - > add_action ( m_editor - > copy_action ( ) ) ;
edit_menu - > add_action ( m_editor - > paste_action ( ) ) ;
edit_menu - > add_action ( m_editor - > delete_action ( ) ) ;
2019-08-22 11:02:03 +02:00
edit_menu - > add_separator ( ) ;
edit_menu - > add_action ( * m_find_action ) ;
2019-08-25 21:35:58 +02:00
edit_menu - > add_action ( * m_find_next_action ) ;
edit_menu - > add_action ( * m_find_previous_action ) ;
2019-07-11 13:52:33 -05:00
menubar - > add_menu ( move ( edit_menu ) ) ;
auto font_menu = make < GMenu > ( " Font " ) ;
GFontDatabase : : the ( ) . for_each_fixed_width_font ( [ & ] ( const StringView & font_name ) {
font_menu - > add_action ( GAction : : create ( font_name , [ this ] ( const GAction & action ) {
m_editor - > set_font ( GFontDatabase : : the ( ) . get_by_name ( action . text ( ) ) ) ;
m_editor - > update ( ) ;
} ) ) ;
} ) ;
2019-08-27 17:01:52 +02:00
auto view_menu = make < GMenu > ( " View " ) ;
view_menu - > add_action ( * m_line_wrapping_setting_action ) ;
2019-08-28 21:13:30 +02:00
view_menu - > add_separator ( ) ;
view_menu - > add_submenu ( move ( font_menu ) ) ;
2019-08-27 17:01:52 +02:00
menubar - > add_menu ( move ( view_menu ) ) ;
2019-07-11 13:52:33 -05:00
auto help_menu = make < GMenu > ( " Help " ) ;
2019-09-02 19:48:47 +02:00
help_menu - > add_action ( GAction : : create ( " About " , [ & ] ( const GAction & ) {
GAboutDialog : : show ( " TextEditor " , load_png ( " /res/icons/32x32/app-texteditor.png " ) , window ( ) ) ;
2019-07-11 13:52:33 -05:00
} ) ) ;
menubar - > add_menu ( move ( help_menu ) ) ;
GApplication : : the ( ) . set_menubar ( move ( menubar ) ) ;
2019-07-25 23:48:39 -05:00
toolbar - > add_action ( * m_new_action ) ;
toolbar - > add_action ( * m_open_action ) ;
toolbar - > add_action ( * m_save_action ) ;
2019-07-11 13:52:33 -05:00
toolbar - > add_separator ( ) ;
toolbar - > add_action ( m_editor - > cut_action ( ) ) ;
toolbar - > add_action ( m_editor - > copy_action ( ) ) ;
toolbar - > add_action ( m_editor - > paste_action ( ) ) ;
toolbar - > add_action ( m_editor - > delete_action ( ) ) ;
toolbar - > add_separator ( ) ;
toolbar - > add_action ( m_editor - > undo_action ( ) ) ;
toolbar - > add_action ( m_editor - > redo_action ( ) ) ;
m_editor - > set_focus ( true ) ;
}
TextEditorWidget : : ~ TextEditorWidget ( )
{
}
2019-07-28 23:45:50 -05:00
void TextEditorWidget : : set_path ( const FileSystemPath & file )
2019-07-24 06:32:30 +02:00
{
2019-07-28 23:45:50 -05:00
m_path = file . string ( ) ;
m_name = file . title ( ) ;
m_extension = file . extension ( ) ;
2019-08-27 20:18:19 +02:00
update_title ( ) ;
}
void TextEditorWidget : : update_title ( )
{
2019-07-24 06:32:30 +02:00
StringBuilder builder ;
builder . append ( " Text Editor: " ) ;
2019-08-27 20:18:19 +02:00
builder . append ( m_path ) ;
if ( m_document_dirty )
builder . append ( " (*) " ) ;
2019-07-24 06:32:30 +02:00
window ( ) - > set_title ( builder . to_string ( ) ) ;
}
2019-07-11 13:52:33 -05:00
void TextEditorWidget : : open_sesame ( const String & path )
{
CFile file ( path ) ;
if ( ! file . open ( CIODevice : : ReadOnly ) ) {
2019-07-16 21:32:10 +02:00
GMessageBox : : show ( String : : format ( " Opening \" %s \" failed: %s " , path . characters ( ) , strerror ( errno ) ) , " Error " , GMessageBox : : Type : : Error , GMessageBox : : InputType : : OK , window ( ) ) ;
2019-08-23 19:10:14 +02:00
return ;
2019-07-11 13:52:33 -05:00
}
2019-08-29 18:54:06 +02:00
m_document_dirty = false ;
2019-08-23 19:04:53 +02:00
m_editor - > set_text ( file . read_all ( ) ) ;
2019-07-28 23:45:50 -05:00
set_path ( FileSystemPath ( path ) ) ;
2019-07-16 21:32:10 +02:00
}
2019-08-27 20:37:41 +02:00
bool TextEditorWidget : : request_close ( )
{
if ( ! m_document_dirty )
return true ;
GMessageBox box ( " The document has been modified. Quit without saving? " , " Quit without saving? " , GMessageBox : : Type : : Warning , GMessageBox : : InputType : : OKCancel , window ( ) ) ;
auto result = box . exec ( ) ;
return result = = GMessageBox : : ExecOK ;
}