FindComponent - usuwanie i "Argument out of range"

FindComponent - usuwanie i "Argument out of range"
maniutek20
  • Rejestracja:około 14 lat
  • Ostatnio:około 5 godzin
  • Postów:138
0

Pozwolę sobie założyć nowy wątek, bo lektura forum nie przyniosła odpowiedzi.
Otóż mam program (galerię obrazów) w którym chciałbym dodać możliwość usuwania wybranych obrazów z widoku. Obrazy są dodawane jako obiekty TImage z przypisanym do nich checkboxem. Poniżej kod procedury usuwania:

Kopiuj
 procedure TForm1.Button5Click(Sender: TObject);
var
  J : Integer;
  S : String;
  NewStr, OldStr : String;
begin
  for J:=0 to componentcount-1 do
    begin
     if (Components[J] is Tcheckbox) and ((Components[J] as Tcheckbox).IsChecked=true)then
          begin
        S:= (components[J].Name);
        oldstr:='czek';
        newstr:='glowny';
        S := StringReplace(S, OldStr, NewStr, [rfReplaceAll]);
      findcomponent(s).free;

           end;
    end;
end;

Kiedy obrazów jest dużo wszystko działa bez zarzutu, ale kiedy schodzę do 3-5 to za każdym razem wyskakuje mi błąd:
"Argument out of range". Mam chyba znaczący brak wiedzy, bo nie mogę tego przeskoczyć.


quidquid Latine dictum sit, altum videtur
Patryk27
Moderator
  • Rejestracja:ponad 17 lat
  • Ostatnio:prawie 2 lata
  • Lokalizacja:Wrocław
  • Postów:13042
0

Musiałbyś zrobić pętlę While i inkrementować tylko wtedy, gdy nie zwalniasz komponentu.


_13th_Dragon
  • Rejestracja:prawie 20 lat
  • Ostatnio:12 dni
0

Kasuj od końca:

Kopiuj
for J:=ComponentCount-1 downto 0 do

Wykonuję programy na zamówienie, pisać na Priv.
Asm/C/C++/Pascal/Delphi/Java/C#/PHP/JS oraz inne języki.
edytowany 1x, ostatnio: _13th_Dragon
GS
  • Rejestracja:ponad 14 lat
  • Ostatnio:minuta
0
Kopiuj
findcomponent(s).free

zmniejsza liczbę komponentów i wymiar tablicy Components, tym samym odwołanie Components[J] może wykraczać poza jej rozmiar

abrakadaber
abrakadaber
  • Rejestracja:ponad 12 lat
  • Ostatnio:8 miesięcy
  • Postów:6610
0

albo pętlę od componentcount - 1 do 0


Chcesz pomocy - pokaż kod - abrakadabra źle działa z techniką.
maniutek20
  • Rejestracja:około 14 lat
  • Ostatnio:około 5 godzin
  • Postów:138
0

Kasując od końca też wywala "Argument out of range". Jestem mocno zdziwiony bo czasami wywala błąd przy 3 ostatnich, czasami przy ostatnim. Jak jest dużo komponentów to się ładnie usuwają, ale przy końcówce zawsze daje błąd. Jest jakaś metoda na wyszukiwanie komponentów wg typu? Tzn. chciałbym znaleźć wszystkie komponenty typu TCheckbox bez znajomości ich nazwy.


quidquid Latine dictum sit, altum videtur
GS
  • Rejestracja:ponad 14 lat
  • Ostatnio:minuta
0

bo 'componentcount' zlicza wszystkie komponenty, w tym również checkboxy

flowCRANE
Poza tym warunek końcowy pętli for obliczany jest tylko raz, co nie ma miejsca w pętli while;
GS
  • Rejestracja:ponad 14 lat
  • Ostatnio:minuta
0

utwórz metodę która będzie kasowała tylko jeden (pierwszy znaleziony) komponent z zaznaczonych do skasowania i zwróci wartość logiczną True jeśli coś skasowano i False jeśli już nic nie było do skasowania

a potem zrób coś takiego :

Kopiuj
while self.kasuj do ;

pętla będzie się wykonywać aż do usunięcia ostatniego komponentu

edytowany 1x, ostatnio: grzegorz_so
maniutek20
  • Rejestracja:około 14 lat
  • Ostatnio:około 5 godzin
  • Postów:138
0

Poszedłem w stronę tego

Kopiuj
 while

, co prawda błąd zniknął ale teraz kasuje mi na raty. Tzn. mam zaznaczone np 9 elementów, najpierw skasuje 5, potem 3 i na końcu jeden. Trochę kodu:

Kopiuj
 begin

J := 0;
while J < componentcount - 1 do
    begin
     if (Components[J] is Tcheckbox) and ((Components[J] as Tcheckbox).IsChecked=true)then
          begin
        S:= (components[J].Name);
        oldstr:='czek';
        newstr:='glowny';
        S := StringReplace(S, OldStr, NewStr, [rfReplaceAll]);
        findcomponent(s).Free;
          end;
       J:=J+1;
          end;
end;

