2

我想要一个 TMemo,它总是以字符串 'SELECT c_name FROM' 开头,我想锁定它,这样用户就不能删除或替换 TMemo 中的这个字符串,他们必须在这个字符串之后写他们的文本。有人可以帮我吗?我尝试了 onChange 事件,但问题是用户可以在 TMemo 的开头单击并在开头进行编辑。

我正在使用德尔福 6。

4

2 回答 2

10

你所要求的是不可能的TMemo。它只是标准 Win32EDIT控件的薄包装,不支持这种功能。

你需要TRichEdit改用。它支持像您描述的那样保护文本。添加所需文本后,使用TRichEdit.SelStartTRichEdit.SelLength属性选择它,然后将TRichEdit.SelAttributes.Protected属性设置为 true。如果用户尝试以任何方式修改受保护的文本,TRichEdit默认情况下将拒绝修改(您可以通过使用TRichEdit.OnProtectChange事件将AllowChange参数设置为 true 来覆盖该决定)。例如:

RichEdit1.Text := 'SELECT c_name FROM ';
RichEdit1.SelStart := 0;
RichEdit1.SelLength := 19;
RichEdit1.SelAttributes.Protected := True;

更新:默认情况下,保护文本也可以防止用户复制它。如果您希望用户能够复制受保护的文本,您必须明确启用它。尽管TRichEdit有一个OnProtectChange事件允许访问受保护的文本,但它并没有告诉您为什么文本请求访问。但是,基础EN_PROTECTED通知确实如此。因此,您可以将 子类化TRichEdit以直接拦截EN_PROTECTED,然后仅在用户复制受保护的文本时才返回 0(允许访问)。

interface

uses
  ...;

type
  TMyForm = class(TForm)
    RichEdit1: TRichEdit;
    procedure FormCreate(Sender: TObject);
  private
    { Private declarations }
    DefRichEditWndProc: TWndMethod;
    procedure RichEditWndProc(var Message: TMessage);
  public
    { Public declarations }
  end;

...

implementation

uses
  Richedit;

procedure TMyForm.FormCreate(Sender: TObject);
begin
  DefRichEditWndProc := RichEdit1.WindowProc;
  RichEdit1.WindowProc := RichEditWndProc;

  RichEdit1.Text := 'SELECT c_name FROM ';
  RichEdit1.SelStart := 0;
  RichEdit1.SelLength := 19;
  RichEdit1.SelAttributes.Protected := True;
end;

procedure TMyForm.RichEditWndProc(var Message: TMessage);
begin
  DefRichEditWndProc(Message);
  if Message.Msg = CN_NOTIFY then
  begin
    if TWMNotify(Message).NMHdr.code = EN_PROTECTED then
    begin
      if PENProtected(Message.lParam).Msg = WM_COPY then
        Message.Result := 0;
    end;
  end;
end;

或者:

interface

uses
  ...;

type
  TRichEdit = class(ComCtrls.TRichEdit)
  private
    procedure CNNotify(var Message: TWMNotify); message CN_NOTIFY;
  end;

  TMyForm = class(TForm)
    RichEdit1: TRichEdit;
    procedure FormCreate(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

...

implementation

uses
  Richedit;

procedure TMyForm.FormCreate(Sender: TObject);
begin
  RichEdit1.Text := 'SELECT c_name FROM ';
  RichEdit1.SelStart := 0;
  RichEdit1.SelLength := 19;
  RichEdit1.SelAttributes.Protected := True;
end;

procedure TRichEdit.CNNotify(var Message: TWMNotify);
begin
  inherited;
  if Message.NMHdr.code = EN_PROTECTED then
  begin
    if PENProtected(Message.NMHdr).Msg = WM_COPY then
      Message.Result := 0;
  end;
end;
于 2015-12-11T18:01:25.633 回答
2

嗯,这是可能的,但我不知道,如果你喜欢这个解决方案。

正如已经说过TMemo的那样,没有实施那种行为。所以你必须对这种行为进行编程。

使用应用程序OnIdle事件和内容的纪念品。在每个空闲消息上验证备忘录内容。如果备忘录内容不是以开头,'SELECT c_name FROM '则分配备忘录值,否则将备忘录内容存储到备忘录。

这是验证和游标保护的示例

type
  TMainForm = class( TForm )
    Memo1: TMemo;
    ApplicationEvents1: TApplicationEvents; // OnIdle = ApplicationEvents1Idle
    procedure ApplicationEvents1Idle( Sender: TObject; var Done: Boolean );
  private
    FMemo1StartWith: string;
    FMemo1Memento  : string;
    procedure ValidateMemo( AMemo: TMemo; const AStartWith: string; var AMemento: string );
  public
    procedure AfterConstruction; override;

  end;

var
  MainForm: TMainForm;

implementation

{$R *.dfm}

procedure TMainForm.AfterConstruction;
begin
  inherited;
  FMemo1StartWith := 'SELECT c_name FROM ';
end;

procedure TMainForm.ApplicationEvents1Idle( Sender: TObject; var Done: Boolean );
begin
  ValidateMemo( Memo1, FMemo1StartWith, FMemo1Memento );
end;

procedure TMainForm.ValidateMemo( AMemo: TMemo; const AStartWith: string; var AMemento: string );
var
  lCurrentContent: string;
begin
  // protect content

  lCurrentContent := AMemo.Text;
  if Pos( AStartWith, lCurrentContent ) = 1
  then
    AMemento := lCurrentContent
  else if Pos( AStartWith, AMemento ) = 1
  then
    AMemo.Text := AMemento
  else
    AMemo.Text := AStartWith;

  // protect cursor position

  if ( AMemo.SelLength = 0 ) and ( AMemo.SelStart < Length( AStartWith ) )
  then
    AMemo.SelStart := Length( AStartWith );
end;
于 2015-12-11T21:09:47.800 回答