diff options
-rw-r--r-- | examples/gui/listviewtest/listviewtest.lpr | 32 | ||||
-rw-r--r-- | src/gui/fpg_listview.pas | 223 |
2 files changed, 222 insertions, 33 deletions
diff --git a/examples/gui/listviewtest/listviewtest.lpr b/examples/gui/listviewtest/listviewtest.lpr index afb1fbdf..e97b86bd 100644 --- a/examples/gui/listviewtest/listviewtest.lpr +++ b/examples/gui/listviewtest/listviewtest.lpr @@ -4,7 +4,7 @@ program listviewtest; uses Classes, sysutils, - fpg_base, fpg_main, fpg_listview, fpg_form, fpg_button, fpg_edit, fpg_checkbox, fpg_splitter, fpg_panel; + fpg_base, fpg_main, fpg_listview, fpg_form, fpg_button, fpg_edit, fpg_checkbox, fpg_splitter, fpg_panel, fpg_imagelist; type @@ -30,6 +30,7 @@ type { TMainForm } + procedure TMainForm.CloseBttn(Sender: TObject); begin Close; @@ -81,12 +82,39 @@ var LVColumn: TfpgLVColumn; TopPanel, BottomPanel: TfpgPanel; + IL: TStringList; + i: Integer; + FImageList: TfpgImageList; + FSelectedImageList: TfpgImageList; + TmpImage: TfpgImage; begin inherited Create(AOwner); WindowTitle := 'ListView Test'; SetPosition(200, 200, 640, 480); + IL := TStringList.Create; + + fpgImages.ListImages(IL); + + FImageList := TfpgImageList.Create; + FSelectedImageList := TfpgImageList.Create; + + for i := 0 to IL.Count-1 do + FImageList.AddImage(fpgImages.GetImage(IL.Strings[i])); + + IL.Free; + + // invert the items for the 'selected' images + FImageList.Item[i].Image.ImageFromSource; + + for i := 0 to FImageList.Count-1 do + begin + TmpImage := FImageList.Item[i].Image.ImageFromSource; + TmpImage.Invert; + FSelectedImageList.AddImage(TmpImage); + + end; BottomPanel := TfpgPanel.Create(Self); BottomPanel.Align := alBottom; @@ -108,6 +136,8 @@ begin OnPaintItem := @PaintItem; OnSelectionChanged := @ItemSelectionChanged; MultiSelect := True; + Images := FImageList; + ImagesSelected := FSelectedImageList; end; FSplitter := TfpgSplitter.Create(TopPanel); diff --git a/src/gui/fpg_listview.pas b/src/gui/fpg_listview.pas index 841f9661..39ebbad9 100644 --- a/src/gui/fpg_listview.pas +++ b/src/gui/fpg_listview.pas @@ -29,7 +29,8 @@ uses fpg_base, fpg_main, fpg_widget, - fpg_scrollbar; + fpg_scrollbar, + fpg_imagelist; type TfpgListView = class; @@ -54,6 +55,7 @@ type FResizable: Boolean; FVisible: Boolean; FWidth: Integer; + Ref: Integer; procedure SetAlignment(const AValue: TAlignment); procedure SetAutoSize(const AValue: Boolean); procedure SetCaption(const AValue: String); @@ -96,8 +98,9 @@ type property Column[AIndex: Integer]: TfpgLVColumn read GetColumn write SetColumn; end; - - TfpgLVItemState = set of (lisFocused, lisSelected, lisHotTrack); + + TfpgLVItemStates = (lisNoState, lisSelected, lisFocused, lisHotTrack); + TfpgLVItemState = set of TfpgLVItemStates; TfpgLVItemPaintPart = set of (lvppBackground, lvppIcon, lvppText, lvppFocused); @@ -152,22 +155,29 @@ type end; + { TfpgLVItem } + TfpgLVItem = class(TObject) private FCaption: String; + FImageIndex: Integer; FItems: TfpgLVItems; FSubItems: TStrings; FUserData: Pointer; function GetSelected(ListView: TfpgListView): Boolean; + function GetSubItems: TStrings; procedure SetCaption(const AValue: String); + procedure SetImageIndex(const AValue: Integer); procedure SetSelected(ListView: TfpgListView; const AValue: Boolean); procedure SubItemsChanged(Sender: TObject); + function SubItemCount: Integer; public constructor Create(Items: TfpgLVItems); virtual; destructor Destroy; override; property Caption: String read FCaption write SetCaption; property UserData: Pointer read FUserData write FUserData; - property SubItems: TStrings read FSubItems; + property ImageIndex: Integer read FImageIndex write SetImageIndex default -1; + property SubItems: TStrings read GetSubItems; property Selected[ListView: TfpgListView]: Boolean read GetSelected write SetSelected; end; @@ -178,6 +188,8 @@ type private procedure SetShiftIsPressed(const AValue: Boolean); private + FImages: array[TfpgLVItemStates] of TfpgImageList; + FSubitemImages: array[TfpgLVItemStates] of TfpgImageList; FItemIndex: Integer; FMultiSelect: Boolean; FOnPaintColumn: TfpgLVPaintColumnEvent; @@ -198,14 +210,20 @@ type FMouseDownPoint: TPoint; FScrollBarNeedsUpdate: Boolean; FShiftIsPressed: Boolean; + function HasImages: Boolean; + function GetImages(AIndex: integer): TfpgImageList; function GetItemHeight: Integer; + procedure SetImages(AIndex: integer; const AValue: TfpgImageList); procedure SetItemIndex(const AValue: Integer); procedure SetItems(const AValue: TfpgLVItems); procedure SetMultiSelect(const AValue: Boolean); procedure SetOnColumnClick(const AValue: TfpgLVColumnClickEvent); procedure SetShowHeaders(const AValue: Boolean); + function SubItemGetImages(AIndex: integer): TfpgImageList; + procedure SubItemSetImages(AIndex: integer; const AValue: TfpgImageList); procedure VScrollChange(Sender: TObject; Position: Integer); procedure HScrollChange(Sender: TObject; Position: Integer); + function FindImageForState(AItemIndex: Integer; AColumnIndex: Integer; AState: TfpgLVItemState): TfpgImage; // interface methods procedure ItemDeleted(AIndex: Integer); procedure ItemAdded(AIndex: Integer); @@ -257,6 +275,10 @@ type property Columns: TfpgLVColumns read FColumns; property Enabled; property HScrollBar: TfpgScrollBar read FHScrollBar; + property Images: TfpgImageList index Ord(lisNoState) read GetImages write SetImages; + property ImagesSelected: TfpgImageList index Ord(lisSelected) read GetImages write SetImages; + property ImagesFocused: TfpgImageList index Ord(lisFocused) read GetImages write SetImages; + property ImagesHotTrack: TfpgImageList index Ord(lisHotTrack) read GetImages write SetImages; property ItemHeight: Integer read GetItemHeight; property ItemIndex: Integer read FItemIndex write SetItemIndex; property Items: TfpgLVItems read FItems write SetItems; @@ -264,6 +286,11 @@ type property MultiSelect: Boolean read FMultiSelect write SetMultiSelect; property ParentShowHint; property SelectionFollowsFocus: Boolean read FSelectionFollowsFocus write FSelectionFollowsFocus; + property SubItemImages: TfpgImageList index Ord(lisNoState) read SubItemGetImages write SubItemSetImages; + property SubItemImagesSelected: TfpgImageList index Ord(lisSelected) read SubItemGetImages write SubItemSetImages; + property SubItemImagesFocused: TfpgImageList index Ord(lisFocused) read SubItemGetImages write SubItemSetImages; + property SubItemImagesHotTrack: TfpgImageList index Ord(lisHotTrack) read SubItemGetImages write SubItemSetImages; + property ShowHeaders: Boolean read FShowHeaders write SetShowHeaders; property ShowHint; property TabOrder; @@ -505,11 +532,29 @@ begin FItems.DoChange(Self); end; +procedure TfpgLVItem.SetImageIndex(const AValue: Integer); +begin + if FImageIndex=AValue then exit; + FImageIndex:=AValue; + if Assigned(FItems) then + FItems.DoChange(Self); +end; + function TfpgLVItem.GetSelected(ListView: TfpgListView): Boolean; begin Result := ListView.ItemGetSelected(Self); end; +function TfpgLVItem.GetSubItems: TStrings; +begin + if FSubItems = nil then + begin + FSubItems := TStringList.Create; + TStringList(FSubItems).OnChange := @SubItemsChanged; + end; + Result := FSubItems; +end; + procedure TfpgLVItem.SetSelected(ListView: TfpgListView; const AValue: Boolean); begin ListView.ItemSetSelected(Self, AValue); @@ -521,22 +566,29 @@ begin FItems.DoChange(Self); end; +function TfpgLVItem.SubItemCount: Integer; +begin + Result := 0; + if FSubItems = nil then + Exit; + Result := FSubItems.Count; +end; + constructor TfpgLVItem.Create(Items: TfpgLVItems); begin FItems := Items; - FSubItems := TStringList.Create; - TStringList(FSubItems).OnChange := @SubItemsChanged; + ImageIndex := -1; end; destructor TfpgLVItem.Destroy; begin - FSubItems.Free; + if Assigned(FSubItems) then + FSubItems.Free; inherited Destroy; end; { TfpgListView } - procedure TfpgListView.SetShowHeaders(const AValue: Boolean); begin if FShowHeaders=AValue then @@ -544,6 +596,22 @@ begin FShowHeaders:=AValue; DoRePaint; end; + +function TfpgListView.SubItemGetImages(AIndex: integer): TfpgImageList; +begin + Result := FSubItemImages[TfpgLVItemStates(AIndex)]; +end; + +procedure TfpgListView.SubItemSetImages(AIndex: integer; + const AValue: TfpgImageList); +begin + if AValue = FSubItemImages[TfpgLVItemStates(AIndex)] then + Exit; // ==> + + FSubItemImages[TfpgLVItemStates(AIndex)] := AValue; + if HasHandle then + Invalidate; +end; procedure TfpgListView.VScrollChange(Sender: TObject; Position: Integer); @@ -556,6 +624,47 @@ begin DoRepaint; end; +function TfpgListView.FindImageForState(AItemIndex: Integer; AColumnIndex: Integer; AState: TfpgLVItemState): TfpgImage; +var + PreferredState: TfpgLVItemStates = lisHotTrack; + State: TfpgLVItemStates; + Item: TfpgLVItem; + ImgList: TfpgImageList; + ImagesArray: Array[TfpgLVItemStates] of TfpgImageList; +begin + Result := nil; + Item := Items.Item[AItemIndex]; + + if AColumnIndex = 0 then + ImagesArray := FImages + else + ImagesArray := FSubItemImages; + + // later we will make the state preference order configurable + while (not (PreferredState in AState)) and (PreferredState <> lisNoState) do + Dec(PreferredState); + + State := lisNoState; //to remove compiler warning + for State := PreferredState downto lisNoState do + begin + if ImagesArray[State] = nil then + continue; + ImgList := ImagesArray[State]; + + if (Item.ImageIndex <> -1) and (Item.ImageIndex < ImgList.Count) then + begin + Result := ImgList.Item[Item.ImageIndex].Image; + end + else + begin + if AItemIndex < ImgList.Count then + Result := ImgList.Item[AItemIndex].Image; + end; + break; + end; + +end; + procedure TfpgListView.SetItems(const AValue: TfpgLVItems); begin if AValue = FItems then @@ -579,7 +688,6 @@ begin FOnColumnClick:=AValue; end; - procedure TfpgListView.SetShiftIsPressed(const AValue: Boolean); begin if AValue = FShiftIsPressed then @@ -602,11 +710,38 @@ begin end; end; +function TfpgListView.HasImages: Boolean; +var + State: TfpgLVItemStates; +begin + Result := False; + for State := lisNoState to lisHotTrack do + if FImages[State] <> nil then + Exit(True); + +end; + function TfpgListView.GetItemHeight: Integer; begin Result := Canvas.Font.Height + 4; end; +function TfpgListView.GetImages(AIndex: integer): TfpgImageList; +begin + Result := FImages[TfpgLVItemStates(AIndex)]; +end; + +procedure TfpgListView.SetImages(AIndex: integer; const AValue: TfpgImageList); +begin + if AValue = FImages[TfpgLVItemStates(AIndex)] then + Exit; // ==> + + FImages[TfpgLVItemStates(AIndex)] := AValue; + if HasHandle then + Invalidate; +end; + + procedure TfpgListView.SetItemIndex(const AValue: Integer); begin if FItemIndex=AValue then @@ -1339,6 +1474,7 @@ var vBottom: Integer; tLeft, tWidth: Integer; + Image, TmpImage: TfpgImage; begin FirstIndex := (FVScrollBar.Position) div ItemHeight; LastIndex := (FVScrollBar.Position+GetItemAreaHeight) div ItemHeight; @@ -1356,6 +1492,7 @@ begin for I := FirstIndex to LastIndex do begin ItemState := []; + Image := nil; PaintPart := [lvppBackground, lvppIcon, lvppText]; ItemRect := ItemGetRect(I); @@ -1389,11 +1526,6 @@ begin if Assigned(FOnPaintItem) then FOnPaintItem(Self, Canvas, Item, I, ItemRect, PaintPart); - if lvppIcon in PaintPart then - begin - { TODO: paint icon } - end; - if lvppFocused in PaintPart then begin if lisSelected in ItemState then @@ -1403,8 +1535,8 @@ begin Canvas.SetLineStyle(1, lsDot); Canvas.DrawRectangle(ItemRect); end; - - if lvppText in PaintPart then + + if (lvppText in PaintPart) or (lvppIcon in PaintPart) then begin if lisSelected in ItemState then Canvas.TextColor := clSelectionText; @@ -1421,25 +1553,47 @@ begin ColumnIndex := FColumns.Column[J].ColumnIndex else ColumnIndex := J; - if ColumnIndex = 0 then - TheText := Item.Caption - else if Item.SubItems.Count >= ColumnIndex then - TheText := Item.SubItems.Strings[ColumnIndex-1] - else - TheText := ''; + if lvppText in PaintPart then begin + if ColumnIndex = 0 then + TheText := Item.Caption + else if Item.SubItemCount >= ColumnIndex then + TheText := Item.SubItems.Strings[ColumnIndex-1] + else + TheText := ''; + end; tLeft := ItemRect.Left; - tWidth := Canvas.Font.TextWidth(TheText); - case FColumns.Column[J].Alignment of - taRightJustify: - Inc(tLeft, FColumns.Column[J].Width - tWidth - 5); - taCenter: - Inc(tLeft, (FColumns.Column[J].Width - tWidth - 5) div 2); - taLeftJustify: - Inc(tLeft, 5); + + Image := FindImageForState(I, J, ItemState); + if (lvppIcon in PaintPart) and Assigned(Image) then + begin + TmpImage := Image; + if not Enabled then + TmpImage := Image.CreateDisabledImage; + Canvas.DrawImage(ItemRect.Left,ItemRect.Top, TmpImage); + if Not Enabled then + TmpImage.Free; + end; + + + if lvppText in PaintPart then + begin + tWidth := Canvas.Font.TextWidth(TheText); + case FColumns.Column[J].Alignment of + taRightJustify: + Inc(tLeft, FColumns.Column[J].Width - tWidth - 5); + taCenter: + Inc(tLeft, (FColumns.Column[J].Width - tWidth - 5) div 2); + taLeftJustify: + begin + Inc(tLeft, 5); + if Image <> nil then + Inc(tLeft, Max(Image.Width, ItemHeight)-5); + end; + end; + fpgStyle.DrawString(Canvas, tLeft, ItemRect.Top+2, TheText, Enabled); end; - fpgStyle.DrawString(Canvas, tLeft, ItemRect.Top+2, TheText, Enabled); Inc(ItemRect.Left, FColumns.Column[J].Width); //WriteLn(ItemRect.Left,' ', ItemRect.Top, ' ', ItemRect.Right, ' ', ItemRect.Bottom); end; @@ -1450,7 +1604,8 @@ begin Canvas.TextColor := TheTextColor; end; - + + vBottom := Height - 2; if FHScrollBar.Visible then Dec(vBottom, FHScrollBar.Height); @@ -1666,6 +1821,7 @@ end; destructor TfpgLVColumns.Destroy; begin + Clear; // needed since Colums can be share between ListViews FColumns.Free; inherited Destroy; end; @@ -1687,12 +1843,15 @@ end; procedure TfpgLVColumns.Delete(AIndex: Integer); begin + Dec(Column[AIndex].Ref); + FColumns.OwnsObjects := Column[AIndex].Ref < 1; FColumns.Delete(AIndex); end; procedure TfpgLVColumns.Insert(AColumn: TfpgLVColumn; AIndex: Integer); begin FColumns.Insert(AIndex, AColumn); + Inc(AColumn.Ref); end; function TfpgLVColumns.Count: Integer; |