2012-09-26 14 views
12

Ho bisogno di un componente di editor di tag VCL per Delphi o C++ Builder, simile a ciò che è disponibile per JavaScript: ad es. this one o this one o l'editor di tag di StackOverflow.Componente editor di tag per Delphi/C++ Builder

È disponibile qualcosa del genere o devo elaborarlo da zero?

Alcune cose specifiche che ho bisogno sono:

  • Editor dovrebbe consentire sia lo scorrimento o diventare multi-linea, se più tag sono presenti alla larghezza del editor permette. Se multipiattaforma, dovrebbe esserci un'opzione per definire un'altezza massima comunque, impedendo che diventi troppo alta
  • Opzione per selezionare se i tag vengono creati premendo il tasto spazio o virgola
  • Testo prompt nell'editor, quando non è focalizzata (ad esempio "Aggiungi nuovo tag")
  • Idealmente, si dovrebbe essere in grado di muoversi tra i tag (evidenziandoli) utilizzando le frecce della tastiera, in modo da poter eliminare qualsiasi tag utilizzando la tastiera unica
+1

Forse puoi modificare [TagCloud per VCL] (http://www.be-precision.com/products/tagcloud/) per fare ciò che vuoi. "La definizione degli elementi di tag semplici, sia in fase di progettazione che in fase di esecuzione" è molto promettente. –

risposta

27

Certo che lo vuoi fare da solo! Scrivere controlli GUI è divertente e gratificante!

Si potrebbe fare qualcosa di simile

unit TagEditor; 

interface 

uses 
    Windows, Messages, SysUtils, Classes, Controls, StdCtrls, Forms, Graphics, 
    Types, Menus; 

type 
    TClickInfo = cardinal; 
    GetTagIndex = word; 

const TAG_LOW = 0; 
const TAG_HIGH = MAXWORD - 2; 
const EDITOR = MAXWORD - 1; 
const NOWHERE = MAXWORD; 

const PART_BODY = $00000000; 
const PART_REMOVE_BUTTON = $00010000; 

function GetTagPart(ClickInfo: TClickInfo): cardinal; 

type 
    TTagClickEvent = procedure(Sender: TObject; TagIndex: integer; 
    const TagCaption: string) of object; 
    TRemoveConfirmEvent = procedure(Sender: TObject; TagIndex: integer; 
    const TagCaption: string; var CanRemove: boolean) of object; 
    TTagEditor = class(TCustomControl) 
    private 
    { Private declarations } 
    FTags: TStringList; 
    FEdit: TEdit; 
    FBgColor: TColor; 
    FBorderColor: TColor; 
    FTagBgColor: TColor; 
    FTagBorderColor: TColor; 
    FSpacing: integer; 
    FTextColor: TColor; 
    FLefts, FRights, FWidths, 
    FTops, FBottoms: array of integer; 
    FCloseBtnLefts, FCloseBtnTops: array of integer; 
    FCloseBtnWidth: integer; 
    FSpaceAccepts: boolean; 
    FCommaAccepts: boolean; 
    FSemicolonAccepts: boolean; 
    FTrimInput: boolean; 
    FNoLeadingSpaceInput: boolean; 
    FTagClickEvent: TTagClickEvent; 
    FAllowDuplicates: boolean; 
    FPopupMenu: TPopupMenu; 
    FMultiLine: boolean; 
    FTagHeight: integer; 
    FEditPos: TPoint; 
    FActualTagHeight: integer; 
    FShrunk: boolean; 
    FEditorColor: TColor; 
    FTagAdded: TNotifyEvent; 
    FTagRemoved: TNotifyEvent; 
    FOnChange: TNotifyEvent; 
    FOnRemoveConfirm: TRemoveConfirmEvent; 
    FMouseDownClickInfo: TClickInfo; 
    FCaretVisible: boolean; 
    FDragging: boolean; 
    FAutoHeight: boolean; 
    FNumRows: integer; 
    procedure SetBorderColor(const Value: TColor); 
    procedure SetTagBgColor(const Value: TColor); 
    procedure SetTagBorderColor(const Value: TColor); 
    procedure SetSpacing(const Value: integer); 
    procedure TagChange(Sender: TObject); 
    procedure SetTags(const Value: TStringList); 
    procedure SetTextColor(const Value: TColor); 
    procedure ShowEditor; 
    procedure HideEditor; 
    procedure EditKeyPress(Sender: TObject; var Key: Char); 
    procedure mnuDeleteItemClick(Sender: TObject); 
    procedure SetMultiLine(const Value: boolean); 
    procedure SetTagHeight(const Value: integer); 
    procedure EditExit(Sender: TObject); 
    function Accept: boolean; 
    procedure SetBgColor(const Value: TColor); 
    function GetClickInfoAt(X, Y: integer): TClickInfo; 
    function GetSeparatorIndexAt(X, Y: integer): integer; 
    procedure CreateCaret; 
    procedure DestroyCaret; 
    function IsFirstOnRow(TagIndex: integer): boolean; inline; 
    function IsLastOnRow(TagIndex: integer): boolean; 
    procedure SetAutoHeight(const Value: boolean); 
    protected 
    { Protected declarations } 
    procedure Paint; override; 
    procedure MouseDown(Button: TMouseButton; Shift: TShiftState; X: Integer; 
     Y: Integer); override; 
    procedure MouseMove(Shift: TShiftState; X: Integer; Y: Integer); override; 

    procedure KeyPress(var Key: Char); override; 
    procedure WndProc(var Message: TMessage); override; 
    procedure KeyDown(var Key: Word; Shift: TShiftState); override; 
    procedure MouseUp(Button: TMouseButton; Shift: TShiftState; X: Integer; 
     Y: Integer); override; 
    public 
    { Public declarations } 
    constructor Create(AOwner: TComponent); override; 
    destructor Destroy; override; 
    published 
    { Published declarations } 
    property TabOrder; 
    property TabStop; 
    property Color; 
    property Anchors; 
    property Align; 
    property Tag; 
    property Cursor; 
    property BgColor: TColor read FBgColor write SetBgColor; 
    property BorderColor: TColor read FBorderColor write SetBorderColor; 
    property TagBgColor: TColor read FTagBgColor write SetTagBgColor; 
    property TagBorderColor: TColor read FTagBorderColor 
     write SetTagBorderColor; 
    property Spacing: integer read FSpacing write SetSpacing; 
    property Tags: TStringList read FTags write SetTags; 
    property TextColor: TColor read FTextColor write SetTextColor; 
    property SpaceAccepts: boolean read FSpaceAccepts write FSpaceAccepts 
     default true; 
    property CommaAccepts: boolean read FCommaAccepts write FCommaAccepts 
     default true; 
    property SemicolonAccepts: boolean read FSemicolonAccepts 
     write FSemicolonAccepts default true; 
    property TrimInput: boolean read FTrimInput write FTrimInput default true; 
    property NoLeadingSpaceInput: boolean read FNoLeadingSpaceInput 
     write FNoLeadingSpaceInput default true; 
    property AllowDuplicates: boolean read FAllowDuplicates write FAllowDuplicates 
     default false; 
    property MultiLine: boolean read FMultiLine write SetMultiLine default false; 
    property TagHeight: integer read FTagHeight write SetTagHeight default 32; 
    property EditorColor: TColor read FEditorColor write FEditorColor 
     default clWindow; 
    property AutoHeight: boolean read FAutoHeight write SetAutoHeight; 
    property OnTagClick: TTagClickEvent read FTagClickEvent write FTagClickEvent; 
    property OnTagAdded: TNotifyEvent read FTagAdded write FTagAdded; 
    property OnTagRemoved: TNotifyEvent read FTagRemoved write FTagRemoved; 
    property OnChange: TNotifyEvent read FOnChange write FOnChange; 
    property OnRemoveConfirm: TRemoveConfirmEvent read FOnRemoveConfirm 
     write FOnRemoveConfirm; 
    end; 

procedure Register; 

implementation 

uses Math, Clipbrd; 

procedure Register; 
begin 
    RegisterComponents('Rejbrand 2009', [TTagEditor]); 
end; 

function IsKeyDown(const VK: integer): boolean; 
begin 
    IsKeyDown := GetKeyState(VK) and $8000 <> 0; 
end; 

function GetTagPart(ClickInfo: TClickInfo): cardinal; 
begin 
    result := ClickInfo and $FFFF0000; 
end; 

{ TTagEditor } 

constructor TTagEditor.Create(AOwner: TComponent); 
var 
    mnuItem: TMenuItem; 
begin 
    inherited; 
    FEdit := TEdit.Create(Self); 
    FEdit.Parent := Self; 
    FEdit.BorderStyle := bsNone; 
    FEdit.Visible := false; 
    FEdit.OnKeyPress := EditKeyPress; 
    FEdit.OnExit := EditExit; 

    FTags := TStringList.Create; 
    FTags.OnChange := TagChange; 

    FBgColor := clWindow; 
    FBorderColor := clWindowFrame; 
    FTagBgColor := clSkyBlue; 
    FTagBorderColor := clNavy; 
    FSpacing := 8; 
    FTextColor := clWhite; 
    FSpaceAccepts := true; 
    FCommaAccepts := true; 
    FSemicolonAccepts := true; 
    FTrimInput := true; 
    FNoLeadingSpaceInput := true; 
    FAllowDuplicates := false; 
    FMultiLine := false; 
    FTagHeight := 32; 
    FShrunk := false; 
    FEditorColor := clWindow; 
    FCaretVisible := false; 
    FDragging := false; 

    FPopupMenu := TPopupMenu.Create(Self); 
    mnuItem := TMenuItem.Create(PopupMenu); 
    mnuItem.Caption := 'Delete'; 
    mnuItem.OnClick := mnuDeleteItemClick; 
    mnuItem.Hint := 'Deletes the selected tag.'; 
    FPopupMenu.Items.Add(mnuItem); 

    TabStop := true; 
end; 

procedure TTagEditor.EditExit(Sender: TObject); 
begin 
    if FEdit.Text <> '' then 
    Accept 
    else 
    HideEditor; 
end; 

procedure TTagEditor.mnuDeleteItemClick(Sender: TObject); 
begin 
    if Sender is TMenuItem then 
    begin 
    FTags.Delete(TMenuItem(Sender).Tag); 
    if Assigned(FTagRemoved) then 
     FTagRemoved(Self); 
    end; 
end; 

procedure TTagEditor.TagChange(Sender: TObject); 
begin 
    Invalidate; 
    if Assigned(FOnChange) then 
    FOnChange(Self); 
end; 

procedure TTagEditor.WndProc(var Message: TMessage); 
begin 
    inherited; 
    case Message.Msg of 
    WM_SETFOCUS: 
     Invalidate; 
    WM_KILLFOCUS: 
     begin 
     if FCaretVisible then DestroyCaret; 
     FDragging := false; 
     Invalidate; 
     end; 
    WM_COPY: 
     Clipboard.AsText := FTags.DelimitedText; 
    WM_CLEAR: 
     FTags.Clear; 
    WM_CUT: 
     begin 
     Clipboard.AsText := FTags.DelimitedText; 
     FTags.Clear; 
     end; 
    WM_PASTE: 
     begin 
     if Clipboard.HasFormat(CF_TEXT) then 
      if FTags.Count = 0 then 
      FTags.DelimitedText := Clipboard.AsText 
      else 
      FTags.DelimitedText := FTags.DelimitedText + ',' + Clipboard.AsText; 
     end; 
    end; 
end; 

function TTagEditor.Accept: boolean; 
begin 
    Assert(FEdit.Visible); 
    result := false; 
    if FTrimInput then 
    FEdit.Text := Trim(FEdit.Text); 
    if (FEdit.Text = '') or 
    ((not AllowDuplicates) and (FTags.IndexOf(FEdit.Text) <> -1)) then 
    begin 
    beep; 
    Exit; 
    end; 
    FTags.Add(FEdit.Text); 
    result := true; 
    HideEditor; 
    if Assigned(FTagAdded) then 
    FTagAdded(Self); 
    Invalidate; 
end; 

procedure TTagEditor.EditKeyPress(Sender: TObject; var Key: Char); 
begin 

    if (Key = chr(VK_SPACE)) and (FEdit.Text = '') and FNoLeadingSpaceInput then 
    begin 
    Key := #0; 
    Exit; 
    end; 

    if ((Key = chr(VK_SPACE)) and FSpaceAccepts) or 
    ((Key = ',') and FCommaAccepts) or 
    ((Key = ';') and FSemicolonAccepts) then 
    Key := chr(VK_RETURN); 

    case ord(Key) of 
    VK_RETURN: 
     begin 
     Accept; 
     ShowEditor; 
     Key := #0; 
     end; 
    VK_BACK: 
     begin 
     if (FEdit.Text = '') and (FTags.Count > 0) then 
     begin 
      FTags.Delete(FTags.Count - 1); 
      if Assigned(FTagRemoved) then 
      FTagRemoved(Sender); 
     end; 
     end; 
    VK_ESCAPE: 
     begin 
     HideEditor; 
     Self.SetFocus; 
     Key := #0; 
     end; 
    end; 

end; 

destructor TTagEditor.Destroy; 
begin 
    FPopupMenu.Free; 
    FTags.Free; 
    FEdit.Free; 
    inherited; 
end; 

procedure TTagEditor.HideEditor; 
begin 
    FEdit.Text := ''; 
    FEdit.Hide; 
// SetFocus; 
end; 


procedure TTagEditor.KeyDown(var Key: Word; Shift: TShiftState); 
begin 
    inherited; 
    case Key of 
    VK_END: 
     ShowEditor; 
    VK_DELETE: 
     Perform(WM_CLEAR, 0, 0); 
    VK_INSERT: 
     Perform(WM_PASTE, 0, 0); 
    end; 
end; 

procedure TTagEditor.KeyPress(var Key: Char); 
begin 
    inherited; 

    case Key of 
    ^C: 
     begin 
     Perform(WM_COPY, 0, 0); 
     Key := #0; 
     Exit; 
     end; 
    ^X: 
     begin 
     Perform(WM_CUT, 0, 0); 
     Key := #0; 
     Exit; 
     end; 
    ^V: 
     begin 
     Perform(WM_PASTE, 0, 0); 
     Key := #0; 
     Exit; 
     end; 
    end; 

    ShowEditor; 
    FEdit.Perform(WM_CHAR, ord(Key), 0); 
end; 

function TTagEditor.GetClickInfoAt(X, Y: integer): TClickInfo; 
var 
    i: integer; 
begin 
    result := NOWHERE; 
    if (X >= FEditPos.X) and (Y >= FEditPos.Y) then 
    Exit(EDITOR); 

    for i := 0 to FTags.Count - 1 do 
    if InRange(X, FLefts[i], FRights[i]) and InRange(Y, FTops[i], FBottoms[i]) then 
    begin 
     result := i; 
     if InRange(X, FCloseBtnLefts[i], FCloseBtnLefts[i] + FCloseBtnWidth) and 
     InRange(Y, FCloseBtnTops[i], FCloseBtnTops[i] + FActualTagHeight) and 
     not FShrunk then 
     result := result or PART_REMOVE_BUTTON; 
     break; 
    end; 
end; 

function TTagEditor.IsFirstOnRow(TagIndex: integer): boolean; 
begin 
    result := (TagIndex = 0) or (FTops[TagIndex] > FTops[TagIndex-1]); 
end; 

function TTagEditor.IsLastOnRow(TagIndex: integer): boolean; 
begin 
    result := (TagIndex = FTags.Count - 1) or (FTops[TagIndex] < FTops[TagIndex+1]); 
end; 

function TTagEditor.GetSeparatorIndexAt(X, Y: integer): integer; 
var 
    i: Integer; 
begin 
    result := FTags.Count; 
    Y := Max(Y, FSpacing + 1); 
    for i := FTags.Count - 1 downto 0 do 
    begin 
    if Y < FTops[i] then Continue; 
    if (IsLastOnRow(i) and (X >= FRights[i])) or 
     ((X < FRights[i]) and (IsFirstOnRow(i) or (FRights[i-1] < X))) then 
    begin 
     result := i; 
     if (IsLastOnRow(i) and (X >= FRights[i])) then inc(result); 
     Exit; 
    end; 
    end; 
end; 

procedure TTagEditor.MouseDown(Button: TMouseButton; Shift: TShiftState; X, 
    Y: Integer); 
begin 
    FMouseDownClickInfo := GetClickInfoAt(X, Y); 
    if GetTagIndex(FMouseDownClickInfo) <> EDITOR then 
    SetFocus; 
end; 

procedure TTagEditor.CreateCaret; 
begin 
    if not FCaretVisible then 
    FCaretVisible := Windows.CreateCaret(Handle, 0, 0, FActualTagHeight); 
end; 

procedure TTagEditor.DestroyCaret; 
begin 
    if not FCaretVisible then Exit; 
    Windows.DestroyCaret; 
    FCaretVisible := false; 
end; 

procedure TTagEditor.MouseMove(Shift: TShiftState; X, Y: Integer); 
var 
    SepIndex: integer; 
begin 
    inherited; 

    if IsKeyDown(VK_LBUTTON) and 
    InRange(GetTagIndex(FMouseDownClickInfo), TAG_LOW, TAG_HIGH) then 
    begin 
    FDragging := true; 
    Screen.Cursor := crDrag; 
    SepIndex := GetSeparatorIndexAt(X, Y); 
    TForm(Parent).Caption := IntToStr(SepIndex); 
    CreateCaret; 
    if SepIndex = FTags.Count then 
     SetCaretPos(FLefts[SepIndex - 1] + FWidths[SepIndex - 1] + FSpacing div 2, 
     FTops[SepIndex - 1]) 
    else 
     SetCaretPos(FLefts[SepIndex] - FSpacing div 2, FTops[SepIndex]); 
    ShowCaret(Handle); 
    Exit; 
    end; 

    case GetTagIndex(GetClickInfoAt(X,Y)) of 
    NOWHERE: Cursor := crArrow; 
    EDITOR: Cursor := crIBeam; 
    TAG_LOW..TAG_HIGH: Cursor := crHandPoint; 
    end; 

end; 

procedure TTagEditor.MouseUp(Button: TMouseButton; Shift: TShiftState; X, 
    Y: Integer); 
var 
    pnt: TPoint; 
    CanRemove: boolean; 
    ClickInfo: TClickInfo; 
    i: word; 
    p: cardinal; 
    SepIndex: integer; 
begin 
    inherited; 

    if FDragging then 
    begin 
    DestroyCaret; 
    FDragging := false; 
    Screen.Cursor := crDefault; 
    SepIndex := GetSeparatorIndexAt(X, Y); 
    if not InRange(SepIndex, GetTagIndex(FMouseDownClickInfo), 
     GetTagIndex(FMouseDownClickInfo) + 1) then 
     FTags.Move(GetTagIndex(FMouseDownClickInfo), SepIndex - 
     IfThen(SepIndex > GetTagIndex(FMouseDownClickInfo), 1, 0)); 
    Exit; 
    end; 

    ClickInfo := GetClickInfoAt(X, Y); 

    if ClickInfo <> FMouseDownClickInfo then Exit; 

    i := GetTagIndex(ClickInfo); 
    p := GetTagPart(ClickInfo); 

    case i of 
    EDITOR: 
     ShowEditor; 
    NOWHERE: ; 
    else 
    case Button of 
     mbLeft: 
     begin 
      case p of 
      PART_BODY: 
       if Assigned(FTagClickEvent) then 
       FTagClickEvent(Self, i, FTags[i]); 
      PART_REMOVE_BUTTON: 
       begin 
       if Assigned(FOnRemoveConfirm) then 
       begin 
        CanRemove := false; 
        FOnRemoveConfirm(Self, i, FTags[i], CanRemove); 
        if not CanRemove then Exit; 
       end; 
       FTags.Delete(i); 
       if Assigned(FTagRemoved) then 
        FTagRemoved(Self); 
       end; 
      end; 
     end; 
     mbRight: 
     begin 
      FPopupMenu.Items[0].Tag := i; 
      pnt := ClientToScreen(Point(X,Y)); 
      FPopupMenu.Items[0].Caption := 'Delete tag "' + FTags[i] + '"'; 
      FPopupMenu.Popup(pnt.X, pnt.Y); 
     end; 
    end; 
    end; 

end; 

procedure TTagEditor.Paint; 
var 
    i: integer; 
    w: integer; 
    x, y: integer; 
    R: TRect; 
    MeanWidth: integer; 
    S: string; 
    DesiredHeight: integer; 
begin 
    inherited; 
    Canvas.Brush.Color := FBgColor; 
    Canvas.Pen.Color := FBorderColor; 
    Canvas.Rectangle(ClientRect); 
    Canvas.Font.Assign(Self.Font); 
    SetLength(FLefts, FTags.Count); 
    SetLength(FRights, FTags.Count); 
    SetLength(FTops, FTags.Count); 
    SetLength(FBottoms, FTags.Count); 
    SetLength(FWidths, FTags.Count); 
    SetLength(FCloseBtnLefts, FTags.Count); 
    SetLength(FCloseBtnTops, FTags.Count); 
    FCloseBtnWidth := Canvas.TextWidth('×'); 
    FShrunk := false; 

    // Do metrics 
    FNumRows := 1; 
    if FMultiLine then 
    begin 
    FActualTagHeight := FTagHeight; 
    x := FSpacing; 
    y := FSpacing; 
    for i := 0 to FTags.Count - 1 do 
    begin 
     FWidths[i] := Canvas.TextWidth(FTags[i] + ' ×') + 2*FSpacing; 
     FLefts[i] := x; 
     FRights[i] := x + FWidths[i]; 
     FTops[i] := y; 
     FBottoms[i] := y + FTagHeight; 

     if x + FWidths[i] + FSpacing > ClientWidth then 
    { no need to make room for the editor, since it can reside on the next row! } 
     begin 
     x := FSpacing; 
     inc(y, FTagHeight + FSpacing); 
     inc(FNumRows); 
     FLefts[i] := x; 
     FRights[i] := x + FWidths[i]; 
     FTops[i] := y; 
     FBottoms[i] := y + FTagHeight; 
     end; 

     FCloseBtnLefts[i] := x + FWidths[i] - FCloseBtnWidth - FSpacing; 
     FCloseBtnTops[i] := y; 

     inc(x, FWidths[i] + FSpacing); 
    end; 
    end 
    else // i.e., not FMultiLine 
    begin 
    FActualTagHeight := ClientHeight - 2*FSpacing; 
    x := FSpacing; 
    y := FSpacing; 
    for i := 0 to FTags.Count - 1 do 
    begin 
     FWidths[i] := Canvas.TextWidth(FTags[i] + ' ×') + 2*FSpacing; 
     FLefts[i] := x; 
     FRights[i] := x + FWidths[i]; 
     FTops[i] := y; 
     FBottoms[i] := y + FActualTagHeight; 
     inc(x, FWidths[i] + FSpacing); 
     FCloseBtnLefts[i] := FRights[i] - FCloseBtnWidth - FSpacing; 
     FCloseBtnTops[i] := y; 
    end; 
    FShrunk := x + 64 {FEdit} > ClientWidth; 
    if FShrunk then 
    begin 

     // Enough to remove close buttons? 
     x := FSpacing; 
     y := FSpacing; 
     for i := 0 to FTags.Count - 1 do 
     begin 
     FWidths[i] := Canvas.TextWidth(FTags[i]) + 2*FSpacing; 
     FLefts[i] := x; 
     FRights[i] := x + FWidths[i]; 
     FTops[i] := y; 
     FBottoms[i] := y + FActualTagHeight; 
     inc(x, FWidths[i] + FSpacing); 
     FCloseBtnLefts[i] := FRights[i] - FCloseBtnWidth - FSpacing; 
     FCloseBtnTops[i] := y; 
     end; 

     if x + 64 {FEdit} > ClientWidth then // apparently no 
     begin 
     MeanWidth := (ClientWidth - 2*FSpacing - 64 {FEdit}) div FTags.Count - FSpacing; 
     x := FSpacing; 
     for i := 0 to FTags.Count - 1 do 
     begin 
      FWidths[i] := Min(FWidths[i], MeanWidth); 
      FLefts[i] := x; 
      FRights[i] := x + FWidths[i]; 
      inc(x, FWidths[i] + FSpacing); 
     end; 
     end; 
    end; 
    end; 

    FEditPos := Point(FSpacing, FSpacing + (FActualTagHeight - FEdit.Height) div 2); 
    if FTags.Count > 0 then 
    FEditPos := Point(FRights[FTags.Count - 1] + FSpacing, 
     FTops[FTags.Count - 1] + (FActualTagHeight - FEdit.Height) div 2); 
    if FMultiLine and (FEditPos.X + 64 > ClientWidth) and (FTags.Count > 0) then 
    begin 
    FEditPos := Point(FSpacing, 
     FTops[FTags.Count - 1] + FTagHeight + FSpacing + 
     (FActualTagHeight - FEdit.Height) div 2); 
    inc(FNumRows); 
    end; 

    DesiredHeight := FSpacing + FNumRows*(FTagHeight+FSpacing); 
    if FMultiLine and FAutoHeight and (ClientHeight <> DesiredHeight) then 
    begin 
    ClientHeight := DesiredHeight; 
    Invalidate; 
    Exit; 
    end; 

    // Draw 
    for i := 0 to FTags.Count - 1 do 
    begin 
    x := FLefts[i]; 
    y := FTops[i]; 
    w := FWidths[i]; 
    R := Rect(x, y, x + w, y + FActualTagHeight); 
    Canvas.Brush.Color := FTagBgColor; 
    Canvas.Pen.Color := FTagBorderColor; 
    Canvas.Rectangle(R); 
    Canvas.Font.Color := FTextColor; 
    Canvas.Brush.Style := bsClear; 
    R.Left := R.Left + FSpacing; 
    S := FTags[i]; 
    if not FShrunk then 
     S := S + ' ×'; 
    DrawText(Canvas.Handle, PChar(S), -1, R, DT_SINGLELINE or DT_VCENTER or 
     DT_LEFT or DT_END_ELLIPSIS or DT_NOPREFIX); 
    Canvas.Brush.Style := bsSolid; 
    end; 

    if FEdit.Visible then 
    begin 
    FEdit.Left := FEditPos.X; 
    FEdit.Top := FEditPos.Y; 
    FEdit.Width := ClientWidth - FEdit.Left - FSpacing; 
    end; 
    if Focused then 
    begin 
    R := Rect(2, 2, ClientWidth - 2, ClientHeight - 2); 
    SetBkColor(Canvas.Handle, clWhite); 
    SetTextColor(clBlack); 
    Canvas.DrawFocusRect(R); 
    end; 
end; 

procedure TTagEditor.SetAutoHeight(const Value: boolean); 
begin 
    if FAutoHeight <> Value then 
    begin 
    FAutoHeight := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetBgColor(const Value: TColor); 
begin 
    if FBgColor <> Value then 
    begin 
    FBgColor := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetBorderColor(const Value: TColor); 
begin 
    if FBorderColor <> Value then 
    begin 
    FBorderColor := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetMultiLine(const Value: boolean); 
begin 
    if FMultiLine <> Value then 
    begin 
    FMultiLine := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetTagBgColor(const Value: TColor); 
begin 
    if FTagBgColor <> Value then 
    begin 
    FTagBgColor := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetTagBorderColor(const Value: TColor); 
begin 
    if FTagBorderColor <> Value then 
    begin 
    FTagBorderColor := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetTagHeight(const Value: integer); 
begin 
    if FTagHeight <> Value then 
    begin 
    FTagHeight := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.SetTags(const Value: TStringList); 
begin 
    FTags.Assign(Value); 
    Invalidate; 
end; 

procedure TTagEditor.SetTextColor(const Value: TColor); 
begin 
    if FTextColor <> Value then 
    begin 
    FTextColor := Value; 
    Invalidate; 
    end; 
end; 

procedure TTagEditor.ShowEditor; 
begin 
    FEdit.Left := FEditPos.X; 
    FEdit.Top := FEditPos.Y; 
    FEdit.Width := ClientWidth - FEdit.Left - FSpacing; 
    FEdit.Color := FEditorColor; 
    FEdit.Text := ''; 
    FEdit.Show; 
    FEdit.SetFocus; 
end; 

procedure TTagEditor.SetSpacing(const Value: integer); 
begin 
    if FSpacing <> Value then 
    begin 
    FSpacing := Value; 
    Invalidate; 
    end; 
end; 

initialization 
    Screen.Cursors[crHandPoint] := LoadCursor(0, IDC_HAND); // Get the normal hand cursor 

end. 

che produce

Screenshot http://privat.rejbrand.se/tageditor.png

Sample video

Demo (Compiled EXE)

Se ho più tempo più tardi oggi I wil eseguo ancora un po 'di lavoro su questo controllo, ad esempio, evidenziazione del pulsante al passaggio del mouse, evento click tag, larghezza massima pulsante ecc.

Aggiornamento: Aggiunte molte funzionalità.

Aggiornamento: Aggiunta funzione multi-linea.

Aggiornamento: Altre caratteristiche.

Aggiornamento: Aggiunto interfaccia appunti, corretti alcuni problemi, ecc

Aggiornamento: Aggiunto il drag-and-drop riordino e corretti alcuni problemi minori. A proposito, questa è l'ultima versione che posterò qui. Le versioni successive (se ce ne saranno) verranno pubblicate su http://specials.rejbrand.se/dev/controls/.

Aggiornamento: È stata aggiunta la proprietà AutoHeight, la casella di modifica è stata centrata verticalmente e il cursore di trascinamento è stato modificato. (Sì, non ho potuto resistere a fare ancora un altro aggiornamento.)

+0

Potresti aggiungere un link al file pas scaricabile? È un PITA per ottenere il codice da SO nell'editor Delphi. ;-) –

+5

+1, _Molto bello! –

+0

È eccellente! Lo userò come base per il mio lavoro. Grazie mille! Non sono molto bravo con Delphi, ma se riesco a migliorare il componente, aggiornerò la tua risposta. –