Отображать дополнительный текст в узлах дерева, а не только в файле node.text.

У меня есть TTreeView в Delphi, с узлами на трех уровнях.

Я использую данные узла для хранения другой метки, кроме текста узла.

Type
  TNodeData = class
    ExtraNodeLabel: WideString;
    //... other members
  end;

У меня есть событие OnAdvancedCustomDrawItem, где я хочу отобразить это ExtraNodeLabel перед текстом узла. Я хочу добиться этого:

  • Синий текст будет дополнительной меткой.
  • выделенный элемент: первые два слова также являются дополнительной меткой

Что я получил так далеко, это:

Проблемы:

  1. По какой-то причине я не могу нарисовать текст с другим стилем, если я использую DrawText/drawTextW (Мне нужен drawtextW из-за данных Unicode)
  2. Другая проблема заключается в том, что все, что находится за пределами прямоугольника с точечным фокусом, является не щелкаемым

Что нужно решить:

  1. Как я могу нарисовать текст с другим стилем, используя DrawText/DrawtextW
  2. Как я могу сделать весь текст кликабельным?

Код:

procedure TMainForm.TntTreeView1AdvancedCustomDrawItem(
  Sender: TCustomTreeView; Node: TTreeNode; State: TCustomDrawState;
  Stage: TCustomDrawStage; var PaintImages, DefaultDraw: Boolean);
var
  txtrect, fullrect : TRect;
  DC: HDC;
  fs: integer;
  fc: TColor;
  ExtralabelRect: TRect;
  nData: TNodeData;
begin
  nData := nil;

  if assigned(Node.Data) then begin
    nData := TNodeData(Node.Data);
  end;

  DC := TntTreeView1.canvas.Handle;
  txtRect := Node.DisplayRect(True);    
  fullrect := Node.DisplayRect(False);

  if stage = cdPostPaint then begin
    TntTreeView1.Canvas.FillRect(txtRect);
    if (cdsFocused In State) And (cdsSelected in State) then begin
      DrawFocusRect(DC,txtRect);
    end;

    txtRect.Left := txtRect.Left + 1;
    txtRect.Top := txtRect.Top + 1;
    txtRect.Right := txtRect.Right - 1;
    txtRect.Bottom := txtRect.Bottom - 1;

    ExtralabelRect := txtRect;

    fs := TntTreeView1.Canvas.Font.size;
    fc := TntTreeView1.Canvas.Font.Color;

    if (nData <> nil) And (nData.ExtraNodeLabel <> '') then begin
      TntTreeView1.Canvas.Font.Size := 7;
      TntTreeView1.Canvas.Font.color := clBlue;
      DrawTextW(
        DC,
        PWideChar(nData.ExtraNodeLabel),
        Length(nData.ExtraNodeLabel),
        ExtraLabelRect,
        DT_LEFT or DT_CALCRECT or DT_VCENTER
      );

      DrawTextW(
        DC,
        PWideChar(nData.ExtraNodeLabel),
        Length(nData.ExtraNodeLabel),
        ExtraLabelRect,
        DT_LEFT or DT_VCENTER
      );

      txtRect.right := txtRect.Right + ExtraLabelRect.Right + 5;
      txtRect.Left := ExtraLabelRect.Right + 5;
    end;

    TntTreeView1.Canvas.Font.Size := fs;
    TntTreeView1.Canvas.Font.color := fc;

    DrawTextW(
      DC,
      PWideChar((Node as TTntTreeNode).Text),
      -1,
      txtRect,
      DT_LEFT or DT_VCENTER
    );
  end;
end;

1 ответ

Решение

Решение ОП

Мне удалось частично решить пользовательский рисунок, определив переменную TFont и используя SelectObject а также setTextColor, Установка цвета и стиля шрифта работает, но установка размера шрифта - нет.

var 
  nFont: TFont;
begin
  DC := TntTreeView1.Canvas.Handle;
  NFont := TFont.Create;

  // rest of the code here ...

  // i tried to set nFont.Size, but it doesn't seem to work
  nFont.Size := 7;
  nFont.Color := colorToRGB(clBlue);
  nFont.Style := TntTreeview1.Font.Style + [fsBold];

  SelectObject(DC,NFont.Handle);
  SetTextColor(DC,colortoRGB(clBlue));

  DrawTextW(
    DC,
    PWideChar(nData.nodeLabel),
    Length(nData.nodeLabel),
    ExtraLabelRect,
    DT_LEFT or DT_VCENTER
  );

  // rest of the code here
end;

Источник: я использовал идею отсюда


Обновление 2

Я решил вторую проблему, установив вид дерева RowSelect свойство к истине. Для этого, чтобы работать, я должен был установить ShowLines свойство false, а пользовательские рисуют линии и кнопки. Это работает сейчас.


Обновление 3

Я улучшил решение для первой проблемы, не создавая новый шрифт, а выбрав шрифт холста для отображения текста, и таким образом я смог изменить любой аспект шрифта, а также были применены системные параметры cleartype:

// set font size for the canvas font (font style can be set the same time)
TntTreeView1.Canvas.Font.Size := 7;

// select canvas font for DC
SelectObject(DC,TntTreeView1.Canvas.Font.Handle);

// set font color
SetTextColor(DC,colortoRGB(clBlue));
Другие вопросы по тегам