Перейти к содержанию
  • Регистрация
  • 0
dmitry

Работа с ListView

Вопрос

Добрый день.

Использую Delphi DX 10 Update 1.

Есть маленькая задача по использованию компонента ListView.

Необходимо, если в Item содержится текст = 'Signal', то цвет у шрифта установить в красный, иначе в черный.

Написал следующий код вставки данных в ListView:

procedure TForm3.Button1Click(Sender: TObject);
Var
  Item: TListViewItem;
  TextObject: TListItemText;
begin
Item := ListView1.Items.Add;
Item.Text := '...';
TextObject := TListItemText.Create(Item);
TextObject.Text := 'Signal';
TextObject.Height := 30;
TextObject.PlaceOffset.Point := TPointF.Create(1, 0);
TextObject.Name := 'STATUS';
TextObject.Font.Size := 20;

Item := ListView1.Items.Add;
Item.Text := '...';
TextObject := TListItemText.Create(Item);
TextObject.Text := 'Normal';
TextObject.Height := 30;
TextObject.PlaceOffset.Point := TPointF.Create(1, 0);
TextObject.Name := 'STATUS';
TextObject.Font.Size := 20;
end;

 

В событии UpdateObjects компонента написал:

procedure TForm3.ListView1UpdateObjects(const Sender: TObject;
  const AItem: TListViewItem);
Var
  TextObject: TListItemText;
begin
TextObject := AItem.Objects.FindDrawable('STATUS') As TListItemText;
If TextObject <> Nil Then
Begin
  If TextObject.Text = 'Signal' Then
    TextObject.TextColor := TAlphaColorRec.Red
  Else
    TextObject.TextColor := TAlphaColorRec.Black;
End;
end;

 

В итоге Item, где есть Signal не устанавливается цвет текста в красный и более того TextObject = Nil.

Помогите разобраться с проблемой.

Тестовый проект прилагаю

LVBag.rar

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Рекомендуемые сообщения

  • 0

А почему сразу не указывать цвет текста при создании ListItem

item :=  ListView.Items.Add;
item.Objects.TextObject.TextColor := ваш цвет;

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0

Конечно, можно и сразу цвет текста присваивать. Но задача мной описана очень кратко. В реальной задаче, в одном Item будет находится более одного TListItemText и отрисовываться они будут по разному.

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0

именно в UpdateObjects нужно создавать нужные Items

при создании нового Item  (Items.add) сразу вызывается OnUpdateObjects

в это время ничего еще не создано, поэтому - NIL

поменяйте логику и все встанет на место

Изменено пользователем krapotkin

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0
procedure TForm3.Button1Click(Sender: TObject);
begin
  with ListView1.Items.Add do
  begin
    Text := '...';
    Data['STATUS'] := 'Signal';
  end;

  with ListView1.Items.Add do
  begin
    Text := '...';
    Data['STATUS'] := 'Normal';
  end;
  ListView1.Resize;
end;

procedure TForm3.ListView1UpdateObjects(const Sender: TObject; const AItem: TListViewItem);
var
  TextObject: TListItemText;
begin
  TextObject := AItem.Objects.FindDrawable('STATUS') As TListItemText;
  if TextObject = nil then
  begin
    TextObject := TListItemText.Create(AItem);
    TextObject.Name := 'STATUS';
  end;
  if TextObject.Text = 'Signal' Then
    TextObject.TextColor := TAlphaColorRec.Red
  else
    TextObject.TextColor := TAlphaColorRec.Black;
end;

остальное сами настроете

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0

Тогда я просто не понимаю, для чего нужно событие UpdateObjects.

Сейчас сделал эксперимент, как предложил Уважаемый Krapotkin.

Стиль для Item = ImageListItemBottomDetailRightButton.

Отдельно создал константы:

Const
  sdStatusSignal = 'sdStatusSignal';
  sdStatusNormal = 'sdStatusNormal';

Заполняю Items по кнопке:

procedure TForm1.Button1Click(Sender: TObject);
Var
  Item: TListViewItem;
