{----------------------------------------------------------------------------- The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/MPL-1.1.html Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is: JvDBComb.PAS, released on 2002-07-04. The Initial Developers of the Original Code are: Fedor Koshevnikov, Igor Pavluk and Serge Korolev Copyright (c) 1997, 1998 Fedor Koshevnikov, Igor Pavluk and Serge Korolev Copyright (c) 2001,2002 SGB Software All Rights Reserved. Contributor(s): Polaris Software You may retrieve the latest version of this file at the Project JEDI's JVCL home page, located at http://jvcl.sourceforge.net Known Issues: -----------------------------------------------------------------------------} // $Id: JvDBCombobox.pas 11299 2007-05-23 16:27:35Z ahuser $ unit JvDBCombobox; {$I jvcl.inc} interface uses {$IFDEF UNITVERSIONING} JclUnitVersioning, {$ENDIF UNITVERSIONING} {$IFDEF VCL} Windows, Messages, {$ENDIF VCL} {$IFDEF VisualCLX} QWindows, {$ENDIF VisualCLX} Classes, Graphics, Controls, StdCtrls, DB, DBCtrls, JvExStdCtrls; type TJvCustomDBComboBox = class(TJvExCustomComboBox) private FDataLink: TFieldDataLink; FPaintControl: TPaintControl; FBeepOnError: Boolean; procedure DataChange(Sender: TObject); procedure EditingChange(Sender: TObject); function GetDataField: string; function GetDataSource: TDataSource; function GetField: TField; function GetReadOnly: Boolean; procedure SetDataField(const Value: string); procedure SetDataSource(Value: TDataSource); procedure SetEditReadOnly; {$IFDEF COMPILER5} procedure SetItems(const Value: TStrings); {$ENDIF COMPILER5} procedure SetReadOnly(Value: Boolean); procedure UpdateData(Sender: TObject); function GetComboText: string; virtual; procedure SetComboText(const Value: string); virtual; procedure CMGetDataLink(var Msg: TMessage); message CM_GETDATALINK; procedure WMPaint(var Msg: TWMPaint); message WM_PAINT; protected procedure DoExit; override; procedure Change; override; procedure Click; override; procedure Reset; // This may cause trouble with BCB because it uses a HWND parameter // but as it is defined in the VCL itself, we can't do much. procedure ComboWndProc(var Msg: TMessage; ComboWnd: HWND; ComboProc: Pointer); override; procedure CreateWnd; override; procedure DropDown; override; function GetPaintText: string; virtual; procedure KeyDown(var Key: Word; Shift: TShiftState); override; procedure KeyPress(var Key: Char); override; procedure Loaded; override; procedure Notification(AComponent: TComponent; Operation: TOperation); override; procedure SetStyle(Value: TComboBoxStyle); override; {$IFDEF COMPILER6_UP} procedure SetItems(const Value: TStrings); override; {$ENDIF COMPILER6_UP} procedure WndProc(var Msg: TMessage); override; property BeepOnError: Boolean read FBeepOnError write FBeepOnError default False; property ComboText: string read GetComboText write SetComboText; property DataField: string read GetDataField write SetDataField; property DataSource: TDataSource read GetDataSource write SetDataSource; property ReadOnly: Boolean read GetReadOnly write SetReadOnly default False; public constructor Create(AOwner: TComponent); override; destructor Destroy; override; function ExecuteAction(Action: TBasicAction): Boolean; override; function UpdateAction(Action: TBasicAction): Boolean; override; function UseRightToLeftAlignment: Boolean; override; property Field: TField read GetField; property Items write SetItems; property Text; end; TJvDBComboBox = class(TJvCustomDBComboBox) private FValues: TStringList; FEnableValues: Boolean; procedure SetEnableValues(Value: Boolean); function GetValues: TStrings; procedure SetValues(Value: TStrings); procedure ValuesChanged(Sender: TObject); protected procedure SetStyle(Value: TComboBoxStyle); override; function GetComboText: string; override; function GetPaintText: string; override; procedure SetComboText(const Value: string); override; public constructor Create(AOwner: TComponent); override; destructor Destroy; override; published property Align; // Polaris property AutoSize; property Style { must be published before Items } default csDropDownList; // Polaris property BeepOnError; {$IFDEF COMPILER6_UP} property BevelEdges; property BevelInner; property BevelKind default bkNone; property BevelOuter; {$ENDIF COMPILER6_UP} property Color; property DataField; property DataSource; property DragMode; property DragCursor; property DropDownCount; property Enabled; property EnableValues: Boolean read FEnableValues write SetEnableValues default True; property Font; property Anchors; property BiDiMode; property Constraints; property DragKind; property ParentBiDiMode; property ImeMode; property ImeName; property ItemHeight; property Items; property ParentColor; property ParentFont; property ParentShowHint; property PopupMenu; property ReadOnly; property ShowHint; property Sorted; property TabOrder; property TabStop; property Values: TStrings read GetValues write SetValues; property Visible; property OnChange; property OnClick; property OnDblClick; property OnDragDrop; property OnDragOver; property OnDrawItem; property OnDropDown; property OnEndDrag; property OnEnter; property OnExit; property OnKeyDown; property OnKeyPress; property OnKeyUp; property OnMeasureItem; property OnStartDrag; property OnContextPopup; property OnEndDock; property OnStartDock; end; {$IFDEF UNITVERSIONING} const UnitVersioning: TUnitVersionInfo = ( RCSfile: '$URL: https://jvcl.svn.sourceforge.net/svnroot/jvcl/tags/JVCL3_32/run/JvDBCombobox.pas $'; Revision: '$Revision: 11299 $'; Date: '$Date: 2007-05-23 18:27:35 +0200 (mer., 23 mai 2007) $'; LogPath: 'JVCL\run' ); {$ENDIF UNITVERSIONING} implementation uses {$IFDEF COMPILER6_UP} VDBConsts, {$ELSE} DBConsts, {$ENDIF COMPILER6_UP} SysUtils, JvConsts, JvDBUtils; //=== { TJvCustomDBComboBox } ================================================ constructor TJvCustomDBComboBox.Create(AOwner: TComponent); begin inherited Create(AOwner); ControlStyle := ControlStyle + [csReplicatable]; FDataLink := TFieldDataLink.Create; FDataLink.Control := Self; FDataLink.OnDataChange := DataChange; FDataLink.OnUpdateData := UpdateData; FDataLink.OnEditingChange := EditingChange; FPaintControl := TPaintControl.Create(Self, 'COMBOBOX'); FBeepOnError := False; end; destructor TJvCustomDBComboBox.Destroy; begin FPaintControl.Free; FDataLink.OnDataChange := nil; FDataLink.OnUpdateData := nil; FDataLink.Free; FDataLink := nil; inherited Destroy; end; procedure TJvCustomDBComboBox.Loaded; begin inherited Loaded; if csDesigning in ComponentState then DataChange(Self); end; procedure TJvCustomDBComboBox.Notification(AComponent: TComponent; Operation: TOperation); begin inherited Notification(AComponent, Operation); if (Operation = opRemove) and (FDataLink <> nil) and (AComponent = DataSource) then DataSource := nil; end; procedure TJvCustomDBComboBox.CreateWnd; begin inherited CreateWnd; SetEditReadOnly; DataChange(Self); end; procedure TJvCustomDBComboBox.DataChange(Sender: TObject); begin if not HandleAllocated or DroppedDown then Exit; if FDataLink.Field <> nil then SetComboText(FDataLink.Field.Text) else if csDesigning in ComponentState then ComboText := Name else ComboText := ''; end; procedure TJvCustomDBComboBox.UpdateData(Sender: TObject); begin FDataLink.Field.Text := ComboText; end; procedure TJvCustomDBComboBox.SetComboText(const Value: string); var I: Integer; Redraw: Boolean; begin if Value <> ComboText then begin if Style <> csDropDown then begin Redraw := (Style <> csSimple) and HandleAllocated; if Redraw then SendMessage(Handle, WM_SETREDRAW, 0, 0); try if Value = '' then I := -1 else I := Items.IndexOf(Value); ItemIndex := I; finally if Redraw then begin SendMessage(Handle, WM_SETREDRAW, 1, 0); Invalidate; end; end; if I >= 0 then Exit; end; if Style in [csDropDown, csSimple] then Text := Value; end; end; function TJvCustomDBComboBox.GetComboText: string; var I: Integer; begin if Style in [csDropDown, csSimple] then Result := Text else begin I := ItemIndex; if I < 0 then Result := '' else Result := Items[I]; end; end; procedure TJvCustomDBComboBox.Change; begin FDataLink.Edit; inherited Change; FDataLink.Modified; end; procedure TJvCustomDBComboBox.Click; begin FDataLink.Edit; inherited Click; FDataLink.Modified; end; procedure TJvCustomDBComboBox.DropDown; begin inherited DropDown; end; function TJvCustomDBComboBox.GetDataSource: TDataSource; begin Result := FDataLink.DataSource; end; procedure TJvCustomDBComboBox.SetDataSource(Value: TDataSource); begin if not (FDataLink.DataSourceFixed and (csLoading in ComponentState)) then FDataLink.DataSource := Value; if Value <> nil then Value.FreeNotification(Self); end; function TJvCustomDBComboBox.GetDataField: string; begin Result := FDataLink.FieldName; end; procedure TJvCustomDBComboBox.SetDataField(const Value: string); begin FDataLink.FieldName := Value; end; function TJvCustomDBComboBox.GetReadOnly: Boolean; begin Result := FDataLink.ReadOnly; end; procedure TJvCustomDBComboBox.SetReadOnly(Value: Boolean); begin FDataLink.ReadOnly := Value; end; function TJvCustomDBComboBox.GetField: TField; begin Result := FDataLink.Field; end; procedure TJvCustomDBComboBox.KeyDown(var Key: Word; Shift: TShiftState); begin inherited KeyDown(Key, Shift); if Key in [VK_BACK, VK_DELETE, VK_UP, VK_DOWN, 32..255] then if not FDataLink.Edit and (Key in [VK_UP, VK_DOWN]) then Key := 0; end; procedure TJvCustomDBComboBox.KeyPress(var Key: Char); begin inherited KeyPress(Key); if (Key in [#32..#255]) and (FDataLink.Field <> nil) and not FDataLink.Field.IsValidChar(Key) then begin if BeepOnError then SysUtils.Beep; Key := #0; end; case Key of CtrlH, CtrlV, CtrlX, #32..#255: FDataLink.Edit; Esc: begin FDataLink.Reset; SelectAll; end; end; end; procedure TJvCustomDBComboBox.EditingChange(Sender: TObject); begin SetEditReadOnly; end; procedure TJvCustomDBComboBox.SetEditReadOnly; begin if (Style in [csDropDown, csSimple]) and HandleAllocated then SendMessage(EditHandle, EM_SETREADONLY, Ord(not FDataLink.Editing), 0); end; procedure TJvCustomDBComboBox.Reset; begin DataChange(Self); {Restore text} end; procedure TJvCustomDBComboBox.WndProc(var Msg: TMessage); begin if not (csDesigning in ComponentState) then case Msg.Msg of WM_COMMAND: if TWMCommand(Msg).NotifyCode = CBN_SELCHANGE then begin try if not FDataLink.Edit then begin if Style <> csSimple then PostMessage(Handle, CB_SHOWDROPDOWN, 0, 0); Exit; end else Reset; except Reset; raise; end; end; CB_SHOWDROPDOWN: if Msg.WParam <> 0 then begin try FDataLink.Edit; except Reset; raise; end; end else if not FDataLink.Editing then Reset; WM_CREATE, WM_WINDOWPOSCHANGED, CM_FONTCHANGED: FPaintControl.DestroyHandle; end; inherited WndProc(Msg); end; procedure TJvCustomDBComboBox.ComboWndProc(var Msg: TMessage; ComboWnd: HWND; ComboProc: Pointer); begin if not (csDesigning in ComponentState) then case Msg.Msg of WM_LBUTTONDOWN: if (Style = csSimple) and (ComboWnd <> EditHandle) then if not FDataLink.Edit then Exit; end; inherited ComboWndProc(Msg, ComboWnd, ComboProc); end; procedure TJvCustomDBComboBox.DoExit; begin try FDataLink.UpdateRecord; except SelectAll; if CanFocus then SetFocus; raise; end; inherited DoExit; end; procedure TJvCustomDBComboBox.CMGetDataLink(var Msg: TMessage); begin Msg.Result := Longint(FDataLink); end; procedure TJvCustomDBComboBox.WMPaint(var Msg: TWMPaint); var S: string; R: TRect; P: TPoint; Child: HWND; begin if csPaintCopy in ControlState then begin S := GetPaintText; if Style = csDropDown then begin SendMessage(FPaintControl.Handle, WM_SETTEXT, 0, Longint(PChar(S))); SendMessage(FPaintControl.Handle, WM_PAINT, Msg.DC, 0); Child := GetWindow(FPaintControl.Handle, GW_CHILD); if Child <> 0 then begin Windows.GetClientRect(Child, R); Windows.MapWindowPoints(Child, FPaintControl.Handle, R.TopLeft, 2); GetWindowOrgEx(Msg.DC, P); SetWindowOrgEx(Msg.DC, P.X - R.Left, P.Y - R.Top, nil); IntersectClipRect(Msg.DC, 0, 0, R.Right - R.Left, R.Bottom - R.Top); SendMessage(Child, WM_PAINT, Msg.DC, 0); end; end else begin SendMessage(FPaintControl.Handle, CB_RESETCONTENT, 0, 0); if Items.IndexOf(S) <> -1 then begin SendMessage(FPaintControl.Handle, CB_ADDSTRING, 0, Longint(PChar(S))); SendMessage(FPaintControl.Handle, CB_SETCURSEL, 0, 0); end; SendMessage(FPaintControl.Handle, WM_PAINT, Msg.DC, 0); end; end else inherited; end; function TJvCustomDBComboBox.GetPaintText: string; begin if FDataLink.Field <> nil then Result := FDataLink.Field.Text else Result := ''; end; procedure TJvCustomDBComboBox.SetItems(const Value: TStrings); begin {$IFDEF COMPILER6_UP} inherited SetItems(Value); {$ELSE} { TODO : (rb) This was incorrectly // Can't test } Items.Assign(Value); {$ENDIF COMPILER6_UP} DataChange(Self); end; procedure TJvCustomDBComboBox.SetStyle(Value: TComboBoxStyle); begin if (Value = csSimple) and Assigned(FDataLink) and FDataLink.DataSourceFixed then _DBError(SNotReplicatable); inherited SetStyle(Value); end; function TJvCustomDBComboBox.UseRightToLeftAlignment: Boolean; begin Result := DBUseRightToLeftAlignment(Self, Field); end; function TJvCustomDBComboBox.ExecuteAction(Action: TBasicAction): Boolean; begin Result := inherited ExecuteAction(Action) or (FDataLink <> nil) and FDataLink.ExecuteAction(Action); end; function TJvCustomDBComboBox.UpdateAction(Action: TBasicAction): Boolean; begin Result := inherited UpdateAction(Action) or (FDataLink <> nil) and FDataLink.UpdateAction(Action); end; //=== { TJvDBComboBox } ====================================================== constructor TJvDBComboBox.Create(AOwner: TComponent); begin inherited Create(AOwner); FValues := TStringList.Create; FValues.OnChange := ValuesChanged; FEnableValues := True; Style := csDropDownList; end; destructor TJvDBComboBox.Destroy; begin FValues.OnChange := nil; FValues.Free; inherited Destroy; end; procedure TJvDBComboBox.ValuesChanged(Sender: TObject); begin if FEnableValues then DataChange(Self); end; function TJvDBComboBox.GetPaintText: string; var I: Integer; begin Result := ''; if FDataLink.Field <> nil then if FEnableValues then begin I := Values.IndexOf(FDataLink.Field.AsString); if I >= 0 then Result := Items.Strings[I] end else Result := FDataLink.Field.Text; end; function TJvDBComboBox.GetComboText: string; var I: Integer; begin if (Style in [csDropDown, csSimple]) and not FEnableValues then Result := Text else begin I := ItemIndex; if (I < 0) or (FEnableValues and (FValues.Count < I + 1)) then Result := '' else if FEnableValues then Result := FValues[I] else Result := Items[I]; end; end; procedure TJvDBComboBox.SetComboText(const Value: string); var I: Integer; Redraw: Boolean; begin if Value <> ComboText then begin if Style <> csDropDown then begin Redraw := (Style <> csSimple) and HandleAllocated; if Redraw then SendMessage(Handle, WM_SETREDRAW, 0, 0); try if Value = '' then I := -1 else begin I := Items.IndexOf(Value); if (I = -1) and FEnableValues then I := Values.IndexOf(Value); end; if I >= Items.Count then I := -1; ItemIndex := I; finally if Redraw then begin SendMessage(Handle, WM_SETREDRAW, 1, 0); Invalidate; end; end; if I >= 0 then Exit; end; if Style in [csDropDown, csSimple] then Text := Value; end; end; procedure TJvDBComboBox.SetEnableValues(Value: Boolean); begin if FEnableValues <> Value then begin if Value and (Style in [csDropDown, csSimple]) then Style := csDropDownList; FEnableValues := Value; DataChange(Self); end; end; function TJvDBComboBox.GetValues: TStrings; begin Result := FValues; end; procedure TJvDBComboBox.SetValues(Value: TStrings); begin FValues.Assign(Value); end; procedure TJvDBComboBox.SetStyle(Value: TComboBoxStyle); begin if (Value in [csSimple, csDropDown]) and FEnableValues then FEnableValues := False; inherited SetStyle(Value); end; {$IFDEF UNITVERSIONING} initialization RegisterUnitVersion(HInstance, UnitVersioning); finalization UnregisterUnitVersion(HInstance); {$ENDIF UNITVERSIONING} end.