@snabcentr/client-core 2.65.1 → 2.65.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,5 @@
1
1
  import { inject, Injectable } from '@angular/core';
2
+ import { isArray } from 'lodash-es';
2
3
  import { ScIdOrSlugPipe } from '../../pipes/sc-id-or-slug.pipe';
3
4
  import { ScMediaImageTransformerPipe } from '../../pipes/sc-media-image-transformer.pipe';
4
5
  import { SC_COMPANY_INFO } from '../../tokens/sc-company-info';
@@ -87,7 +88,7 @@ export class SchemaOrgFactory {
87
88
  // Основной контент страницы - список товаров и подкатегорий.
88
89
  mainEntity: this.createCategoryMainEntity(category, products, categoryPath, productPath),
89
90
  // Навигационные хлебные крошки (Schema.org: breadcrumb).
90
- breadcrumb: breadcrumbs,
91
+ breadcrumb: this.createCatalogBreadcrumbsList(breadcrumbs, categoryPath),
91
92
  // Дополнительные метаданные для SEO.
92
93
  about: {
93
94
  '@type': 'CategoryCode',
@@ -257,6 +258,44 @@ export class SchemaOrgFactory {
257
258
  ],
258
259
  };
259
260
  }
261
+ /**
262
+ * Преобразует массив LinkData в BreadcrumbList для каталога.
263
+ *
264
+ * @param breadcrumbs Массив хлебных крошек.
265
+ * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.
266
+ */
267
+ createCatalogBreadcrumbsList(breadcrumbs, categoryPath) {
268
+ const breadcrumbItems = [
269
+ // Всегда добавляем главную страницу.
270
+ {
271
+ '@type': 'ListItem',
272
+ position: 1,
273
+ name: 'Главная',
274
+ item: this.urls.siteUrl,
275
+ },
276
+ // Добавляем каталог.
277
+ {
278
+ '@type': 'ListItem',
279
+ position: 2,
280
+ name: 'Каталог',
281
+ item: `${this.urls.siteUrl}/${categoryPath}`,
282
+ },
283
+ ];
284
+ // Добавляем существующие breadcrumbs с корректировкой позиций.
285
+ breadcrumbs.forEach((item, index) => {
286
+ const link = isArray(item.routerLink) ? item.routerLink.join('/') : item.routerLink;
287
+ breadcrumbItems.push({
288
+ '@type': 'ListItem',
289
+ position: index + 3, // Начинаем с позиции 3 (после Главная и Каталог).
290
+ name: item.label,
291
+ item: link ? `${this.urls.siteUrl}/${link}` : undefined,
292
+ });
293
+ });
294
+ return {
295
+ '@type': 'BreadcrumbList',
296
+ itemListElement: breadcrumbItems,
297
+ };
298
+ }
260
299
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SchemaOrgFactory, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
261
300
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SchemaOrgFactory, providedIn: 'root' }); }
262
301
  }
@@ -266,4 +305,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
266
305
  providedIn: 'root',
267
306
  }]
268
307
  }] });
269
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema-org-factory.js","sourceRoot":"","sources":["../../../../../projects/core/seo/classes/schema-org-factory.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { CollectionPage, Product, WithContext } from 'schema-dts';\n\nimport { ScCategory } from '../../catalog/dto/sc-category';\nimport { ScProduct } from '../../catalog/dto/sc-product';\nimport { ScVirtualCategory } from '../../catalog/dto/sc-virtual-category';\nimport { ScICategory } from '../../catalog/interfaces/sc-i-category';\nimport { ScIProduct } from '../../catalog/interfaces/sc-i-product';\nimport { ScIVirtualCategory } from '../../catalog/interfaces/sc-i-virtual-category';\nimport { ScIdOrSlugPipe } from '../../pipes/sc-id-or-slug.pipe';\nimport { ScMediaImageTransformerPipe } from '../../pipes/sc-media-image-transformer.pipe';\nimport { SC_COMPANY_INFO } from '../../tokens/sc-company-info';\nimport { SC_URLS } from '../../tokens/sc-urls';\nimport { ScISchemaOrgBreadcrumbList } from '../interfaces/sc-i-schema-org-breadcrumb-list';\n\n/**\n * Механизмы генерации схем данных Schema.org.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class SchemaOrgFactory {\n    /**\n     * Данные о компании.\n     */\n    private readonly companyInfo = inject(SC_COMPANY_INFO);\n\n    /**\n     * Список ссылок приложения.\n     */\n    private readonly urls = inject(SC_URLS);\n\n    /**\n     * Пайп, возвращающий идентификатор или символьное обозначение (slug) исходя из настроек окружения.\n     */\n    private idOrSlugPipe: ScIdOrSlugPipe = inject(ScIdOrSlugPipe);\n\n    /**\n     * Пайп для формирования полного URI изображения на media сервере.\n     */\n    private readonly mediaImageTransformerPipe = inject(ScMediaImageTransformerPipe);\n\n    /**\n     * Генерирует и возвращает схему данных для указанного продукта.\n     *\n     * @param product Данные о продукте.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    public generateForProductItem(product: ScIProduct, productPath: string): WithContext<Product> {\n        const productUrl = `${this.urls.siteUrl}${productPath.replace(':id', this.idOrSlugPipe.transform(product).toString())}`;\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        const image = this.mediaImageTransformerPipe.transform((product as ScProduct)?.getImagePreview() ?? product.image, true);\n        const description = product.properties?.description;\n        const priceValidUntil = this.normalizePriceValidUntil(product.discount?.expiredAt);\n\n        return {\n            '@context': 'https://schema.org',\n            '@type': 'Product',\n            name: product.name,\n            ...(image && { image }),\n            ...(description && { description }),\n            mpn: product.code,\n            url: productUrl,\n            offers: [\n                {\n                    '@type': 'Offer',\n                    priceCurrency: 'RUB',\n                    ...(priceValidUntil && { priceValidUntil }),\n                    price: product.costRub,\n                    availability: this.getAvailability(product),\n                    itemCondition: 'https://schema.org/NewCondition',\n                    seller: {\n                        '@type': 'Organization',\n                        name: this.companyInfo.name,\n                    },\n                    url: productUrl,\n                },\n            ],\n        } as WithContext<Product>;\n    }\n\n    /**\n     * Генерирует и возвращает схему данных для указанного продукта.\n     *\n     * @param category Данные о категории.\n     * @param products Данные о продуктах категории.\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     * @param breadcrumbs Данные о хлебных крошках для категории.\n     */\n    public generateForCategoryItem(\n        category: ScICategory | ScIVirtualCategory,\n        products: ScIProduct[],\n        categoryPath: string,\n        productPath: string,\n        breadcrumbs: ScISchemaOrgBreadcrumbList\n    ): WithContext<CollectionPage> {\n        const description = category.seo?.description ?? category.properties?.description;\n        const image = this.mediaImageTransformerPipe.transform(category.image, true);\n\n        return {\n            '@context': 'https://schema.org',\n            '@type': 'CollectionPage',\n            name: category.seo?.title ?? category.name,\n            ...(description && { description }),\n            url: this.getCategoryCanonicalUrl(category, categoryPath),\n            ...(image && { image }),\n            // SEO данные.\n            ...(category.seo?.keywords && { keywords: category.seo.keywords }),\n            // Основной контент страницы - список товаров и подкатегорий.\n            mainEntity: this.createCategoryMainEntity(category, products, categoryPath, productPath),\n            // Навигационные хлебные крошки (Schema.org: breadcrumb).\n            breadcrumb: breadcrumbs,\n            // Дополнительные метаданные для SEO.\n            about: {\n                '@type': 'CategoryCode',\n                codeValue: category.slug,\n                name: category.name,\n                ...(description && { description }),\n            },\n            // Информация о сайте.\n            isPartOf: {\n                '@type': 'WebSite',\n                name: this.companyInfo.name,\n                url: this.urls.siteUrl,\n            },\n            // Связанные страницы.\n            relatedLink: [\n                `${this.urls.siteUrl}${categoryPath.replace(':id', '')}`,\n                ...(this.isVirtualCategory(category) || !category.parentCategory ? [] : [this.getCategoryCanonicalUrl(category.parentCategory, categoryPath)]),\n            ],\n            // Дополнительные свойства категории (не изображения).\n            additionalProperty: [\n                {\n                    '@type': 'PropertyValue',\n                    name: 'Количество товаров',\n                    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                    value: (products?.length ?? 0).toString(),\n                },\n                {\n                    '@type': 'PropertyValue',\n                    name: 'Slug категории',\n                    value: category.slug,\n                },\n                // Добавляем SEO данные если есть.\n                ...(category.seo?.keywords\n                    ? [\n                          {\n                              '@type': 'PropertyValue',\n                              name: 'SEO Keywords',\n                              value: category.seo.keywords,\n                          },\n                      ]\n                    : []),\n                ...(category.seo?.title\n                    ? [\n                          {\n                              '@type': 'PropertyValue',\n                              name: 'SEO Title',\n                              value: category.seo.title,\n                          },\n                      ]\n                    : []),\n            ],\n        } as WithContext<CollectionPage>;\n    }\n\n    /**\n     * Нормализует дату окончания цены в формат Schema.org (YYYY-MM-DD).\n     *\n     * @param expiredAt Дата для нормализации\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private normalizePriceValidUntil(expiredAt?: string): string | undefined {\n        if (!expiredAt) {\n            return undefined;\n        }\n\n        const date = expiredAt.slice(0, 10);\n\n        return /^\\d{4}-\\d{2}-\\d{2}$/.test(date) ? date : undefined;\n    }\n\n    /**\n     * Определяет доступность товара по данным продукта.\n     *\n     * @param product Данные о продукте.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private getAvailability(product: ScIProduct): 'InStock' | 'OutOfStock' | 'PreOrder' {\n        if (product.onOrder) {\n            return 'PreOrder';\n        }\n\n        return product.stockCount?.length ? 'InStock' : 'OutOfStock';\n    }\n\n    /**\n     * Проверяет, является ли категория виртуальной.\n     *\n     * @param category Данные о категории для проверки.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private isVirtualCategory(category: ScICategory | ScIVirtualCategory): boolean {\n        return 'childCategories' in category && 'productCategories' in category;\n    }\n\n    /**\n     * Возвращает URL для указанной категории.\n     *\n     * @param category Данные о категории\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     */\n    private getCategoryCanonicalUrl(category: ScICategory | ScIVirtualCategory, categoryPath: string): string {\n        const id = this.isVirtualCategory(category) ? category.slug : this.idOrSlugPipe.transform(category).toString();\n        const path = categoryPath.replace(':id', id);\n\n        return `${this.urls.siteUrl}${path}`;\n    }\n\n    /**\n     * Создаёт элемент списка ItemList с кратким Product для страницы категории.\n     *\n     * @param product Данные о продукте.\n     * @param position Позиция в списке.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    private createProductsListItem(product: ScIProduct, position: number, productPath: string): object {\n        const productUrl = `${this.urls.siteUrl}${productPath.replace(':id', this.idOrSlugPipe.transform(product).toString())}`;\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        const image = this.mediaImageTransformerPipe.transform((product as ScProduct)?.getImagePreview?.() ?? product.image, true);\n        const description = product.seo?.description ?? product.properties?.description;\n        const priceValidUntil = this.normalizePriceValidUntil(product.discount?.expiredAt);\n\n        return {\n            '@type': 'ListItem',\n            position,\n            item: {\n                '@type': 'Product',\n                name: product.name,\n                url: productUrl,\n                ...(image && { image }),\n                ...(description && { description }),\n                offers: [\n                    {\n                        '@type': 'Offer',\n                        priceCurrency: 'RUB',\n                        ...(priceValidUntil && { priceValidUntil }),\n                        price: product.costRub,\n                        availability: this.getAvailability(product),\n                        itemCondition: 'https://schema.org/NewCondition',\n                        seller: {\n                            '@type': 'Organization',\n                            name: this.companyInfo.name,\n                        },\n                        url: productUrl,\n                    },\n                ],\n            },\n        };\n    }\n\n    /**\n     * Создает mainEntity для указанной категории товаров.\n     *\n     * @param category Данные о категории.\n     * @param products Данные о товарах категории.\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    private createCategoryMainEntity(category: ScICategory | ScIVirtualCategory, products: ScIProduct[], categoryPath: string, productPath: string): object {\n        const childCategories = this.isVirtualCategory(category)\n            ? [...((category as ScVirtualCategory).childCategories ?? []), ...((category as ScVirtualCategory).productCategories ?? [])]\n            : ((category as ScCategory).categories ?? []);\n\n        return {\n            '@type': 'ItemList',\n            numberOfItems: products.length + childCategories.length,\n            itemListOrder: 'https://schema.org/ItemListUnordered',\n            itemListElement: [\n                // Добавляем подкатегории в начало списка.\n                ...childCategories.slice(0, 5).map((subCategory, index) => ({\n                    '@type': 'ListItem',\n                    position: index + 1,\n                    name: subCategory.name,\n                    url: this.getCategoryCanonicalUrl(subCategory, categoryPath),\n                    image: this.mediaImageTransformerPipe.transform(subCategory.image, true),\n                    description: subCategory.seo?.description ?? subCategory.properties?.description,\n                })),\n                // Добавляем товары после подкатегорий.\n                ...products.slice(0, 10).map((product, index) => this.createProductsListItem(product, childCategories.length + index + 1, productPath)),\n            ],\n        };\n    }\n}\n"]}
308
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema-org-factory.js","sourceRoot":"","sources":["../../../../../projects/core/seo/classes/schema-org-factory.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { isArray } from 'lodash-es';\nimport { BreadcrumbList, CollectionPage, ListItem, Product, WithContext } from 'schema-dts';\n\nimport { ScCategory } from '../../catalog/dto/sc-category';\nimport { ScProduct } from '../../catalog/dto/sc-product';\nimport { ScVirtualCategory } from '../../catalog/dto/sc-virtual-category';\nimport { ScICategory } from '../../catalog/interfaces/sc-i-category';\nimport { ScIProduct } from '../../catalog/interfaces/sc-i-product';\nimport { ScIVirtualCategory } from '../../catalog/interfaces/sc-i-virtual-category';\nimport { ScIdOrSlugPipe } from '../../pipes/sc-id-or-slug.pipe';\nimport { ScMediaImageTransformerPipe } from '../../pipes/sc-media-image-transformer.pipe';\nimport { SC_COMPANY_INFO } from '../../tokens/sc-company-info';\nimport { SC_URLS } from '../../tokens/sc-urls';\nimport { ScIBreadcrumbItem } from '../interfaces';\n\n/**\n * Механизмы генерации схем данных Schema.org.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class SchemaOrgFactory {\n    /**\n     * Данные о компании.\n     */\n    private readonly companyInfo = inject(SC_COMPANY_INFO);\n\n    /**\n     * Список ссылок приложения.\n     */\n    private readonly urls = inject(SC_URLS);\n\n    /**\n     * Пайп, возвращающий идентификатор или символьное обозначение (slug) исходя из настроек окружения.\n     */\n    private idOrSlugPipe: ScIdOrSlugPipe = inject(ScIdOrSlugPipe);\n\n    /**\n     * Пайп для формирования полного URI изображения на media сервере.\n     */\n    private readonly mediaImageTransformerPipe = inject(ScMediaImageTransformerPipe);\n\n    /**\n     * Генерирует и возвращает схему данных для указанного продукта.\n     *\n     * @param product Данные о продукте.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    public generateForProductItem(product: ScIProduct, productPath: string): WithContext<Product> {\n        const productUrl = `${this.urls.siteUrl}${productPath.replace(':id', this.idOrSlugPipe.transform(product).toString())}`;\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        const image = this.mediaImageTransformerPipe.transform((product as ScProduct)?.getImagePreview() ?? product.image, true);\n        const description = product.properties?.description;\n        const priceValidUntil = this.normalizePriceValidUntil(product.discount?.expiredAt);\n\n        return {\n            '@context': 'https://schema.org',\n            '@type': 'Product',\n            name: product.name,\n            ...(image && { image }),\n            ...(description && { description }),\n            mpn: product.code,\n            url: productUrl,\n            offers: [\n                {\n                    '@type': 'Offer',\n                    priceCurrency: 'RUB',\n                    ...(priceValidUntil && { priceValidUntil }),\n                    price: product.costRub,\n                    availability: this.getAvailability(product),\n                    itemCondition: 'https://schema.org/NewCondition',\n                    seller: {\n                        '@type': 'Organization',\n                        name: this.companyInfo.name,\n                    },\n                    url: productUrl,\n                },\n            ],\n        } as WithContext<Product>;\n    }\n\n    /**\n     * Генерирует и возвращает схему данных для указанного продукта.\n     *\n     * @param category Данные о категории.\n     * @param products Данные о продуктах категории.\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     * @param breadcrumbs Данные о хлебных крошках для категории.\n     */\n    public generateForCategoryItem(\n        category: ScICategory | ScIVirtualCategory,\n        products: ScIProduct[],\n        categoryPath: string,\n        productPath: string,\n        breadcrumbs: ScIBreadcrumbItem[]\n    ): WithContext<CollectionPage> {\n        const description = category.seo?.description ?? category.properties?.description;\n        const image = this.mediaImageTransformerPipe.transform(category.image, true);\n\n        return {\n            '@context': 'https://schema.org',\n            '@type': 'CollectionPage',\n            name: category.seo?.title ?? category.name,\n            ...(description && { description }),\n            url: this.getCategoryCanonicalUrl(category, categoryPath),\n            ...(image && { image }),\n            // SEO данные.\n            ...(category.seo?.keywords && { keywords: category.seo.keywords }),\n            // Основной контент страницы - список товаров и подкатегорий.\n            mainEntity: this.createCategoryMainEntity(category, products, categoryPath, productPath),\n            // Навигационные хлебные крошки (Schema.org: breadcrumb).\n            breadcrumb: this.createCatalogBreadcrumbsList(breadcrumbs, categoryPath),\n            // Дополнительные метаданные для SEO.\n            about: {\n                '@type': 'CategoryCode',\n                codeValue: category.slug,\n                name: category.name,\n                ...(description && { description }),\n            },\n            // Информация о сайте.\n            isPartOf: {\n                '@type': 'WebSite',\n                name: this.companyInfo.name,\n                url: this.urls.siteUrl,\n            },\n            // Связанные страницы.\n            relatedLink: [\n                `${this.urls.siteUrl}${categoryPath.replace(':id', '')}`,\n                ...(this.isVirtualCategory(category) || !category.parentCategory ? [] : [this.getCategoryCanonicalUrl(category.parentCategory, categoryPath)]),\n            ],\n            // Дополнительные свойства категории (не изображения).\n            additionalProperty: [\n                {\n                    '@type': 'PropertyValue',\n                    name: 'Количество товаров',\n                    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                    value: (products?.length ?? 0).toString(),\n                },\n                {\n                    '@type': 'PropertyValue',\n                    name: 'Slug категории',\n                    value: category.slug,\n                },\n                // Добавляем SEO данные если есть.\n                ...(category.seo?.keywords\n                    ? [\n                          {\n                              '@type': 'PropertyValue',\n                              name: 'SEO Keywords',\n                              value: category.seo.keywords,\n                          },\n                      ]\n                    : []),\n                ...(category.seo?.title\n                    ? [\n                          {\n                              '@type': 'PropertyValue',\n                              name: 'SEO Title',\n                              value: category.seo.title,\n                          },\n                      ]\n                    : []),\n            ],\n        } as WithContext<CollectionPage>;\n    }\n\n    /**\n     * Нормализует дату окончания цены в формат Schema.org (YYYY-MM-DD).\n     *\n     * @param expiredAt Дата для нормализации\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private normalizePriceValidUntil(expiredAt?: string): string | undefined {\n        if (!expiredAt) {\n            return undefined;\n        }\n\n        const date = expiredAt.slice(0, 10);\n\n        return /^\\d{4}-\\d{2}-\\d{2}$/.test(date) ? date : undefined;\n    }\n\n    /**\n     * Определяет доступность товара по данным продукта.\n     *\n     * @param product Данные о продукте.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private getAvailability(product: ScIProduct): 'InStock' | 'OutOfStock' | 'PreOrder' {\n        if (product.onOrder) {\n            return 'PreOrder';\n        }\n\n        return product.stockCount?.length ? 'InStock' : 'OutOfStock';\n    }\n\n    /**\n     * Проверяет, является ли категория виртуальной.\n     *\n     * @param category Данные о категории для проверки.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    private isVirtualCategory(category: ScICategory | ScIVirtualCategory): boolean {\n        return 'childCategories' in category && 'productCategories' in category;\n    }\n\n    /**\n     * Возвращает URL для указанной категории.\n     *\n     * @param category Данные о категории\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     */\n    private getCategoryCanonicalUrl(category: ScICategory | ScIVirtualCategory, categoryPath: string): string {\n        const id = this.isVirtualCategory(category) ? category.slug : this.idOrSlugPipe.transform(category).toString();\n        const path = categoryPath.replace(':id', id);\n\n        return `${this.urls.siteUrl}${path}`;\n    }\n\n    /**\n     * Создаёт элемент списка ItemList с кратким Product для страницы категории.\n     *\n     * @param product Данные о продукте.\n     * @param position Позиция в списке.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    private createProductsListItem(product: ScIProduct, position: number, productPath: string): object {\n        const productUrl = `${this.urls.siteUrl}${productPath.replace(':id', this.idOrSlugPipe.transform(product).toString())}`;\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        const image = this.mediaImageTransformerPipe.transform((product as ScProduct)?.getImagePreview?.() ?? product.image, true);\n        const description = product.seo?.description ?? product.properties?.description;\n        const priceValidUntil = this.normalizePriceValidUntil(product.discount?.expiredAt);\n\n        return {\n            '@type': 'ListItem',\n            position,\n            item: {\n                '@type': 'Product',\n                name: product.name,\n                url: productUrl,\n                ...(image && { image }),\n                ...(description && { description }),\n                offers: [\n                    {\n                        '@type': 'Offer',\n                        priceCurrency: 'RUB',\n                        ...(priceValidUntil && { priceValidUntil }),\n                        price: product.costRub,\n                        availability: this.getAvailability(product),\n                        itemCondition: 'https://schema.org/NewCondition',\n                        seller: {\n                            '@type': 'Organization',\n                            name: this.companyInfo.name,\n                        },\n                        url: productUrl,\n                    },\n                ],\n            },\n        };\n    }\n\n    /**\n     * Создает mainEntity для указанной категории товаров.\n     *\n     * @param category Данные о категории.\n     * @param products Данные о товарах категории.\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     * @param productPath Url путь к товару. В пути должен присутствовать шаблон ':id', обозначающий идентификатор товара для подстановки.\n     */\n    private createCategoryMainEntity(category: ScICategory | ScIVirtualCategory, products: ScIProduct[], categoryPath: string, productPath: string): object {\n        const childCategories = this.isVirtualCategory(category)\n            ? [...((category as ScVirtualCategory).childCategories ?? []), ...((category as ScVirtualCategory).productCategories ?? [])]\n            : ((category as ScCategory).categories ?? []);\n\n        return {\n            '@type': 'ItemList',\n            numberOfItems: products.length + childCategories.length,\n            itemListOrder: 'https://schema.org/ItemListUnordered',\n            itemListElement: [\n                // Добавляем подкатегории в начало списка.\n                ...childCategories.slice(0, 5).map((subCategory, index) => ({\n                    '@type': 'ListItem',\n                    position: index + 1,\n                    name: subCategory.name,\n                    url: this.getCategoryCanonicalUrl(subCategory, categoryPath),\n                    image: this.mediaImageTransformerPipe.transform(subCategory.image, true),\n                    description: subCategory.seo?.description ?? subCategory.properties?.description,\n                })),\n                // Добавляем товары после подкатегорий.\n                ...products.slice(0, 10).map((product, index) => this.createProductsListItem(product, childCategories.length + index + 1, productPath)),\n            ],\n        };\n    }\n\n    /**\n     * Преобразует массив LinkData в BreadcrumbList для каталога.\n     *\n     * @param breadcrumbs Массив хлебных крошек.\n     * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.\n     */\n    private createCatalogBreadcrumbsList(breadcrumbs: ScIBreadcrumbItem[], categoryPath: string): BreadcrumbList {\n        const breadcrumbItems: ListItem[] = [\n            // Всегда добавляем главную страницу.\n            {\n                '@type': 'ListItem',\n                position: 1,\n                name: 'Главная',\n                item: this.urls.siteUrl,\n            },\n            // Добавляем каталог.\n            {\n                '@type': 'ListItem',\n                position: 2,\n                name: 'Каталог',\n                item: `${this.urls.siteUrl}/${categoryPath}`,\n            },\n        ];\n\n        // Добавляем существующие breadcrumbs с корректировкой позиций.\n        breadcrumbs.forEach((item, index) => {\n            const link = isArray(item.routerLink) ? item.routerLink.join('/') : item.routerLink;\n\n            breadcrumbItems.push({\n                '@type': 'ListItem',\n                position: index + 3, // Начинаем с позиции 3 (после Главная и Каталог).\n                name: item.label,\n                item: link ? `${this.urls.siteUrl}/${link}` : undefined,\n            });\n        });\n\n        return {\n            '@type': 'BreadcrumbList',\n            itemListElement: breadcrumbItems,\n        } as BreadcrumbList;\n    }\n}\n"]}
@@ -1,5 +1,5 @@
1
- export * from './sc-i-schema-org-breadcrumb-list';
1
+ export * from './sc-i-breadcrumb-item';
2
2
  export * from './sc-i-default-page-meta';
3
3
  export * from './sc-i-seo';
4
4
  export * from './sc-i-seo-resource';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3Nlby9pbnRlcmZhY2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLFlBQVksQ0FBQztBQUMzQixjQUFjLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zYy1pLXNjaGVtYS1vcmctYnJlYWRjcnVtYi1saXN0JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtaS1kZWZhdWx0LXBhZ2UtbWV0YSc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWktc2VvJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtaS1zZW8tcmVzb3VyY2UnO1xuIl19
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3Nlby9pbnRlcmZhY2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLFlBQVksQ0FBQztBQUMzQixjQUFjLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zYy1pLWJyZWFkY3J1bWItaXRlbSc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWktZGVmYXVsdC1wYWdlLW1ldGEnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1pLXNlbyc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWktc2VvLXJlc291cmNlJztcbiJdfQ==
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Данные об элементе навигационной цепочки.
3
+ */
4
+ export class ScIBreadcrumbItem {
5
+ constructor() {
6
+ /**
7
+ * Формат дополнительного заголовка.
8
+ */
9
+ this.subLabelFormat = 'base';
10
+ }
11
+ }
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtaS1icmVhZGNydW1iLWl0ZW0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3Nlby9pbnRlcmZhY2VzL3NjLWktYnJlYWRjcnVtYi1pdGVtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQUE5QjtRQWlCSTs7V0FFRztRQUNILG1CQUFjLEdBQTBCLE1BQU0sQ0FBQztJQU1uRCxDQUFDO0NBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqINCU0LDQvdC90YvQtSDQvtCxINGN0LvQtdC80LXQvdGC0LUg0L3QsNCy0LjQs9Cw0YbQuNC+0L3QvdC+0Lkg0YbQtdC/0L7Rh9C60LguXG4gKi9cbmV4cG9ydCBjbGFzcyBTY0lCcmVhZGNydW1iSXRlbSB7XG4gICAgLyoqXG4gICAgICog0KHRgdGL0LvQutCwINC40LvQuCDQvtCx0YrQtdC60YIg0L7Qv9C40YHRi9Cy0LDRjtGJ0LjQuSDRgdGB0YvQu9C60YMuXG4gICAgICovXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1leHBsaWNpdC1hbnlcbiAgICByb3V0ZXJMaW5rPzogc3RyaW5nIHwgc3RyaW5nW107XG5cbiAgICAvKipcbiAgICAgKiDQnNC10YLQutCwINC40LvQuCDQt9Cw0LPQvtC70L7QstC+0Log0YHRgdGL0LvQutC4LlxuICAgICAqL1xuICAgIGxhYmVsOiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiDQlNC+0L/QvtC70L3QuNGC0LXQu9GM0L3Ri9C5INC30LDQs9C+0LvQvtCy0L7QuiDRgdGB0YvQu9C60LguXG4gICAgICovXG4gICAgc3ViTGFiZWw/OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiDQpNC+0YDQvNCw0YIg0LTQvtC/0L7Qu9C90LjRgtC10LvRjNC90L7Qs9C+INC30LDQs9C+0LvQvtCy0LrQsC5cbiAgICAgKi9cbiAgICBzdWJMYWJlbEZvcm1hdD86ICdzbScgfCAnYmFzZScgfCAnbGcnID0gJ2Jhc2UnO1xuXG4gICAgLyoqXG4gICAgICog0J/RgNC40LfQvdCw0Log0L3QtdC+0LHRhdC+0LTQuNC80L7RgdGC0Lgg0YHQutGA0YvRgtGMINGB0YHRi9C70LrRgyDQtNC70Y8g0LPQvtGB0YLQtdC5LlxuICAgICAqL1xuICAgIGhpZGVGb3JHdWVzdHM/OiBib29sZWFuO1xufVxuIl19
@@ -112,7 +112,7 @@ export class ScSeoService {
112
112
  * @param urlPath Url путь к элементу каталога. В пути должен присутствовать шаблон ':id', обозначающий идентификатор элемента каталога для подстановки.
113
113
  * @param schemaOrgData Данные Schema.org элемента каталога
114
114
  */
115
- setForCategoryItem(item, urlPath, schemaOrgData) {
115
+ setForCatalogItem(item, urlPath, schemaOrgData) {
116
116
  const url = `${this.urls.siteUrl}${urlPath.replace(':id', this.idOrSlugPipe.transform(item).toString())}`;
117
117
  const title = item.seo?.title ?? `${item.name} - Купить в ${this.companyInfo.name}`;
118
118
  const description = item.seo?.description ?? `${item.name} - Купить по выгодным ценам в ${this.companyInfo.name}`;
@@ -188,9 +188,18 @@ export class ScSeoService {
188
188
  * @param data Данные Schema.org для встраивания на странице.
189
189
  */
190
190
  setSchemaOrgData(data) {
191
+ const projectAttributeName = 'data-sc-schema-org';
192
+ // Удаляем все ранее созданные скрипты Schema.org (по уникальному атрибуту проекта).
193
+ const existingScripts = this.document.head.querySelectorAll(`script[${projectAttributeName}]`);
194
+ existingScripts.forEach((existingScript) => {
195
+ if (existingScript.parentNode) {
196
+ this.renderer.removeChild(existingScript.parentNode, existingScript);
197
+ }
198
+ });
191
199
  // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
192
200
  const script = this.renderer.createElement('script');
193
201
  this.renderer.setAttribute(script, 'type', 'application/ld+json');
202
+ this.renderer.setAttribute(script, projectAttributeName, '');
194
203
  this.renderer.appendChild(script, this.document.createTextNode(JSON.stringify(data)));
195
204
  this.renderer.appendChild(this.document.head, script);
196
205
  }
@@ -203,4 +212,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
203
212
  providedIn: 'root',
204
213
  }]
205
214
  }] });
206
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-seo.service.js","sourceRoot":"","sources":["../../../../../projects/core/seo/services/sc-seo.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport { inject, Injectable, Renderer2, RendererFactory2 } from '@angular/core';\nimport { Meta, Title } from '@angular/platform-browser';\nimport { Router } from '@angular/router';\nimport { Thing, WithContext } from 'schema-dts';\n\nimport { ScICatalogItem } from '../../catalog/interfaces/sc-i-catalog-item';\nimport { ScICategory } from '../../catalog/interfaces/sc-i-category';\nimport { ScIdOrSlugPipe } from '../../pipes/sc-id-or-slug.pipe';\nimport { ScMediaImageTransformerPipe } from '../../pipes/sc-media-image-transformer.pipe';\nimport { SC_COMPANY_INFO } from '../../tokens/sc-company-info';\nimport { SC_CATALOG_PAGE_META, SC_DEFAULT_PAGE_META } from '../../tokens/sc-default-page-meta';\nimport { SC_ID_OR_SLUG_IN_ROUTE } from '../../tokens/sc-id-or-slug-in-route';\nimport { SC_URLS } from '../../tokens/sc-urls';\nimport { ScISeo } from '../interfaces/sc-i-seo';\nimport { RobotsTagType } from '../types/robots-tag-type';\n\n/**\n * Сервис управления SEO-параметрами.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class ScSeoService {\n    /**\n     * Данные о компании.\n     */\n    private readonly companyInfo = inject(SC_COMPANY_INFO);\n\n    /**\n     * Метаданные по умолчанию.\n     */\n    private readonly defaultSeo = inject(SC_DEFAULT_PAGE_META);\n\n    /**\n     * Метаданные страницы каталога по умолчанию.\n     */\n    private readonly catalogDefaultSeo = inject(SC_CATALOG_PAGE_META);\n\n    /**\n     * Признак необходимости использовать Id или Slug при маршрутизации.\n     */\n    private readonly idOrSlugInRoute = inject(SC_ID_OR_SLUG_IN_ROUTE);\n\n    /**\n     * Список ссылок приложения.\n     */\n    private readonly urls = inject(SC_URLS);\n\n    /**\n     * Пайп, возвращающий идентификатор или символьное обозначение (slug) исходя из настроек окружения.\n     */\n    private idOrSlugPipe: ScIdOrSlugPipe = inject(ScIdOrSlugPipe);\n\n    /**\n     * Пайп для формирования полного URI изображения на media сервере.\n     */\n    private readonly mediaImageTransformerPipe = inject(ScMediaImageTransformerPipe);\n\n    /**\n     * Сервис установки метаданных страницы.\n     */\n    private readonly metaService = inject(Meta);\n\n    /**\n     * Сервис маршрутизации.\n     */\n    private readonly router = inject(Router);\n\n    /**\n     * Сервис управления заголовком страницы.\n     */\n    private readonly titleService = inject(Title);\n\n    /**\n     * Объект {@link Document}, предоставляющий доступ к DOM страницы.\n     */\n    private readonly document = inject(DOCUMENT);\n\n    /**\n     * Фабрика для создания экземпляров {@link Renderer2}.\n     */\n    private readonly rendererFactory = inject(RendererFactory2);\n\n    /**\n     * Экземпляр {@link Renderer2} для безопасной работы с DOM.\n     */\n    private readonly renderer: Renderer2 = this.rendererFactory.createRenderer(null, null);\n\n    /**\n     * Устанавливает seo параметры на основе указанного ресурса. Если какого-то тега нет, будет установлено значение по умолчанию.\n     *\n     * @param resource Ресурс на основе которого необходимо установить мета-теги.\n     * @param img Изображение. Если не пришло, то будет установлено значение по умолчанию.\n     * @param robots Список robots-тегов для страницы.\n     */\n    public setFromResource(resource?: ScISeo, img: string = this.urls.logoUrl, robots: RobotsTagType[] = ['index', 'follow']): void {\n        const updatedResource = {\n            title: resource?.title ?? this.defaultSeo.title,\n            description: resource?.description ?? this.defaultSeo.description,\n            keywords: resource?.keywords ?? this.defaultSeo.keywords,\n            meta: resource?.meta ?? [],\n        };\n        const url = `${this.urls.siteUrl}${this.router.url}`;\n\n        this.setMetaTags(updatedResource.title, updatedResource.description, updatedResource.keywords, updatedResource.meta, robots);\n        this.setOpenGraphMetaTags(updatedResource.title, updatedResource.description, url, img);\n    }\n\n    /**\n     * Устанавливает seo параметры со значением по умолчанию.\n     *\n     * @param title Заголовок страницы.\n     * @param robots Список robots тегов для страницы.\n     */\n    public setByDefault(title?: string, robots: RobotsTagType[] = ['index', 'follow']): void {\n        this.setFromResource(\n            {\n                title: `${title ?? this.defaultSeo.title} - ${this.companyInfo.name}`,\n                description: `${title}, ${this.defaultSeo.description}`,\n            },\n            this.urls.logoUrl,\n            robots\n        );\n    }\n\n    /**\n     * Устанавливает seo параметры со значением по умолчанию. Устанавливает значение \"не индексировать\" для поисковых роботов.\n     *\n     * @param title Заголовок страницы.\n     */\n    public setByDefaultAsNoIndexed(title?: string): void {\n        this.setByDefault(title, ['noindex', 'nofollow', 'noarchive']);\n    }\n\n    /**\n     * Устанавливает seo-параметры для указанного элемента каталога.\n     *\n     * @param item Данные об элементе каталога.\n     * @param urlPath Url путь к элементу каталога. В пути должен присутствовать шаблон ':id', обозначающий идентификатор элемента каталога для подстановки.\n     * @param schemaOrgData Данные Schema.org элемента каталога\n     */\n    public setForCategoryItem(item: ScICatalogItem<ScICategory>, urlPath: string, schemaOrgData?: WithContext<Thing>): void {\n        const url = `${this.urls.siteUrl}${urlPath.replace(':id', this.idOrSlugPipe.transform(item).toString())}`;\n        const title = item.seo?.title ?? `${item.name} - Купить в ${this.companyInfo.name}`;\n        const description = item.seo?.description ?? `${item.name} - Купить по выгодным ценам в ${this.companyInfo.name}`;\n        const keywords = item.seo?.keywords ?? `${item.name}, ${this.catalogDefaultSeo.keywords}`;\n        const meta = item.seo?.meta ?? [];\n\n        // Устанавливаем стандартные seo-параметры.\n        this.setMetaTags(title, description, keywords, meta, ['index', 'follow']);\n        this.setOpenGraphMetaTags(title, description, url, this.mediaImageTransformerPipe.transform(item.image, true));\n\n        // В том случае, если используется slug для формирования ссылок, то необходимо сформировать кононическую\n        // ссылку на страницу в которой используется slug (а не id).\n        if (this.idOrSlugInRoute === 'slug' && item.id && item.slug) {\n            this.setCanonicalLink(url);\n        }\n\n        // Устанавливаем Schema.org параметры.\n        if (schemaOrgData) {\n            this.setSchemaOrgData(schemaOrgData);\n        }\n    }\n\n    /**\n     * Устанавливает все meta-теги страницы. Если какой-либо из параметров не указан, для него устанавливается значение по умолчанию.\n     *\n     * @param title Заголовок страницы.\n     * @param description Описание страницы.\n     * @param keywords Ключевые слова страницы.\n     * @param meta Массив дополнительных значений метатегов.\n     * @param robots Список robots тегов для страницы.\n     */\n    private setMetaTags(\n        title: string = this.defaultSeo.title,\n        description: string = this.defaultSeo.description,\n        keywords: string = this.defaultSeo.keywords,\n        meta: ScISeo['meta'] = [],\n        robots: RobotsTagType[] = ['index', 'follow']\n    ): void {\n        this.titleService.setTitle(title);\n\n        this.metaService.updateTag({ name: 'description', content: description });\n        this.metaService.updateTag({ name: 'keywords', content: keywords });\n        this.metaService.updateTag({ name: 'robots', content: robots.join(', ') });\n\n        meta.forEach((tag) => {\n            this.metaService.updateTag({ name: tag.name, content: tag.content });\n        });\n    }\n\n    /**\n     * Устанавливает Open Graph meta-теги.\n     *\n     * @param ogTitle Заголовок.\n     * @param ogDescription Описание.\n     * @param ogUrl Url.\n     * @param ogImg Ссылка на изображение.\n     */\n    private setOpenGraphMetaTags(ogTitle: string, ogDescription: string, ogUrl: string, ogImg: string): void {\n        this.metaService.updateTag({ property: 'og:type', content: 'website' });\n        this.metaService.updateTag({ property: 'og:title', content: ogTitle });\n        this.metaService.updateTag({ property: 'og:description', content: ogDescription });\n        this.metaService.updateTag({ property: 'og:url', content: ogUrl });\n        this.metaService.updateTag({ property: 'og:image', content: ogImg });\n    }\n\n    /**\n     * Устанавливает тег канонической страницы <link rel=\"canonical\" href=\"...\" />.\n     *\n     * @param url URL который необходимо использовать в качестве ссылки.\n     */\n    private setCanonicalLink(url: string): void {\n        // Удаляем существующий link canonical.\n        const existingLink = this.document.querySelector('link[rel=\"canonical\"]');\n\n        if (existingLink?.parentNode) {\n            this.renderer.removeChild(existingLink.parentNode, existingLink);\n        }\n\n        // Создаём и устанавливаем новый тег.\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n        const link = this.renderer.createElement('link');\n        this.renderer.setAttribute(link, 'rel', 'canonical');\n        this.renderer.setAttribute(link, 'href', url);\n        this.renderer.appendChild(this.document.head, link);\n    }\n\n    /**\n     * Устанавливает Schema.org объект страницы.\n     *\n     * @param data Данные Schema.org для встраивания на странице.\n     */\n    private setSchemaOrgData<T extends Thing>(data: WithContext<T>): void {\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n        const script = this.renderer.createElement('script');\n        this.renderer.setAttribute(script, 'type', 'application/ld+json');\n        this.renderer.appendChild(script, this.document.createTextNode(JSON.stringify(data)));\n        this.renderer.appendChild(this.document.head, script);\n    }\n}\n"]}
215
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-seo.service.js","sourceRoot":"","sources":["../../../../../projects/core/seo/services/sc-seo.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport { inject, Injectable, Renderer2, RendererFactory2 } from '@angular/core';\nimport { Meta, Title } from '@angular/platform-browser';\nimport { Router } from '@angular/router';\nimport { Thing, WithContext } from 'schema-dts';\n\nimport { ScICatalogItem } from '../../catalog/interfaces/sc-i-catalog-item';\nimport { ScICategory } from '../../catalog/interfaces/sc-i-category';\nimport { ScIVirtualCategory } from '../../catalog/interfaces/sc-i-virtual-category';\nimport { ScIdOrSlugPipe } from '../../pipes/sc-id-or-slug.pipe';\nimport { ScMediaImageTransformerPipe } from '../../pipes/sc-media-image-transformer.pipe';\nimport { SC_COMPANY_INFO } from '../../tokens/sc-company-info';\nimport { SC_CATALOG_PAGE_META, SC_DEFAULT_PAGE_META } from '../../tokens/sc-default-page-meta';\nimport { SC_ID_OR_SLUG_IN_ROUTE } from '../../tokens/sc-id-or-slug-in-route';\nimport { SC_URLS } from '../../tokens/sc-urls';\nimport { ScISeo } from '../interfaces/sc-i-seo';\nimport { RobotsTagType } from '../types/robots-tag-type';\n\n/**\n * Сервис управления SEO-параметрами.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class ScSeoService {\n    /**\n     * Данные о компании.\n     */\n    private readonly companyInfo = inject(SC_COMPANY_INFO);\n\n    /**\n     * Метаданные по умолчанию.\n     */\n    private readonly defaultSeo = inject(SC_DEFAULT_PAGE_META);\n\n    /**\n     * Метаданные страницы каталога по умолчанию.\n     */\n    private readonly catalogDefaultSeo = inject(SC_CATALOG_PAGE_META);\n\n    /**\n     * Признак необходимости использовать Id или Slug при маршрутизации.\n     */\n    private readonly idOrSlugInRoute = inject(SC_ID_OR_SLUG_IN_ROUTE);\n\n    /**\n     * Список ссылок приложения.\n     */\n    private readonly urls = inject(SC_URLS);\n\n    /**\n     * Пайп, возвращающий идентификатор или символьное обозначение (slug) исходя из настроек окружения.\n     */\n    private idOrSlugPipe: ScIdOrSlugPipe = inject(ScIdOrSlugPipe);\n\n    /**\n     * Пайп для формирования полного URI изображения на media сервере.\n     */\n    private readonly mediaImageTransformerPipe = inject(ScMediaImageTransformerPipe);\n\n    /**\n     * Сервис установки метаданных страницы.\n     */\n    private readonly metaService = inject(Meta);\n\n    /**\n     * Сервис маршрутизации.\n     */\n    private readonly router = inject(Router);\n\n    /**\n     * Сервис управления заголовком страницы.\n     */\n    private readonly titleService = inject(Title);\n\n    /**\n     * Объект {@link Document}, предоставляющий доступ к DOM страницы.\n     */\n    private readonly document = inject(DOCUMENT);\n\n    /**\n     * Фабрика для создания экземпляров {@link Renderer2}.\n     */\n    private readonly rendererFactory = inject(RendererFactory2);\n\n    /**\n     * Экземпляр {@link Renderer2} для безопасной работы с DOM.\n     */\n    private readonly renderer: Renderer2 = this.rendererFactory.createRenderer(null, null);\n\n    /**\n     * Устанавливает seo параметры на основе указанного ресурса. Если какого-то тега нет, будет установлено значение по умолчанию.\n     *\n     * @param resource Ресурс на основе которого необходимо установить мета-теги.\n     * @param img Изображение. Если не пришло, то будет установлено значение по умолчанию.\n     * @param robots Список robots-тегов для страницы.\n     */\n    public setFromResource(resource?: ScISeo, img: string = this.urls.logoUrl, robots: RobotsTagType[] = ['index', 'follow']): void {\n        const updatedResource = {\n            title: resource?.title ?? this.defaultSeo.title,\n            description: resource?.description ?? this.defaultSeo.description,\n            keywords: resource?.keywords ?? this.defaultSeo.keywords,\n            meta: resource?.meta ?? [],\n        };\n        const url = `${this.urls.siteUrl}${this.router.url}`;\n\n        this.setMetaTags(updatedResource.title, updatedResource.description, updatedResource.keywords, updatedResource.meta, robots);\n        this.setOpenGraphMetaTags(updatedResource.title, updatedResource.description, url, img);\n    }\n\n    /**\n     * Устанавливает seo параметры со значением по умолчанию.\n     *\n     * @param title Заголовок страницы.\n     * @param robots Список robots тегов для страницы.\n     */\n    public setByDefault(title?: string, robots: RobotsTagType[] = ['index', 'follow']): void {\n        this.setFromResource(\n            {\n                title: `${title ?? this.defaultSeo.title} - ${this.companyInfo.name}`,\n                description: `${title}, ${this.defaultSeo.description}`,\n            },\n            this.urls.logoUrl,\n            robots\n        );\n    }\n\n    /**\n     * Устанавливает seo параметры со значением по умолчанию. Устанавливает значение \"не индексировать\" для поисковых роботов.\n     *\n     * @param title Заголовок страницы.\n     */\n    public setByDefaultAsNoIndexed(title?: string): void {\n        this.setByDefault(title, ['noindex', 'nofollow', 'noarchive']);\n    }\n\n    /**\n     * Устанавливает seo-параметры для указанного элемента каталога.\n     *\n     * @param item Данные об элементе каталога.\n     * @param urlPath Url путь к элементу каталога. В пути должен присутствовать шаблон ':id', обозначающий идентификатор элемента каталога для подстановки.\n     * @param schemaOrgData Данные Schema.org элемента каталога\n     */\n    public setForCatalogItem(item: ScICatalogItem<ScICategory | ScIVirtualCategory>, urlPath: string, schemaOrgData?: WithContext<Thing>): void {\n        const url = `${this.urls.siteUrl}${urlPath.replace(':id', this.idOrSlugPipe.transform(item).toString())}`;\n        const title = item.seo?.title ?? `${item.name} - Купить в ${this.companyInfo.name}`;\n        const description = item.seo?.description ?? `${item.name} - Купить по выгодным ценам в ${this.companyInfo.name}`;\n        const keywords = item.seo?.keywords ?? `${item.name}, ${this.catalogDefaultSeo.keywords}`;\n        const meta = item.seo?.meta ?? [];\n\n        // Устанавливаем стандартные seo-параметры.\n        this.setMetaTags(title, description, keywords, meta, ['index', 'follow']);\n        this.setOpenGraphMetaTags(title, description, url, this.mediaImageTransformerPipe.transform(item.image, true));\n\n        // В том случае, если используется slug для формирования ссылок, то необходимо сформировать кононическую\n        // ссылку на страницу в которой используется slug (а не id).\n        if (this.idOrSlugInRoute === 'slug' && item.id && item.slug) {\n            this.setCanonicalLink(url);\n        }\n\n        // Устанавливаем Schema.org параметры.\n        if (schemaOrgData) {\n            this.setSchemaOrgData(schemaOrgData);\n        }\n    }\n\n    /**\n     * Устанавливает все meta-теги страницы. Если какой-либо из параметров не указан, для него устанавливается значение по умолчанию.\n     *\n     * @param title Заголовок страницы.\n     * @param description Описание страницы.\n     * @param keywords Ключевые слова страницы.\n     * @param meta Массив дополнительных значений метатегов.\n     * @param robots Список robots тегов для страницы.\n     */\n    private setMetaTags(\n        title: string = this.defaultSeo.title,\n        description: string = this.defaultSeo.description,\n        keywords: string = this.defaultSeo.keywords,\n        meta: ScISeo['meta'] = [],\n        robots: RobotsTagType[] = ['index', 'follow']\n    ): void {\n        this.titleService.setTitle(title);\n\n        this.metaService.updateTag({ name: 'description', content: description });\n        this.metaService.updateTag({ name: 'keywords', content: keywords });\n        this.metaService.updateTag({ name: 'robots', content: robots.join(', ') });\n\n        meta.forEach((tag) => {\n            this.metaService.updateTag({ name: tag.name, content: tag.content });\n        });\n    }\n\n    /**\n     * Устанавливает Open Graph meta-теги.\n     *\n     * @param ogTitle Заголовок.\n     * @param ogDescription Описание.\n     * @param ogUrl Url.\n     * @param ogImg Ссылка на изображение.\n     */\n    private setOpenGraphMetaTags(ogTitle: string, ogDescription: string, ogUrl: string, ogImg: string): void {\n        this.metaService.updateTag({ property: 'og:type', content: 'website' });\n        this.metaService.updateTag({ property: 'og:title', content: ogTitle });\n        this.metaService.updateTag({ property: 'og:description', content: ogDescription });\n        this.metaService.updateTag({ property: 'og:url', content: ogUrl });\n        this.metaService.updateTag({ property: 'og:image', content: ogImg });\n    }\n\n    /**\n     * Устанавливает тег канонической страницы <link rel=\"canonical\" href=\"...\" />.\n     *\n     * @param url URL который необходимо использовать в качестве ссылки.\n     */\n    private setCanonicalLink(url: string): void {\n        // Удаляем существующий link canonical.\n        const existingLink = this.document.querySelector('link[rel=\"canonical\"]');\n\n        if (existingLink?.parentNode) {\n            this.renderer.removeChild(existingLink.parentNode, existingLink);\n        }\n\n        // Создаём и устанавливаем новый тег.\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n        const link = this.renderer.createElement('link');\n        this.renderer.setAttribute(link, 'rel', 'canonical');\n        this.renderer.setAttribute(link, 'href', url);\n        this.renderer.appendChild(this.document.head, link);\n    }\n\n    /**\n     * Устанавливает Schema.org объект страницы.\n     *\n     * @param data Данные Schema.org для встраивания на странице.\n     */\n    private setSchemaOrgData<T extends Thing>(data: WithContext<T>): void {\n        const projectAttributeName = 'data-sc-schema-org';\n        // Удаляем все ранее созданные скрипты Schema.org (по уникальному атрибуту проекта).\n        const existingScripts = this.document.head.querySelectorAll(`script[${projectAttributeName}]`);\n        existingScripts.forEach((existingScript) => {\n            if (existingScript.parentNode) {\n                this.renderer.removeChild(existingScript.parentNode, existingScript);\n            }\n        });\n\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n        const script = this.renderer.createElement('script');\n        this.renderer.setAttribute(script, 'type', 'application/ld+json');\n        this.renderer.setAttribute(script, projectAttributeName, '');\n        this.renderer.appendChild(script, this.document.createTextNode(JSON.stringify(data)));\n        this.renderer.appendChild(this.document.head, script);\n    }\n}\n"]}
@@ -6915,6 +6915,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
6915
6915
  args: [SC_URLS]
6916
6916
  }] }] });
6917
6917
 
6918
+ /**
6919
+ * Данные об элементе навигационной цепочки.
6920
+ */
6921
+ class ScIBreadcrumbItem {
6922
+ constructor() {
6923
+ /**
6924
+ * Формат дополнительного заголовка.
6925
+ */
6926
+ this.subLabelFormat = 'base';
6927
+ }
6928
+ }
6929
+
6918
6930
  /**
6919
6931
  * Механизмы генерации схем данных Schema.org.
6920
6932
  */
@@ -6998,7 +7010,7 @@ class SchemaOrgFactory {
6998
7010
  // Основной контент страницы - список товаров и подкатегорий.
6999
7011
  mainEntity: this.createCategoryMainEntity(category, products, categoryPath, productPath),
7000
7012
  // Навигационные хлебные крошки (Schema.org: breadcrumb).
7001
- breadcrumb: breadcrumbs,
7013
+ breadcrumb: this.createCatalogBreadcrumbsList(breadcrumbs, categoryPath),
7002
7014
  // Дополнительные метаданные для SEO.
7003
7015
  about: {
7004
7016
  '@type': 'CategoryCode',
@@ -7168,6 +7180,44 @@ class SchemaOrgFactory {
7168
7180
  ],
7169
7181
  };
7170
7182
  }
7183
+ /**
7184
+ * Преобразует массив LinkData в BreadcrumbList для каталога.
7185
+ *
7186
+ * @param breadcrumbs Массив хлебных крошек.
7187
+ * @param categoryPath Url путь к категориям товаров. В пути должен присутствовать шаблон ':id', обозначающий идентификатор категории для подстановки.
7188
+ */
7189
+ createCatalogBreadcrumbsList(breadcrumbs, categoryPath) {
7190
+ const breadcrumbItems = [
7191
+ // Всегда добавляем главную страницу.
7192
+ {
7193
+ '@type': 'ListItem',
7194
+ position: 1,
7195
+ name: 'Главная',
7196
+ item: this.urls.siteUrl,
7197
+ },
7198
+ // Добавляем каталог.
7199
+ {
7200
+ '@type': 'ListItem',
7201
+ position: 2,
7202
+ name: 'Каталог',
7203
+ item: `${this.urls.siteUrl}/${categoryPath}`,
7204
+ },
7205
+ ];
7206
+ // Добавляем существующие breadcrumbs с корректировкой позиций.
7207
+ breadcrumbs.forEach((item, index) => {
7208
+ const link = isArray(item.routerLink) ? item.routerLink.join('/') : item.routerLink;
7209
+ breadcrumbItems.push({
7210
+ '@type': 'ListItem',
7211
+ position: index + 3, // Начинаем с позиции 3 (после Главная и Каталог).
7212
+ name: item.label,
7213
+ item: link ? `${this.urls.siteUrl}/${link}` : undefined,
7214
+ });
7215
+ });
7216
+ return {
7217
+ '@type': 'BreadcrumbList',
7218
+ itemListElement: breadcrumbItems,
7219
+ };
7220
+ }
7171
7221
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SchemaOrgFactory, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
7172
7222
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SchemaOrgFactory, providedIn: 'root' }); }
7173
7223
  }