begin
//Первый item
Item := ListView1.Items.Add;
Item.Bitmap.Assign(Image1.Bitmap);
Item.Objects.TextObject.Text := 'Task 1';
Item.Objects.DetailObject.Text := '3';
Item.Data[sdStatusSignal] := 'SIGNAL';
//второй item
Item := ListView1.Items.Add;
Item.Bitmap.Assign(Image1.Bitmap);
Item.Objects.TextObject.Text := 'Task 2';
Item.Objects.DetailObject.Text := '0';
Item.Data[sdStatusNormal] := 'NORMAL';
end;

И сам обработчик UpdateObjects, в котором рисую новый объект:

procedure TForm1.ListView1UpdateObjects(const Sender: TObject;
  const AItem: TListViewItem);
Var
  StatusSignal: TListItemText;
  StatusNormal: TListItemText;
begin
StatusSignal := AItem.Objects.FindDrawable(sdStatusSignal) As TListItemText;
If StatusSignal = Nil Then
Begin
  StatusSignal := TListItemText.Create(AItem);
  StatusSignal.Name := sdStatusSignal;
  StatusSignal.Width := 50;
  StatusSignal.Height := 20;
  StatusSignal.PlaceOffset.Point := TPointF.Create(70, 0);
  StatusSignal.TextColor := TAlphaColorRec.Red;
End;
StatusNormal := AItem.Objects.FindDrawable(sdStatusNormal) As TListItemText;
If StatusNormal = Nil Then
Begin
  StatusNormal := TListItemText.Create(AItem);
  StatusNormal.Name := sdStatusNormal;
  StatusNormal.Width := 50;
  StatusNormal.Height := 20;
  StatusNormal.PlaceOffset.Point := TPointF.Create(70, 0);
  StatusNormal.TextColor := TAlphaColorRec.Black;
End;
end;

Только совсем не понимаю как этот обработчик работает. StatusSingal и StatusNormal всегда будут для Item = Nil, что приведет к созданию новых TListItemText. Очень интересный момент получается при выводе, показывает все правильно (в приложении к ответу).

При создании первого Item я должен создать только sdStatusSingal, но в отладчике видно что также создается и sdStatusNormal хотя и не отображается.

Может мне кто-нибудь этот момент объяснить?

P.S. С ListView только пробую работать, есть пока что некое непонимае.

 

 

Безымянный.png

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0

компоненты же для каждого Item создаются отдельно.

OnUpdateObjects - это и есть место для создания всяких своих дополнительных ListViewItems Возможно еще когда-то вызываются, можно по текстам поискать

по слухам, разработчики готовят что-то новенькое к апрелю, как раз для облегчения этого процесса, т.к. ListView - один из основных компонентов для мобильных приложений

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
  • 0

Zuby и Krapotkin Спасибо Вам за помощь.

Zuby, сделал как Вы и описывали

procedure TForm1.Button1Click(Sender: TObject);
Var
  Item: TListViewItem;
begin
//Первый item
Item := ListView1.Items.Add;
Item.Bitmap.Assign(Image1.Bitmap);
Item.Objects.TextObject.Text := 'Task 1';
Item.Objects.DetailObject.Text := '3';
//Item.Data[sdStatusSignal] := 'SIGNAL';
Item.Data['STATUS'] := 'SIGNAL';
//второй item
Item := ListView1.Items.Add;
Item.Bitmap.Assign(Image1.Bitmap);
Item.Objects.TextObject.Text := 'Task 2';
Item.Objects.DetailObject.Text := '0';
//Item.Data[sdStatusNormal] := 'NORMAL';
Item.Data['STATUS'] := 'NORMAL';
ListView1.Resize;
end;

procedure TForm1.ListView1UpdateObjects(const Sender: TObject;
  const AItem: TListViewItem);
Var
  //StatusSignal: TListItemText;
  //StatusNormal: TListItemText;
  TextObject: TListItemText;
begin
TextObject := AItem.Objects.FindDrawable('STATUS') As TListItemText;
If TextObject = Nil Then
Begin
  TextObject := TListItemText.Create(AItem);
  TextObject.Name := 'STATUS';
