Версия для печати темы

Нажмите сюда для просмотра этой темы в обычном формате

Форум «Всё о Паскале» _ Задачи _ Продолжение экспериментов со строкой

Автор: Tan 10.04.2007 21:06

Недавно 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 21: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 21:51

спасибо огромное, у меня с большими программами всегда такая путанница, буду переучиваться, вы очень помогли!

Автор: Tan 10.04.2007 22:24

Кстати очень люботно, а как минимальными изменениями добиться такого же результата, но если учесть, что до вызова процедуры уже задано место gotoXY, где должна выводиться строка. Просто получается что в привидённом выше примере всё взаимозависимо, сначала я подумал, что просто принять за х начальный wherex за y начальный wherey и там уже при сравнении с длиной отнимать от active x (забавно получилось ActiveX smile.gif ). Но не особо представляю как до конца это сделать.

Автор: volvo 10.04.2007 22:41

Такие изменения считаются минимальными? (Показать/Скрыть)