2023-05-19 13:39:35 -07:00
/*
* Copyright ( C ) 2023 Kyle Reed
2024-02-10 09:56:50 -06:00
* Copyright ( C ) 2023 Mark Thompson
2023-05-19 13:39:35 -07:00
*
* This file is part of PortaPack .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 , or ( at your option )
* any later version .
*
* 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 . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; see the file COPYING . If not , write to
* the Free Software Foundation , Inc . , 51 Franklin Street ,
* Boston , MA 02110 - 1301 , USA .
*/
# include "ui_fileman.hpp"
# include "ui_text_editor.hpp"
2023-05-26 01:02:17 -07:00
# include "ui_textentry.hpp"
2023-05-22 13:08:59 -07:00
# include "log_file.hpp"
2023-05-19 13:39:35 -07:00
# include "string_format.hpp"
2024-02-07 16:07:30 +08:00
# include "portapack_persistent_memory.hpp"
2023-05-19 13:39:35 -07:00
using namespace portapack ;
namespace fs = std : : filesystem ;
namespace ui {
2023-05-26 01:02:17 -07:00
/* TextViewer *******************************************************/
2023-05-22 13:08:59 -07:00
2023-05-26 01:02:17 -07:00
TextViewer : : TextViewer ( Rect parent_rect )
2023-06-29 12:55:25 -05:00
: Widget ( parent_rect ) {
2023-05-19 13:39:35 -07:00
set_focusable ( true ) ;
2023-06-29 12:55:25 -05:00
set_font_zoom ( false ) ;
2023-05-22 21:40:03 -07:00
}
2023-05-26 01:02:17 -07:00
void TextViewer : : paint ( Painter & painter ) {
2023-05-19 13:39:35 -07:00
auto first_line = paint_state_ . first_line ;
auto first_col = paint_state_ . first_col ;
2023-05-26 01:02:17 -07:00
if ( ! has_file ( ) )
2023-05-19 13:39:35 -07:00
return ;
2023-05-22 13:08:59 -07:00
// Move the viewport vertically.
2023-05-19 13:39:35 -07:00
if ( cursor_ . line < first_line )
first_line = cursor_ . line ;
else if ( cursor_ . line > = first_line + max_line )
first_line = cursor_ . line - max_line + 1 ;
2023-05-22 13:08:59 -07:00
// Move the viewport horizontally.
2023-05-19 13:39:35 -07:00
if ( cursor_ . col < first_col )
first_col = cursor_ . col ;
if ( cursor_ . col > = first_col + max_col )
first_col = cursor_ . col - max_col + 1 ;
2023-05-22 13:08:59 -07:00
// Viewport updated? Redraw text.
2023-05-19 13:39:35 -07:00
if ( first_line ! = paint_state_ . first_line | |
first_col ! = paint_state_ . first_col ) {
paint_state_ . first_line = first_line ;
paint_state_ . first_col = first_col ;
paint_state_ . redraw_text = true ;
2023-07-27 09:14:02 -05:00
paint_state_ . line = UINT32_MAX ; // forget old cursor position when overwritten
2023-05-19 13:39:35 -07:00
}
if ( paint_state_ . redraw_text ) {
paint_text ( painter , first_line , first_col ) ;
paint_state_ . redraw_text = false ;
}
2023-11-30 15:57:22 -05:00
if ( paint_state_ . redraw_marked ) {
paint_marked ( painter ) ;
paint_state_ . redraw_marked = false ;
}
2023-05-19 13:39:35 -07:00
paint_cursor ( painter ) ;
}
2023-05-26 01:02:17 -07:00
bool TextViewer : : on_key ( const KeyEvent key ) {
2023-05-19 13:39:35 -07:00
int16_t delta_col = 0 ;
int16_t delta_line = 0 ;
if ( key = = KeyEvent : : Left )
delta_col = - 1 ;
else if ( key = = KeyEvent : : Right )
delta_col = 1 ;
else if ( key = = KeyEvent : : Up )
delta_line = - 1 ;
else if ( key = = KeyEvent : : Down )
delta_line = 1 ;
2023-05-26 01:02:17 -07:00
else if ( key = = KeyEvent : : Select & & on_select ) {
on_select ( ) ;
return true ;
}
2023-05-19 13:39:35 -07:00
// Always allow cursor direction to be updated.
cursor_ . dir = delta_col ! = 0 ? ScrollDirection : : Horizontal : ScrollDirection : : Vertical ;
auto updated = apply_scrolling_constraints ( delta_line , delta_col ) ;
if ( updated )
2023-05-26 01:02:17 -07:00
redraw ( ) ;
2023-05-19 13:39:35 -07:00
return updated ;
}
2023-05-26 01:02:17 -07:00
bool TextViewer : : on_encoder ( EncoderEvent delta ) {
2023-05-19 13:39:35 -07:00
bool updated = false ;
if ( cursor_ . dir = = ScrollDirection : : Horizontal )
updated = apply_scrolling_constraints ( 0 , delta ) ;
2023-06-15 00:45:13 -07:00
else {
delta * = 16 ;
2023-05-19 13:39:35 -07:00
updated = apply_scrolling_constraints ( delta , 0 ) ;
2023-06-15 00:45:13 -07:00
}
2023-05-19 13:39:35 -07:00
if ( updated )
2023-05-26 01:02:17 -07:00
redraw ( ) ;
2023-05-19 13:39:35 -07:00
return updated ;
}
2023-11-30 15:57:22 -05:00
void TextViewer : : redraw ( bool redraw_text , bool redraw_marked ) {
2023-06-01 15:45:55 -07:00
paint_state_ . redraw_text = redraw_text ;
2023-11-30 15:57:22 -05:00
paint_state_ . redraw_marked = redraw_marked ;
2023-06-01 15:45:55 -07:00
set_dirty ( ) ;
}
uint32_t TextViewer : : offset ( ) const {
auto range = file_ - > line_range ( cursor_ . line ) ;
if ( range )
return range - > start + col ( ) ;
return 0 ;
}
2023-06-29 13:17:31 -07:00
void TextViewer : : cursor_home ( ) {
cursor_ . col = 0 ;
redraw ( ) ;
}
void TextViewer : : cursor_end ( ) {
cursor_ . col = line_length ( ) - 1 ;
redraw ( ) ;
}
2023-11-30 15:57:22 -05:00
void TextViewer : : cursor_set ( uint16_t line , uint16_t col ) {
cursor_ . line = line ;
cursor_ . col = col ;
}
void TextViewer : : cursor_mark_selected ( ) {
LineColPair newMarker = std : : make_pair ( cursor_ . line , cursor_ . col ) ;
2023-12-02 02:05:29 -05:00
auto it = std : : find ( lineColPair . begin ( ) , lineColPair . end ( ) , newMarker ) ;
2023-11-30 15:57:22 -05:00
2023-12-02 02:05:29 -05:00
if ( it ! = lineColPair . end ( ) ) {
lineColPair . erase ( it ) ;
2023-11-30 15:57:22 -05:00
} else {
2023-12-02 02:05:29 -05:00
lineColPair . push_back ( newMarker ) ;
2023-11-30 15:57:22 -05:00
}
// Mark pending change.
cursor_ . mark_change = false ;
redraw ( ) ;
}
2023-12-02 02:05:29 -05:00
void TextViewer : : cursor_clear_marked ( ) {
lineColPair . clear ( ) ;
redraw ( true , true ) ;
}
2023-06-01 15:45:55 -07:00
uint16_t TextViewer : : line_length ( ) {
return file_ - > line_length ( cursor_ . line ) ;
}
2023-05-26 01:02:17 -07:00
bool TextViewer : : apply_scrolling_constraints ( int16_t delta_line , int16_t delta_col ) {
if ( ! has_file ( ) )
return false ;
2023-05-19 13:39:35 -07:00
int32_t new_line = cursor_ . line + delta_line ;
int32_t new_col = cursor_ . col + delta_col ;
2023-05-26 01:02:17 -07:00
int32_t new_line_length = file_ - > line_length ( new_line ) ;
2023-05-19 13:39:35 -07:00
if ( new_col < 0 )
- - new_line ;
2023-05-22 13:08:59 -07:00
else if ( new_col > = new_line_length & & delta_line = = 0 ) {
// Only wrap if moving horizontally.
2023-05-19 13:39:35 -07:00
new_col = 0 ;
+ + new_line ;
}
2023-06-29 13:17:31 -07:00
// Snap to first/last line to make navigating easier.
if ( new_line < 0 & & cursor_ . line > 0 ) {
2023-05-22 21:40:03 -07:00
new_line = 0 ;
2023-05-26 01:02:17 -07:00
} else if ( new_line > = ( int32_t ) file_ - > line_count ( ) ) {
auto last_line = file_ - > line_count ( ) - 1 ;
2023-05-22 21:40:03 -07:00
2023-06-29 13:17:31 -07:00
if ( cursor_ . line < last_line )
2023-05-22 21:40:03 -07:00
new_line = last_line ;
}
2023-05-26 01:02:17 -07:00
if ( new_line < 0 | | ( uint32_t ) new_line > = file_ - > line_count ( ) )
2023-05-19 13:39:35 -07:00
return false ;
2023-05-26 01:02:17 -07:00
new_line_length = file_ - > line_length ( new_line ) ;
2023-05-19 13:39:35 -07:00
// Wrap or clamp column.
if ( new_line_length = = 0 )
new_col = 0 ;
2023-05-22 13:08:59 -07:00
else if ( new_col > = new_line_length | | new_col < 0 )
new_col = new_line_length - 1 ;
2023-05-19 13:39:35 -07:00
cursor_ . line = new_line ;
cursor_ . col = new_col ;
2023-05-26 01:02:17 -07:00
if ( on_cursor_moved )
on_cursor_moved ( ) ;
2023-05-19 13:39:35 -07:00
return true ;
}
2023-05-26 01:02:17 -07:00
void TextViewer : : paint_text ( Painter & painter , uint32_t line , uint16_t col ) {
2023-05-19 13:39:35 -07:00
auto r = screen_rect ( ) ;
2023-06-01 15:45:55 -07:00
char buffer [ max_col + 1 ] ;
2023-05-19 13:39:35 -07:00
// Draw the lines from the file
2023-05-22 13:08:59 -07:00
for ( auto i = 0u ; i < max_line ; + + i ) {
2023-05-26 01:02:17 -07:00
if ( line + i > = file_ - > line_count ( ) )
2023-05-22 13:08:59 -07:00
break ;
2023-05-19 13:39:35 -07:00
2023-06-01 15:45:55 -07:00
auto result = file_ - > get_text ( line + i , col , buffer , max_col ) ;
2023-05-19 13:39:35 -07:00
2023-06-01 15:45:55 -07:00
if ( result & & * result > 0 )
2023-05-19 13:39:35 -07:00
painter . draw_string (
2023-05-26 01:02:17 -07:00
{ 0 , r . top ( ) + ( int ) i * char_height } ,
2023-06-29 12:55:25 -05:00
style ( ) , { buffer , * result } ) ;
2023-05-22 13:08:59 -07:00
2023-05-28 08:44:21 -07:00
// Clear empty line sections. This is less visually jarring than full clear.
2023-06-01 15:45:55 -07:00
int32_t clear_width = max_col - ( result ? * result : 0 ) ;
2023-05-22 13:08:59 -07:00
if ( clear_width > 0 )
painter . fill_rectangle (
{ ( max_col - clear_width ) * char_width ,
2023-05-26 01:02:17 -07:00
r . top ( ) + ( int ) i * char_height ,
2023-05-22 13:08:59 -07:00
clear_width * char_width , char_height } ,
2023-06-29 12:55:25 -05:00
style ( ) . background ) ;
2023-07-27 09:14:02 -05:00
// if cursor line got overwritten, disable XOR of old cursor when displaying new cursor
if ( i = = paint_state_ . line )
paint_state_ . line = UINT32_MAX ;
2023-05-19 13:39:35 -07:00
}
}
2023-05-26 01:02:17 -07:00
void TextViewer : : paint_cursor ( Painter & painter ) {
if ( ! has_focus ( ) )
return ;
2023-07-27 09:14:02 -05:00
auto xor_cursor = [ this , & painter ] ( int32_t line , uint16_t col ) {
int cursor_width = char_width + 1 ;
int x = ( col - paint_state_ . first_col ) * char_width - 1 ;
if ( x < 0 ) { // cursor is one pixel narrower when in left column
cursor_width - - ;
x = 0 ;
}
int y = screen_rect ( ) . top ( ) + ( line - paint_state_ . first_line ) * char_height ;
// Converting one row at a time to reduce buffer size
auto pbuf8 = cursor_ . pixel_buffer8 ;
auto pbuf = cursor_ . pixel_buffer ;
for ( auto col = 0 ; col < char_height ; col + + ) {
// Annoyingly, read_pixels uses a 24-bit pixel format vs draw_pixels which uses 16-bit
portapack : : display . read_pixels ( { x , y + col , cursor_width , 1 } , pbuf8 , cursor_width ) ;
for ( auto i = 0 ; i < cursor_width ; i + + )
pbuf [ i ] = Color ( pbuf8 [ i ] . r , pbuf8 [ i ] . g , pbuf8 [ i ] . b ) . v ^ 0xFFFF ;
portapack : : display . draw_pixels ( { x , y + col , cursor_width , 1 } , pbuf , cursor_width ) ;
}
2023-05-19 13:39:35 -07:00
} ;
2023-07-27 09:14:02 -05:00
if ( paint_state_ . line ! = UINT32_MAX ) // only XOR old cursor if it still appears on the screen
2023-11-30 15:57:22 -05:00
{
// Only reset previous cursor if we aren't marking.
if ( paint_state_ . mark_change ) {
xor_cursor ( paint_state_ . line , paint_state_ . col ) ;
}
}
2023-07-27 09:14:02 -05:00
xor_cursor ( cursor_ . line , cursor_ . col ) ;
2023-05-19 13:39:35 -07:00
paint_state_ . line = cursor_ . line ;
paint_state_ . col = cursor_ . col ;
2023-11-30 15:57:22 -05:00
paint_state_ . mark_change = cursor_ . mark_change ;
// Reset marking and wait for new change.
cursor_ . mark_change = true ;
}
void TextViewer : : paint_marked ( Painter & painter ) {
auto xor_cursor = [ this , & painter ] ( int32_t line , uint16_t col ) {
int cursor_width = char_width + 1 ;
int x = ( col - paint_state_ . first_col ) * char_width - 1 ;
if ( x < 0 ) { // cursor is one pixel narrower when in left column
cursor_width - - ;
x = 0 ;
}
int y = screen_rect ( ) . top ( ) + ( line - paint_state_ . first_line ) * char_height ;
// Converting one row at a time to reduce buffer size
auto pbuf8 = cursor_ . pixel_buffer8 ;
auto pbuf = cursor_ . pixel_buffer ;
for ( auto col = 0 ; col < char_height ; col + + ) {
// Annoyingly, read_pixels uses a 24-bit pixel format vs draw_pixels which uses 16-bit
portapack : : display . read_pixels ( { x , y + col , cursor_width , 1 } , pbuf8 , cursor_width ) ;
for ( auto i = 0 ; i < cursor_width ; i + + )
pbuf [ i ] = Color ( pbuf8 [ i ] . r , pbuf8 [ i ] . g , pbuf8 [ i ] . b ) . v ^ 0xFFFF ;
portapack : : display . draw_pixels ( { x , y + col , cursor_width , 1 } , pbuf , cursor_width ) ;
}
} ;
2023-12-02 02:05:29 -05:00
auto it = lineColPair . begin ( ) ;
2023-11-30 15:57:22 -05:00
2023-12-02 02:05:29 -05:00
while ( it ! = lineColPair . end ( ) ) {
2023-11-30 15:57:22 -05:00
LineColPair entry = ( LineColPair ) * it ;
xor_cursor ( entry . first , entry . second ) ;
it + + ;
}
2023-05-19 13:39:35 -07:00
}
2023-05-26 01:02:17 -07:00
void TextViewer : : reset_file ( FileWrapper * file ) {
file_ = file ;
paint_state_ . first_line = 0 ;
paint_state_ . first_col = 0 ;
cursor_ . line = 0 ;
cursor_ . col = 0 ;
redraw ( true ) ;
}
2023-06-29 15:07:39 -05:00
void TextViewer : : set_font_zoom ( bool zoom ) {
font_zoom = zoom ;
font_style = font_zoom ? & Styles : : white : & Styles : : white_small ;
char_height = style ( ) . font . line_height ( ) ;
char_width = style ( ) . font . char_width ( ) ;
max_line = ( uint8_t ) ( parent_rect ( ) . height ( ) / char_height ) ;
max_col = ( uint8_t ) ( parent_rect ( ) . width ( ) / char_width ) ;
}
2023-05-26 01:02:17 -07:00
/* TextEditorMenu ***************************************************/
TextEditorMenu : : TextEditorMenu ( )
: View { { 7 * 4 , 9 * 4 , 25 * 8 , 25 * 8 } } {
add_children (
{
& rect_frame ,
2023-06-29 13:17:31 -07:00
& button_home ,
& button_end ,
2023-06-29 22:58:10 +02:00
& button_zoom ,
2023-05-26 01:02:17 -07:00
& button_delline ,
& button_edit ,
& button_addline ,
& button_open ,
& button_save ,
& button_exit ,
} ) ;
}
void TextEditorMenu : : on_show ( ) {
hide_children ( false ) ;
button_edit . focus ( ) ;
}
void TextEditorMenu : : on_hide ( ) {
hide_children ( true ) ;
}
void TextEditorMenu : : hide_children ( bool hidden ) {
for ( auto child : children ( ) ) {
child - > hidden ( hidden ) ;
}
}
/* TextEditorView ***************************************************/
2023-07-13 09:38:40 -07:00
static fs : : path get_temp_path ( const fs : : path & path ) {
if ( ! path . empty ( ) )
return path + " ~ " ;
return { } ;
}
static void delete_temp_file ( const fs : : path & path ) {
auto temp_path = get_temp_path ( path ) ;
if ( ! temp_path . empty ( ) ) {
delete_file ( temp_path ) ;
}
}
static void save_temp_file ( const fs : : path & path ) {
delete_file ( path ) ;
copy_file ( get_temp_path ( path ) , path ) ;
}
static void show_save_prompt (
NavigationView & nav ,
std : : function < void ( ) > on_save ,
std : : function < void ( ) > continuation ) {
nav . display_modal (
2023-07-30 00:36:57 -07:00
" Save? " , " Save changes? " , YESNO ,
2023-07-13 09:38:40 -07:00
[ on_save ] ( bool choice ) {
if ( choice & & on_save )
on_save ( ) ;
} ) ;
nav . set_on_pop ( continuation ) ;
}
2023-05-26 01:02:17 -07:00
TextEditorView : : TextEditorView ( NavigationView & nav )
: nav_ { nav } {
add_children (
{
& viewer ,
& menu ,
& button_menu ,
& text_position ,
& text_size ,
} ) ;
2023-08-18 12:35:41 -07:00
viewer . set_font_zoom ( enable_zoom ) ;
2023-05-26 01:02:17 -07:00
viewer . on_select = [ this ] ( ) {
// Treat as if menu button was pressed.
if ( button_menu . on_select )
button_menu . on_select ( ) ;
} ;
viewer . on_cursor_moved = [ this ] ( ) {
update_position ( ) ;
} ;
menu . hidden ( true ) ;
2023-06-29 13:17:31 -07:00
menu . on_home ( ) = [ this ] ( ) {
viewer . cursor_home ( ) ;
hide_menu ( true ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-29 13:17:31 -07:00
menu . on_end ( ) = [ this ] ( ) {
viewer . cursor_end ( ) ;
2023-06-29 12:55:25 -05:00
hide_menu ( true ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-06-29 22:58:10 +02:00
menu . on_zoom ( ) = [ this ] ( ) {
2023-08-18 12:35:41 -07:00
enable_zoom = viewer . toggle_font_zoom ( ) ;
2023-06-29 22:58:10 +02:00
refresh_ui ( ) ;
hide_menu ( true ) ;
2023-06-29 13:17:31 -07:00
} ;
2023-05-26 01:02:17 -07:00
menu . on_delete_line ( ) = [ this ] ( ) {
2023-06-01 15:45:55 -07:00
prepare_for_write ( ) ;
file_ - > delete_line ( viewer . line ( ) ) ;
refresh_ui ( ) ;
hide_menu ( true ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
menu . on_edit_line ( ) = [ this ] ( ) {
2023-06-01 15:45:55 -07:00
show_edit_line ( ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
menu . on_add_line ( ) = [ this ] ( ) {
2023-06-01 15:45:55 -07:00
prepare_for_write ( ) ;
if ( viewer . offset ( ) < file_ - > size ( ) - 1 )
file_ - > insert_line ( viewer . line ( ) ) ;
else
file_ - > insert_line ( - 1 ) ; // Add after last line.
refresh_ui ( ) ;
hide_menu ( true ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
menu . on_open ( ) = [ this ] ( ) {
2023-09-27 12:03:02 -07:00
show_save_prompt ( [ this ] ( ) {
2023-06-01 15:45:55 -07:00
show_file_picker ( ) ;
2023-09-27 12:03:02 -07:00
} ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
menu . on_save ( ) = [ this ] ( ) {
2023-06-01 15:45:55 -07:00
save_temp_file ( ) ;
hide_menu ( true ) ;
2023-05-26 01:02:17 -07:00
} ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
menu . on_exit ( ) = [ this ] ( ) {
2023-07-13 09:38:40 -07:00
nav_ . pop ( ) ;
2023-05-26 01:02:17 -07:00
} ;
button_menu . on_select = [ this ] ( ) {
if ( file_ ) {
// Toggle menu.
hide_menu ( ! menu . hidden ( ) ) ;
} else {
show_file_picker ( ) ;
}
} ;
}
TextEditorView : : TextEditorView ( NavigationView & nav , const fs : : path & path )
: TextEditorView ( nav ) {
open_file ( path ) ;
}
2023-06-01 15:45:55 -07:00
TextEditorView : : ~ TextEditorView ( ) {
2023-07-13 09:38:40 -07:00
// NB: Be careful here. The UI will render after this instance
// has been destroyed. Everything needed to render the UI
// and perform the save actions must be value captured.
if ( file_dirty_ ) {
ui : : show_save_prompt (
nav_ ,
[ p = path_ ] ( ) { ui : : save_temp_file ( p ) ; } ,
[ p = std : : move ( path_ ) ] ( ) { delete_temp_file ( p ) ; } ) ;
}
2023-06-01 15:45:55 -07:00
}
2023-05-26 01:02:17 -07:00
void TextEditorView : : on_show ( ) {
if ( file_ )
viewer . focus ( ) ;
else
button_menu . focus ( ) ;
}
void TextEditorView : : open_file ( const fs : : path & path ) {
2023-06-01 15:45:55 -07:00
file_ . reset ( ) ;
viewer . clear_file ( ) ;
2023-07-13 09:38:40 -07:00
delete_temp_file ( path_ ) ;
2023-06-01 15:45:55 -07:00
path_ = { } ;
file_dirty_ = false ;
has_temp_file_ = false ;
2023-07-30 00:36:57 -07:00
auto result = FileWrapper : : open (
path , false , [ ] ( uint32_t value , uint32_t total ) {
Painter p ;
auto percent = ( value * 100 ) / total ;
auto width = ( percent * screen_width ) / 100 ;
p . draw_hline ( { 0 , 16 } , width , Color : : yellow ( ) ) ;
} ) ;
2023-05-26 01:02:17 -07:00
2023-05-28 08:44:21 -07:00
if ( ! result ) {
nav_ . display_modal ( " Read Error " , " Cannot open file: \n " + result . error ( ) . what ( ) ) ;
2023-06-01 15:45:55 -07:00
2023-05-26 01:02:17 -07:00
} else {
2023-06-01 15:45:55 -07:00
file_ = * std : : move ( result ) ;
path_ = path ;
2023-05-26 01:02:17 -07:00
viewer . set_file ( * file_ ) ;
}
2024-02-07 16:07:30 +08:00
portapack : : persistent_memory : : set_apply_fake_brightness ( false ) ; // work around to resolve the display issue in notepad app. not elegant i know, so TODO.
2023-05-26 01:02:17 -07:00
refresh_ui ( ) ;
}
void TextEditorView : : refresh_ui ( ) {
if ( file_ ) {
update_position ( ) ;
text_size . set (
" Lines: " + to_string_dec_uint ( file_ - > line_count ( ) ) +
" ( " + to_string_file_size ( file_ - > size ( ) ) + " ) " ) ;
} else {
text_position . set ( " " ) ;
text_size . set ( " " ) ;
}
}
void TextEditorView : : update_position ( ) {
if ( viewer . has_file ( ) ) {
text_position . set (
" Ln " + to_string_dec_uint ( viewer . line ( ) + 1 ) +
" , Col " + to_string_dec_uint ( viewer . col ( ) + 1 ) ) ;
}
}
void TextEditorView : : hide_menu ( bool hidden ) {
menu . hidden ( hidden ) ;
// Only let the viewer be focused when the menu is
// not shown, otherwise menu focus gets confusing.
viewer . set_focusable ( hidden ) ;
if ( hidden )
viewer . focus ( ) ;
viewer . redraw ( true ) ;
set_dirty ( ) ;
}
2023-09-27 12:03:02 -07:00
void TextEditorView : : show_file_picker ( ) {
auto open_view = nav_ . push < FileLoadView > ( " " ) ;
open_view - > on_changed = [ this ] ( std : : filesystem : : path path ) {
// Can't update the UI focus while the FileLoadView is still up.
// Do this on a continuation instead of in on_changed.
nav_ . set_on_pop ( [ this , p = std : : move ( path ) ] ( ) {
open_file ( p ) ;
2023-06-01 15:45:55 -07:00
hide_menu ( ) ;
2023-09-27 12:03:02 -07:00
} ) ;
} ;
2023-06-01 15:45:55 -07:00
}
void TextEditorView : : show_edit_line ( ) {
auto str = file_ - > get_text ( viewer . line ( ) , 0 , viewer . line_length ( ) ) ;
if ( ! str ) {
nav_ . display_modal ( " Error " , " Failed to get line text. " ) ;
return ;
}
edit_line_buffer_ = * std : : move ( str ) ;
text_prompt (
nav_ ,
edit_line_buffer_ ,
viewer . col ( ) ,
max_edit_length ,
[ this ] ( std : : string & buffer ) {
auto range = file_ - > line_range ( viewer . line ( ) ) ;
if ( ! range )
return ;
prepare_for_write ( ) ;
file_ - > replace_range ( * range , buffer ) ;
} ) ;
nav_ . set_on_pop ( [ this ] ( ) {
edit_line_buffer_ . clear ( ) ;
refresh_ui ( ) ;
hide_menu ( true ) ;
} ) ;
2023-05-26 01:02:17 -07:00
}
2023-06-01 15:45:55 -07:00
void TextEditorView : : show_save_prompt ( std : : function < void ( ) > continuation ) {
if ( ! file_dirty_ ) {
if ( continuation )
continuation ( ) ;
return ;
}
2023-07-13 09:38:40 -07:00
ui : : show_save_prompt (
nav_ ,
[ this ] ( ) { save_temp_file ( ) ; } ,
continuation ) ;
2023-06-01 15:45:55 -07:00
}
void TextEditorView : : prepare_for_write ( ) {
file_dirty_ = true ;
if ( has_temp_file_ )
return ;
2023-07-30 00:36:57 -07:00
// TODO: This would be nice to have but it causes a stack overflow in an ISR?
// Painter p;
// p.draw_string({2, 48}, Styles::yellow, "Creating temporary file...");
2023-06-01 15:45:55 -07:00
// Copy to temp file on write.
has_temp_file_ = true ;
2023-07-13 09:38:40 -07:00
delete_temp_file ( path_ ) ;
copy_file ( path_ , get_temp_path ( path_ ) ) ;
file_ - > assume_file ( get_temp_path ( path_ ) ) ;
2023-06-01 15:45:55 -07:00
}
void TextEditorView : : save_temp_file ( ) {
if ( file_dirty_ ) {
2023-07-13 09:38:40 -07:00
ui : : save_temp_file ( path_ ) ;
2023-06-01 15:45:55 -07:00
file_dirty_ = false ;
}
}
} // namespace ui