ASP.NET MVC: История одного проекта Поиск (часть 10)
Содержание
ASP.NET MVC: История одного проекта "Готовимся к старту" (часть 1)
ASP.NET MVC: История одного проекта "Всё ради данных" (часть 2)
ASP.NET MVC: История одного проекта "Шаблоны и внешний вид" (часть 3)
ASP.NET MVC: История одного проекта "Еще немного классов" (часть 4)
ASP.NET MVC: История одного проекта "UI - всё для пользователя" (часть 5)
ASP.NET MVC: История одного проекта "UI - Добавление экспоната" (часть 6)
ASP.NET MVC: История одного проекта "UI - Редактирование экспоната" (часть 7)
ASP.NET MVC: История одного проекта "Обработка ошибок" (часть 8)
ASP.NET MVC: История одного проекта "Фильтрация" (часть 9)
ASP.NET MVC: История одного проекта "Поиск" (часть 10)
ASP.NET MVC: История одного проекта "Облако тегов" (часть 11)
ASP.NET MVC: История одного проекта "Главная страница" (часть 12)
Задача
Мне требуется создать универсальный поиск, с возможностью расширения списка типов сущностей, по которым можно осуществлять поиск. Поиск должен осуществляться через autocomplete-поле на всех страницах сайта. Для этого надо все типы сущностей: экспонат, лента, метка "засунуть" в один список, который через AJAX будет возвращать JSON-объекты в выпадающий список.
Поле ввода запроса
Куда положить поле ввода, это риторический вопрос. Конечно же в шаблон (_LayoutExtended), чтобы поле было доступно с любой страницы сайта (отключение поиска на избранных страницах, например, ввод логина и пароля, пока не будем обсуждать, но это в планах). Я создам UserControl, который будет отображать само поле ввода и содержать скрипты (у нас же ajax), которые будут обслуживать это поле:
Вот код разметки:
@Html.TextBox("searcher", null, new { data_autocomplete = @Url.Content("/site/search/") }) @Html.ScriptBlock( @<script type="text/javascript" src="@Url.Content("/scripts/search.mvc.helpers.js")"> </script>) @Html.ScriptBlock( @<script type="text/javascript" src="@Url.Content("/scripts/jquery.blockUI.js")"> </script>)
Как видно из разметки есть поле ввода и два скрипта: один - работает с поле ввода, второй - "прячет" от пользователя контент сайта, на время, пока происходит перенаправление на выбранный из списка элемент. Показывать второй скрипт я не буду, вы можете найти подобного рода скриптов на jQuery огромное количество в интернете или, в конце-концов, написать свой скрипт с таким функционалом. А вот первый я создам в папке Scripts, он пока пустой займемся им позже.
Откуда контент-то? Дык, JsonResult есть!
Поместить метод поиска Search разумнее всего в основной для сайта контролер, чтобы доступность метода не перекрывалась необходимостью авторизации и аутентификации. Значит в моем случае, в контролле SiteController:
public JsonResult Search() { return Json("", JsonRequestBehavior.AllowGet); }
Пока он тоже "пустой". А прежде чем начать писать что-то, я бы хотел объяснить, что я собираюсь сделать. Разложить, так сказать, по полочкам вариант реализации.
SearchItem базовый класс для поиска
Чтобы получить возможность наполнить выпадающий список autocomplete, надо чтобы сущности были одного типа (я не буду сейчас говорить про Tuple), иначе придется выдавать из метода Search (JsonResult) несколько списков, а потом еще и обрабатывать их по разному, в зависимости от полей и свойств, которые есть в этих классах. Я упрощу себе задачу, создам один абстрактный базовый класс SearchItem. От этого класса унаследую все классы, которые и будут иметь возможность отображаться в списке при поиске:
/// <summary>
/// Базовый класс для всех сущностей,
/// от которого должна быть унаследована сущность
/// чтобы иметь возможность поиска по ней
/// </summary>
public abstract class SearchItem {
/// <summary>
/// Идентификатор
/// </summary>
public abstract int Id { get; }
/// <summary>
/// Название CSS стиля для отображения
/// </summary>
public abstract string CssClass { get; }
/// <summary>
/// Наименование типа,
/// используется для перенаправления
/// пользователя
/// </summary>
public abstract string TypeName { get; }
/// <summary>
/// Заголовок, отображается в списке autocomplete
/// </summary>
public abstract string Title { get; }
}
Я постарался описать в summary всю информацию по свойствам. Надеюсь, всё понятно. Теперь унаследую от этого класса новый класс ExhibitSearch:
/// <summary> /// Класс используется при формировании /// результатов поиска по сущности Exhibit /// </summary> public class ExhibitSearch : SearchItem { Exhibit Model; public ExhibitSearch(Exhibit model) { this.Model = model; } public override int Id { get { return this.Model.Id; } } public override string Title { get { return Model.Title; } } public override string CssClass { get { return "exhibitsearch"; } } public override string TypeName { get { return "экспонат:"; } } }
Обратите внимание на CssClass, я создал этот стиль в файле каскадных стилей. В нем ничего особенного, просто хочется чтобы заголовок был под цвет шаблона в выпадающем списке. Такие стили я создал для каждого из наслеников класса SearchItem. А теперь класс для меток для поиска:
/// <summary> /// Класс используется при формировании /// результатов поиска по сущности Tag /// </summary> public class TagSearch : SearchItem { Tag Model; public TagSearch(Tag model) { this.Model = model; } public override int Id { get { return Model.Id; } } public override string Title { get { return Model.Name; } } public override string CssClass { get { return "tagsearch"; } } public override string TypeName { get { return "метка:"; } } }
И напоследок, остался еще класс для поиска по сущности "Лента":
/// <summary> /// Класс используется при формировании /// результатов поиска по сущности Tag /// </summary> public class LentaSearch : SearchItem { Lenta Model; public LentaSearch(Lenta model) { this.Model = model; } public override int Id { get { return Model.Id; } } public override string Title { get { return Model.Content.CutLongText(100, "..."); } } public override string CssClass { get { return "lentasearch"; } } public override string TypeName { get { return "лента:"; } } }
Пока хватит новых классов. Если потребуется осуществлять поиск по какой-либо еще сущности, я добавлю ее позже.
Метод Search
Возвращаемся к методу поиска:
public JsonResult Search(string term) { List<SearchItem> result = new List<SearchItem>(); var exhibits = exhibitRepository.All.Where(x => x.Title.Contains(term) || x.Content.Contains(term)) .Take(5) .ToList() .Select(x => new ExhibitSearch(x)); if (exhibits.Any()) { result.AddRange(exhibits); } var tags = tagRepository.All .Where(x => x.Name.Contains(term)) .Take(5) .ToList() .Select(x => new TagSearch(x)); if (tags.Any()) { result.AddRange(tags); } var lentas = lentaRepository.All .Where(x => x.Content.Contains(term)) .Take(5) .ToList() .Select(x => new LentaSearch(x)); if (lentas.Any()) { result.AddRange(lentas); } return Json(result.OrderBy(x => x.Title).ToArray(), JsonRequestBehavior.AllowGet); }
Находим все записи в разных типах сущностей, которые подходят по критерию отбора и складываем их в один список (Можно было бы использовать Linq и в частности Uinion(), но я сделал "по старинке"). И если теперь есть метод, можно написать скрипт, который будет его вызывать в файле search.mvc.helpers.js:
$(document).ready(function () { var url = $("#searcher").attr("data-autocomplete"); $("#searcher").autocomplete({ minLength: 2, source: function (request, response) { $.getJSON(url, { term: request.term }, response); }, select: function (event, ui) { block(); if (ui.item.CssClass == 'tagsearch') window.location = '/museum?t=' ui.item.Title; if (ui.item.CssClass == 'exhibitsearch') window.location = '/museum/show/' ui.item.Id; if (ui.item.CssClass == 'lentasearch') window.location = '/lenta?t=' ui.item.Title; return false; } }).data('autocomplete')._renderItem = function (ul, item) { return $('<li style="font-size:.7em; max-width:700px;"></li>') .data('item.autocomplete', item) .append('<a><span class="searchtype">' item.TypeName '</span><span class="searchtitle ' item.CssClass '">' item.Title '</span></a>') .appendTo(ul); }; });
Стоит еще показать как это работает. Autocomplete-поле теперь выбрасывает вот такой спискок:
Я набрал в строке поиска "го" и это результат работы поиска. Мне осталось только предложить скачать проект и попрощаться. Спасибо за внимание - пишите комментарии.
Подробнее: http://feedproxy.google.com/~r/blogmusor/~3/UpMOEN7bMTQ/85
Дайджест новых статей по интернет-маркетингу на ваш email
Новые статьи и публикации
- 2024-11-26 » Капитан грузового судна, или Как начать использовать Docker в своих проектах
- 2024-11-26 » Обеспечение безопасности ваших веб-приложений с помощью PHP OOP и PDO
- 2024-11-22 » Ошибки в Яндекс Вебмастере: как найти и исправить
- 2024-11-22 » Ошибки в Яндекс Вебмастере: как найти и исправить
- 2024-11-15 » Перенос сайта на WordPress с одного домена на другой
- 2024-11-08 » OSPanel 6: быстрый старт
- 2024-11-08 » Как установить PhpMyAdmin в Open Server Panel
- 2024-09-30 » Как быстро запустить Laravel на Windows
- 2024-09-25 » Next.js
- 2024-09-05 » OpenAI рассказал, как запретить ChatGPT использовать содержимое сайта для обучения
- 2024-08-28 » Чек-лист: как увеличить конверсию интернет-магазина на примере спортпита
- 2024-08-01 » WebSocket
- 2024-07-26 » Интеграция с Яндекс Еда
- 2024-07-26 » Интеграция с Эквайринг
- 2024-07-26 » Интеграция с СДЕК
- 2024-07-26 » Интеграция с Битрикс-24
- 2024-07-26 » Интеграция с Travelline
- 2024-07-26 » Интеграция с Iiko
- 2024-07-26 » Интеграция с Delivery Club
- 2024-07-26 » Интеграция с CRM
- 2024-07-26 » Интеграция с 1C-Бухгалтерия
- 2024-07-24 » Что такое сторителлинг: техники и примеры
- 2024-07-17 » Ошибка 404: что это такое и как ее использовать для бизнеса
- 2024-07-03 » Размещайте прайс-листы на FarPost.ru и продавайте товары быстро и выгодно
- 2024-07-01 » Профилирование кода в PHP
- 2024-06-28 » Изучаем ABC/XYZ-анализ: что это такое и какие решения с помощью него принимают
- 2024-06-17 » Зачем вам знать потребности клиента
- 2024-06-11 » Что нового в работе Яндекс Метрики: полный обзор обновления
- 2024-06-11 » Поведенческие факторы ранжирования в Яндексе
- 2024-06-11 » Скорость загрузки сайта: почему это важно и как влияет на ранжирование
Есть три способа отвечать на вопросы: сказать необходимое, отвечать с приветливостью и – наговорить лишнего Плутарх - (ок. 46 — ок.120) - древнегреческий писатель, историк |
Мы создаем сайты, которые работают! Профессионально обслуживаем и продвигаем их , а также по всей России и ближнему зарубежью с 2006 года!
Как мы работаем
Заявка
Позвоните или оставьте заявку на сайте.
Консультация
Обсуждаем что именно Вам нужно и помогаем определить как это лучше сделать!
Договор
Заключаем договор на оказание услуг, в котором прописаны условия и обязанности обеих сторон.
Выполнение работ
Непосредственно оказание требующихся услуг и работ по вашему заданию.
Поддержка
Сдача выполненых работ, последующие корректировки и поддержка при необходимости.