End;
If TextObject.Text = 'SIGNAL' Then
  TextObject.TextColor := TAlphaColorRec.Red
Else
  TextObject.TextColor := TAlphaColorRec.Black;

Только в последнем обработчике когда идет сравнение TextObject.Text = 'SIGNAL', свойство Text равно пустому значению всегда.  

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.

Гость
Ответить на вопрос...

×   Вставлено с форматированием.   Вставить как обычный текст

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.


  • Похожий контент

    • От krapotkin
      Всем привет!
      После долгого творческого запоя я запилил альфу либы, которая позволяет юзать обычный, не-генномодифицированный ListView.
      Данные подаются в виде модели данных, описание раскладки итема лежит в JSON.

      Для работы пишется примитивный наследник класса-адаптера, который биндит данные модели на элементы из JSON. Я решил, что руками делать это дешевле чем через RTTI
      { TMyAdapter } procedure TMyAdapter.SetupDrawableContent(const ADrawable: TListItemDrawable; const AData: TMyData); begin if SameText( ADrawable.Name, 'text') then begin (ADrawable as TListItemText).Text := AData.Text; end else if SameText( ADrawable.Name, 'detail') then begin (ADrawable as TListItemText).Text := AData.Detail; end else if SameText( ADrawable.Name, 'balance') then begin (ADrawable as TListItemText).Text := FormatFloat('0.00', AData.Balance); end else if SameText( ADrawable.Name, 'reserved') then begin (ADrawable as TListItemText).Text := FormatFloat('0.00', AData.Reserved); end end; Главная хитрость и отличие от стандартного DynamicAppearance+LiveBinding - переменная высота итемов ListView и использование арифметики в описании раскладки.
      Можно указать Detail.Y  = Text.Bottom + 5, ItemHeight = Detail.Bottom + 10, Detail.W = ItemWidth/2 - X
      {"Kind":"rect","Name":"BonusRect","Value":"", "Place":{"X":"itemwidth*3/4","Y":"Text.Y","W":"ItemWidth/4-5","H":"50"}, "BorderColor":"#FF005500", "Color":"lime", "LineWidth":3}, {"TextHAlign":2,"TextVAlign":1,"Kind":"text","Name":"Balance","Value":"", "Place":{"X":"BonusRect.x+5","Y":"Text.Y","W":"BonusRect.w-10","H":"50"}, "WordWrap":true, "Color" :"Black", "Font":{"Size":18,"Style":""}} ], "ItemHeight":"detail.bottom+10", Это дает довольно гибкую систему. Не на все случаи жизни, но все, что нужно, можно после автоматической раскладки дополнительно приписать в OnUpdateObjects 
      Код для работы примерно такой
      procedure TForm1.FormCreate(Sender: TObject); begin data:= TMyDataList.CreateFromFile(ExePath()+'data.json'); // загрузка в модель данных Adapter := TMyAdapter.Create(lvWallets, data.Items, ExePath()+'pattern.json'); // создание адаптера и загрузка шаблона Adapter.Pattern.SetupListView(lvWallets); // задать отступы и разную мелочь Adapter.ResetView(); // здесь в цикле из модели данных создается нужное количество итемов ListView end; procedure TForm1.lvWalletsUpdatingObjects(const Sender: TObject; const AItem: TListViewItem; var AHandled: Boolean); begin Adapter.SetupContent(AItem); // загрузить текст, картинки и другое содержимое в элементы итема Adapter.Pattern.DoLayout(AItem); // поправить раскладку в соответствии с содержимым элементов итема end; Для использования кроме стандартных - текст, картинка, кнопка - создано несколько дополнительных элементов итема - прямоугольник, круг, уголок, линия.
      Отличие от ModernLV - , 
      1) все происходит без правки системных файлов
      2) пока нет колонок и других особых изысков.
      На результат работы смотреть тут. Исходники пока не причесаны, будут чуть позже
       
      ListViewFramework.7z
    • От Dmitry Stolyarov
      Добрый день, подскажите, пжл, новичку...
      Есть ListView и ImageList с двумя картинками. В ListView Item добавлен Image. ListView заполняю из БД и пытаюсь вывести картинку в зависимости от значения поля БД, но в независимости от значения БД выводится одна и та же картинка.. Такое нельзя провернуть?
      procedure TForm6.ListViewNAMEUpdateObjects(const Sender: TObject;
        const AItem: TListViewItem);
        var IItem:TListItemImage;
      begin
       IItem:= aItem.Objects.FindDrawable('Image4') as TListItemImage;
      if (IItem<>nil) then
      with FDQuery1 do
       begin
       First;
       while not Eof do
        begin
         if FDQuery1FK_PRICE.AsInteger = 1 then begin
          IItem.ImageIndex:=1
         end else
            begin
               IItem.OwnsBitmap := True;
           IItem.ImageIndex:=0;
            end;
         Next;
        end;
       end;
      end;
       
    • От gresaggr
      Добрый день.
      Как под FMX в ListView сделать подчеркивание Detail в рантайме?
      Пытался делать так:
      LV.ItemAppearanceObjects.ItemObjects.Detail.Font.Style := [TFontStyle.fsUnderline]
      но не срабатывает.
      P.S. Возможно это применить не ко всем строкам, а только к некоторым?
    • От ice donkey
      Заполняю ListView в рантайме:
       
       
      LiveBindings не подходит, поэтому есть вопрос: а как задать фиксированный заголовок - чтобы в списке все Items, начинающиеся с буквы А были как бы объединены под одним заголовком А? Который  в свою очередь смещался бы только заголовком Б (с Items, которые на букву Б) и так далее.
       
      Пример:
       
    • От Olexander Sagaydak
      Летом 2016 вопрос отображения объектов в ListView при создании Items уже обсуждался. Хочу ещё раз обсудить этот вопрос. А именно:
      Простой код 
      ListViewItem := ListView.Items.Add;
      ListViewItem.Objects.AccessoryObject.Visible := false; (или true, не важно)
      вторая строка не работает если до выполнения, ListView ни одного раза не отображался. Если же отображался хотя бы один раз - всё нормально.
      В предыдущих обсуждениях Krapotkin рекомендовал Resize и OnUpdateObjects. Если нет идей проще, то, конечно, воспользуюсь.  
       
    • От x11
      Если у элементов, идущих подряд одинаковый текст, то текст скрывается.
      Как это это отключить?
       
    • От x11
      Не понять, ни в справке найти, ни примеров, как получить текст из header.
      Как получить текст из какого-то элемента, есть пример:
      lvContacts.Items[li.Index].Objects.FindObjectT<TListItemText>('textTel').Text; а для заголовка ил подвала не нахожу примеров
    • От Erlan_krg
      Всем привет! Встретился с одной проблемой. Удаляю Item у ListView по нажатию кнопки, которая находится на ListViewItem
      procedure TfmAppointment.lvSelectedServicesButtonClick(const Sender: TObject;
        const AItem: TListItem; const AObject: TListItemSimpleControl);
      begin
          lvSelectedServices.BeginUpdate;
          if lvSelectedServices.ItemCount > 0 then
              lvSelectedServices.Items.Delete(AItem.Index);
          lvSelectedServices.EndUpdate;
      end;
      итем удаляется, но потом выскакивает ошибка Access violation at address 60BCB318 и то се бо се
       
      у ListView ItemAppearance выставлен на DynamicAppearance
      помимо родного объекта Text добавил еще один объект Text и кнопку
       
      но фишка в чем, если убрать дополнительный объект Text с ListWiewItem и удалить итем то ошибка не выскакивает)
       
      что за прикол?
    • От sinuke
      Доброго дня, форумчане
      Возникла проблема с тем, чтобы поменять местами два итема в ListView. ListView.Items не имеет метода Exchange, сами итемы не имеют метода Assign чтобы можно было один итем клонировать в другой / удалить / вставить
      У кого какие мысли? Как реализовать?
      Delphi 10.2 Tokyo
       
    • От x11
      Не могу понять, как записать текст в Footer text
       
       

  • Последние посетители   0 пользователей онлайн

    Ни одного зарегистрированного пользователя не просматривает данную страницу

×
×
  • Создать...