@@ -7281,7 +7331,7 @@ class ScSeoService {
7281
7331
  * @param urlPath Url путь к элементу каталога. В пути должен присутствовать шаблон ':id', обозначающий идентификатор элемента каталога для подстановки.
7282
7332
  * @param schemaOrgData Данные Schema.org элемента каталога
7283
7333
  */
7284
- setForCategoryItem(item, urlPath, schemaOrgData) {
7334
+ setForCatalogItem(item, urlPath, schemaOrgData) {
7285
7335
  const url = `${this.urls.siteUrl}${urlPath.replace(':id', this.idOrSlugPipe.transform(item).toString())}`;
7286
7336
  const title = item.seo?.title ?? `${item.name} - Купить в ${this.companyInfo.name}`;
7287
7337
  const description = item.seo?.description ?? `${item.name} - Купить по выгодным ценам в ${this.companyInfo.name}`;
@@ -7357,9 +7407,18 @@ class ScSeoService {
7357
7407
  * @param data Данные Schema.org для встраивания на странице.
7358
7408
  */
7359
7409
  setSchemaOrgData(data) {
7410
+ const projectAttributeName = 'data-sc-schema-org';
7411
+ // Удаляем все ранее созданные скрипты Schema.org (по уникальному атрибуту проекта).
7412
+ const existingScripts = this.document.head.querySelectorAll(`script[${projectAttributeName}]`);
7413
+ existingScripts.forEach((existingScript) => {
7414
+ if (existingScript.parentNode) {
7415
+ this.renderer.removeChild(existingScript.parentNode, existingScript);
7416
+ }
7417
+ });
7360
7418
  // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
7361
7419
  const script = this.renderer.createElement('script');
7362
7420
  this.renderer.setAttribute(script, 'type', 'application/ld+json');
7421
+ this.renderer.setAttribute(script, projectAttributeName, '');
7363
7422
  this.renderer.appendChild(script, this.document.createTextNode(JSON.stringify(data)));
7364
7423
  this.renderer.appendChild(this.document.head, script);
7365
7424
  }
@@ -7752,5 +7811,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
7752
7811
  * Generated bundle index. Do not edit.
7753
7812
  */
7754
7813
 
7755
- export { CATEGORY_PROVIDERS, EMPTY_CART, ERROR_HANDLER, IS_BROWSER, IS_RUNNING_ON_TERMINAL, IS_SERVER, RESPONSE, SC_ACCESS_AUTH_TOKEN_STORAGE_KEY, SC_API_KEYS, SC_AUTH_ADD_HEADER_REQUIRED, SC_AUTH_AS_CLIENT_DEFAULT_OPTIONS, SC_AUTH_AS_CLIENT_OPTIONS, SC_CACHE_LIFETIME, SC_CACHE_SETTINGS, SC_CATALOG_PAGE_META, SC_CATEGORY_INFO, SC_COMPANY_INFO, SC_COMPANY_NAME, SC_CONFIGURATOR_COMPONENTS, SC_CONFIGURATOR_SETTINGS, SC_COUNT_FIRST_DISCOUNTED_PRODUCTS, SC_COUNT_LAST_NEWS, SC_DATE_FORMAT, SC_DEFAULT_PAGE_META, SC_FEEDBACK_API, SC_GUEST_ENDPOINTS_TOKEN, SC_GUEST_PARAMETER_NAME_TOKEN, SC_GUEST_TOKEN_STORAGE_KEY, SC_ID_OR_SLUG_IN_ROUTE, SC_IS_HEADER_REQUIRED, SC_IS_HIDDEN_ERROR_ALERT, SC_IS_HIDDEN_ERROR_ALERT_HTTP_CONTEXT, SC_IS_LOGOUT_REQUEST, SC_IS_REFRESH_REQUIRED, SC_LINEAR_VALUES, SC_LINEAR_VALUES_TOKEN, SC_MAX_LENGTH_SEARCH_TERM, SC_MIN_LENGTH_SEARCH_TERM, SC_NEXT_PAGE_PAGINATION_CLICK, SC_PATH_IMAGE_NOT_FOUND, SC_PRODUCT_PAGINATION_DEFAULT_OPTIONS, SC_PRODUCT_PAGINATION_OPTIONS, SC_REFRESH_AUTH_TOKEN_STORAGE_KEY, SC_RELEASE, SC_UPDATE_INTERVAL, SC_URLS, SC_USER_INFO, SC_VACANCIES_DATA_SOURCE, SC_VIRTUAL_CATEGORY_INFO, SC_VIRTUAL_CATEGORY_PROVIDERS, SEARCH_TERM, SEARCH_TERM_PROVIDERS, ScAuthAsClientGuard, ScAuthInterceptor, ScAuthService, ScBanner, ScBannerService, ScCacheInterceptor, ScCacheSettings, ScCachedData, ScCachedItem, ScCart, ScCartItem, ScCartService, ScCatalogFilterService, ScCatalogFormat, ScCatalogItem, ScCatalogService, ScCatalogableItem, ScCategory, ScClientType, ScCodedIdentity, ScConfiguratorService, ScContactsService, ScContragentService, ScConvertInterceptor, ScConvertersService, ScDateFormatInterceptor, ScDeletableNamedIdentity, ScDeliveryAddressService, ScDeliveryCost, ScDeliveryType, ScDocumentInfoNode, ScDocumentInfoTypesEnum, ScErrorsInterceptor, ScFavoriteService, ScFeedbackForms, ScFeedbackService, ScFilesService, ScFrequentlyAskedQuestionsService, ScGuestInterceptor, ScHiddenCatalogableItem, ScISalesDirectionCart, ScISuggestionType, ScIconTypesEnum, ScIdOrSlugGuard, ScIdOrSlugPipe, ScIdentity, ScImage, ScImageHelper, ScLocationsService, ScMediaImageTransformerPipe, ScMimeTypes, ScNamedIdentity, ScNews, ScNewsService, ScNotificationActionTypes, ScNotificationLevelNames, ScNotificationsService, ScOpfList, ScOptionsInterceptor, ScOrder, ScOrderShort, ScOrderStateStatus, ScOrdersService, ScPaginationService, ScPaymentStatus, ScPaymentType, ScPhoneService, ScPrimaryCatalogableItem, ScProduct, ScProductTileType, ScQuestionnaireService, ScQuestionnaireStatusEnum, ScReclamationService, ScReclamationStatus, ScRecommendationService, ScReference, ScReferenceName, ScReferencesService, ScRequisitesService, ScRouteKeys, ScSearchService, ScSeoService, ScSocialType, ScSuggestionService, ScSum, ScTokenService, ScUIService, ScUTMService, ScUnitsHelper, ScUpdatableNamedIdentity, ScUploadedFile, ScUser, ScUserMetadata, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService, ScUserType, ScVCardService, ScVacanciesList, ScVacanciesService, ScVacancy, ScVerificationService, ScVirtualCategory, ScWarehouseService, SchemaOrgFactory, TERMINAL_PROVIDERS, USER_AGENT_TERMINAL, filterChangedByKey, runningOnTerminalFactory, searchTermFactory };
7814
+ export { CATEGORY_PROVIDERS, EMPTY_CART, ERROR_HANDLER, IS_BROWSER, IS_RUNNING_ON_TERMINAL, IS_SERVER, RESPONSE, SC_ACCESS_AUTH_TOKEN_STORAGE_KEY, SC_API_KEYS, SC_AUTH_ADD_HEADER_REQUIRED, SC_AUTH_AS_CLIENT_DEFAULT_OPTIONS, SC_AUTH_AS_CLIENT_OPTIONS, SC_CACHE_LIFETIME, SC_CACHE_SETTINGS, SC_CATALOG_PAGE_META, SC_CATEGORY_INFO, SC_COMPANY_INFO, SC_COMPANY_NAME, SC_CONFIGURATOR_COMPONENTS, SC_CONFIGURATOR_SETTINGS, SC_COUNT_FIRST_DISCOUNTED_PRODUCTS, SC_COUNT_LAST_NEWS, SC_DATE_FORMAT, SC_DEFAULT_PAGE_META, SC_FEEDBACK_API, SC_GUEST_ENDPOINTS_TOKEN, SC_GUEST_PARAMETER_NAME_TOKEN, SC_GUEST_TOKEN_STORAGE_KEY, SC_ID_OR_SLUG_IN_ROUTE, SC_IS_HEADER_REQUIRED, SC_IS_HIDDEN_ERROR_ALERT, SC_IS_HIDDEN_ERROR_ALERT_HTTP_CONTEXT, SC_IS_LOGOUT_REQUEST, SC_IS_REFRESH_REQUIRED, SC_LINEAR_VALUES, SC_LINEAR_VALUES_TOKEN, SC_MAX_LENGTH_SEARCH_TERM, SC_MIN_LENGTH_SEARCH_TERM, SC_NEXT_PAGE_PAGINATION_CLICK, SC_PATH_IMAGE_NOT_FOUND, SC_PRODUCT_PAGINATION_DEFAULT_OPTIONS, SC_PRODUCT_PAGINATION_OPTIONS, SC_REFRESH_AUTH_TOKEN_STORAGE_KEY, SC_RELEASE, SC_UPDATE_INTERVAL, SC_URLS, SC_USER_INFO, SC_VACANCIES_DATA_SOURCE, SC_VIRTUAL_CATEGORY_INFO, SC_VIRTUAL_CATEGORY_PROVIDERS, SEARCH_TERM, SEARCH_TERM_PROVIDERS, ScAuthAsClientGuard, ScAuthInterceptor, ScAuthService, ScBanner, ScBannerService, ScCacheInterceptor, ScCacheSettings, ScCachedData, ScCachedItem, ScCart, ScCartItem, ScCartService, ScCatalogFilterService, ScCatalogFormat, ScCatalogItem, ScCatalogService, ScCatalogableItem, ScCategory, ScClientType, ScCodedIdentity, ScConfiguratorService, ScContactsService, ScContragentService, ScConvertInterceptor, ScConvertersService, ScDateFormatInterceptor, ScDeletableNamedIdentity, ScDeliveryAddressService, ScDeliveryCost, ScDeliveryType, ScDocumentInfoNode, ScDocumentInfoTypesEnum, ScErrorsInterceptor, ScFavoriteService, ScFeedbackForms, ScFeedbackService, ScFilesService, ScFrequentlyAskedQuestionsService, ScGuestInterceptor, ScHiddenCatalogableItem, ScIBreadcrumbItem, ScISalesDirectionCart, ScISuggestionType, ScIconTypesEnum, ScIdOrSlugGuard, ScIdOrSlugPipe, ScIdentity, ScImage, ScImageHelper, ScLocationsService, ScMediaImageTransformerPipe, ScMimeTypes, ScNamedIdentity, ScNews, ScNewsService, ScNotificationActionTypes, ScNotificationLevelNames, ScNotificationsService, ScOpfList, ScOptionsInterceptor, ScOrder, ScOrderShort, ScOrderStateStatus, ScOrdersService, ScPaginationService, ScPaymentStatus, ScPaymentType, ScPhoneService, ScPrimaryCatalogableItem, ScProduct, ScProductTileType, ScQuestionnaireService, ScQuestionnaireStatusEnum, ScReclamationService, ScReclamationStatus, ScRecommendationService, ScReference, ScReferenceName, ScReferencesService, ScRequisitesService, ScRouteKeys, ScSearchService, ScSeoService, ScSocialType, ScSuggestionService, ScSum, ScTokenService, ScUIService, ScUTMService, ScUnitsHelper, ScUpdatableNamedIdentity, ScUploadedFile, ScUser, ScUserMetadata, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService, ScUserType, ScVCardService, ScVacanciesList, ScVacanciesService, ScVacancy, ScVerificationService, ScVirtualCategory, ScWarehouseService, SchemaOrgFactory, TERMINAL_PROVIDERS, USER_AGENT_TERMINAL, filterChangedByKey, runningOnTerminalFactory, searchTermFactory };
7756
7815
  //# sourceMappingURL=snabcentr-client-core.mjs.map