Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Delphi auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen (https://www.delphipraxis.net/200764-auf-komponenteneigenschaften-grijjy-mvvm-starter-kit-zugreifen.html)

jus 22. Mai 2019 23:41


auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen
 
Hallo,

aktuell versuche ich mich in das Grijjy MVVM Starter Kit einzuarbeiten. Ich verwende das Demo von Markus Kinzler als Grundlage. Doch wie kann man im ViewModel auf Komponenteneigenschaften der GUI wie z.B. ItemIndex von einer ComboBox zugreifen?

Wenn ich in der View folgendes mache, akzeptiert der Compiler es zwar klaglos, doch es passiert nichts:
Delphi-Quellcode:
procedure TViewDemo.SetupView;
begin
  inherited;

  //..

  Binder.Bind(ViewModel, 'SpezialItemIndex', ComboBox1, 'ItemIndex');

  //..
end;
Mein ViewModel sieht so aus:
Delphi-Quellcode:
TViewModelDemo = class(TgoObservable)
  private
    //..
    FSpezialItemIndex: Integer;
    //..
  public
     constructor Create(const Model: TModel);

    //..
    { ComboBox }
    property Entries: TEnumerable<TEntry> read GetEntries;
    property SelectedEntry: TEntry read FSelectedEntry write SetSelectedEntry;
    property SpezialItemIndex: Integer read FSpezialItemIndex write FSpezialItemIndex;
  end;
lg,
jus

Schokohase 23. Mai 2019 00:53

AW: auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen
 
Wie bekommt denn die View mitgeteilt, dass sich im ViewModel ein Eigenschafts-Wert geändert hat?

Im Demo passiert das wie folgt:
Delphi-Quellcode:
procedure TViewModelDemo.SetText(const Value: string);
begin
  if (Value <> FsRich) then
  begin
    FsRich := Value;
    PropertyChanged('TextChanged');
  end;
end;
Und in deinem ViewModel? Du schreibst den Wert einfach in das Feld und fertig. Das ist ein bisschen wenig.

jus 24. Mai 2019 00:42

AW: auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen
 
@Schokohase: du hattest recht, was Benachrichtigung seitens Viewmodel betrifft. :thumb:
Das war aber nicht wirklich die Herausforderung. Das Problem war, dass die gebundene TComboBox.ItemIndex View-Eigenschaft im Viewmodel nie aktualisiert/benachrichtigt wurde. In DSharp MVVM konnte man mit "AddElementConvention<TCombobox>" die Binding Eigenschaften selbst erweitern. Da ich den Code von Grijjy MVVM aktuell noch nicht durchschaue habe ich brutal quick&dirty wie folgt die "Grijjy.Mvvm.Controls.Vcl.pas" angepasst:
Delphi-Quellcode:
procedure TComboBox.Change;
begin
  if Assigned(FOnPropertyChanged) then
  begin
    FOnPropertyChanged.Invoke(Self, 'Text');
    FOnPropertyChanged.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
  end;

  if Assigned(FOnPropertyChangeTracking) then
  begin
    FOnPropertyChangeTracking.Invoke(Self, 'Text');
    FOnPropertyChanged.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
  end;

  SetSelectedItem( Items.Objects[ItemIndex]);
  inherited;
end;
Das Viewmodel schaut dann so aus:
Delphi-Quellcode:
TViewModelDemo = class(TgoObservable)
  private
    //..
    FSpezialItemIndex: Integer;
    procedure setSpezialItemIndex(const Value: Integer);
    //..
  public
    //..
    property SpezialItemIndex: Integer read FSpezialItemIndex write setSpezialItemIndex;
  end;
implementation
//..
procedure TViewModelDemo.setSpezialItemIndex(const Value: Integer);
begin
  if FSpezialItemIndex = Value then exit;

  FSpezialItemIndex := Value;
  PropertyChanged('SpezialItemIndex');
end;
Nun die Frage an die Experten, so wie ich die Grijjy.Mvvm.Controls.Vcl.pas angepasst habe, ist es im Sinne des Erfinders oder sollte man eher an einer anderen Stelle ansetzen? :gruebel:

lg,
jus

Stevie 24. Mai 2019 15:21

AW: auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen
 
Zitat:

Zitat von jus (Beitrag 1433007)
Delphi-Quellcode:
  if Assigned(FOnPropertyChangeTracking) then
  begin
    FOnPropertyChangeTracking.Invoke(Self, 'Text');
    FOnPropertyChanged.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
  end;

Das muss wohl
Delphi-Quellcode:
FOnPropertyChangeTracking.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
heißen.

Sieht ansonsten aber nach genau der richtigen Stelle aus. Nur die Control Interposer Klassen können die Notifications Triggern, wenn man diesen Ansatz nimmt um das Feature als Drop-in zu nutzen und nicht extra binding aware Controls bauen möchte. Kurzer Blick in meinen DSharp Code geworfen... jup, ItemIndex und Text sind auch dort die Eigenschaften, über deren Änderung benachrichtig wird im Change.


Soweit ich das sehen kann, ist das Projekt aber auch derzeit nicht über den "netter Prototyp" Status hinausgekommen - mein ich keineswegs negativ.
Richtiges MVVM (und nicht, das was manche MVVM nennen) steht und fällt mit ner guten Databinding Lösung - und die schaut durchaus tauglich aus. Das ist auch zusammen mit dem TreeViewPresenter der Teil, den wir aus DSharp einsetzen.

Ich bin inzwischen der Meinung, dass wenn man zu ambitioniert versucht, MVVM so zu bauen, wie es in anderen Sprachen geht (deklarativ), fällt einem das zu schnell auf die Füße in Delphi, weils entweder wie bei Visual Livebinding einfach nur ein Riesengroßer unwartbarer Klump in den dfm Dateien wird außer man baut ein robustes Tooling drumrum, was massig Arbeit ist, so dass man das zur Designzeit schon auswerten kann - grad wenn man mit CoC arbeitet und Dinge automatisch binden lässt.

jus 24. Mai 2019 21:43

AW: auf Komponenteneigenschaften in Grijjy MVVM Starter Kit zugreifen
 
Zitat:

Zitat von Stevie (Beitrag 1433074)
Zitat:

Zitat von jus (Beitrag 1433007)
Delphi-Quellcode:
  if Assigned(FOnPropertyChangeTracking) then
  begin
    FOnPropertyChangeTracking.Invoke(Self, 'Text');
    FOnPropertyChanged.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
  end;

Das muss wohl
Delphi-Quellcode:
FOnPropertyChangeTracking.Invoke(Self, 'ItemIndex');  // <--- hier erweitert
heißen.

ups, ja sollte so heißen :oops:

Zitat:

Zitat von Stevie (Beitrag 1433074)
Sieht ansonsten aber nach genau der richtigen Stelle aus. Nur die Control Interposer Klassen können die Notifications Triggern, wenn man diesen Ansatz nimmt um das Feature als Drop-in zu nutzen und nicht extra binding aware Controls bauen möchte. Kurzer Blick in meinen DSharp Code geworfen... jup, ItemIndex und Text sind auch dort die Eigenschaften, über deren Änderung benachrichtig wird im Change.


Soweit ich das sehen kann, ist das Projekt aber auch derzeit nicht über den "netter Prototyp" Status hinausgekommen - mein ich keineswegs negativ.
Richtiges MVVM (und nicht, das was manche MVVM nennen) steht und fällt mit ner guten Databinding Lösung - und die schaut durchaus tauglich aus. Das ist auch zusammen mit dem TreeViewPresenter der Teil, den wir aus DSharp einsetzen.

Ich bin inzwischen der Meinung, dass wenn man zu ambitioniert versucht, MVVM so zu bauen, wie es in anderen Sprachen geht (deklarativ), fällt einem das zu schnell auf die Füße in Delphi, weils entweder wie bei Visual Livebinding einfach nur ein Riesengroßer unwartbarer Klump in den dfm Dateien wird außer man baut ein robustes Tooling drumrum, was massig Arbeit ist, so dass man das zur Designzeit schon auswerten kann - grad wenn man mit CoC arbeitet und Dinge automatisch binden lässt.

ok, vielen Dank für die Rückmeldung Stefan! :thumb:

lg,
jus


Alle Zeitangaben in WEZ +1. Es ist jetzt 06:31 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz