Plugin-DLL: Unterschied zwischen den Versionen

Aus LOTUS Wiki DE
Wechseln zu: Navigation, Suche
K (1 Version importiert: Init)
(Updated from LOTUS Lexicon on 2025-01-21)
 
Zeile 1: Zeile 1:
[[Category:Lexikon]]
{{ArticleMetadata
[[Category:Unvollständige Seiten]]
|author=Marcel Kuhnt
Die Plugin-DLL enthält den eigentlichen ausführbaren Code des Plugins.
|created_date=2017-04-11T23:29:56+02:00
|modified_date=2021-09-23T11:09:12+02:00
|source_url=https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/
}}
 
Die Plugin-DLL enthält den eigentlichen ausführbaren Code des Plugins.  
 
Für die Kommunikation mit LOTUS müssen je nach Bedarf festgelegte Methoden exportiert werden. Im Folgenden soll deshalb erläutert werden, wie der Variablen-Zugriff genau von statten geht und welche Methoden zu exportieren sind. Grundsätzlich bestehen aber große Freiheiten beim Anlegen der DLL; beispielsweise ist es möglich, ein zusätzliches Fenster mit VCL-Komponenten darzustellen und zu verwenden.
Für die Kommunikation mit LOTUS müssen je nach Bedarf festgelegte Methoden exportiert werden. Im Folgenden soll deshalb erläutert werden, wie der Variablen-Zugriff genau von statten geht und welche Methoden zu exportieren sind. Grundsätzlich bestehen aber große Freiheiten beim Anlegen der DLL; beispielsweise ist es möglich, ein zusätzliches Fenster mit VCL-Komponenten darzustellen und zu verwenden.


Da – abgesehen vom <code>AOwner</code> – auf die Verwendung von Objekten/Klassen verzichtet wurde, ist es vermutlich möglich, die Plugin-DLL auch in einer anderen Programmiersprache als Object Pascal zu entwickeln. Getestet wurde von mir aber nur Object Pascal (Delphi). Dementsprechend ist das folgende Beispiel auch in dieser Programmiersprache verfasst.
 
 
Da – abgesehen vom "AOwner" – auf die Verwendung von Objekten/Klassen verzichtet wurde, ist es vermutlich möglich, die Plugin-DLL auch in einer anderen Programmiersprache als Object Pascal zu entwickeln. Getestet wurde von mir aber nur Object Pascal (Delphi). Dementsprechend ist das folgende Beispiel auch in dieser Programmiersprache verfasst.
 
 


Welche Methoden aber konkret auf welche Weise deklariert und exportiert werden müssen, um verschiedene Kommunikationsaufgaben zu übernehmen, wird anhand des folgenden Beispiels erläutert. ACHTUNG: Im Abschnitt <code>exports</code> muss dringend auf die korrekte Groß- und Kleinschreibung geachtet werden!
Welche Methoden aber konkret auf welche Weise deklariert und exportiert werden müssen, um verschiedene Kommunikationsaufgaben zu übernehmen, wird anhand des folgenden Beispiels erläutert. **ACHTUNG: Im Abschnitt "exports" muss dringend auf die korrekte Groß- und Kleinschreibung geachtet werden!**
<div class="toccolours mw-collapsible mw-collapsed" overflow:auto;">
<div style="font-weight:bold;line-height:1.6;">Beispielcode</div>
<div class="mw-collapsible-content">
library Test;
uses
  SysUtils,
  Dialogs,
  Classes,
  TestU in 'TestU.pas' {Form1};
{$R *.res}
procedure PluginStart(AOwner: TComponent); stdcall;
begin
  form1 := TForm1.Create( AOwner );
  form1.Show;
end;
procedure PluginFinalize; stdcall;
begin
  form1.Free;
end;
procedure ReceiveVarFloat(varindex: word; value: single); stdcall;
begin
  case varindex of
    0:
    begin
      form1.Label2.Caption := floattostrF( value, ffFixed, 5, 1 );
      form1.Gauge1.Progress := round( value );
    end;
  end;
