Программное развертывание собственной кнопки в надстройке с размещением у поставщика
Это девятая часть серии статей, посвященной основам разработки надстроек SharePoint, размещаемых у поставщика. Для начала вам следует ознакомиться со статьей Надстройки SharePoint и предыдущими статьями этой серии, представленными в разделе Знакомство с созданием надстроек SharePoint, размещаемых у поставщика.
Примечание.
Если вы изучали предыдущие статьи этой серии о размещаемых у поставщика надстройках, то у вас уже есть решение Visual Studio, которое можно использовать для работы с данной статьей. Кроме того, вы можете скачать репозиторий SharePoint_Provider-hosted_Add-Ins_Tutorials и открыть файл BeforeProgrammaticButton.sln.
В этой статьи вы узнаете, как включать настраиваемую кнопку ленты в надстройку SharePoint, когда список, на ленте которого размещается кнопка, также развертывается программным способом в той же надстройке.
Повторное добавление настраиваемой кнопки в проект
Примечание.
Когда решение открывается повторно, для параметров раздела "Запускаемые проекты" в Visual Studio обычно возвращаются значения по умолчанию. После повторного открытия примера решения, который рассматривается в этой серии статей, всегда выполняйте указанные ниже действия.
- В верхней части обозревателя решений щелкните узел решения правой кнопкой мыши и выберите пункт Назначить запускаемые проекты.
- Убедитесь, что в столбце Действие для всех трех проектов указано значение Запуск.
В предыдущей статье вы удалили настраиваемую кнопку ленты AddEmployeeToCorpDB из проекта. Добавьте ее обратно, сделав вот что:
В верхней части обозревателя решений на панели инструментов нажмите кнопку Показать все файлы.
Рис. 1. панель инструментов Обозреватель решений
В проекте ChainStore щелкните правой кнопкой мыши элемент AddEmployeeToCorpDB, а затем выберите пункт Включить в проект.
Снова нажмите кнопку Показать все файлы.
В проекте ChainStore разверните узел AddEmployeeToCorpDB, а затем откройте файл elements.xml.
Описание проблемы и ее решение
В файле elements.xml атрибут RegistrationId элемента CustomAction служит для идентификации списка, на ленту которого вы добавляете кнопку: {$ListId:Lists/Local Employees;}
. Это отлично работало, когда список уже был добавлен на хост-сайт вручную. Но теперь при развертывании списка программным способом в коде, выполняемом при первом запуске, когда SharePoint устанавливает надстройку и пытается развернуть кнопку, списка еще не существует. В этом случае установка надстройки может привести к созданию исключения и завершиться со сбоем.
Развертывание списка в обработчике событий установки, а не в коде, выполняемом при первом запуске, не решит эту проблему. Это связано с тем, что SharePoint развертывает описанные настраиваемые компоненты, например настраиваемую кнопку (и веб-часть надстройки Place Order (Размещение заказа)), прежде чем запустить настраиваемый обработчик, поэтому когда SharePoint пытается развернуть кнопку, списка еще не существует.
Создание настраиваемой кнопки полностью программным способом непрактично по причинам, которые слишком сложны, чтобы обсуждать их здесь. К счастью, это не нужно. Существует относительно безболезненный способ "полупрограммного" создания настраиваемой кнопки и назначения ее настраиваемому списку.
Для этого нужно выполнить вот какие основные действия:
Оставьте описательно определенную кнопку в проекте, но назначьте ее ленте какого-либо приложения, которое всегда имеется на сайтах SharePoint, а не ленте списка, который развертывается программным способом с помощью той же надстройки.
В коде, выполняемом при первом запуске, после программного создания списка добавьте неопределенную кнопку на ленту списка.
Инициализируйте свойства новой кнопки, используя значения свойств исходной кнопки. Сейчас у нас есть две идентичные кнопки. Вторая кнопка назначена ленте списка Local Employees (Местные сотрудники).
Удалите исходную кнопку программным способом.
Регистрация настраиваемой кнопки программным способом
Ниже описывается реализация этой стратегии.
В проекте ChainStore разверните узел AddEmployeeToCorpDB, откройте файл elements.xml, а затем измените значение атрибута RegistrationId элемента CustomAction на число 100. Это идентификатор типа списка. Даже если на веб-сайте нет экземпляров списка этого типа, тип списка есть на всех веб-сайтах SharePoint. Теперь атрибут должен выглядеть так:
RegistrationId="100"
В файле SharePointComponentDeployer.cs добавьте указанную ниже строку в метод DeployChainStoreComponentsToHostWeb сразу же после строки, в которой вызывается метод
CreateLocalEmployeesList
. Вы создадите этот метод на следующем этапе.ChangeCustomActionRegistration();
Добавьте указанный ниже метод в класс
SharePointComponentDeployer
.private static void ChangeCustomActionRegistration() { using (var clientContext = sPContext.CreateUserClientContextForSPHost()) { var query = from action in clientContext.Web.UserCustomActions where action.Name == "{button_GUID} .AddEmployeeToCorpDB" select action; IEnumerable<UserCustomAction> matchingActions = clientContext.LoadQuery(query); clientContext.ExecuteQuery(); UserCustomAction webScopedEmployeeAction = matchingActions.Single(); // TODO8: Get a reference to the (empty) collection of custom actions // that are registered with the custom list. // TODO9: Add a blank custom action to the list's collection. // TODO10: Copy property values from the descriptively deployed // custom action to the new custom action // TODO11: Delete the original custom action. clientContext.ExecuteQuery(); } }
Обратите внимание на следующие особенности этого кода:
Так как дополнительное действие, то есть настраиваемая кнопка, зарегистрировано на ленте для типа списка, область его действия распространяется на весь веб-сайт и находится в коллекции дополнительных действий веб-сайта. Таким образом, код получает его из этой коллекции.
Значение возвращается из атрибута
action.Name
ID элемента CustomAction в файле elements.xml в AddEmployeeToCorpDB.
Важно!
Вам необходимо изменить значение
action.Name
в коде, чтобы оно совпадало со значением в файле elements.xml. Часть GUID имени будет другой. Обратите внимание, что между GUID и остальной частью имени есть символ"."
. Вот пример строки:where action.Name == "4a926a42-3577-4e02-9d06-fef78586b1bc.AddEmployeeToCorpDB"
Замените
TODO8
указанным ниже кодом. Обратите внимание, что если вы отзовете надстройку, созданные ею компоненты не будут удалены. По завершении работы кода, выполняемого при первом запуске, в коллекции UserCustomActions списка появится дополнительное действие, которое не будет отозвано при следующем нажатии клавиши F5. Чтобы избежать путаницы, в последней строке в этом коде методlistActions.Clear();
очищает коллекцию.var queryForList = from list in clientContext.Web.Lists where list.Title == "Local Employees" select list; IEnumerable<List> matchingLists = clientContext.LoadQuery(queryForList); clientContext.ExecuteQuery(); List employeeList = matchingLists.First(); var listActions = employeeList.UserCustomActions; clientContext.Load(listActions); listActions.Clear();
Замените
TODO9
указанной ниже строкой, которая добавляет неопределенное дополнительное действие в список Local Employees (Местные сотрудники).var listScopedEmployeeAction = listActions.Add();
Замените
TODO10
указанным ниже кодом.listScopedEmployeeAction.Title = webScopedEmployeeAction.Title; listScopedEmployeeAction.Location = webScopedEmployeeAction.Location; listScopedEmployeeAction.Sequence = webScopedEmployeeAction.Sequence; listScopedEmployeeAction.CommandUIExtension = webScopedEmployeeAction.CommandUIExtension; listScopedEmployeeAction.Update();
Обратите внимание на следующие особенности этого кода:
Он назначает значения свойств кнопки уровня веб-сайта (которая была развернута с помощью описательной разметки) соответствующим свойствам кнопки уровня списка, так что эти две кнопки становятся идентичными за исключением области их действия.
В свойстве Sequence указывается относительный порядок, в котором кнопка будет отображена в своей области на ленте. В этом случае кнопка находится в разделе Actions (Действия) на вкладке Items (Элементы) ленты. В описательной разметке это значение было равно 10001. Это достаточно большое значение, чтобы кнопка отображалась после (т. е. справа от) всех встроенных кнопок, которые SharePoint самостоятельно помещает в раздел Actions (Действия) ленты.
Замените
TODO11
указанной ниже строкой, в которой удаляется исходная кнопка, определенная описательным способом. Если бы у нас не было этой строки, то каждый список на веб-сайте, для которого используется шаблон списка 100, включал бы настраиваемую кнопку. Так как функциональность кнопки тесно связана со списком Local Employees (Местные сотрудники), то нет никакого смысла размещать эту кнопку в других списках. Кроме того, без этой строки кнопка будет дважды размещена в списке Local Employees (Местные сотрудники), так как для этого списка используется шаблон 100.webScopedEmployeeAction.DeleteObject();
Теперь весь метод должен иметь указанный ниже вид (за исключением того, что заполнитель необходимо заменить на GUID).
private static void ChangeCustomActionRegistration() { using (var clientContext = SPContext.CreateUserClientContextForSPHost()) { var query = from action in clientContext.Web.UserCustomActions where action.Name == "{button_GUID} .AddEmployeeToCorpDB" select action; IEnumerable<UserCustomAction> matchingActions = clientContext.LoadQuery(query); clientContext.ExecuteQuery(); UserCustomAction webScopedEmployeeAction = matchingActions.Single(); var queryForList = from list in clientContext.Web.Lists where list.Title == "Local Employees" select list; IEnumerable<List> matchingLists = clientContext.LoadQuery(queryForList); clientContext.ExecuteQuery(); List employeeList = matchingLists.First(); var listActions = employeeList.UserCustomActions; clientContext.Load(listActions); listActions.Clear(); var listScopedEmployeeAction = listActions.Add(); listScopedEmployeeAction.Title = webScopedEmployeeAction.Title; listScopedEmployeeAction.Location = webScopedEmployeeAction.Location; listScopedEmployeeAction.Sequence = webScopedEmployeeAction.Sequence; listScopedEmployeeAction.CommandUIExtension = webScopedEmployeeAction.CommandUIExtension; listScopedEmployeeAction.Update(); webScopedEmployeeAction.DeleteObject(); clientContext.ExecuteQuery(); } }
Запрос полного контроля над хост-сайтом
Так как теперь надстройка добавляет и удаляет дополнительные действия уровня веб-сайта, необходимо расширить разрешения для нее с уровня "Управление" до уровня "Полный контроль". Для этого сделайте вот что:
В обозревателе решений в проекте ChainStore откройте файл AppManifest.xml.
Откройте вкладку Разрешения. Для поля Область оставьте значение Интернет, а в поле Разрешение выберите из раскрывающегося списка пункт Полный контроль.
Сохраните файл.
Запуск надстройки и тестирование развертывания кнопки
Откройте страницу Site Contents (Содержание сайта) веб-сайта магазина в Гонконге и удалите список Local Employees (Местные сотрудники).
Примечание.
При отзыве надстройки в Visual Studio не будут удалены созданные ею списки, поэтому вам потребуется вручную удалять их каждый раз, когда вы тестируете создающий их код.
Нажмите клавишу F5, чтобы развернуть и запустить надстройку. Редактор Visual Studio размещает удаленное веб-приложение в IIS Express, а базу данных SQL — в SQL Express. Кроме того, он выполняет временную установку надстройки на вашем тестовом сайте SharePoint и сразу же запускает ее. Прежде чем откроется начальная страница надстройки, вам будет предложено предоставить надстройке необходимые разрешения.
Когда откроется начальная страница надстройки, перейдите по ссылке Вернуться на сайт на размещенном в верхней части элементе управления хрома.
Перейдите на страницу Содержимое сайта. На ней будет список Local Employees (Местные сотрудники), так как его добавила логика первого запуска.
Примечание.
Если списка там нет или имеются другие признаки того, что код первого запуска не выполняется, может быть, при нажатии клавиши F5 таблица Клиенты не очищается. Как правило, это связано с тем, что проект ChainCorporateDB больше не задан в качестве запускаемого в Visual Studio. См. примечание в верхней части данной статьи о том, как устранить эту проблему. Кроме того, убедитесь, что вы настроили базу данных так, чтобы она перестраивалась, как описано в разделе Настройка Visual Studio для перестройки корпоративной базы данных при каждом сеансе отладки.
Откройте список и добавьте элемент.
В представлении списка выберите этот элемент и откройте вкладку Item (Элемент) на ленте.
На вкладке Item (Элемент) нажмите кнопку Add to Corporate DB (Добавить в корпоративную базу данных). Сотрудник будет добавлен в корпоративную базу данных, а значение поля Added to Corporate DB (Добавлен в корпоративную базу данных) изменится на Yes (Да).
Вернитесь на страницу Site Contents (Содержание сайта) и выберите Add an add-in (Добавить надстройку).
Добавьте новый настраиваемый список. По умолчанию он будет иметь тип Generic, которому соответствует тип списка 100. После создания списка откройте вкладку Item (Элемент) на ленте. Обратите внимание, что кнопки Add to Corporate DB (Добавить в корпоративную базу данных) нет на ленте. Это связано с тем, что ваш код удалил кнопку уровня веб-сайта.
Чтобы завершить сеанс отладки, закройте окно браузера или остановите отладку в Visual Studio. При каждом нажатии клавиши F5 Visual Studio будет отзывать предыдущую версию надстройки и устанавливать ее последнюю версию.
Вы будете работать с этой надстройкой и решением Visual Studio при изучении других статей, поэтому при перерывах в работе рекомендуем отзывать надстройку. В обозревателе решений щелкните проект правой кнопкой мыши и выберите пункт Отозвать.
Дальнейшие действия
Для событий в списках и элементах списков также можно использовать настраиваемые обработчики в SharePoint. Чтобы узнать, как создать такой обработчик и развернуть его в коде, выполняемом при первом запуске, см. статью Обработка событий элемента списка в надстройке, размещаемой у поставщика.