quidquid Latine dictum sit, altum videtur
GS
bo musisz mieć dwie pętle zagnieżdżone , teraz tę pętle zewnętrzną tworzysz przez wielokrotne naciskanie buttona, i dlatego działa :)
GS
  • Rejestracja:ponad 14 lat
  • Ostatnio:minuta
0

Wariant drugi , to tworzysz dynamiczną tablicę zawierającą wskaźniki do komponentów które mają być usunięte.
W pierwszej pętli tylko wyszukujesz komponenty do skasowania i dodajesz do tablicy odpowiednie wskaźniki, a w kolejnej lecisz po tej tablicy i kasujesz

GS
  • Rejestracja:ponad 14 lat
  • Ostatnio:minuta
0
Kopiuj
var
  I: Integer;
  a: array of tcomponent;
begin
  setlength(a, 0);
  for I := 0 to self.Componentcount - 1 do
    if { tu warunek kasowania }  then
    begin
      setlength(a, length(a) + 1);
      a[ high(a)] := self.Components[I];
    end;
  for I := 0 to high(a) do
    a[I].free;
end;
_13th_Dragon
  • Rejestracja:prawie 20 lat
  • Ostatnio:12 dni
0

Dostałeś odpowiedź w drugim poście ale nadal robisz to samo, masz gotowe poprawne rozwiązanie:

Kopiuj
for J:=ComponentCount-1 downto 0 do
  if (Components[J] is Tcheckbox)and((Components[J] as Tcheckbox).IsChecked) then
    findcomponent(StringReplace(components[J].Name,'czek','glowny',[])).Free;

Wykonuję programy na zamówienie, pisać na Priv.
Asm/C/C++/Pascal/Delphi/Java/C#/PHP/JS oraz inne języki.
maniutek20
  • Rejestracja:około 14 lat
  • Ostatnio:około 5 godzin
  • Postów:138
0

Nie jest to poprawne rozwiązanie, niestety. Taki kod zwraca "Argument out of range". Wersja z odliczaniem w dół to pierwsza którą napisałem.


quidquid Latine dictum sit, altum videtur
_13th_Dragon
  • Rejestracja:prawie 20 lat
  • Ostatnio:12 dni
1
Kopiuj
J:=ComponentCount-1;
while J>=0 do
begin
  if (Components[J] is Tcheckbox)and((Components[J] as Tcheckbox).IsChecked) then
  begin
    findcomponent(StringReplace(components[J].Name,'czek','glowny',[])).Free;
    if J>ComponentCount then J:=ComponentCount;
  end;
  Dec(J);  
end;

Wykonuję programy na zamówienie, pisać na Priv.
Asm/C/C++/Pascal/Delphi/Java/C#/PHP/JS oraz inne języki.
maniutek20
  • Rejestracja:około 14 lat
  • Ostatnio:około 5 godzin
  • Postów:138
0

Dobra ciemny jestem, i nie umiem programować:). Sypię głowę popiołem. _13th_Dragon dzięki wielkie, ten kod działa poprawnie i szybko.
Dzięki wszystkim za pomoc


quidquid Latine dictum sit, altum videtur
_13th_Dragon
  • Rejestracja:prawie 20 lat
  • Ostatnio:12 dni
0

Z tym że powinieneś to przerobić na normalne tablice.


Wykonuję programy na zamówienie, pisać na Priv.
Asm/C/C++/Pascal/Delphi/Java/C#/PHP/JS oraz inne języki.
maniutek20
w sensie dodawać komponenty do tablicy i usuwać z tablicy po kolei?
_13th_Dragon
dokładnie, pamiętaj że nawet nie musisz usuwać, wystarczy że wrzucisz na jakiś panel i zrobisz ten panel niewidocznym.
Patryk27
Moderator
  • Rejestracja:ponad 17 lat
  • Ostatnio:prawie 2 lata
  • Lokalizacja:Wrocław
  • Postów:13042
0
```delphi J := 0;

while J < ComponentsCount do
begin
if (Components[J] is TCheckBox) and (TCheckBox(Components[J]).IsChecked) then
begin
findcomponent(StringReplace(components[J].Name,'czek','glowny',[])).Free;
end else
begin
Inc(J);
end;
end;

Kopiuj
Ale pisanie nieskomplikowanego kodu jest takie mainstreamowe, wiem ;p</del>

edytowany 1x, ostatnio: Patryk27
Zobacz pozostałe 3 komentarze
Patryk27
Mea culpa, zwracam honor. Nie wziąłem pod uwagę tego, że przecież usuwamy inny komponent, a nie ten sprawdzany.
_13th_Dragon
To akurat było jasne od początku, ale to: - "Plus potwierdzone doświadczalnie, że działa" - dla mnie wciąż jest zagadką.
Patryk27
Potwierdzone doświadczalnie wrzuceniem na formatkę trzech czekboksów i wywołaniem tego kodu... tyle że bez StringReplace :P
_13th_Dragon
Bardzo czysty eksperyment, gratulacje ;P
Patryk27
Oj tam, oj tam ;p

Zarejestruj się i dołącz do największej społeczności programistów w Polsce.

Otrzymaj wsparcie, dziel się wiedzą i rozwijaj swoje umiejętności z najlepszymi.