end;
procedure ReceiveVarBool(varindex: word; value: boolean); stdcall;
begin
  case varindex of
    0:
    begin
      form1.Label2.Caption := BoolToStr(value);
    end;
  end;
end;
procedure ReceiveVarInt(varindex: word; value: integer); stdcall;
begin
  case varindex of
    0:
    begin
      form1.Label2.Caption := IntToStr(value);
      form1.Gauge1.Progress := value;
    end;
  end;
end;
procedure OnConnectingVehicle(name: shortstring); stdcall;
begin
  form1.Caption := name;
end;
procedure OnUnconnectingVehicle; stdcall;
begin
  form1.Caption := 'no bus loaded';
end;
function SetButton(eventindex: word): boolean;
begin
  case eventindex of
    0: result := Form1.button1_pressed;
  end;
end;
function SetFloat(eventindex: word): single;
begin
  case eventindex of
    0: result := (Form1.TrackBar1.Position - 15)/15;
  end;
end;
exports
  ReceiveVarInt,
  SetButton,
  SetFloat,
  PluginStart,
  OnConnectingVehicle,
  OnUnconnectingVehicle,
  PluginFinalize;
begin
end.
</div></div>


 
Code
  1. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_1_bef16a|]]library Test;
  2. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_2_bef16a|]]
  3. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_3_bef16a|]]uses
  4. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_4_bef16a|]] SysUtils,
  5. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_5_bef16a|]] Dialogs,
  6. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_6_bef16a|]] Classes,
  7. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_7_bef16a|]] TestU in 'TestU.pas' {Form1};
  8. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_8_bef16a|]]
  9. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_9_bef16a|]]{$R *.res}
  10. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_10_bef16a|]]
  11. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_11_bef16a|]]procedure PluginStart(AOwner: TComponent); stdcall;
  12. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_12_bef16a|]]begin
  13. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_13_bef16a|]] form1 := TForm1.Create( AOwner );
  14. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_14_bef16a|]] form1.Show;
  15. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_15_bef16a|]]end;
  16. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_16_bef16a|]]
  17. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_17_bef16a|]]procedure PluginFinalize; stdcall;
  18. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_18_bef16a|]]begin
  19. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_19_bef16a|]] form1.Free;
  20. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_20_bef16a|]]end;
  21. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_21_bef16a|]]
  22. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_22_bef16a|]]procedure ReceiveVarFloat(varindex: word; value: single); stdcall;
  23. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_23_bef16a|]]begin
  24. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_24_bef16a|]] case varindex of
  25. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_25_bef16a|]] 0:
  26. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_26_bef16a|]] begin
  27. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_27_bef16a|]] form1.Label2.Caption := floattostrF( value, ffFixed, 5, 1 );
  28. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_28_bef16a|]] form1.Gauge1.Progress := round( value );
  29. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_29_bef16a|]] end;
  30. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_30_bef16a|]] end;
  31. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_31_bef16a|]]end;
  32. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_32_bef16a|]]
  33. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_33_bef16a|]]procedure ReceiveVarBool(varindex: word; value: boolean); stdcall;
  34. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_34_bef16a|]]begin
  35. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_35_bef16a|]] case varindex of
  36. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_36_bef16a|]] 0:
  37. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_37_bef16a|]] begin
  38. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_38_bef16a|]] form1.Label2.Caption := BoolToStr(value);
  39. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_39_bef16a|]] end;
  40. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_40_bef16a|]] end;
  41. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_41_bef16a|]]end;
  42. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_42_bef16a|]]
  43. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_43_bef16a|]]procedure ReceiveVarInt(varindex: word; value: integer); stdcall;
  44. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_44_bef16a|]]begin
  45. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_45_bef16a|]] case varindex of
  46. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_46_bef16a|]] 0:
  47. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_47_bef16a|]] begin
  48. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_48_bef16a|]] form1.Label2.Caption := IntToStr(value);
  49. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_49_bef16a|]] form1.Gauge1.Progress := value;
  50. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_50_bef16a|]] end;
  51. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_51_bef16a|]] end;
  52. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_52_bef16a|]]end;
  53. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_53_bef16a|]]
  54. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_54_bef16a|]]procedure OnConnectingVehicle(name: shortstring); stdcall;
  55. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_55_bef16a|]]begin
  56. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_56_bef16a|]] form1.Caption := name;
  57. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_57_bef16a|]]end;
  58. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_58_bef16a|]]
  59. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_59_bef16a|]]procedure OnUnconnectingVehicle; stdcall;
  60. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_60_bef16a|]]begin
  61. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_61_bef16a|]] form1.Caption := 'no bus loaded';
  62. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_62_bef16a|]]end;
  63. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_63_bef16a|]]
  64. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_64_bef16a|]]function SetButton(eventindex: word): boolean;
  65. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_65_bef16a|]]begin
  66. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_66_bef16a|]] case eventindex of
  67. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_67_bef16a|]] 0: result := Form1.button1_pressed;
  68. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_68_bef16a|]] end;
  69. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_69_bef16a|]]end;
  70. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_70_bef16a|]]
  71. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_71_bef16a|]]function SetFloat(eventindex: word): single;
  72. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_72_bef16a|]]begin
  73. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_73_bef16a|]] case eventindex of
  74. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_74_bef16a|]] 0: result := (Form1.TrackBar1.Position - 15)/15;
  75. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_75_bef16a|]] end;
  76. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_76_bef16a|]]end;
  77. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_77_bef16a|]]
  78. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_78_bef16a|]]exports
  79. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_79_bef16a|]] ReceiveVarInt,
  80. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_80_bef16a|]] SetButton,
  81. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_81_bef16a|]] SetFloat,
  82. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_82_bef16a|]] PluginStart,
  83. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_83_bef16a|]] OnConnectingVehicle,
  84. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_84_bef16a|]] OnUnconnectingVehicle,
  85. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_85_bef16a|]] PluginFinalize;
  86. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_86_bef16a|]]begin
  87. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_87_bef16a|]]end.
