Недавно volvo помог доработать программу, которая позволяла при вводе строки перебираться к любому символу, вставлять либо удалять символ в любом месте, теперь я решил усложнить задание и сделать процедуру, суть которой описана чуть ниже. Идея данной процедура очень удобна при работе с записями, например прося пользователя ввести определённое поле. Что - то я сам очень сильно запутался, описал всё с комментами что вам легче было понять то, что я пытался сделать. Программа компилится, но опято что - то не то выводит, снова нуждаюсь в вашей помощи.
uses crt; var s : string; Text : boolean; {Процедура должна обеспечивать ввод строки длиной длина, при этом параметр парам отвечает за то, какая это строка, численная или буквенная. Во время введения строки возможно перемезаться по ней, убирая или прибавляя символ, использовать backspace Stroka - рабочая строка, dlina - величина, которую не может превышать вводимая строка} procedure MyInput(var Stroka : string; dlina : integer; param:boolean); {param - true - буквенный ряд, false - численный} var i,k,len : integer; ch : char; CopyStroka : string; active : integer; begin for i:=1 to dlina do Stroka[i]:=' '; Stroka[0]:=chr(dlina); i:=1; Active:=1; while true do begin ch := ReadKey; if((ch >= 'A') and (ch <= 'Z') or (ch >= 'a') and (ch <= 'z') or (ch = '_') or (Ord(ch) = 32)) and param {условие для ввода букв} or ((ch >= '0') and (ch <= '9') or (ch = ':') or (ch = '.')) and not param then {условие для ввода чисел} begin if(i>dlina) and (Ord(ch) <> 8) then continue; {если нет возможности дописать символ} if(i=1)then ch:=UpCase(ch) {первую букву делаем большой} else if(ch >= 'A') and (ch <= 'Z') then ch:=Chr(Ord(ch)+32); {остальные буквы только малы} write(ch); Stroka[i]:=ch; {Прибавляем символ к строке} i:=i+1; end;
Case readkey of #13 : if i > 1 then break; {если введён 1 символ и ентер то выходим из цикла}
#8: If i> 1 then begin {BACKSPACE} i:=i-1; Stroka[i]:=' '; {вместо последнего символа пробел} gotoxy(Wherex-1, Wherey); write(' '); gotoxy(Wherex-1, Wherey);
end; #75: if Active > 1 then begin { LEFT } dec(Active); gotoXY(Active, WhereY); end; #77: if WhereX <= length(Stroka) then begin { RIGHT } inc(Active); gotoXY(Active, Wherey); end; #83: if Active <= Length(Stroka) then Delete(Stroka, Active, 1); { DELETE }
'a' .. 'z': If param and (i < dlina) then {insert для символьного ряда} begin if Active <= Length(Stroka) then begin Insert(ch, Stroka, active); inc(i); end else begin Stroka[i]:=ch; inc(i); inc(Active); end; end; '1'..'9' : if not param and (i < dlina) then {insert для числового ряда} begin if Active <= Length(Stroka) then begin Insert(ch, Stroka, active); inc(i); end else begin Stroka[i]:=ch; inc(i); inc(Active); end; end; end;
if not param then {уничтожаем пробелы при вводе числового ряда} begin len := 0; for i:= 1 to ord(Stroka[0]) do if Stroka[i] <> ' ' then begin len := len + 1; CopyStroka[len] := Stroka[i] end; CopyStroka[0] := chr(len); Stroka := CopyStroka end
end; end; begin Text:=True; MyInput(S,5,Text); end.
volvo
10.04.2007 17:47
Опять ты нарушил всю структуру... Я же показывал, что Left/Right/Delete имеют расширенные коды, и как их обрабатывать тоже показывал... Смотри, я добавил в предыдущую свою программу несколько строк:
uses crt;
function get_string(max_len: integer; param: boolean): string;
var Active: Integer; s: string; ch: char;
type tset = set of char; const char_set: array[boolean] of tset = ( ['0' .. '9', ':', '.'], ['a' .. 'z', 'A' .. 'Z', '_', #32] );
begin writeln('Vvedite stroku'); s := '';
Active := 1; repeat gotoxy(1, WhereY); clreol; write(s); gotoxy(Active, 1);
ch := readkey; case ch of #0: case readkey of #75: if Active > 1 then begin { LEFT } dec(Active); gotoXY(Active, WhereY); end;
#77: if WhereX <= length(s) then begin { RIGHT } inc(Active); gotoXY(Active, Wherey); end;
#83: if Active <= Length(s) then Delete(s, Active, 1); { DELETE }
end;
#8: if length(s) <> 0 then begin { BS } dec(Active); gotoXY(Active, Wherey); Delete(s, Active, length(s)); end;
#13: break;
else begin
if (length(s) < max_len) and (ch in char_set[param]) then begin
if Active <= Length(s) then Insert(ch, s, active) else s := s + ch; inc(Active);
end;
end;
end;
until false; get_string := s;
end;
var s: string; begin clrscr; writeln('string: '); s := get_string(5, true); writeln; writeln('s = ', s); end.
(true означает текстовую строку, false - числовую)
Tan
10.04.2007 17:51
спасибо огромное, у меня с большими программами всегда такая путанница, буду переучиваться, вы очень помогли!
Tan
10.04.2007 18:24
Кстати очень люботно, а как минимальными изменениями добиться такого же результата, но если учесть, что до вызова процедуры уже задано место gotoXY, где должна выводиться строка. Просто получается что в привидённом выше примере всё взаимозависимо, сначала я подумал, что просто принять за х начальный wherex за y начальный wherey и там уже при сравнении с длиной отнимать от active x (забавно получилось ActiveX ). Но не особо представляю как до конца это сделать.
volvo
10.04.2007 18:41
Такие изменения считаются минимальными?(Показать/Скрыть)
uses crt;
function get_string(max_len: integer; param: boolean): string;
var Active: Integer; s: string; ch: char; start_pos: integer;
type tset = set of char; const char_set: array[boolean] of tset = ( ['0' .. '9', ':', '.'], ['a' .. 'z', 'A' .. 'Z', '_', #32] );
begin start_pos := WhereX; s := '';
Active := 1; repeat gotoxy(start_pos, WhereY); write('':max_len); gotoxy(start_pos, WhereY); write(s);
gotoxy(start_pos + Active - 1, WhereY);
ch := readkey; case ch of #0: case readkey of #75: if Active > 1 then begin { LEFT } dec(Active); gotoXY(start_pos + Active - 1, WhereY); end;
#77: if WhereX <= (start_pos + length(s) - 1) then begin { RIGHT } inc(Active); gotoXY(start_pos + Active - 1, Wherey); end;
#83: if Active <= Length(s) then Delete(s, Active, 1); { DELETE }
end;
#8: if length(s) <> 0 then begin { BS } dec(Active); gotoXY(start_pos + Active - 1, Wherey); Delete(s, Active, 1); end;
#13: break;
else begin
if (length(s) < max_len) and (ch in char_set[param]) then begin
if Active <= Length(s) then Insert(ch, s, active) else s := s + ch; inc(Active);