{----------------------------------------------------------------------------- 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: JvDBLookupEdit.PAS, released on 2003-09-18. The Code was modified to: JvDBLookupComboEdit.PAS, released on 2003-10-20. The Initial Developers of the Original Code are: Michael Habbe Copyright (c) 2003 Michael Habbe All Rights Reserved. Contributor(s): 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: ============= It must inherit from JvLookupEdit, because the new structure in JVCL3 produced (in my environment) stack-overflow-errors, which "kicked" Delphi out of the memory with only the message "Danger. Stack-Overflow. Save your work and restart Delphi.". (The message is in German and i never saw it before!?) I find out the problem in line 286 "or inherited ReadOnly;", when i uncommented it, Delphi works, but i can modify the dataset, although i set ReadOnly to True. As aforesaid, the component works in my Delphi with JVCL2, but as soon as i inherit it with JVCL3 from JvDBLookupEdit, the specified errors occur. Michael Habbe [2003-10-20] -----------------------------------------------------------------------------} // $Id: JvDBLookupComboEdit.pas 10699 2006-06-10 17:49:41Z obones $ unit JvDBLookupComboEdit; {$I jvcl.inc} interface uses {$IFDEF UNITVERSIONING} JclUnitVersioning, {$ENDIF UNITVERSIONING} Windows, {$IFDEF VCL} Messages, {$ENDIF VCL} Classes, Controls, Graphics, DB, DBCtrls, JvDBLookup; type TJvDBLookupComboEdit = class(TJvDBLookupEdit) private FDataLink: TFieldDataLink; FCanvas: TControlCanvas; // FAlignment: TAlignment; FFocused: Boolean; FBeepOnError: Boolean; procedure ActiveChange(Sender: TObject); procedure DataChange(Sender: TObject); procedure EditingChange(Sender: TObject); function GetCanvas: TCanvas; function GetDataField: string; function GetDataSource: TDataSource; function GetField: TField; function GetTextMargins: TPoint; procedure ResetMaxLength; procedure SetDataField(const Value: string); procedure SetDataSource(Value: TDataSource); procedure SetFocused(Value: Boolean); procedure UpdateData(Sender: TObject); procedure WMPaint(var Msg: TWMPaint); message WM_PAINT; procedure CMGetDataLink(var Msg: TMessage); message CM_GETDATALINK; protected procedure WMCut(var Msg: TMessage); message WM_CUT; procedure WMPaste(var Msg: TMessage); message WM_PASTE; procedure WMUndo(var Msg: TMessage); message WM_UNDO; procedure DoEnter; override; procedure DoExit; override; function GetReadOnly: Boolean; override; // suppress the warning procedure SetReadOnly(Value: Boolean); override; procedure Change; override; function EditCanModify: Boolean; override; 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 Reset; override; 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 Canvas: TCanvas read GetCanvas; published property Anchors; property AutoSelect; property AutoSize; property BeepOnError: Boolean read FBeepOnError write FBeepOnError default True; {$IFDEF COMPILER6_UP} property BevelEdges; property BevelInner; property BevelKind default bkNone; property BevelOuter; {$ENDIF COMPILER6_UP} property BiDiMode; property BorderStyle; property CharCase; property Color; property Constraints; property DataField: string read GetDataField write SetDataField; property DataSource: TDataSource read GetDataSource write SetDataSource; property DragCursor; property DragKind; property DragMode; property Enabled; property Font; property ImeMode; property ImeName; property MaxLength; property ParentBiDiMode; property ParentColor; property ParentFont; property ParentShowHint; property PasswordChar; property PopupMenu; property ReadOnly: Boolean read GetReadOnly write SetReadOnly default False; property ShowHint; property TabOrder; property TabStop; property Visible; property OnChange; property OnClick; property OnContextPopup; property OnDblClick; property OnDragDrop; property OnDragOver; property OnEndDock; property OnEndDrag; property OnEnter; property OnExit; property OnKeyDown; property OnKeyPress; property OnKeyUp; property OnMouseDown; property OnMouseMove; property OnMouseUp; property OnStartDock; property OnStartDrag; end; {$IFDEF UNITVERSIONING} const UnitVersioning: TUnitVersionInfo = ( RCSfile: '$URL: https://jvcl.svn.sourceforge.net/svnroot/jvcl/tags/JVCL3_32/run/JvDBLookupComboEdit.pas $'; Revision: '$Revision: 10699 $'; Date: '$Date: 2006-06-10 19:49:41 +0200 (sam., 10 juin 2006) $'; LogPath: 'JVCL\run' ); {$ENDIF UNITVERSIONING} implementation uses SysUtils, Forms, StdCtrls, JvConsts; constructor TJvDBLookupComboEdit.Create(AOwner: TComponent); begin inherited Create(AOwner); // inherited ReadOnly := True; ControlStyle := ControlStyle + [csReplicatable]; FCanvas := TControlCanvas.Create; FCanvas.Control := Self; FDataLink := TFieldDataLink.Create; FDataLink.Control := Self; FDataLink.OnDataChange := DataChange; FDataLink.OnEditingChange := EditingChange; FDataLink.OnUpdateData := UpdateData; FDataLink.OnActiveChange := ActiveChange; FBeepOnError := True; end; destructor TJvDBLookupComboEdit.Destroy; begin FDataLink.Free; FDataLink := nil; inherited Destroy; // (rom) destroy Canvas AFTER inherited Destroy FCanvas.Free; end; procedure TJvDBLookupComboEdit.ResetMaxLength; var F: TField; begin if (MaxLength > 0) and Assigned(DataSource) and Assigned(DataSource.DataSet) then begin F := DataSource.DataSet.FindField(DataField); if Assigned(F) and (F.DataType in [ftString, ftWideString]) and (F.Size = MaxLength) then MaxLength := 0; end; end; procedure TJvDBLookupComboEdit.Loaded; begin inherited Loaded; ResetMaxLength; if csDesigning in ComponentState then DataChange(Self); end; procedure TJvDBLookupComboEdit.Notification(AComponent: TComponent; Operation: TOperation); begin inherited Notification(AComponent, Operation); if (Operation = opRemove) and (FDataLink <> nil) and (AComponent = DataSource) then DataSource := nil; end; //function TJvDBLookupComboEdit.UseRightToLeftAlignment: Boolean; //begin // Result := DBUseRightToLeftAlignment(Self, Field); //end; procedure TJvDBLookupComboEdit.KeyDown(var Key: Word; Shift: TShiftState); begin // inherited KeyDown(Key, Shift); // if (Key = VK_DELETE) or ((Key = VK_INSERT) and (ssShift in Shift)) then // FDataLink.Edit; // new order, because result of inherited KeyDown(...) could be = 0 // so, first set DataSet in Edit-Mode if Key in [VK_BACK, VK_DELETE, VK_UP, VK_DOWN, 32..255] then // taken from TDBComboBox.KeyDown(...) FDataLink.Edit; inherited KeyDown(Key, Shift); end; procedure TJvDBLookupComboEdit.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; // Key := #0; end; end; end; function TJvDBLookupComboEdit.EditCanModify: Boolean; begin Result := FDataLink.Edit; end; procedure TJvDBLookupComboEdit.Reset; begin FDataLink.Reset; SelectAll; end; procedure TJvDBLookupComboEdit.SetFocused(Value: Boolean); begin if FFocused <> Value then begin FFocused := Value; // if (FAlignment <> taLeftJustify) and not IsMasked then Invalidate; FDataLink.Reset; end; end; procedure TJvDBLookupComboEdit.Change; begin FDataLink.Modified; inherited Change; end; function TJvDBLookupComboEdit.GetDataSource: TDataSource; begin Result := FDataLink.DataSource; end; procedure TJvDBLookupComboEdit.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 TJvDBLookupComboEdit.GetDataField: string; begin Result := FDataLink.FieldName; end; procedure TJvDBLookupComboEdit.SetDataField(const Value: string); begin if not (csDesigning in ComponentState) then ResetMaxLength; FDataLink.FieldName := Value; end; function TJvDBLookupComboEdit.GetReadOnly: Boolean; begin Result := FDataLink.ReadOnly or inherited GetReadOnly; end; procedure TJvDBLookupComboEdit.SetReadOnly(Value: Boolean); begin inherited SetReadOnly(Value); FDataLink.ReadOnly := Value; end; function TJvDBLookupComboEdit.GetCanvas: TCanvas; begin Result := FCanvas; end; function TJvDBLookupComboEdit.GetField: TField; begin Result := FDataLink.Field; end; procedure TJvDBLookupComboEdit.ActiveChange(Sender: TObject); begin ResetMaxLength; end; procedure TJvDBLookupComboEdit.DataChange(Sender: TObject); begin if FDataLink.Field <> nil then begin //if FAlignment <> FDataLink.Field.Alignment then //begin // EditText := ''; {forces update} // FAlignment := FDataLink.Field.Alignment; //end; EditMask := FDataLink.Field.EditMask; if not (csDesigning in ComponentState) then if (FDataLink.Field.DataType in [ftString, ftWideString]) and (MaxLength = 0) then MaxLength := FDataLink.Field.Size; if FFocused and FDataLink.CanModify then Text := FDataLink.Field.Text else begin EditText := FDataLink.Field.DisplayText; if FDataLink.Editing then //and FDataLink.FModified || fmodified is private in parent of fdatalink Modified := True; end; end else begin //FAlignment := taLeftJustify; EditMask := ''; if csDesigning in ComponentState then EditText := Name else EditText := ''; end; end; procedure TJvDBLookupComboEdit.EditingChange(Sender: TObject); begin //ReadOnly := not FDataLink.Editing; end; procedure TJvDBLookupComboEdit.UpdateData(Sender: TObject); begin ValidateEdit; FDataLink.Field.Text := Text; end; procedure TJvDBLookupComboEdit.WMUndo(var Msg: TMessage); begin FDataLink.Edit; inherited; end; procedure TJvDBLookupComboEdit.WMPaste(var Msg: TMessage); begin FDataLink.Edit; inherited; end; procedure TJvDBLookupComboEdit.WMCut(var Msg: TMessage); begin FDataLink.Edit; inherited; end; procedure TJvDBLookupComboEdit.DoEnter; begin SetFocused(True); inherited DoEnter; if SysLocale.FarEast and FDataLink.CanModify then inherited ReadOnly := False; end; procedure TJvDBLookupComboEdit.DoExit; begin try FDataLink.UpdateRecord; except SelectAll; SetFocus; raise; end; SetFocused(False); CheckCursor; inherited DoExit; end; procedure TJvDBLookupComboEdit.WMPaint(var Msg: TWMPaint); const AlignStyle: array [Boolean, TAlignment] of DWORD = ((WS_EX_LEFT, WS_EX_RIGHT, WS_EX_LEFT), (WS_EX_RIGHT, WS_EX_LEFT, WS_EX_LEFT)); var Left: Integer; Margins: TPoint; R: TRect; DC: HDC; PS: TPaintStruct; S: string; AAlignment: TAlignment; ExStyle: DWORD; begin if csDestroying in ComponentState then Exit; AAlignment := Alignment; //FAlignment; if UseRightToLeftAlignment then ChangeBiDiModeAlignment(AAlignment); if ((AAlignment = taLeftJustify) or FFocused) and not (csPaintCopy in ControlState) then begin if SysLocale.MiddleEast and HandleAllocated and (IsRightToLeft) then begin { This keeps the right aligned text, right aligned } ExStyle := DWORD(GetWindowLong(Handle, GWL_EXSTYLE)) and (not WS_EX_RIGHT) and (not WS_EX_RTLREADING) and (not WS_EX_LEFTSCROLLBAR); if UseRightToLeftReading then ExStyle := ExStyle or WS_EX_RTLREADING; if UseRightToLeftScrollbar then ExStyle := ExStyle or WS_EX_LEFTSCROLLBAR; ExStyle := ExStyle or AlignStyle[UseRightToLeftAlignment, AAlignment]; if DWORD(GetWindowLong(Handle, GWL_EXSTYLE)) <> ExStyle then SetWindowLong(Handle, GWL_EXSTYLE, ExStyle); end; inherited; Exit; end; { Since edit controls do not handle justification unless multi-line (and then only poorly) we will draw right and center justify manually unless the edit has the focus. } DC := Msg.DC; if DC = 0 then DC := BeginPaint(Handle, PS); FCanvas.Handle := DC; try FCanvas.Font := Font; with FCanvas do begin R := ClientRect; if not (NewStyleControls and Ctl3D) and (BorderStyle = bsSingle) then begin Brush.Color := clWindowFrame; FrameRect(R); InflateRect(R, -1, -1); end; Brush.Color := Color; if not Enabled then Font.Color := clGrayText; if (csPaintCopy in ControlState) and (FDataLink.Field <> nil) then begin S := FDataLink.Field.DisplayText; case CharCase of ecUpperCase: S := AnsiUpperCase(S); ecLowerCase: S := AnsiLowerCase(S); end; end else S := EditText; if PasswordChar <> #0 then FillChar(S[1], Length(S), PasswordChar); Margins := GetTextMargins; case AAlignment of taLeftJustify: Left := Margins.X; taRightJustify: Left := ClientWidth - TextWidth(S) - Margins.X - 1; else Left := (ClientWidth - TextWidth(S)) div 2; end; if SysLocale.MiddleEast then UpdateTextFlags; TextRect(R, Left, Margins.Y, S); end; finally FCanvas.Handle := 0; if Msg.DC = 0 then EndPaint(Handle, PS); end; end; procedure TJvDBLookupComboEdit.CMGetDataLink(var Msg: TMessage); begin Msg.Result := Integer(FDataLink); end; function TJvDBLookupComboEdit.GetTextMargins: TPoint; var DC: HDC; SaveFont: HFont; I: Integer; SysMetrics, Metrics: TTextMetric; begin if NewStyleControls then begin if BorderStyle = bsNone then I := 0 else if Ctl3D then I := 1 else I := 2; Result.X := SendMessage(Handle, EM_GETMARGINS, 0, 0) and $0000FFFF + I; Result.Y := I; end else begin if BorderStyle = bsNone then I := 0 else begin DC := GetDC(HWND_DESKTOP); GetTextMetrics(DC, SysMetrics); SaveFont := SelectObject(DC, Font.Handle); GetTextMetrics(DC, Metrics); SelectObject(DC, SaveFont); ReleaseDC(HWND_DESKTOP, DC); I := SysMetrics.tmHeight; if I > Metrics.tmHeight then I := Metrics.tmHeight; I := I div 4; end; Result.X := I; Result.Y := I; end; end; function TJvDBLookupComboEdit.ExecuteAction(Action: TBasicAction): Boolean; begin Result := inherited ExecuteAction(Action) or (FDataLink <> nil) and FDataLink.ExecuteAction(Action); end; function TJvDBLookupComboEdit.UpdateAction(Action: TBasicAction): Boolean; begin Result := inherited UpdateAction(Action) or (FDataLink <> nil) and FDataLink.UpdateAction(Action); end; {$IFDEF UNITVERSIONING} initialization RegisterUnitVersion(HInstance, UnitVersioning); finalization UnregisterUnitVersion(HInstance); {$ENDIF UNITVERSIONING} end.