Alles anzeigen
Typen- und Funktionsdefinitionen für C, C++ und C#
Code
  1. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_1_e6e3a0|]]typedef void* TComponent;
  2. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_2_e6e3a0|]]typedef byte Boolean;
  3. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_3_e6e3a0|]]typedef float single, Single, *PSingle;
  4. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_4_e6e3a0|]]typedef struct _ShortString {
  5. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_5_e6e3a0|]] byte Length; // == min(sizeof(ShortString.String), 256)
  6. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_6_e6e3a0|]] char String[]; // Nicht-nullterminiert! NUR strncpy o.Ä. anwenden, rennt sonst direkt in unbekannten Speicher!
  7. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_7_e6e3a0|]]} ShortString, *PShortString;
  8. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_8_e6e3a0|]]typedef float Float, *PFloat;
  9. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_9_e6e3a0|]]typedef unsigned short word;
  10. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_10_e6e3a0|]]
  11. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_11_e6e3a0|]]__declspec(dllexport) void __stdcall PluginStart(TComponent Owner);
  12. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_12_e6e3a0|]]__declspec(dllexport) void __stdcall PluginFinalize(void);
  13. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_13_e6e3a0|]]__declspec(dllexport) void __stdcall ReceiveVarFloat(word varindex, single value);
  14. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_14_e6e3a0|]]__declspec(dllexport) void __stdcall ReceiveVarBool(word VarIndex, Boolean Value);
  15. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_15_e6e3a0|]]__declspec(dllexport) void __stdcall ReceiveVarInt(word VarIndex, int Value);
  16. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_16_e6e3a0|]]__declspec(dllexport) void __stdcall OnConnectingVehicle(ShortString Name);
  17. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_17_e6e3a0|]]__declspec(dllexport) void __stdcall OnUnconnectingVehicle(void);
  18. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_18_e6e3a0|]]__declspec(dllexport) Boolean __stdcall SetButton(single Eventindex);
  19. [[https://www.lotus-simulator.de/lexikon/index.php?entry/8-plugin-dll/#codeLine_19_e6e3a0|]]__declspec(dllexport) PFloat __stdcall SetFloat(single EventIndex);
Alles anzeigen


<div class="toccolours mw-collapsible mw-collapsed" overflow:auto;">
<div style="font-weight:bold;line-height:1.6;">Typen- und Funktionsdefintionen für C, C++ und C#</div>
<div class="mw-collapsible-content">
typedef void* TComponent; // Nicht verwendbar
typedef byte Boolean;
typedef unsigned short Single, *PSingle;
typedef struct _ShortString {
  byte Length; // == min(sizeof(ShortString.String), 256)
  char String[]; // Nicht-nullterminiert! NUR strncpy o.Ä. anwenden, rennt sonst direkt in unbekannten Speicher!
  } ShortString, *PShortString;
typedef float Float, *PFloat;
__declspec(dllexport) void __stdcall PluginStart(TComponent Owner);
__declspec(dllexport) void __stdcall PluginFinalize(void);
__declspec(dllexport) void __stdcall ReceiveVarFloat(word varindex, single value);
__declspec(dllexport) void __stdcall ReceiveVarBool(word VarIndex, Boolean Value);
__declspec(dllexport) void __stdcall ReceiveVarInt(word VarIndex, int Value);
__declspec(dllexport) void __stdcall OnConnectingVehicle(ShortString Name);
__declspec(dllexport) void __stdcall OnUnconnectingVehicle(void);
__declspec(dllexport) Boolean __stdcall SetButton(single Eventindex);
__declspec(dllexport) PFloat __stdcall SetFloat(single EventIndex);
</div></div>
Vielen Dank an Feder für den Code!
Vielen Dank an Feder für den Code!


== Initialisierung und Finalisierung ==
== Initialisierung und Finalisierung ==
Nach dem Laden der DLL wird <code>PluginStart</code> aufgerufen. In diesem Fall soll eine Form erstellt werden; für diesen Zweck bietet die Prozedur den AOwner als Parameter an.


Vor dem Entladen der DLL beim Schließen von LOTUS wird <code>PluginFinalize</code>" aufgerufen. In diesem Beispiel wird die Form aus dem Speicher gelöscht.
Nach dem Laden der DLL wird "PluginStart" aufgerufen. In diesem Fall soll eine Form erstellt werden; für diesen Zweck bietet die Prozedur den AOwner als Parameter an.


Unmittelbar nachdem ein Fahrzeug in den Fokus genommen wird – also nach dem Start der Simulation, nach dem Neuplatzieren eines Fahrzeuges oder beim Wechsel auf ein anderes Fahrzeug – und somit die DLL an das neu ausgewählte Fahrzeug andocken soll, wird die Prozedur <code>OnConnectingVehicle</code> ausgeführt. Diese übergibt als Parameter den Namen des neu ausgewählten Fahrzeuges. In diesem Beispiel wird der Name des Fahrzeuges als Titel auf die Form geschrieben.
 


Bevor ein Fahrzeug aus dem Fokus genommen werden soll (ebenfalls beim Wechsel, beim Löschen oder bei Rückkehr zum Hauptmenü) wird <code>OnUnconnectingVehicle</code> ausgeführt.
Vor dem Entladen der DLL beim Schließen von LOTUS wird "PluginFinalize" aufgerufen. In diesem Beispiel wird die Form aus dem Speicher gelöscht.


== Lesezugriff ==
 
Der Lesezugriff erfolgt mit den Prozeduren <code>ReceiveVarBool</code>, <code>ReceiveVarInt</code>, <code>ReceiveVarFloat</code> und <code>ReceiveVarString</code>. Der erste Parameter ist jeweils <code>varindex: word</code>, der zweite ist <code>value: boolean</code>, <code>integer</code>, <code>single</code> oder <code>PWideChar</code>, je nach Prozedur.


Der Ablauf ist folgender: LOTUS durchläuft die jeweilige Variablenliste in der Plugin-Ini und ruft jeweils die entsprechende <code>ReceiveVar...</code>-Prozedur auf und übergibt ihr den Index der Variable in der *.ini-Liste und den aktuellen Wert. Im Normalfall wird dann in der Prozedur ein <code>Case</code>-Konstrukt verwendet, um die Werte der Variablen entsprechend unterschiedlich zu verarbeiten.
Unmittelbar nachdem ein Fahrzeug in den Fokus genommen wird – also nach dem Start der Simulation, nach dem Neuplatzieren eines Fahrzeuges oder beim Wechsel auf ein anderes Fahrzeug – und somit die DLL an das neu ausgewählte Fahrzeug andocken soll, wird die Prozedur "OnConnectingVehicle" ausgeführt. Diese übergibt als Parameter den Namen des neu ausgewählten Fahrzeuges. In diesem Beispiel wird der Name des Fahrzeuges als Titel auf die Form geschrieben.


Im Beispiel wird in der Prozedur <code>ReceiveVarFloat</code> dafür gesorgt, dass der Wert der Variable, die in der Plugin-Ini im <code>[ReadingVarsFloat]</code>-Abschnitt unter <code>var.0</code> eingetragen ist, in das Label2 der Form geschrieben wird und die Gauge1 entsprechend ausschlägt.
 


==  Schreibzugriff ==
Bevor ein Fahrzeug aus dem Fokus genommen werden soll (ebenfalls beim Wechsel, beim Löschen oder bei Rückkehr zum Hauptmenü) wird "OnUnconnectingVehicle" ausgeführt.
Der Schreibzugriff erfolgt dahingehend, dass Tastatur-/Gamecontroller-Events ausgelöst werden (und damit der Druck von Tastenkombinationen oder Joystick-Knöpfen simuliert wird) und dass Float-Events ausgelöst werden, denen der neue Float-Wert beigefügt ist (was somit der Verarbeitung von Joystick-Achsen entspricht).
 
== 2  Lesezugriff  ==
 
Der Lesezugriff erfolgt mit den Prozeduren "ReceiveVarBool", "ReceiveVarInt", "ReceiveVarFloat" und "ReceiveVarString". Der erste Parameter ist jeweils "varindex: word", der zweite ist "value: boolean", "integer", "single" oder "PWideChar", je nach Prozedur.
 
 
 
Der Ablauf ist folgender: LOTUS durchläuft die jeweilige Variablenliste in der Plugin-Ini und ruft jeweils die entsprechende "ReceiveVar..."-Prozedur auf und übergibt ihr den Index der Variable in der *.ini-Liste und den aktuellen Wert. Im Normalfall wird dann in der Prozedur ein Case-Konstrukt verwendet, um die Werte der Variablen entsprechend unterschiedlich zu verarbeiten.
 
 
 
Im Beispiel wird in der Prozedur "ReceiveVarFloat" dafür gesorgt, dass der Wert der Variable, die in der Plugin-Ini im [ReadingVarsFloat]-Abschnitt unter "var.0" eingetragen ist, in das Label2 der Form geschrieben wird und die Gauge1 entsprechend ausschlägt.
 
== 3 Schreibzugriff ==
 
Der Schreibzugriff erfolgt dahingehend, dass Tastatur-/Gamecontroller-Events ausgelöst werden (und damit der Druck von [[https://www.lotus-simulator.de/lexikon/index.php?entry/5-tastenkombinationen/|Tastenkombinationen]] oder Joystick-Knöpfen simuliert wird) und dass Float-Events ausgelöst werden, denen der neue Float-Wert beigefügt ist (was somit der Verarbeitung von Joystick-Achsen entspricht).
 
 


Anders gesagt: Die Plugin-Schnittstelle funktioniert dahingehend wie ein frei konfigurierbarer und beliebig komplexer Gamecontroller. Die Event-Namen sind dieselben wie bei der Konfiguration von Tastatur und Gamecontroller.
Anders gesagt: Die Plugin-Schnittstelle funktioniert dahingehend wie ein frei konfigurierbarer und beliebig komplexer Gamecontroller. Die Event-Namen sind dieselben wie bei der Konfiguration von Tastatur und Gamecontroller.


Für die Events gibt es die beiden Funktionen <code>SetButton</code> und <code>SetFloat</code>. Beide übergeben als Parameter den Index (aus der Plugin-Ini). Die Funktion kann dann je nach Index den gewünschten Wert zurückgeben. Für diesen Zweck wird wiederum üblicherweise ein <code>Case</code>-Konstrukt verwendet. In diesem Beispiel wird in <code>SetButton</code> das erste Tastatur-Event in der Liste in der *.ini-Datei von der Variable <code>Form1.button1_pressed</code> gesteuert. Außerdem wird das erste Float-Event in der Liste in der *.ini-Datei in <code>SetFloat</code> entsprechend der Stellung des Reglers <code>TrackBar1</code> gesetzt.
 
 
Für die Events gibt es die beiden Funktionen "SetButton" und "SetFloat". Beide übergeben als Parameter den Index (aus der Plugin-Ini). Die Funktion kann dann je nach Index den gewünschten Wert zurückgeben. Für diesen Zweck wird wiederum üblicherweise ein Case-Konstrukt verwendet. In diesem Beispiel wird in "SetButton" das erste Tastatur-Event in der Liste in der *.ini-Datei von der Variable "Form1.button1_pressed" gesteuert. Außerdem wird das erste Float-Event in der Liste in der *.ini-Datei in "SetFloat" entsprechend der Stellung des Reglers "TrackBar1" gesetzt.

Aktuelle Version vom 21. Januar 2025, 22:56 Uhr

Vorlage:ArticleMetadata

Die Plugin-DLL enthält den eigentlichen ausführbaren Code des Plugins.

Für die Kommunikation mit LOTUS müssen je nach Bedarf festgelegte Methoden exportiert werden. Im Folgenden soll deshalb erläutert werden, wie der Variablen-Zugriff genau von statten geht und welche Methoden zu exportieren sind. Grundsätzlich bestehen aber große Freiheiten beim Anlegen der DLL; beispielsweise ist es möglich, ein zusätzliches Fenster mit VCL-Komponenten darzustellen und zu verwenden.


Da – abgesehen vom "AOwner" – auf die Verwendung von Objekten/Klassen verzichtet wurde, ist es vermutlich möglich, die Plugin-DLL auch in einer anderen Programmiersprache als Object Pascal zu entwickeln. Getestet wurde von mir aber nur Object Pascal (Delphi). Dementsprechend ist das folgende Beispiel auch in dieser Programmiersprache verfasst.


Welche Methoden aber konkret auf welche Weise deklariert und exportiert werden müssen, um verschiedene Kommunikationsaufgaben zu übernehmen, wird anhand des folgenden Beispiels erläutert. **ACHTUNG: Im Abschnitt "exports" muss dringend auf die korrekte Groß- und Kleinschreibung geachtet werden!**


Code

 1. [[1]]library Test;
 2. [[2]]
 3. [[3]]uses
 4. [[4]] SysUtils,
 5. [[5]] Dialogs,
 6. [[6]] Classes,
 7. [[7]] TestU in 'TestU.pas' {Form1};
 8. [[8]]
 9. [[9]]{$R *.res}
 10. [[10]]
 11. [[11]]procedure PluginStart(AOwner: TComponent); stdcall;
 12. [[12]]begin
 13. [[13]] form1 := TForm1.Create( AOwner );
 14. [[14]] form1.Show;
 15. [[15]]end;
 16. [[16]]
 17. [[17]]procedure PluginFinalize; stdcall;
 18. [[18]]begin
 19. [[19]] form1.Free;
 20. [[20]]end;
 21. [[21]]
 22. [[22]]procedure ReceiveVarFloat(varindex: word; value: single); stdcall;
 23. [[23]]begin
 24. [[24]] case varindex of
 25. [[25]] 0:
 26. [[26]] begin
 27. [[27]] form1.Label2.Caption := floattostrF( value, ffFixed, 5, 1 );
 28. [[28]] form1.Gauge1.Progress := round( value );
 29. [[29]] end;
 30. [[30]] end;
 31. [[31]]end;
 32. [[32]]
 33. [[33]]procedure ReceiveVarBool(varindex: word; value: boolean); stdcall;
 34. [[34]]begin
 35. [[35]] case varindex of
 36. [[36]] 0:
 37. [[37]] begin
 38. [[38]] form1.Label2.Caption := BoolToStr(value);
 39. [[39]] end;
 40. [[40]] end;
 41. [[41]]end;
 42. [[42]]
 43. [[43]]procedure ReceiveVarInt(varindex: word; value: integer); stdcall;
 44. [[44]]begin
 45. [[45]] case varindex of
 46. [[46]] 0:
 47. [[47]] begin
 48. [[48]] form1.Label2.Caption := IntToStr(value);
 49. [[49]] form1.Gauge1.Progress := value;
 50. [[50]] end;
 51. [[51]] end;
 52. [[52]]end;
 53. [[53]]
 54. [[54]]procedure OnConnectingVehicle(name: shortstring); stdcall;
 55. [[55]]begin
 56. [[56]] form1.Caption := name;
 57. [[57]]end;
 58. [[58]]
 59. [[59]]procedure OnUnconnectingVehicle; stdcall;
 60. [[60]]begin
 61. [[61]] form1.Caption := 'no bus loaded';
 62. [[62]]end;
 63. [[63]]
 64. [[64]]function SetButton(eventindex: word): boolean;
 65. [[65]]begin
 66. [[66]] case eventindex of
 67. [[67]] 0: result := Form1.button1_pressed;
 68. [[68]] end;
 69. [[69]]end;
 70. [[70]]
 71. [[71]]function SetFloat(eventindex: word): single;
 72. [[72]]begin
 73. [[73]] case eventindex of
 74. [[74]] 0: result := (Form1.TrackBar1.Position - 15)/15;
 75. [[75]] end;
 76. [[76]]end;
 77. [[77]]
 78. [[78]]exports
 79. [[79]] ReceiveVarInt,
 80. [[80]] SetButton,
 81. [[81]] SetFloat,
 82. [[82]] PluginStart,
 83. [[83]] OnConnectingVehicle,
 84. [[84]] OnUnconnectingVehicle,
 85. [[85]] PluginFinalize;
 86. [[86]]begin
 87. [[87]]end.

Alles anzeigen

Typen- und Funktionsdefinitionen für C, C++ und C#

Code

 1. [[88]]typedef void* TComponent;
 2. [[89]]typedef byte Boolean;
 3. [[90]]typedef float single, Single, *PSingle;
 4. [[91]]typedef struct _ShortString {
 5. [[92]] byte Length; // == min(sizeof(ShortString.String), 256)
 6. [[93]] char String[]; // Nicht-nullterminiert! NUR strncpy o.Ä. anwenden, rennt sonst direkt in unbekannten Speicher!
 7. [[94]]} ShortString, *PShortString;
 8. [[95]]typedef float Float, *PFloat;
 9. [[96]]typedef unsigned short word;
 10. [[97]]
 11. [[98]]__declspec(dllexport) void __stdcall PluginStart(TComponent Owner);
 12. [[99]]__declspec(dllexport) void __stdcall PluginFinalize(void);
 13. [[100]]__declspec(dllexport) void __stdcall ReceiveVarFloat(word varindex, single value);
 14. [[101]]__declspec(dllexport) void __stdcall ReceiveVarBool(word VarIndex, Boolean Value);
 15. [[102]]__declspec(dllexport) void __stdcall ReceiveVarInt(word VarIndex, int Value);
 16. [[103]]__declspec(dllexport) void __stdcall OnConnectingVehicle(ShortString Name);
 17. [[104]]__declspec(dllexport) void __stdcall OnUnconnectingVehicle(void);
 18. [[105]]__declspec(dllexport) Boolean __stdcall SetButton(single Eventindex);
 19. [[106]]__declspec(dllexport) PFloat __stdcall SetFloat(single EventIndex);

Alles anzeigen

Vielen Dank an Feder für den Code!

1 Initialisierung und Finalisierung

Nach dem Laden der DLL wird "PluginStart" aufgerufen. In diesem Fall soll eine Form erstellt werden; für diesen Zweck bietet die Prozedur den AOwner als Parameter an.


Vor dem Entladen der DLL beim Schließen von LOTUS wird "PluginFinalize" aufgerufen. In diesem Beispiel wird die Form aus dem Speicher gelöscht.


Unmittelbar nachdem ein Fahrzeug in den Fokus genommen wird – also nach dem Start der Simulation, nach dem Neuplatzieren eines Fahrzeuges oder beim Wechsel auf ein anderes Fahrzeug – und somit die DLL an das neu ausgewählte Fahrzeug andocken soll, wird die Prozedur "OnConnectingVehicle" ausgeführt. Diese übergibt als Parameter den Namen des neu ausgewählten Fahrzeuges. In diesem Beispiel wird der Name des Fahrzeuges als Titel auf die Form geschrieben.


Bevor ein Fahrzeug aus dem Fokus genommen werden soll (ebenfalls beim Wechsel, beim Löschen oder bei Rückkehr zum Hauptmenü) wird "OnUnconnectingVehicle" ausgeführt.

2 Lesezugriff

Der Lesezugriff erfolgt mit den Prozeduren "ReceiveVarBool", "ReceiveVarInt", "ReceiveVarFloat" und "ReceiveVarString". Der erste Parameter ist jeweils "varindex: word", der zweite ist "value: boolean", "integer", "single" oder "PWideChar", je nach Prozedur.


Der Ablauf ist folgender: LOTUS durchläuft die jeweilige Variablenliste in der Plugin-Ini und ruft jeweils die entsprechende "ReceiveVar..."-Prozedur auf und übergibt ihr den Index der Variable in der *.ini-Liste und den aktuellen Wert. Im Normalfall wird dann in der Prozedur ein Case-Konstrukt verwendet, um die Werte der Variablen entsprechend unterschiedlich zu verarbeiten.


Im Beispiel wird in der Prozedur "ReceiveVarFloat" dafür gesorgt, dass der Wert der Variable, die in der Plugin-Ini im [ReadingVarsFloat]-Abschnitt unter "var.0" eingetragen ist, in das Label2 der Form geschrieben wird und die Gauge1 entsprechend ausschlägt.

3 Schreibzugriff

Der Schreibzugriff erfolgt dahingehend, dass Tastatur-/Gamecontroller-Events ausgelöst werden (und damit der Druck von [[107]] oder Joystick-Knöpfen simuliert wird) und dass Float-Events ausgelöst werden, denen der neue Float-Wert beigefügt ist (was somit der Verarbeitung von Joystick-Achsen entspricht).


Anders gesagt: Die Plugin-Schnittstelle funktioniert dahingehend wie ein frei konfigurierbarer und beliebig komplexer Gamecontroller. Die Event-Namen sind dieselben wie bei der Konfiguration von Tastatur und Gamecontroller.


Für die Events gibt es die beiden Funktionen "SetButton" und "SetFloat". Beide übergeben als Parameter den Index (aus der Plugin-Ini). Die Funktion kann dann je nach Index den gewünschten Wert zurückgeben. Für diesen Zweck wird wiederum üblicherweise ein Case-Konstrukt verwendet. In diesem Beispiel wird in "SetButton" das erste Tastatur-Event in der Liste in der *.ini-Datei von der Variable "Form1.button1_pressed" gesteuert. Außerdem wird das erste Float-Event in der Liste in der *.ini-Datei in "SetFloat" entsprechend der Stellung des Reglers "TrackBar1" gesetzt.