Verarbeiten von Listenelementereignissen in anbietergehosteten Add-Ins
Dies ist der zehnte einer Reihe von Artikeln über die Grundlagen der Entwicklung von vom Anbieter gehosteten SharePoint-Add-Ins. Machen Sie sich zunächst mit SharePoint-Add-Ins und den vorherigen Artikeln dieser Reihe vertraut, die Sie unter Erste Schritte beim Erstellen von von einem Anbieter gehosteten SharePoint-Add-Ins finden.
Hinweis
Wenn Sie unsere Artikelreihe zum Thema anbietergehostete Add-Ins durchgearbeitet haben, haben Sie bereits eine Visual Studio-Lösung, die Sie für diesen Artikel verwenden können. Sie können auch das Repository unter SharePoint_Provider-hosted_Add-Ins_Tutorials herunterladen und die Datei BeforeRER.sln öffnen.
In einem vorherigen Artikel dieser Reihe haben Sie gesehen, dass ein Auftrag bei der Auftragserteilung zur Aufträge-Tabelle der Unternehmensdatenbank und ein Element für diesen Auftrag automatisch zu der Erwartete Lieferungen-Liste hinzugefügt wird. Wenn die Lieferung im lokalen Geschäft eintrifft, setzt ein Benutzer die Spalte Angekommen auf Ja. Das Ändern eines Feldwerts für ein Element erstellt ein Element aktualisiert-Ereignis, für das Sie einen benutzerdefinierten Handler hinzufügen können.
In diesem Artikel erstellen Sie einen Handler für dieses Listenelement-Ereignis und stellen es dann programmgesteuert in der ersten Ausführungslogik des SharePoint-Add-Ins bereit. Ihr Handler fügt das Element zur Bestand-Tabelle der Unternehmensdatenbank hinzu. Dann wird die Zum Bestand hinzugefügt-Spalte der Erwartete Lieferungen-Liste auf Ja gesetzt. Sie erfahren auch, wie Sie das zweite Elementaktualisierungsereignis vermeiden, indem Sie eine unbegrenzte Serie Elementaktualisierungsereignisse auslösen.
Programmgesteuertes Bereitstellen der Liste „Erwartete Lieferungen“
Hinweis
Die Einstellungen für Startprojekte in Visual Studio werden in der Regel nach jedem erneuten Öffnen der Lösung wieder auf die Standardwerte zurückgesetzt. Wann immer Sie beim Durcharbeiten dieser Artikelreihe die Beispiellösung erneut öffnen, müssen Sie umgehend die folgenden Schritte durchführen:
- Klicken Sie oben im Projektmappen-Explorer mit der rechten Maustaste auf den Lösungsknoten, und wählen Sie die Option Startprojekte festlegen aus.
- Stellen Sie sicher, dass alle drei Projekte in der Spalte Aktion auf Start festgelegt sind.
Öffnen Sie im Projektmappen-Explorer die Datei Utilities\SharePointComponentDeployer.cs im Projekt ChainStoreWeb. Fügen Sie der Klasse
SharePointComponentDeployer
die folgende Methode hinzu:private static void CreateExpectedShipmentsList() { using (var clientContext = sPContext.CreateUserClientContextForSPHost()) { var query = from list in clientContext.Web.Lists where list.Title == "Expected Shipments" select list; IEnumerable<List> matchingLists = clientContext.LoadQuery(query); clientContext.ExecuteQuery(); if (matchingLists.Count() == 0) { ListCreationInformation listInfo = new ListCreationInformation(); listInfo.Title = "Expected Shipments"; listInfo.TemplateType = (int)ListTemplateType.GenericList; listInfo.Url = "Lists/ExpectedShipments"; List expectedShipmentsList = clientContext.Web.Lists.Add(listInfo); Field field = expectedShipmentsList.Fields.GetByInternalNameOrTitle("Title"); field.Title = "Product"; field.Update(); expectedShipmentsList.Fields.AddFieldAsXml("<Field DisplayName='Supplier'" + " Type='Text' />", true, AddFieldOptions.DefaultValue); expectedShipmentsList.Fields.AddFieldAsXml("<Field DisplayName='Quantity'" + " Type='Number'" + " Required='TRUE' >" + "<Default>1</Default></Field>", true, AddFieldOptions.DefaultValue); expectedShipmentsList.Fields.AddFieldAsXml("<Field DisplayName='Arrived'" + " Type='Boolean'" + " ShowInNewForm='FALSE'>" + "<Default>FALSE</Default></Field>", true, AddFieldOptions.DefaultValue); expectedShipmentsList.Fields.AddFieldAsXml("<Field DisplayName='Added to Inventory'" + " Type='Boolean'" + " ShowInNewForm='FALSE'>" + "<Default>FALSE</Default></Field>", true, AddFieldOptions.DefaultValue); clientContext.ExecuteQuery(); } } }
Der Code fügt nur Funktionen hinzu, die Sie bereits in vorherigen Artikeln dieser Reihe kennengelernt haben, beachten Sie jedoch Folgendes:
Das Attribut Required des Felds Menge wird auf TRUE festgelegt, sodass das Feld immer einen Wert enthalten muss. Der Standardwert wird dann auf 1 festgelegt.
Die Felder Angekommen und Zum Bestand hinzugefügt sind im Formular für ein neues Element ausgeblendet.
Idealerweise wäre das Feld Zum Bestand hinzugefügt auch auf dem Formular Element bearbeiten ausgeblendet, da es nur in Ja geändert werden sollte, wenn der aktualisierte Ereignishandler das Element zum ersten Mal zu der Tabelle Bestand hinzufügt. Aus technischen Gründen, die wir in einem späteren Schritt erläutern möchten, muss ein Feld im Formular Element bearbeiten angezeigt werden, wenn es in einem Elementaktualisierungsereignis-Handler programmgesteuert ausgefüllt werden soll.
Fügen Sie in der Methode DeployChainStoreComponentsToHostWeb die folgende Zeile direkt über der Zeile
RemoteTenantVersion = localTenantVersion
ein.CreateExpectedShipmentsList();
Erstellen des Listenelement-Ereignisempfängers
Hinweis
Wenn Sie diese Reihe von Artikeln durchgearbeitet haben, haben Sie bereits Ihre Entwicklungsumgebung für das Debuggen von Remote-Ereignisempfängern konfiguriert. Falls dies nicht der Fall ist, lesen Sie bitte Konfigurieren der Lösung für das Debuggen des Ereignisempfängers, bevor Sie mit diesem Thema fortfahren.
Die Office Developer Tools für Visual Studio enthalten eine Remote-Ereignisempfänger-Element, das zu einer SharePoint-Add-In-Lösung hinzugefügt werden kann. Als dieser Artikel geschrieben wurde, nahm dieses Element jedoch an, dass sich die Liste (mit der der Empfänger registriert wird) im Add-In-Web befindet, und daher erstellen die Tools ein Add-In-Web mit einigen SharePoint-Artefakten. Der Empfänger für das Chain Store-Add-In wird (in einem späteren Schritt) mit der Liste Erwartete Lieferungen im Hostweb registriert, sodass das Add-In kein Add-In-Web benötigt. (Eine Auffrischung zur Unterscheidung zwischen Add-In-Webs und Hostwebs finden Sie unter SharePoint-Add-Ins.)
Hinweis
Listen- und Listenelement-Ereignisempfänger heißen Remoteereignisempfänger (RER), da sich ihr Code remote von SharePoint in der Cloud oder auf einem lokalen Server außerhalb der SharePoint-Farm befindet. Allerdings befinden sich die Ereignisse, die diese auslösen, in SharePoint.
Klicken Sie im Projektmappen-Explorer mit der rechten Maustaste auf den Ordner Dienste im Projekt ChainStoreWeb, und wählen Sie Hinzufügen>WCF-Dienst Service aus.
Wenn Sie aufgefordert werden, nennen Sie den Dienst RemoteEventReceiver1, und wählen Sie dann OK.
Die Tools erstellen eine Schnittstellendatei, eine *.svc-Datei und eine CodeBehind-Datei. Die Benutzeroberflächendatei IRemoteEventReceiver1.cs ist nicht erforderlich und kann gelöscht werden. (Möglicherweise wurde sie von den Tools automatisch geöffnet, falls dies der Fall ist, schließen und löschen Sie sie.)
Hinweis
Beim Erstellen des Add-In-Ereignisempfängers für die installierten und deinstallierten Ereignisse in einem früheren Artikel dieser Reihe wurden die URLs von den Office Developer Tools für Visual Studio zu der Manifestdatei hinzugefügt. Listen- und Listenelement-Ereignisempfänger werden nicht im App-Manifest registriert. Stattdessen werden sie (in einem vom Anbieter gehosteten Add-In) programmgesteuert registriert. Dies wird in einem späteren Schritt behandelt.
Öffnen Sie die CodeBehind-Datei RemoteEventReceiver1.svc.cs. Ersetzen Sie den gesamten Inhalt durch den folgenden Code.
using System; using System.Collections.Generic; using Microsoft.SharePoint.Client; using Microsoft.SharePoint.Client.EventReceivers; using System.Data.SqlClient; using System.Data; using ChainStoreWeb.Utilities; namespace ChainStoreWeb.Services { public class RemoteEventReceiver1 : IRemoteEventService { /// <summary> /// Handles events that occur before an action occurs, /// such as when a user is adding or deleting a list item. /// </summary> /// <param name="properties">Holds information about the remote event.</param> /// <returns>Holds information returned from the remote event.</returns> public SPRemoteEventResult ProcessEvent(SPRemoteEventProperties properties) { throw new NotImplementedException(); } /// <summary> /// Handles events that occur after an action occurs, /// such as after a user adds an item to a list or deletes an item from a list. /// </summary> /// <param name="properties">Holds information about the remote event.</param> public void ProcessOneWayEvent(SPRemoteEventProperties properties) { } } }
Beachten Sie Folgendes zu diesem Code:
Die Schnittstelle
IRemoteEventService
ist im Namespace Microsoft.SharePoint.Client.EventReceivers definiert.Im ChainStore-Add-In werden keine „Bevor“-Ereignisse verarbeitet, aber die Methode ProcessEvent ist für die Schnittstelle
IRemoteEventService
erforderlich.
Fügen Sie der ProcessOneWayEvent-Methode den folgenden Code hinzu. Beachten Sie, dass das ItemUpdated-Ereignis das einzige Ereignis ist, das in diesem Beispiel behandelt wird. Daher hätten wir anstelle eines Schalters eine einfache if-Struktur verwenden können. Ereignisempfänger behandeln jedoch in der Regel mehrere Ereignisse. Daher möchten wir, dass Sie das Muster sehen, das Sie am häufigsten in Ihren Ereignishandlern als SharePoint-Add-In-Entwickler verwenden.
switch (properties.EventType) { case SPRemoteEventType.ItemUpdated: // TODO12: Handle the item updated event. break; }
Ersetzen Sie
TODO12
durch den folgenden Code. Auch hier verwenden wir eine switch-Struktur, obwohl eine einfache if-Struktur ausreichend wäre, da Sie das typische Muster in SharePoint-Ereignisempfängern sehen sollen.switch (properties.ItemEventProperties.ListTitle) { case "Expected Shipments": // TODO13: Handle the arrival of a shipment. break; }
Der Code, der auf den Eingang einer Lieferung reagiert, sollten zwei Aufgaben ausführen:
Hinzufügen des Elements, das im Geschäft eingetroffen ist, zum Unternehmensbestand
Setzen Sie das Zum Bestand hinzugefügt-Feld der Erwartete Lieferungen-Liste auf Ja. Dies geschieht normalerweise nur, wenn das Element erfolgreich zum Bestand hinzugefügt wurde.
Fügen Sie den folgenden Code anstelle von
TODO13
hinzu. Die beiden MethodenTryUpdateInventory
undRecordInventoryUpdateLocally
werden in späteren Schritten erstellt.bool updateComplete = TryUpdateInventory(properties); if (updateComplete) { RecordInventoryUpdateLocally(properties); }
Die ProcessOneWayEvent-Methode sollte jetzt wie folgt aussehen:
public void ProcessOneWayEvent(SPRemoteEventProperties properties) { switch (properties.EventType) { case SPRemoteEventType.ItemUpdated: switch (properties.ItemEventProperties.ListTitle) { case "Expected Shipments": bool updateComplete = UpdateInventory(properties); if (updateComplete) { RecordInventoryUpdateLocally(properties); } break; } break; } }
Fügen Sie der Klasse
RemoteEventReceiver1
die unten aufgeführte Methode hinzu.private bool TryUpdateInventory(SPRemoteEventProperties properties) { bool successFlag = false; // TODO14: Test whether the list item is changing because the product has arrived // or for some other reason. If the former, add it to the inventory and set the success flag // to true. return successFlag; }
Es gibt fünf Spalten in der Lieferung erwartet-Liste, aber bei den meisten Arten der Updates möchten wir nicht, dass der Handler reagiert. Wenn ein Benutzer z.B. die Schreibweise des Namens eines Lieferanten korrigiert, wird das Elementaktualisierungsereignis ausgelöst, aber unser Handler sollte nicht reagieren. Der Handler sollte nur reagieren, wenn das Angekommen-Feld auf Ja gesetzt wurde.
Es gibt eine weitere Bedingung, die getestet werden muss. Nehmen wir einmal an, Angekommen wurde auf Ja festgelegt und das Produkt im Element wird zum Bestand hinzugefügt (und Zum Bestand hinzugefügt ist auf Ja) festgelegt. Später ändert ein Benutzer jedoch versehentlich das Angekommen-Feld einer Lieferung zurück auf Nein. Später behebt er den Fehler, indem er es erneut auf Ja setzt. Sowohl der Fehler als auch die Korrektur lösen das Elementaktualisierungsereignis aus. Der Handler reagiert nicht auf den Fehler, da er nur reagiert, wenn Angekommen auf Ja festgelegt ist, aber er reagiert auf die Korrektur, bei der Angekommen zurück auf Ja gesetzt wird, sodass das gleiche Produkt und die gleiche Menge ein zweites Mal zum Bestand hinzugefügt würden. Aus diesem Grund sollte der Handler nur reagieren, wenn der Wert Zum Bestand hinzugefügt auf Nein festgelegt ist.
Daher muss der Handler wissen, welche Werte diese Felder besitzen, nachdem der Benutzer das Element aktualisiert hat. Das SPRemoteEventProperties-Objekt verfügt über die ItemEventProperties-Eigenschaft. Es enthält aber auch eine indizierte AfterProperties-Eigenschaft mit den Werten der Felder des aktualisierten Elements. Der folgende Code verwendet diese Eigenschaften, um zu testen, ob der Handler reagieren soll. Setzen Sie dies an die Stelle von
TODO14
.var arrived = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Arrived"]); var addedToInventory = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Added_x0020_to_x0020_Inventory"]); if (arrived && !addedToInventory) { // TODO15: Add the item to inventory successFlag = true; }
Ersetzen Sie
TODO15
durch den folgenden Code.using (SqlConnection conn = SQLAzureUtilities.GetActiveSqlConnection()) using (SqlCommand cmd = conn.CreateCommand()) { conn.Open(); cmd.CommandText = "UpdateInventory"; cmd.CommandType = CommandType.StoredProcedure; SqlParameter tenant = cmd.Parameters.Add("@Tenant", SqlDbType.NVarChar); tenant.Value = properties.ItemEventProperties.WebUrl + "/"; SqlParameter product = cmd.Parameters.Add("@ItemName", SqlDbType.NVarChar, 50); product.Value = properties.ItemEventProperties.AfterProperties["Title"]; // not "Product" SqlParameter quantity = cmd.Parameters.Add("@Quantity", SqlDbType.SmallInt); quantity.Value = Convert.ToUInt16(properties.ItemEventProperties.AfterProperties["Quantity"]); cmd.ExecuteNonQuery(); }
Da es sich hierbei hauptsächlich um SQL- und ASP.NET-Programmierung handelt, werden wir nicht näher darauf eingehen, aber beachten Sie Folgendes:
Wir verwenden die Eigenschaft ItemEventProperties.WebUrl, um den Mandantennamen abzurufen, der der Hostweb-URL entspricht.
Wir verwenden erneut AfterProperties, um die Werte für Produktname und Menge abzurufen.
Wir bezeichnen das Produktnamenfeld als "Titel", obwohl der Anzeigename in "Product" geändert wurde (in der CreateExpectedShipmentsList-Methode ), da Felder immer durch ihre internen Namen referenziert werden.
Wir sind noch nicht mit der TryUpdateInventory-Methode fertig, aber an diesem Punkt sollte sie wie folgt aussehen.
private bool TryUpdateInventory(SPRemoteEventProperties properties) { bool successFlag = false; var arrived = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Arrived"]); var addedToInventory = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Added_x0020_to_x0020_Inventory"]); if (arrived && !addedToInventory) { using (SqlConnection conn = SQLAzureUtilities.GetActiveSqlConnection()) using (SqlCommand cmd = conn.CreateCommand()) { conn.Open(); cmd.CommandText = "UpdateInventory"; cmd.CommandType = CommandType.StoredProcedure; SqlParameter tenant = cmd.Parameters.Add("@Tenant", SqlDbType.NVarChar); tenant.Value = properties.ItemEventProperties.WebUrl + "/"; SqlParameter product = cmd.Parameters.Add("@ItemName", SqlDbType.NVarChar, 50); product.Value = properties.ItemEventProperties.AfterProperties["Title"]; // not "Product" SqlParameter quantity = cmd.Parameters.Add("@Quantity", SqlDbType.SmallInt); quantity.Value = Convert.ToUInt16(properties.ItemEventProperties.AfterProperties["Quantity"]); cmd.ExecuteNonQuery(); } successFlag = true; } return successFlag; }
Wenn die TryUpdateInventory-Methode true zurückgibt, ruft der Handler eine Methode (noch nicht geschrieben) auf, die das gleiche Element in der Lieferung erwartet-Liste aktualisiert, indem das Feld Zum Bestand hinzugefügt auf Ja gesetzt wird. Dies ist wiederum ein Elementaktualisierungsereignis, sodass der Handler erneut aufgerufen wird. (Die Tatsache, dass das Zum Bestand hinzugefügt-Feld nun auf Ja gesetzt ist, verhindert, dass der Handler die gleiche Lieferung ein zweites Mal zum Bestand hinzufügt, aber der Handler wird trotzdem aufgerufen.)
SharePoint verhält sich etwas anders, wenn das Elementaktualisierungsereignis durch ein programmgesteuertes Update ausgelöst wird: Es enthält nur die Felder, die sich in der Aktualisierung geändert haben, in den AfterProperties . Daher ist das Feld "Eingetroffen " nicht vorhanden, da sich nur das Feld Zu Bestand hinzugefügt geändert hat.
Die Zeile –
var arrived = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Arrived"]);
löst eine KeyNotFoundException aus.
Es gibt mehre Möglichkeiten dieses Problem zu beheben. In diesem Beispiel fangen wir die Ausnahme ab und verwenden den catch-Block, um sicherzustellen, dass
successFlag
auf false festgelegt ist. Auf diese Weise wird sichergestellt, dass das Element nicht ein drittes Mal aktualisiert wird.Platzieren Sie alles, was sich in der Methode zwischen der ersten Zeile,
bool successFlag = false;
, und der letzten Zeile,return successFlag;
, befindet, in einem try-Block.Fügen Sie den folgenden catch-Block direkt unterhalb des try-Blocks hinzu.
catch (KeyNotFoundException) { successFlag = false; }
Hinweis
Die KeyNotFoundException ist auch der Grund, warum das Feld Zu Bestand hinzugefügt im Formular zum Bearbeiten des Elements sichtbar sein muss. SharePoint enthält keine Felder, die auf dem Formular zum Bearbeiten in AfterProperties ausgeblendet sind.
Die gesamte Methode sollte jetzt wie folgt aussehen.
private bool TryUpdateInventory(SPRemoteEventProperties properties) { bool successFlag = false; try { var arrived = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Arrived"]); var addedToInventory = Convert.ToBoolean(properties.ItemEventProperties.AfterProperties["Added_x0020_to_x0020_Inventory"]); if (arrived && !addedToInventory) { using (SqlConnection conn = SQLAzureUtilities.GetActiveSqlConnection()) using (SqlCommand cmd = conn.CreateCommand()) { conn.Open(); cmd.CommandText = "UpdateInventory"; cmd.CommandType = CommandType.StoredProcedure; SqlParameter tenant = cmd.Parameters.Add("@Tenant", SqlDbType.NVarChar); tenant.Value = properties.ItemEventProperties.WebUrl + "/"; SqlParameter product = cmd.Parameters.Add("@ItemName", SqlDbType.NVarChar, 50); product.Value = properties.ItemEventProperties.AfterProperties["Title"]; // not "Product" SqlParameter quantity = cmd.Parameters.Add("@Quantity", SqlDbType.SmallInt); quantity.Value = Convert.ToUInt16(properties.ItemEventProperties.AfterProperties["Quantity"]); cmd.ExecuteNonQuery(); } successFlag = true; } } catch (KeyNotFoundException) { successFlag = false; } return successFlag; }
Fügen Sie der Klasse
RemoteEventReceiver1
die unten aufgeführte Methode hinzu.private void RecordInventoryUpdateLocally(SPRemoteEventProperties properties) { using (ClientContext clientContext = TokenHelper.CreateRemoteEventReceiverClientContext(properties)) { List expectedShipmentslist = clientContext.Web.Lists.GetByTitle(properties.ItemEventProperties.ListTitle); ListItem arrivedItem = expectedShipmentslist.GetItemById(properties.ItemEventProperties.ListItemId); arrivedItem["Added_x0020_to_x0020_Inventory"] = true; arrivedItem.Update(); clientContext.ExecuteQuery(); } }
Bis hierhin ist das Codemuster aus früheren Artikeln dieser Reihe bekannt. Beachten Sie jedoch einen Unterschied:
Der Code erhält das ClientContext- Objekt durch Aufrufen der TokenHelper.CreateRemoteEventReceiverClientContext-Methode anstelle der SharePointContext.CreateUserClientContextForSPHost-Methode, wie wir sie für den Code verwendet haben, der SharePoint über Seiten abgerufen hat, wie beispielsweise die Seite "EmployeeAdder".
Der Hauptgrund für die beiden verschiedenen Methoden zum Abrufen eines ClientContext-Objekts ist, dass SharePoint die für die Erstellung eines solchen Objekts benötigten Informationen anders zu Ereignisempfängern weiterleitet als wenn SharePoint diese Informationen an Seiten übergibt. Für Ereignisempfänger gibt SharePoint ein SPRemoteEventProperties-Objekt weiter, aber für Seiten übergibt SharePoint ein spezielles Feld, das als Kontexttoken bezeichnet wird, in den Textkörper der Anforderung, die die Add-In-Seite startet.
Speichern und schließen Sie die Datei mit dem Empfängercode.
Registrieren des Empfängers
Die letzte Aufgabe besteht darin, SharePoint mitzuteilen, dass wir einen benutzerdefinierten Empfänger haben, den SharePoint immer dann aufrufen soll, wenn ein Element in der Liste Erwartete Lieferungen aktualisiert wird.
Öffnen Sie die Datei „SharePointComponentDeployer.cs“ und fügen Sie die folgende Zeile zur Methode DeployChainStoreComponentsToHostWeb hinzu, und zwar direkt unterhalb der Zeile, die die Erwartete Lieferungen-Liste erstellt (wir fügen diese Methode im nächsten Schritt hinzu). Beachten Sie, dass wir die Methode an das HttpRequest-Objekt übergeben, das die Add-In-Startseite an die Methode DeployChainStoreComponentsToHostWeb übergeben hat.
RegisterExpectedShipmentsEventHandler(request);
Fügen Sie der Klasse
SharePointComponentDeployer
die folgende Methode hinzu.private static void RegisterExpectedShipmentsEventHandler(HttpRequest request) { using (var clientContext = sPContext.CreateUserClientContextForSPHost()) { var query = from list in clientContext.Web.Lists where list.Title == "Expected Shipments" select list; IEnumerable<List> matchingLists = clientContext.LoadQuery(query); clientContext.ExecuteQuery(); List expectedShipmentsList = matchingLists.Single(); // TODO16: Add the event receiver to the list's collection of event receivers. clientContext.ExecuteQuery(); } }
Ersetzen Sie
TODO16
durch die folgenden Zeilen. Beachten Sie, dass es eine einfache CreationInformation-Klasse für Ereignisempfänger ebenso wie für Listen und Listenelemente gibt.EventReceiverDefinitionCreationInformation receiver = new EventReceiverDefinitionCreationInformation(); receiver.ReceiverName = "ExpectedShipmentsItemUpdated"; receiver.EventType = EventReceiverType.ItemUpdated; // TODO17: Set the URL of the receiver. expectedShipmentsList.EventReceivers.Add(receiver);
Jetzt müssen Sie SharePoint die URL des Ereignisempfängers mitteilen. In der Produktion befindet sie sich in der gleichen Domäne wie die Remote-Seiten. Der Pfad lautet /Services/RemoteEventReceiver1.svc. Da der Handler in der ersten Ausführungslogik über die Add-In-Startseite registriert wird, befindet sich die Domäne im Hostheader des HttpRequest-Objekts der Anforderung, die die Seite aufgerufen hat. Unser Code hat dieses Objekt von der Seite an die DeployChainStoreComponentsToHostWeb-Methode übergeben, die es wiederum an die RegisterExpectedShipmentsEventHandler-Methode weitergegeben hat. Daher können wir die Empfänger-URL mit dem folgenden Code angeben.
receiver.ReceiverUrl = "https://" + request.Headers["Host"] + "/Services/RemoteEventReceiver1.svc";
Dies funktioniert leider nicht, wenn Sie das Add-In in Visual Studio debuggen. Beim Debuggen wird der Empfänger in der Azure Service Bus gehostet und nicht in der Localhost-URL, in der die Remote-Seiten gehostet werden. Je nachdem, ob wir debuggen oder nicht, müssen wir unterschiedliche URLs für den Empfänger festlegen. Ersetzen Sie also
TODO17
durch die folgende Struktur, die C#-Compiler-Richtlinien verwendet. Beachten Sie, dass im Debugmodus die URL des Empfängers aus einer web.config Einstellung gelesen wird (Sie erstellen diese Einstellung in einem späteren Schritt).#if DEBUG receiver.ReceiverUrl = WebConfigurationManager.AppSettings["RERdebuggingServiceBusUrl"].ToString(); #else receiver.ReceiverUrl = "https://" + request.Headers["Host"] + "/Services/RemoteEventReceiver1.svc"; #endif
Die gesamte RegisterExpectedShipmentsEventHandler-Methode sollte nun wie folgt aussehen.
private static void RegisterExpectedShipmentsEventHandler(HttpRequest request) { using (var clientContext = sPContext.CreateUserClientContextForSPHost()) { var query = from list in clientContext.Web.Lists where list.Title == "Expected Shipments" select list; IEnumerable<List> matchingLists = clientContext.LoadQuery(query); clientContext.ExecuteQuery(); List expectedShipmentsList = matchingLists.Single(); EventReceiverDefinitionCreationInformation receiver = new EventReceiverDefinitionCreationInformation(); receiver.ReceiverName = "ExpectedShipmentsItemUpdated"; receiver.EventType = EventReceiverType.ItemUpdated; #if DEBUG receiver.ReceiverUrl = WebConfigurationManager.AppSettings["RERdebuggingServiceBusUrl"].ToString(); #else receiver.ReceiverUrl = "https://" + request.Headers["Host"] + "/Services/RemoteEventReceiver1.svc"; #endif expectedShipmentsList.EventReceivers.Add(receiver); clientContext.ExecuteQuery(); } }
Fügen Sie die folgenden using-Anweisung an den Anfang der Datei hinzu.
using System.Web.Configuration;
Führen Sie die folgende Unterprozedur aus, um sicherzustellen, dass
DEBUG
dies nur true ist, wenn das Add-In debuggt wird:Klicken Sie Projektmappen-Explorer mit der rechten Maustaste auf das Projekt ChainStoreWeb, und wählen Sie Eigenschaften aus.
Öffnen Sie die Registerkarte Build der Eigenschaften, und wählen Sie dann oben in der Dropdownliste Konfiguration die Option Debuggen aus.
Stellen Sie sicher, dass das Kontrollkästchen Debugkonstante definieren aktiviert ist (in der Regel standardmäßig). Der folgende Screenshot zeigt die richtige Einstellung.
Abbildung 1: Unterregisterkarte "Build" der Registerkarte "Eigenschaften" in Visual Studio
Ändern Sie die Dropdownliste Konfiguration in Release, und stellen Sie dann sicher, dass das Kontrollkästchen Debugkonstante definierennicht aktiviert ist (dies ist in der Regel nicht standardmäßig der Fall). Der folgende Screenshot zeigt die richtige Einstellung.
Abbildung 2: Unterregisterkarte "Build" der Registerkarte "Eigenschaften" mit deaktiviertem Kontrollkästchen
Wenn Sie Änderungen vorgenommen haben, speichern Sie diese, und schließen Sie dann die Registerkarte Eigenschaften.
Öffnen Sie die web.config-Datei, und fügen Sie das folgende Markup als untergeordnetes Element des appSettings-Elements hinzu (der Wert der Einstellung wird im nächsten Abschnitt abgerufen).
<add key="RERdebuggingServiceBusUrl" value="" />
Abrufen der Empfänger-URL für das Debuggen
Die Ereignisempfänger für Add-In-Ereignisse und Listenelemente sind WCF-Dienste (Windows Communication Service), und jeder WCF-Dienst kennt seinen eigenen Endpunkt und speichert ihn an mehreren Stellen, einschließlich des System.ServiceModel.OperationContext.Current.Channel.LocalAddress.Uri-Objekts .
Beim Debuggen wird der Add-In-Empfänger an einem Azure Service Bus Endpunkt gehostet, der fast mit dem Endpunkt für den Listenelementempfänger identisch ist. Der Unterschied besteht darin, dass die URL des Add-In-Endpunkts mit "AppEventReceiver.svc" endet, die URL des Listenelementempfängers jedoch mit "RemoteEventReceiver1.svc" endet. Damit wir die URL des Endpunkts im Add-In-Empfänger abrufen, eine kleine Änderung am Ende vornehmen und sie dann als Wert unserer web.config RERdebuggingServiceBusUrl-Einstellung verwenden können.
Öffnen Sie die Datei „AppEventReceiver.svc.cs“ im Ordner Dienste des Projekts ChainStoreWeb.
Fügen Sie Folgendes als erste Zeile in der Methode ProcessEvent hinzu.
string debugEndpoint = System.ServiceModel.OperationContext.Current.Channel.LocalAddress.Uri.ToString();
Fügen Sie einen Haltepunkt in der nächsten Zeile der Methode hinzu.
Wählen Sie F5, um das Add-In zu debuggen. Da web.config geöffnet ist und Office Developer Tools für Visual Studio bei jeder Auswahl von F5 eine Einstellung ändert, werden Sie aufgefordert, sie erneut zu laden. Wählen Sie Ja.
Wenn der Haltepunkt erreicht wird, zeigen Sie mit dem Cursor auf die
debugEndpoint
Variable. Wenn der Visual Studio-Datentipp angezeigt wird, wählen Sie den Pfeil nach unten und dann Textschnellansicht aus.Abbildung 3: Visual Studio-Textschnellansicht mit einer Azure Service Bus URL
Kopieren Sie den Zeichenfolgenwert aus der Schnellansicht und fügen Sie ihn an einer beliebigen Stelle ein.
Schließen Sie die Schnellansicht, und beenden Sie dann das Debuggen in Visual Studio.
Löschen Sie die Zeile, die Sie im zweiten Schritt dieses Verfahrens hinzugefügt haben, oder kommentieren Sie die Zeile aus, und löschen Sie auch den Haltepunkt.
Ersetzen Sie in der Zeichenfolge, die Sie kopiert haben, „AppEventReceiver.svc“ am Ende durch „RemoteEventReceiver1.svc“.
Kopieren Sie die geänderte URL und fügen Sie sie als Wert für den Schlüssel RERdebuggingServiceBusUrl in der web.config-Datei ein.
Hinweis
Das manuelle Kopieren der Service Bus-URL und das Einfügen (einer geänderten Version von) in den web.config ist nicht die einzige Möglichkeit, eine andere URL beim Debuggen eines Remoteereignisempfängers zu beheben, wenn er in der Produktion ausgeführt wird. Wir könnten den Wert von System.ServiceModel.OperationContext.Current.Channel.LocalAddress.Uri programmgesteuert an einer beliebigen Stelle in SharePoint oder in der Remotedatenbank speichern und dann den Code für die erste Ausführung lesen und der receiver.ReceiverUrl
Eigenschaft zuweisen lassen. Der Listenelement-Ereignisempfänger könnte als Teil des installierten Add-In-Ereignishandlers registriert werden. Anschließend könnten wir System.ServiceModel.OperationContext.Current.Channel.LocalAddress.Uri programmgesteuert lesen, ändern und zuweisen, receiver.ReceiverUrl
ohne sie irgendwo speichern zu müssen.
Diese Strategie erfordert, dass die Liste Erwartete Lieferungen auch im installierten Add-In-Ereignishandler erstellt wird, da sie vorhanden sein muss, bevor der Handler bei ihr registriert werden kann.
Beachten Sie auch, dass wir unseren Add-In-Ereignisempfänger und listenelement-Ereignisempfänger in einem einzelnen Empfänger (d. h. den gleichen .svc- und .svc.cs-Dateien) kombinieren können. In diesem Fall ist keine Änderung der URL erforderlich, bevor sie als Wert von receiver.ReceiverUrl
verwendet wird.
Ausführen des Add-Ins und Testen des Listenelementempfängers
Öffnen Sie die Seite Websiteinhalte der Website des Hongkong-Stores, und entfernen Sie die Liste Erwartete Lieferungen , falls vorhanden.
Drücken Sie auf die F5-TASTE, um Ihr Add-In bereitzustellen und auszuführen. Visual Studio hostet die Remotewebanwendung in IIS Express und die SQL-Datenbank in SQL Express. Außerdem wird eine temporäre Installation des Add-Ins auf Ihrer SharePoint-Testwebsite durchgeführt, und das Add-In wird sofort ausgeführt. Sie werden aufgefordert, Berechtigungen für das Add-In zu erteilen, bevor die Startseite geöffnet wird.
Wenn die Startseite des Add-Ins geöffnet wird, wählen Sie oben im Chromsteuerelement die Schaltfläche Zurück zur Website aus.
Wechseln Sie auf der Startseite des Hongkong-Stores zur Seite Websiteinhalte , und öffnen Sie die Liste Erwartete Lieferungen .
Erstellen Sie ein Element, und beachten Sie, dass die Felder "Eingetroffen " und " Zum Bestand hinzugefügt " nicht angezeigt werden.
Nachdem das Element erstellt wurde, öffnen Sie es erneut zur Bearbeitung. Aktivieren Sie das Kontrollkästchen Angekommen , und speichern Sie das Element. Dadurch wird das Element aktualisiert.This triggers the item updated event. Das Element wird dem Bestand hinzugefügt, und der Wert des Felds Zu Bestand hinzugefügt ändert sich in Ja (Möglicherweise müssen Sie die Seite aktualisieren, um die Änderung an Zu Bestand hinzugefügt zu sehen).
Verwenden Sie die Zurück-Schaltfläche des Browsers, bis Sie wieder auf der Startseite des Chain Store-Add-Ins sind, und wählen Sie dann die Schaltfläche Bestand anzeigen aus. Das Element, das Sie als Angekommen markiert haben, ist jetzt aufgelistet.
Zurück zur Liste Erwartete Lieferungen und fügen Sie einen weiteren Artikel mit genau demselben Produktnamen und Lieferantennamen, aber einer anderen Menge hinzu.
Nachdem das Element erstellt wurde, öffnen Sie es erneut zur Bearbeitung. Ändern Sie den Wert für Angekommen auf Ja, und speichern Sie das Element.
Verwenden Sie die Zurück-Schaltfläche des Browsers, bis Sie wieder auf der Startseite des Chain Store-Add-Ins sind, und wählen Sie dann die Schaltfläche Bestand anzeigen aus. Es ist immer noch nur ein Element für den Produktnamen und den Anbieter vorhanden, aber die Menge ist jetzt die Summe der zwei Elemente in der Liste Erwartete Lieferungen.
Schließen Sie zum Beenden der Debugsitzung das Browserfenster, oder beenden Sie das Debuggen in Visual Studio. Jedes Mal, wenn Sie F5 drücken, zieht Visual Studio die vorherige Version des Add-Ins zurück und installiert die neueste.
Da Sie mit diesem Add-In und dieser Visual Studio-Lösung in anderen Artikeln arbeiten werden, hat es sich bewährt, das Add-In ein letztes Mal zurückzuziehen, wenn Sie Ihre Arbeit daran für eine Weile abgeschlossen haben. Klicken Sie im Projektmappen-Explorer mit der rechten Maustaste auf das Projekt, und wählen Sie die Option Zurückziehen aus.
Nächste Schritte
Informationen zum Veröffentlichen Ihres Add-Ins auf einer SharePoint-Website finden Sie unter Bereitstellen und Installieren von SharePoint-Add-Ins: Methoden und Optionen. Auf den folgenden Seiten können Sie auch erweiterte Arbeiten in der SharePoint-Add-In-Entwicklung fortsetzen: