blob: 5f6f04d51b36e5abefffd55dcbf363066df4390a [file] [log] [blame]
/*
* Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
*
* 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 of the
* License, or 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; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
FILE_LICENCE ( GPL2_OR_LATER );
#include <assert.h>
#include <string.h>
#include <ctype.h>
#include <ipxe/keys.h>
#include <ipxe/editstring.h>
/** @file
*
* Editable strings
*
*/
static void insert_delete ( struct edit_string *string, size_t delete_len,
const char *insert_text )
__attribute__ (( nonnull (1) ));
static void insert_character ( struct edit_string *string,
unsigned int character ) __nonnull;
static void delete_character ( struct edit_string *string ) __nonnull;
static void backspace ( struct edit_string *string ) __nonnull;
static void previous_word ( struct edit_string *string ) __nonnull;
static void kill_word ( struct edit_string *string ) __nonnull;
static void kill_sol ( struct edit_string *string ) __nonnull;
static void kill_eol ( struct edit_string *string ) __nonnull;
/**
* Insert and/or delete text within an editable string
*
* @v string Editable string
* @v delete_len Length of text to delete from current cursor position
* @v insert_text Text to insert at current cursor position, or NULL
*/
static void insert_delete ( struct edit_string *string, size_t delete_len,
const char *insert_text ) {
size_t old_len, max_delete_len, insert_len, max_insert_len, new_len;
/* Calculate lengths */
old_len = strlen ( string->buf );
assert ( string->cursor <= old_len );
max_delete_len = ( old_len - string->cursor );
if ( delete_len > max_delete_len )
delete_len = max_delete_len;
insert_len = ( insert_text ? strlen ( insert_text ) : 0 );
max_insert_len = ( ( string->len - 1 ) - ( old_len - delete_len ) );
if ( insert_len > max_insert_len )
insert_len = max_insert_len;
new_len = ( old_len - delete_len + insert_len );
/* Fill in edit history */
string->mod_start = string->cursor;
string->mod_end = ( ( new_len > old_len ) ? new_len : old_len );
/* Move data following the cursor */
memmove ( ( string->buf + string->cursor + insert_len ),
( string->buf + string->cursor + delete_len ),
( max_delete_len + 1 - delete_len ) );
/* Copy inserted text to cursor position */
memcpy ( ( string->buf + string->cursor ), insert_text, insert_len );
string->cursor += insert_len;
}
/**
* Insert character at current cursor position
*
* @v string Editable string
* @v character Character to insert
*/
static void insert_character ( struct edit_string *string,
unsigned int character ) {
char insert_text[2] = { character, '\0' };
insert_delete ( string, 0, insert_text );
}
/**
* Delete character at current cursor position
*
* @v string Editable string
*/
static void delete_character ( struct edit_string *string ) {
insert_delete ( string, 1, NULL );
}
/**
* Delete character to left of current cursor position
*
* @v string Editable string
*/
static void backspace ( struct edit_string *string ) {
if ( string->cursor > 0 ) {
string->cursor--;
delete_character ( string );
}
}
/**
* Move to start of previous word
*
* @v string Editable string
*/
static void previous_word ( struct edit_string *string ) {
while ( string->cursor &&
isspace ( string->buf[ string->cursor - 1 ] ) ) {
string->cursor--;
}
while ( string->cursor &&
( ! isspace ( string->buf[ string->cursor - 1 ] ) ) ) {
string->cursor--;
}
}
/**
* Delete to end of previous word
*
* @v string Editable string
*/
static void kill_word ( struct edit_string *string ) {
size_t old_cursor = string->cursor;
previous_word ( string );
insert_delete ( string, ( old_cursor - string->cursor ), NULL );
}
/**
* Delete to start of line
*
* @v string Editable string
*/
static void kill_sol ( struct edit_string *string ) {
size_t old_cursor = string->cursor;
string->cursor = 0;
insert_delete ( string, old_cursor, NULL );
}
/**
* Delete to end of line
*
* @v string Editable string
*/
static void kill_eol ( struct edit_string *string ) {
insert_delete ( string, ~( ( size_t ) 0 ), NULL );
}
/**
* Replace editable string
*
* @v string Editable string
* @v replacement Replacement string
*/
void replace_string ( struct edit_string *string, const char *replacement ) {
string->cursor = 0;
insert_delete ( string, ~( ( size_t ) 0 ), replacement );
}
/**
* Edit editable string
*
* @v string Editable string
* @v key Key pressed by user
* @ret key Key returned to application, or zero
*
* Handles keypresses and updates the content of the editable string.
* Basic line editing facilities (delete/insert/cursor) are supported.
* If edit_string() understands and uses the keypress it will return
* zero, otherwise it will return the original key.
*
* This function does not update the display in any way.
*
* The string's edit history will be updated to allow the caller to
* efficiently bring the display into sync with the string content.
*/
int edit_string ( struct edit_string *string, int key ) {
int retval = 0;
size_t len = strlen ( string->buf );
/* Prepare edit history */
string->last_cursor = string->cursor;
string->mod_start = string->cursor;
string->mod_end = string->cursor;
/* Interpret key */
if ( ( key >= 0x20 ) && ( key <= 0x7e ) ) {
/* Printable character; insert at current position */
insert_character ( string, key );
} else switch ( key ) {
case KEY_BACKSPACE:
/* Backspace */
backspace ( string );
break;
case KEY_DC:
case CTRL_D:
/* Delete character */
delete_character ( string );
break;
case CTRL_W:
/* Delete word */
kill_word ( string );
break;
case CTRL_U:
/* Delete to start of line */
kill_sol ( string );
break;
case CTRL_K:
/* Delete to end of line */
kill_eol ( string );
break;
case KEY_HOME:
case CTRL_A:
/* Start of line */
string->cursor = 0;
break;
case KEY_END:
case CTRL_E:
/* End of line */
string->cursor = len;
break;
case KEY_LEFT:
case CTRL_B:
/* Cursor left */
if ( string->cursor > 0 )
string->cursor--;
break;
case KEY_RIGHT:
case CTRL_F:
/* Cursor right */
if ( string->cursor < len )
string->cursor++;
break;
default:
retval = key;
break;
}
return retval;
}