--upgrade scripts from nopCommerce 3.00 to 3.10 --new locale resources declare @resources xml --a resource will be delete if its value is empty set @resources=' App ID/API Key Enter your app ID/API key here. You can find it on your FaceBook application page. App Secret Enter your app secret here. You can find it on your FaceBook application page. Created on Date and time when this product was created. Updated on Date and time when this product was updated. Share button code A page share button code. By default, we''re using AddThis service. Number of associated products ({0}) ({0}) You have no items to compare. Home Home Store All stores Store If an asterisk is selected, then this shipping rate will apply to all stores. Date and time should entered in Coordinated Universal Time (UTC) Date and time should entered in Coordinated Universal Time (UTC) Limited to stores Determines whether the plugin is available only at certain stores. Stores Select stores for which the plugin will be used. Stores Plugin Info Tier prices You need to save the product before you can add tier prices for this product page. Store All stores Customer role All customer roles Quantity Price Discounts No discounts available. Create at least one discount before mapping. Product attributes You need to save the product before you can add attributes for this page. No product attributes available. Create at least one product attribute before mapping. Attribute combinations Select new combination and enter details below Add combination Attribute combinations are useful when your ''Manage inventory method'' is set to ''Track inventory by product attributes'' Attributes Allow out of stock A value indicating whether to allow orders when out of stock. Stock quantity The current stock quantity of this combination. Sku Product stock keeping unit (SKU). Your internal unique identifier that can be used to track this attribute combination. Manufacturer part number The manufacturer''s part number for this attribute combination. GTIN Enter global trade item number (GTIN). These identifiers include UPC (in North America), EAN (in Europe), JAN (in Japan), and ISBN (for books). Attributes Attribute Text prompt Is Required Control type Display order Values Add a new value back to product details Edit value Name The attribute value name e.g. ''Blue'' for Color attributes. Please provide a name. RGB color Choose color to be used with the color squares attribute control. Price adjustment The price adjustment applied when choosing this attribute value e.g. ''10'' to add 10 dollars. Weight adjustment The weight adjustment applied when choosing this attribute value Is pre-selected Determines whether this attribute value is pre selected for the customer Display order The display order of the attribute value. 1 represents the first item in attribute value list. Add/Edit values for [{0}] attribute. Product: {1} View/Edit value (Total: {0}) Additional shipping charge The additional shipping charge. Allow back in stock subscriptions Allow customers to subscribe to a notification list for a product that has gone out of stock. Allowed quantities Enter a comma separated list of quantities you want this product to be restricted to. Instead of a quantity textbox that allows them to enter any quantity, they will receive a dropdown list of the values you enter here. Automatically add these products to the cart Check to automatically add these products to the cart. Available end date The end of the product''s availability in Coordinated Universal Time (UTC). Available for pre-order Check if this item is available for Pre-Order. It also displays "Pre-order" button instead of "Add to cart". Available start date The start of the product''s availability in Coordinated Universal Time (UTC). Backorders Select backorder mode. Call for price Check to show "Call for Pricing" or "Call for quote" instead of price Customer enters price An option indicating whether customer should enter price. Disable buy button Check to disable the buy button for this product. This may be necessary for products that are ''available upon request''. Disable wishlist button Check to disable the wishlist button for this product. Display stock availability Check to display stock availability. When enabled, customers will see stock availability. Display stock quantity Check to display stock quantity. When enabled, customers will see stock quantity. Download activation type A value indicating when download links will be enabled. Number of days The number of days during customers keeps access to the file (e.g. 14). Leave this field blank to allow continuous downloads. Download file The download file. Gift card type Select gift card type. WARNING: not recommended to change in production environment. GTIN (global trade item number) Enter global trade item number (GTIN). These identifiers include UPC (in North America), EAN (in Europe), JAN (in Japan), and ISBN (for books). Has sample download file Check if this product has a sample download file that can be downloaded before checkout. Has user agreement Check if the product has a user agreement. Height The height of the product. ID The product identifier. Downloadable product Check if this product is a downloadable product. When a customer purchases a download product, they can download the item direct from your store by viewing their completed order. Free shipping Check if this product comes with FREE shipping. Is gift card Check if it is a gift card Recurring product Check if this product is a recurring product. Shipping enabled Determines whether the product can be shipped. Tax exempt Determines whether this product is tax exempt (tax will not be applied to this product at checkout). Length The length of the product. Low stock activity Action to be taken when your current stock quantity falls below the ''Minimum stock quantity''. Manage inventory method Select manage inventory method. When enabled, stock quantities are automatically adjusted when a customer makes a purchase. You can also set low stock activity actions and receive notifications. Manufacturer part number The manufacturer''s part number for this product. Maximum amount Enter a maximum amount. Max. downloads The maximum number of downloads. Minimum amount Enter a minimum amount. Minimum stock quantity If you have enabled ''Manage Stock'' you can perform a number of different actions when the current stock quantity falls below your minimum stock quantity. Notify admin for quantity below When the current stock quantity falls below this quantity, the storekeeper (admin) will receive a notification. Old price The old price of the product. If you set an old price, this will display alongside the current price on the product page to show the difference in price. Maximum cart quantity Set the maximum quantity allowed in a customer''s shopping cart e.g. set to 5 to only allow customers to purchase 5 of this product. Minimum cart quantity Set the minimum quantity allowed in a customer''s shopping cart e.g. set to 3 to only allow customers to purchase 3 or more of this product. Price The price of the product. Product cost The product cost is the cost of all the different components which make up the product. This may either be the purchase price if the components are bought from outside suppliers, or the combined cost of materials and manufacturing processes if the component is made in-house. Required product IDs Specify comma separated list of required product IDs. NOTE: Ensure that there are no circular references (for example, A requires B, and B requires A). Require other products added to the cart Check if this product requires that other products are added to the cart. Sample download file The sample download file. SKU Product stock keeping unit (SKU). Your internal unique identifier that can be used to track this product. Special price Set a special price for the product. New price will be valid between start and end dates. Leave empty to ignore field. Special price start date The start date of the special price in Coordinated Universal Time (UTC). Special price end date The end date of the special price in Coordinated Universal Time (UTC). Stock quantity The current stock quantity of this product. Total cycles Enter total cycles. Cycle length Enter cycle length. Cycle period Select cycle period. Tax category The tax classification for this product. You can manage product tax classifications from Configuration : Tax : Tax Classes User agreement text The text of the user agreement Unlimited downloads When a customer purchases a download product, they can download the item unlimited number of times. Weight The weight of the product. Can be used in shipping calculations. Width The width of the product. Assigned to products A list of products to which the discount is to be applied. You can assign this discount on a product details page. No products selected Restricted products The comma-separated list of product identifiers (e.g. 77, 123, 156). You can find a product ID on its details page. You can also specify the comma-separated list of product identifiers with quantities ({Product ID}:{Quantity}. for example, 77:1, 123:2, 156:3). And you can also specify the comma-separated list of product identifiers with quantity range ({Product ID}:{Min quantity}-{Max quantity}. for example, 77:1-3, 123:2-5, 156:3-8). Restricted products The comma-separated list of product identifiers (e.g. 77, 123, 156). You can find a product ID on its details page. You can also specify the comma-separated list of product identifiers with quantities ({Product ID}:{Quantity}. for example, 77:1, 123:2, 156:3). And you can also specify the comma-separated list of product identifiers with quantity range ({Product ID}:{Min quantity}-{Max quantity}. for example, 77:1-3, 123:2-5, 156:3-8). Product type Choose your product type. Simple product Grouped product Bulk edit products Click on interested product The type of discount. Assigned to products Choose a product template. This template defines how this product will be displayed in public store. Associated product image size The default size (pixels) for associated product images (part of ''grouped'' products). Associated products Add new associated product Product Associated products are used only with "grouped" products. A product could be associated to only one "grouped" product. You need to save the product before you can add associated products for this product page. Visible individually Check it if you want this product to be visible in catalog or search results. You can use this field (just uncheck) to hide associated products from catalog and make them accessible only from a parent "grouped" product details page. This product is sold out Product type Search by a product type. Display order Tracking number Search by a specific tracking number. Picture Choose a picture associated to this attribute value. No picture Associated to product A "grouped" parent product which this one is associated to. Attribute value type Choose your attribute value type. Associated product Associated product. Simple Associated to product {0}, {1}: {2} Associate a product Select To prevent a full-text index from becoming bloated, Microsoft SQL Server has a mechanism that discards commonly occurring words that do not help the search. These words are called noise words, or stop words. Noise words are listed in the locale specific noise word files. For example, in the English locale, words such as "a," "and," "is," and "the" are in the English noise word file and are left out of the full-text index since they are empirically known to be useless to a search. Please contact your SQL Server administrator to get more information about it. Message Search in title and review text. Open link in new window Select locale No customer account found Customer is deleted Account is not active Account is not registered Automatically search sub categories Check to search in sub categories. {0} review(s) Your cart has auto-ship (recurring) items with conflicting shipment schedules. Only one auto-ship schedule is allowed per order. Force SSL for all site pages By default not all site pages are SSL protected. Check to force SSL for the entire site. This setting is useful only when you have SSL enabled on your store details pages. Disable "Order completed" page When disabled, customers will be automatically redirected to the order details page. Continue Generate all possible combinations ' CREATE TABLE #LocaleStringResourceTmp ( [ResourceName] [nvarchar](200) NOT NULL, [ResourceValue] [nvarchar](max) NOT NULL ) INSERT INTO #LocaleStringResourceTmp (ResourceName, ResourceValue) SELECT nref.value('@Name', 'nvarchar(200)'), nref.value('Value[1]', 'nvarchar(MAX)') FROM @resources.nodes('//Language/LocaleResource') AS R(nref) --do it for each existing language DECLARE @ExistingLanguageID int DECLARE cur_existinglanguage CURSOR FOR SELECT [ID] FROM [Language] OPEN cur_existinglanguage FETCH NEXT FROM cur_existinglanguage INTO @ExistingLanguageID WHILE @@FETCH_STATUS = 0 BEGIN DECLARE @ResourceName nvarchar(200) DECLARE @ResourceValue nvarchar(MAX) DECLARE cur_localeresource CURSOR FOR SELECT ResourceName, ResourceValue FROM #LocaleStringResourceTmp OPEN cur_localeresource FETCH NEXT FROM cur_localeresource INTO @ResourceName, @ResourceValue WHILE @@FETCH_STATUS = 0 BEGIN IF (EXISTS (SELECT 1 FROM [LocaleStringResource] WHERE LanguageID=@ExistingLanguageID AND ResourceName=@ResourceName)) BEGIN UPDATE [LocaleStringResource] SET [ResourceValue]=@ResourceValue WHERE LanguageID=@ExistingLanguageID AND ResourceName=@ResourceName END ELSE BEGIN INSERT INTO [LocaleStringResource] ( [LanguageId], [ResourceName], [ResourceValue] ) VALUES ( @ExistingLanguageID, @ResourceName, @ResourceValue ) END IF (@ResourceValue is null or @ResourceValue = '') BEGIN DELETE [LocaleStringResource] WHERE LanguageID=@ExistingLanguageID AND ResourceName=@ResourceName END FETCH NEXT FROM cur_localeresource INTO @ResourceName, @ResourceValue END CLOSE cur_localeresource DEALLOCATE cur_localeresource --fetch next language identifier FETCH NEXT FROM cur_existinglanguage INTO @ExistingLanguageID END CLOSE cur_existinglanguage DEALLOCATE cur_existinglanguage DROP TABLE #LocaleStringResourceTmp GO --more indexes IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Category_LimitedToStores' and object_id=object_id(N'[Category]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Category_LimitedToStores] ON [Category] ([LimitedToStores] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Manufacturer_LimitedToStores' and object_id=object_id(N'[Manufacturer]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Manufacturer_LimitedToStores] ON [Manufacturer] ([LimitedToStores] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_LimitedToStores' and object_id=object_id(N'[Product]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Product_LimitedToStores] ON [Product] ([LimitedToStores] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Category_SubjectToAcl' and object_id=object_id(N'[Category]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Category_SubjectToAcl] ON [Category] ([SubjectToAcl] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Manufacturer_SubjectToAcl' and object_id=object_id(N'[Manufacturer]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Manufacturer_SubjectToAcl] ON [Manufacturer] ([SubjectToAcl] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_SubjectToAcl' and object_id=object_id(N'[Product]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Product_SubjectToAcl] ON [Product] ([SubjectToAcl] ASC) END GO --recaptcha theme name IF NOT EXISTS (SELECT 1 FROM [Setting] WHERE [name] = N'captchasettings.recaptchatheme') BEGIN INSERT [Setting] ([Name], [Value], [StoreId]) VALUES (N'captchasettings.recaptchatheme', N'', 0) END GO IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductLoadAllPaged]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [ProductLoadAllPaged] GO CREATE PROCEDURE [dbo].[ProductLoadAllPaged] ( @CategoryIds nvarchar(MAX) = null, --a list of category IDs (comma-separated list). e.g. 1,2,3 @ManufacturerId int = 0, @StoreId int = 0, @VendorId int = 0, @ProductTagId int = 0, @FeaturedProducts bit = null, --0 featured only , 1 not featured only, null - load all products @PriceMin decimal(18, 4) = null, @PriceMax decimal(18, 4) = null, @Keywords nvarchar(4000) = null, @SearchDescriptions bit = 0, --a value indicating whether to search by a specified "keyword" in product descriptions @SearchProductTags bit = 0, --a value indicating whether to search by a specified "keyword" in product tags @UseFullTextSearch bit = 0, @FullTextMode int = 0, --0 using CONTAINS with , 5 - using CONTAINS and OR with , 10 - using CONTAINS and AND with @FilteredSpecs nvarchar(MAX) = null, --filter by attributes (comma-separated list). e.g. 14,15,16 @LanguageId int = 0, @OrderBy int = 0, --0 position, 5 - Name: A to Z, 6 - Name: Z to A, 10 - Price: Low to High, 11 - Price: High to Low, 15 - creation date @AllowedCustomerRoleIds nvarchar(MAX) = null, --a list of customer role IDs (comma-separated list) for which a product should be shown (if a subjet to ACL) @PageIndex int = 0, @PageSize int = 2147483644, @ShowHidden bit = 0, @LoadFilterableSpecificationAttributeOptionIds bit = 0, --a value indicating whether we should load the specification attribute option identifiers applied to loaded products (all pages) @FilterableSpecificationAttributeOptionIds nvarchar(MAX) = null OUTPUT, --the specification attribute option identifiers applied to loaded products (all pages). returned as a comma separated list of identifiers @TotalRecords int = null OUTPUT ) AS BEGIN /* Products that filtered by keywords */ CREATE TABLE #KeywordProducts ( [ProductId] int NOT NULL ) DECLARE @SearchKeywords bit, @sql nvarchar(max), @sql_orderby nvarchar(max) SET NOCOUNT ON --filter by keywords SET @Keywords = isnull(@Keywords, '') SET @Keywords = rtrim(ltrim(@Keywords)) IF ISNULL(@Keywords, '') != '' BEGIN SET @SearchKeywords = 1 IF @UseFullTextSearch = 1 BEGIN --remove wrong chars (' ") SET @Keywords = REPLACE(@Keywords, '''', '') SET @Keywords = REPLACE(@Keywords, '"', '') --full-text search IF @FullTextMode = 0 BEGIN --0 - using CONTAINS with SET @Keywords = ' "' + @Keywords + '*" ' END ELSE BEGIN --5 - using CONTAINS and OR with --10 - using CONTAINS and AND with --clean multiple spaces WHILE CHARINDEX(' ', @Keywords) > 0 SET @Keywords = REPLACE(@Keywords, ' ', ' ') DECLARE @concat_term nvarchar(100) IF @FullTextMode = 5 --5 - using CONTAINS and OR with BEGIN SET @concat_term = 'OR' END IF @FullTextMode = 10 --10 - using CONTAINS and AND with BEGIN SET @concat_term = 'AND' END --now let's build search string declare @fulltext_keywords nvarchar(4000) set @fulltext_keywords = N'' declare @index int set @index = CHARINDEX(' ', @Keywords, 0) -- if index = 0, then only one field was passed IF(@index = 0) set @fulltext_keywords = ' "' + @Keywords + '*" ' ELSE BEGIN DECLARE @first BIT SET @first = 1 WHILE @index > 0 BEGIN IF (@first = 0) SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' ELSE SET @first = 0 SET @fulltext_keywords = @fulltext_keywords + '"' + SUBSTRING(@Keywords, 1, @index - 1) + '*"' SET @Keywords = SUBSTRING(@Keywords, @index + 1, LEN(@Keywords) - @index) SET @index = CHARINDEX(' ', @Keywords, 0) end -- add the last field IF LEN(@fulltext_keywords) > 0 SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' + '"' + SUBSTRING(@Keywords, 1, LEN(@Keywords)) + '*"' END SET @Keywords = @fulltext_keywords END END ELSE BEGIN --usual search by PATINDEX SET @Keywords = '%' + @Keywords + '%' END --PRINT @Keywords --product name SET @sql = ' INSERT INTO #KeywordProducts ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Name]) > 0 ' --product variant name SET @sql = @sql + ' UNION SELECT pv.ProductId FROM ProductVariant pv with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pv.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pv.[Name]) > 0 ' --SKU SET @sql = @sql + ' UNION SELECT pv.ProductId FROM ProductVariant pv with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pv.[Sku], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pv.[Sku]) > 0 ' --localized product name SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' IF @SearchDescriptions = 1 BEGIN --product short description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[ShortDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[ShortDescription]) > 0 ' --product full description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[FullDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[FullDescription]) > 0 ' --product variant description SET @sql = @sql + ' UNION SELECT pv.ProductId FROM ProductVariant pv with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pv.[Description], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pv.[Description]) > 0 ' --localized product short description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''ShortDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' --localized product full description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''FullDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END IF @SearchProductTags = 1 BEGIN --product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM Product_ProductTag_Mapping pptm with(NOLOCK) INNER JOIN ProductTag pt with(NOLOCK) ON pt.Id = pptm.ProductTag_Id WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pt.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pt.[Name]) > 0 ' --localized product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM LocalizedProperty lp with (NOLOCK) INNER JOIN Product_ProductTag_Mapping pptm with(NOLOCK) ON lp.EntityId = pptm.ProductTag_Id WHERE lp.LocaleKeyGroup = N''ProductTag'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END --PRINT (@sql) EXEC sp_executesql @sql, N'@Keywords nvarchar(4000)', @Keywords END ELSE BEGIN SET @SearchKeywords = 0 END --filter by category IDs SET @CategoryIds = isnull(@CategoryIds, '') CREATE TABLE #FilteredCategoryIds ( CategoryId int not null ) INSERT INTO #FilteredCategoryIds (CategoryId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@CategoryIds, ',') DECLARE @CategoryIdsCount int SET @CategoryIdsCount = (SELECT COUNT(1) FROM #FilteredCategoryIds) --filter by attributes SET @FilteredSpecs = isnull(@FilteredSpecs, '') CREATE TABLE #FilteredSpecs ( SpecificationAttributeOptionId int not null ) INSERT INTO #FilteredSpecs (SpecificationAttributeOptionId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@FilteredSpecs, ',') DECLARE @SpecAttributesCount int SET @SpecAttributesCount = (SELECT COUNT(1) FROM #FilteredSpecs) --filter by customer role IDs (access control list) SET @AllowedCustomerRoleIds = isnull(@AllowedCustomerRoleIds, '') CREATE TABLE #FilteredCustomerRoleIds ( CustomerRoleId int not null ) INSERT INTO #FilteredCustomerRoleIds (CustomerRoleId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@AllowedCustomerRoleIds, ',') --paging DECLARE @PageLowerBound int DECLARE @PageUpperBound int DECLARE @RowsToReturn int SET @RowsToReturn = @PageSize * (@PageIndex + 1) SET @PageLowerBound = @PageSize * @PageIndex SET @PageUpperBound = @PageLowerBound + @PageSize + 1 CREATE TABLE #DisplayOrderTmp ( [Id] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) SET @sql = ' INSERT INTO #DisplayOrderTmp ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK)' IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Category_Mapping pcm with (NOLOCK) ON p.Id = pcm.ProductId' END IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Manufacturer_Mapping pmm with (NOLOCK) ON p.Id = pmm.ProductId' END IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_ProductTag_Mapping pptm with (NOLOCK) ON p.Id = pptm.Product_Id' END IF @ShowHidden = 0 OR @PriceMin > 0 OR @PriceMax > 0 OR @OrderBy = 10 /* Price: Low to High */ OR @OrderBy = 11 /* Price: High to Low */ BEGIN SET @sql = @sql + ' LEFT JOIN ProductVariant pv with (NOLOCK) ON p.Id = pv.ProductId' END --searching by keywords IF @SearchKeywords = 1 BEGIN SET @sql = @sql + ' JOIN #KeywordProducts kp ON p.Id = kp.ProductId' END SET @sql = @sql + ' WHERE p.Deleted = 0' --filter by category IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' AND pcm.CategoryId IN (SELECT CategoryId FROM #FilteredCategoryIds)' IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pcm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by manufacturer IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' AND pmm.ManufacturerId = ' + CAST(@ManufacturerId AS nvarchar(max)) IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pmm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by vendor IF @VendorId > 0 BEGIN SET @sql = @sql + ' AND p.VendorId = ' + CAST(@VendorId AS nvarchar(max)) END --filter by product tag IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' AND pptm.ProductTag_Id = ' + CAST(@ProductTagId AS nvarchar(max)) END --show hidden IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND p.Published = 1 AND pv.Published = 1 AND pv.Deleted = 0 AND (getutcdate() BETWEEN ISNULL(pv.AvailableStartDateTimeUtc, ''1/1/1900'') and ISNULL(pv.AvailableEndDateTimeUtc, ''1/1/2999''))' END --min price IF @PriceMin > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (pv.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(pv.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(pv.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (pv.SpecialPrice >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (pv.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(pv.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(pv.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (pv.Price >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) )' END --max price IF @PriceMax > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (pv.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(pv.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(pv.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (pv.SpecialPrice <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (pv.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(pv.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(pv.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (pv.Price <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) )' END --show hidden and ACL IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND (p.SubjectToAcl = 0 OR EXISTS ( SELECT 1 FROM #FilteredCustomerRoleIds [fcr] WHERE [fcr].CustomerRoleId IN ( SELECT [acl].CustomerRoleId FROM [AclRecord] acl with (NOLOCK) WHERE [acl].EntityId = p.Id AND [acl].EntityName = ''Product'' ) ))' END --show hidden and filter by store IF @StoreId > 0 BEGIN SET @sql = @sql + ' AND (p.LimitedToStores = 0 OR EXISTS ( SELECT 1 FROM [StoreMapping] sm with (NOLOCK) WHERE [sm].EntityId = p.Id AND [sm].EntityName = ''Product'' and [sm].StoreId=' + CAST(@StoreId AS nvarchar(max)) + ' ))' END --filter by specs IF @SpecAttributesCount > 0 BEGIN SET @sql = @sql + ' AND NOT EXISTS ( SELECT 1 FROM #FilteredSpecs [fs] WHERE [fs].SpecificationAttributeOptionId NOT IN ( SELECT psam.SpecificationAttributeOptionId FROM Product_SpecificationAttribute_Mapping psam with (NOLOCK) WHERE psam.AllowFiltering = 1 AND psam.ProductId = p.Id ) )' END --sorting SET @sql_orderby = '' IF @OrderBy = 5 /* Name: A to Z */ SET @sql_orderby = ' p.[Name] ASC' ELSE IF @OrderBy = 6 /* Name: Z to A */ SET @sql_orderby = ' p.[Name] DESC' ELSE IF @OrderBy = 10 /* Price: Low to High */ SET @sql_orderby = ' pv.[Price] ASC' ELSE IF @OrderBy = 11 /* Price: High to Low */ SET @sql_orderby = ' pv.[Price] DESC' ELSE IF @OrderBy = 15 /* creation date */ SET @sql_orderby = ' p.[CreatedOnUtc] DESC' ELSE /* default sorting, 0 (position) */ BEGIN --category position (display order) IF @CategoryIdsCount > 0 SET @sql_orderby = ' pcm.DisplayOrder ASC' --manufacturer position (display order) IF @ManufacturerId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' pmm.DisplayOrder ASC' END --name IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[Name] ASC' END SET @sql = @sql + ' ORDER BY' + @sql_orderby --PRINT (@sql) EXEC sp_executesql @sql DROP TABLE #FilteredCategoryIds DROP TABLE #FilteredSpecs DROP TABLE #FilteredCustomerRoleIds DROP TABLE #KeywordProducts CREATE TABLE #PageIndex ( [IndexId] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) INSERT INTO #PageIndex ([ProductId]) SELECT ProductId FROM #DisplayOrderTmp GROUP BY ProductId ORDER BY min([Id]) --total records SET @TotalRecords = @@rowcount DROP TABLE #DisplayOrderTmp --prepare filterable specification attribute option identifier (if requested) IF @LoadFilterableSpecificationAttributeOptionIds = 1 BEGIN CREATE TABLE #FilterableSpecs ( [SpecificationAttributeOptionId] int NOT NULL ) INSERT INTO #FilterableSpecs ([SpecificationAttributeOptionId]) SELECT DISTINCT [psam].SpecificationAttributeOptionId FROM [Product_SpecificationAttribute_Mapping] [psam] with (NOLOCK) WHERE [psam].[AllowFiltering] = 1 AND [psam].[ProductId] IN (SELECT [pi].ProductId FROM #PageIndex [pi]) --build comma separated list of filterable identifiers SELECT @FilterableSpecificationAttributeOptionIds = COALESCE(@FilterableSpecificationAttributeOptionIds + ',' , '') + CAST(SpecificationAttributeOptionId as nvarchar(4000)) FROM #FilterableSpecs DROP TABLE #FilterableSpecs END --return products SELECT TOP (@RowsToReturn) p.* FROM #PageIndex [pi] INNER JOIN Product p with (NOLOCK) on p.Id = [pi].[ProductId] WHERE [pi].IndexId > @PageLowerBound AND [pi].IndexId < @PageUpperBound ORDER BY [pi].IndexId DROP TABLE #PageIndex END GO --remove obsolete setting DELETE FROM [Setting] WHERE [name] = N'SecuritySettings.HideAdminMenuItemsBasedOnPermissions' GO --new column IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[TierPrice]') and NAME='StoreId') BEGIN ALTER TABLE [TierPrice] ADD [StoreId] int NULL END GO UPDATE [TierPrice] SET [StoreId] = 0 WHERE [StoreId] IS NULL GO ALTER TABLE [TierPrice] ALTER COLUMN [StoreId] int NOT NULL GO --shipping by weight plugin IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ShippingByWeight]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN --new [StoreId] column EXEC ('IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id(''[ShippingByWeight]'') and NAME=''StoreId'') BEGIN ALTER TABLE [ShippingByWeight] ADD [StoreId] int NULL exec(''UPDATE [ShippingByWeight] SET [StoreId] = 0'') EXEC (''ALTER TABLE [ShippingByWeight] ALTER COLUMN [StoreId] int NOT NULL'') END') END GO --rename ShipmentOrderProductVariant to ShipmentItem IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[Shipment_OrderProductVariant]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN EXEC sp_rename 'Shipment_OrderProductVariant', 'ShipmentItem'; END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ShipmentOrderProductVariant_Shipment' AND parent_object_id = Object_id('ShipmentItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN EXEC sp_rename 'ShipmentOrderProductVariant_Shipment', 'ShipmentItem_Shipment'; END GO --rename OrderProductVariant to OrderItem IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[GiftCard]') and NAME='PurchasedWithOrderProductVariantId') BEGIN EXEC sp_rename 'GiftCard.PurchasedWithOrderProductVariantId', 'PurchasedWithOrderItemId', 'COLUMN'; END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'GiftCard_PurchasedWithOrderProductVariant' AND parent_object_id = Object_id('GiftCard') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN EXEC sp_rename 'GiftCard_PurchasedWithOrderProductVariant', 'GiftCard_PurchasedWithOrderItem'; END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[OrderProductVariant]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN EXEC sp_rename 'OrderProductVariant', 'OrderItem'; END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'OrderProductVariant_Order' AND parent_object_id = Object_id('OrderItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN EXEC sp_rename 'OrderProductVariant_Order', 'OrderItem_Order'; END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'OrderProductVariant_ProductVariant' AND parent_object_id = Object_id('OrderItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN EXEC sp_rename 'OrderProductVariant_ProductVariant', 'OrderItem_ProductVariant'; END GO IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_OrderProductVariant_OrderId' and object_id=object_id(N'[OrderItem]')) BEGIN EXEC sp_rename 'OrderItem.IX_OrderProductVariant_OrderId', 'IX_OrderItem_OrderId', 'INDEX'; END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ReturnRequest]') and NAME='OrderProductVariantId') BEGIN EXEC sp_rename 'ReturnRequest.OrderProductVariantId', 'OrderItemId', 'COLUMN'; END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ShipmentItem]') and NAME='OrderProductVariantId') BEGIN EXEC sp_rename 'ShipmentItem.OrderProductVariantId', 'OrderItemId', 'COLUMN'; END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='OrderProductVariantGuid') BEGIN EXEC sp_rename 'OrderItem.OrderProductVariantGuid', 'OrderItemGuid', 'COLUMN'; END GO --revise product/product variant logic DELETE FROM [ActivityLogType] WHERE [SystemKeyword] = N'AddNewProductVariant' GO DELETE FROM [ActivityLogType] WHERE [SystemKeyword] = N'DeleteProductVariant' GO DELETE FROM [ActivityLogType] WHERE [SystemKeyword] = N'EditProductVariant' GO IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductLoadAllPaged]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [ProductLoadAllPaged] GO CREATE PROCEDURE [dbo].[ProductLoadAllPaged] ( @CategoryIds nvarchar(MAX) = null, --a list of category IDs (comma-separated list). e.g. 1,2,3 @ManufacturerId int = 0, @StoreId int = 0, @VendorId int = 0, @ParentProductId int = 0, @ProductTagId int = 0, @FeaturedProducts bit = null, --0 featured only , 1 not featured only, null - load all products @PriceMin decimal(18, 4) = null, @PriceMax decimal(18, 4) = null, @Keywords nvarchar(4000) = null, @SearchDescriptions bit = 0, --a value indicating whether to search by a specified "keyword" in product descriptions @SearchProductTags bit = 0, --a value indicating whether to search by a specified "keyword" in product tags @UseFullTextSearch bit = 0, @FullTextMode int = 0, --0 using CONTAINS with , 5 - using CONTAINS and OR with , 10 - using CONTAINS and AND with @FilteredSpecs nvarchar(MAX) = null, --filter by attributes (comma-separated list). e.g. 14,15,16 @LanguageId int = 0, @OrderBy int = 0, --0 position, 5 - Name: A to Z, 6 - Name: Z to A, 10 - Price: Low to High, 11 - Price: High to Low, 15 - creation date @AllowedCustomerRoleIds nvarchar(MAX) = null, --a list of customer role IDs (comma-separated list) for which a product should be shown (if a subjet to ACL) @PageIndex int = 0, @PageSize int = 2147483644, @ShowHidden bit = 0, @LoadFilterableSpecificationAttributeOptionIds bit = 0, --a value indicating whether we should load the specification attribute option identifiers applied to loaded products (all pages) @FilterableSpecificationAttributeOptionIds nvarchar(MAX) = null OUTPUT, --the specification attribute option identifiers applied to loaded products (all pages). returned as a comma separated list of identifiers @TotalRecords int = null OUTPUT ) AS BEGIN /* Products that filtered by keywords */ CREATE TABLE #KeywordProducts ( [ProductId] int NOT NULL ) DECLARE @SearchKeywords bit, @sql nvarchar(max), @sql_orderby nvarchar(max) SET NOCOUNT ON --filter by keywords SET @Keywords = isnull(@Keywords, '') SET @Keywords = rtrim(ltrim(@Keywords)) IF ISNULL(@Keywords, '') != '' BEGIN SET @SearchKeywords = 1 IF @UseFullTextSearch = 1 BEGIN --remove wrong chars (' ") SET @Keywords = REPLACE(@Keywords, '''', '') SET @Keywords = REPLACE(@Keywords, '"', '') --full-text search IF @FullTextMode = 0 BEGIN --0 - using CONTAINS with SET @Keywords = ' "' + @Keywords + '*" ' END ELSE BEGIN --5 - using CONTAINS and OR with --10 - using CONTAINS and AND with --clean multiple spaces WHILE CHARINDEX(' ', @Keywords) > 0 SET @Keywords = REPLACE(@Keywords, ' ', ' ') DECLARE @concat_term nvarchar(100) IF @FullTextMode = 5 --5 - using CONTAINS and OR with BEGIN SET @concat_term = 'OR' END IF @FullTextMode = 10 --10 - using CONTAINS and AND with BEGIN SET @concat_term = 'AND' END --now let's build search string declare @fulltext_keywords nvarchar(4000) set @fulltext_keywords = N'' declare @index int set @index = CHARINDEX(' ', @Keywords, 0) -- if index = 0, then only one field was passed IF(@index = 0) set @fulltext_keywords = ' "' + @Keywords + '*" ' ELSE BEGIN DECLARE @first BIT SET @first = 1 WHILE @index > 0 BEGIN IF (@first = 0) SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' ELSE SET @first = 0 SET @fulltext_keywords = @fulltext_keywords + '"' + SUBSTRING(@Keywords, 1, @index - 1) + '*"' SET @Keywords = SUBSTRING(@Keywords, @index + 1, LEN(@Keywords) - @index) SET @index = CHARINDEX(' ', @Keywords, 0) end -- add the last field IF LEN(@fulltext_keywords) > 0 SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' + '"' + SUBSTRING(@Keywords, 1, LEN(@Keywords)) + '*"' END SET @Keywords = @fulltext_keywords END END ELSE BEGIN --usual search by PATINDEX SET @Keywords = '%' + @Keywords + '%' END --PRINT @Keywords --product name SET @sql = ' INSERT INTO #KeywordProducts ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Name]) > 0 ' --SKU SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Sku], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Sku]) > 0 ' --localized product name SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' IF @SearchDescriptions = 1 BEGIN --product short description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[ShortDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[ShortDescription]) > 0 ' --product full description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[FullDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[FullDescription]) > 0 ' --localized product short description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''ShortDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' --localized product full description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''FullDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END IF @SearchProductTags = 1 BEGIN --product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM Product_ProductTag_Mapping pptm with(NOLOCK) INNER JOIN ProductTag pt with(NOLOCK) ON pt.Id = pptm.ProductTag_Id WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pt.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pt.[Name]) > 0 ' --localized product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM LocalizedProperty lp with (NOLOCK) INNER JOIN Product_ProductTag_Mapping pptm with(NOLOCK) ON lp.EntityId = pptm.ProductTag_Id WHERE lp.LocaleKeyGroup = N''ProductTag'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END --PRINT (@sql) EXEC sp_executesql @sql, N'@Keywords nvarchar(4000)', @Keywords END ELSE BEGIN SET @SearchKeywords = 0 END --filter by category IDs SET @CategoryIds = isnull(@CategoryIds, '') CREATE TABLE #FilteredCategoryIds ( CategoryId int not null ) INSERT INTO #FilteredCategoryIds (CategoryId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@CategoryIds, ',') DECLARE @CategoryIdsCount int SET @CategoryIdsCount = (SELECT COUNT(1) FROM #FilteredCategoryIds) --filter by attributes SET @FilteredSpecs = isnull(@FilteredSpecs, '') CREATE TABLE #FilteredSpecs ( SpecificationAttributeOptionId int not null ) INSERT INTO #FilteredSpecs (SpecificationAttributeOptionId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@FilteredSpecs, ',') DECLARE @SpecAttributesCount int SET @SpecAttributesCount = (SELECT COUNT(1) FROM #FilteredSpecs) --filter by customer role IDs (access control list) SET @AllowedCustomerRoleIds = isnull(@AllowedCustomerRoleIds, '') CREATE TABLE #FilteredCustomerRoleIds ( CustomerRoleId int not null ) INSERT INTO #FilteredCustomerRoleIds (CustomerRoleId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@AllowedCustomerRoleIds, ',') --paging DECLARE @PageLowerBound int DECLARE @PageUpperBound int DECLARE @RowsToReturn int SET @RowsToReturn = @PageSize * (@PageIndex + 1) SET @PageLowerBound = @PageSize * @PageIndex SET @PageUpperBound = @PageLowerBound + @PageSize + 1 CREATE TABLE #DisplayOrderTmp ( [Id] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) SET @sql = ' INSERT INTO #DisplayOrderTmp ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK)' IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Category_Mapping pcm with (NOLOCK) ON p.Id = pcm.ProductId' END IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Manufacturer_Mapping pmm with (NOLOCK) ON p.Id = pmm.ProductId' END IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_ProductTag_Mapping pptm with (NOLOCK) ON p.Id = pptm.Product_Id' END --searching by keywords IF @SearchKeywords = 1 BEGIN SET @sql = @sql + ' JOIN #KeywordProducts kp ON p.Id = kp.ProductId' END SET @sql = @sql + ' WHERE p.Deleted = 0' --filter by category IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' AND pcm.CategoryId IN (SELECT CategoryId FROM #FilteredCategoryIds)' IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pcm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by manufacturer IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' AND pmm.ManufacturerId = ' + CAST(@ManufacturerId AS nvarchar(max)) IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pmm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by vendor IF @VendorId > 0 BEGIN SET @sql = @sql + ' AND p.VendorId = ' + CAST(@VendorId AS nvarchar(max)) END --filter by parent product identifer IF @ParentProductId > 0 BEGIN SET @sql = @sql + ' AND p.ParentProductId = ' + CAST(@ParentProductId AS nvarchar(max)) END --filter by product tag IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' AND pptm.ProductTag_Id = ' + CAST(@ProductTagId AS nvarchar(max)) END --show hidden IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND p.Published = 1 AND p.Deleted = 0 AND (getutcdate() BETWEEN ISNULL(p.AvailableStartDateTimeUtc, ''1/1/1900'') and ISNULL(p.AvailableEndDateTimeUtc, ''1/1/2999''))' END --min price IF @PriceMin > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) )' END --max price IF @PriceMax > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) )' END --show hidden and ACL IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND (p.SubjectToAcl = 0 OR EXISTS ( SELECT 1 FROM #FilteredCustomerRoleIds [fcr] WHERE [fcr].CustomerRoleId IN ( SELECT [acl].CustomerRoleId FROM [AclRecord] acl with (NOLOCK) WHERE [acl].EntityId = p.Id AND [acl].EntityName = ''Product'' ) ))' END --show hidden and filter by store IF @StoreId > 0 BEGIN SET @sql = @sql + ' AND (p.LimitedToStores = 0 OR EXISTS ( SELECT 1 FROM [StoreMapping] sm with (NOLOCK) WHERE [sm].EntityId = p.Id AND [sm].EntityName = ''Product'' and [sm].StoreId=' + CAST(@StoreId AS nvarchar(max)) + ' ))' END --filter by specs IF @SpecAttributesCount > 0 BEGIN SET @sql = @sql + ' AND NOT EXISTS ( SELECT 1 FROM #FilteredSpecs [fs] WHERE [fs].SpecificationAttributeOptionId NOT IN ( SELECT psam.SpecificationAttributeOptionId FROM Product_SpecificationAttribute_Mapping psam with (NOLOCK) WHERE psam.AllowFiltering = 1 AND psam.ProductId = p.Id ) )' END --sorting SET @sql_orderby = '' IF @OrderBy = 5 /* Name: A to Z */ SET @sql_orderby = ' p.[Name] ASC' ELSE IF @OrderBy = 6 /* Name: Z to A */ SET @sql_orderby = ' p.[Name] DESC' ELSE IF @OrderBy = 10 /* Price: Low to High */ SET @sql_orderby = ' p.[Price] ASC' ELSE IF @OrderBy = 11 /* Price: High to Low */ SET @sql_orderby = ' p.[Price] DESC' ELSE IF @OrderBy = 15 /* creation date */ SET @sql_orderby = ' p.[CreatedOnUtc] DESC' ELSE /* default sorting, 0 (position) */ BEGIN --category position (display order) IF @CategoryIdsCount > 0 SET @sql_orderby = ' pcm.DisplayOrder ASC' --manufacturer position (display order) IF @ManufacturerId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' pmm.DisplayOrder ASC' END --name IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[Name] ASC' END SET @sql = @sql + ' ORDER BY' + @sql_orderby --PRINT (@sql) EXEC sp_executesql @sql DROP TABLE #FilteredCategoryIds DROP TABLE #FilteredSpecs DROP TABLE #FilteredCustomerRoleIds DROP TABLE #KeywordProducts CREATE TABLE #PageIndex ( [IndexId] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) INSERT INTO #PageIndex ([ProductId]) SELECT ProductId FROM #DisplayOrderTmp GROUP BY ProductId ORDER BY min([Id]) --total records SET @TotalRecords = @@rowcount DROP TABLE #DisplayOrderTmp --prepare filterable specification attribute option identifier (if requested) IF @LoadFilterableSpecificationAttributeOptionIds = 1 BEGIN CREATE TABLE #FilterableSpecs ( [SpecificationAttributeOptionId] int NOT NULL ) INSERT INTO #FilterableSpecs ([SpecificationAttributeOptionId]) SELECT DISTINCT [psam].SpecificationAttributeOptionId FROM [Product_SpecificationAttribute_Mapping] [psam] with (NOLOCK) WHERE [psam].[AllowFiltering] = 1 AND [psam].[ProductId] IN (SELECT [pi].ProductId FROM #PageIndex [pi]) --build comma separated list of filterable identifiers SELECT @FilterableSpecificationAttributeOptionIds = COALESCE(@FilterableSpecificationAttributeOptionIds + ',' , '') + CAST(SpecificationAttributeOptionId as nvarchar(4000)) FROM #FilterableSpecs DROP TABLE #FilterableSpecs END --return products SELECT TOP (@RowsToReturn) p.* FROM #PageIndex [pi] INNER JOIN Product p with (NOLOCK) on p.Id = [pi].[ProductId] WHERE [pi].IndexId > @PageLowerBound AND [pi].IndexId < @PageUpperBound ORDER BY [pi].IndexId DROP TABLE #PageIndex END GO --remove obsolete setting DELETE FROM [Setting] WHERE [name] = N'MediaSettings.ProductVariantPictureSize' GO IF NOT EXISTS (SELECT 1 FROM [Setting] WHERE [name] = N'mediasettings.associatedproductpicturesize') BEGIN INSERT [Setting] ([Name], [Value], [StoreId]) VALUES (N'mediasettings.associatedproductpicturesize', N'125', 0) END GO --update some message template tokens UPDATE [MessageTemplate] SET [Subject] = REPLACE([Subject], 'ProductVariant.ID', 'Product.ID'), [Body] = REPLACE([Body], 'ProductVariant.ID', 'Product.ID') GO UPDATE [MessageTemplate] SET [Subject] = REPLACE([Subject], 'ProductVariant.FullProductName', 'Product.Name'), [Body] = REPLACE([Body], 'ProductVariant.FullProductName', 'Product.Name') GO UPDATE [MessageTemplate] SET [Subject] = REPLACE([Subject], 'ProductVariant.StockQuantity', 'Product.StockQuantity'), [Body] = REPLACE([Body], 'ProductVariant.StockQuantity', 'Product.StockQuantity') GO --new setting IF NOT EXISTS (SELECT 1 FROM [Setting] WHERE [name] = N'mediasettings.ProductThumbPerRowOnProductDetailsPage') BEGIN INSERT [Setting] ([Name], [Value], [StoreId]) VALUES (N'mediasettings.ProductThumbPerRowOnProductDetailsPage', N'4', 0) END GO --update product templates UPDATE [ProductTemplate] SET [Name] = N'Grouped product', [ViewPath] = N'ProductTemplate.Grouped', [DisplayOrder] = 100 WHERE [ViewPath] = N'ProductTemplate.VariantsInGrid' GO UPDATE [ProductTemplate] SET [Name] = N'Simple product', [ViewPath] = N'ProductTemplate.Simple', [DisplayOrder] = 10 WHERE [ViewPath] = N'ProductTemplate.SingleVariant' GO IF (NOT EXISTS(SELECT 1 FROM [ProductTemplate] WHERE [ViewPath] = N'ProductTemplate.Grouped')) BEGIN INSERT INTO [ProductTemplate] ([Name],[ViewPath],[DisplayOrder]) VALUES (N'Grouped product',N'ProductTemplate.Grouped',100) END GO IF (NOT EXISTS(SELECT 1 FROM [ProductTemplate] WHERE [ViewPath] = N'ProductTemplate.Simple')) BEGIN INSERT INTO [ProductTemplate] ([Name],[ViewPath],[DisplayOrder]) VALUES (N'Simple product',N'ProductTemplate.Simple',10) END GO --delete products without variants IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductVariant]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN DELETE FROM [Product] WHERE [Id] NOT IN (SELECT [ProductId] FROM [ProductVariant]) END GO --move records from Product to ProductVariant IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ProductTypeId') BEGIN ALTER TABLE [Product] ADD [ProductTypeId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentGroupedProductId') BEGIN ALTER TABLE [Product] ADD [ParentProductId] int NULL END END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='SKU') BEGIN ALTER TABLE [Product] ADD [SKU] nvarchar(400) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ManufacturerPartNumber') BEGIN ALTER TABLE [Product] ADD [ManufacturerPartNumber] nvarchar(400) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Gtin') BEGIN ALTER TABLE [Product] ADD [Gtin] nvarchar(400) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsGiftCard') BEGIN ALTER TABLE [Product] ADD [IsGiftCard] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='GiftCardTypeId') BEGIN ALTER TABLE [Product] ADD [GiftCardTypeId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='RequireOtherProducts') BEGIN ALTER TABLE [Product] ADD [RequireOtherProducts] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='RequiredProductIds') BEGIN ALTER TABLE [Product] ADD [RequiredProductIds] nvarchar(1000) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AutomaticallyAddRequiredProducts') BEGIN ALTER TABLE [Product] ADD [AutomaticallyAddRequiredProducts] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsDownload') BEGIN ALTER TABLE [Product] ADD [IsDownload] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DownloadId') BEGIN ALTER TABLE [Product] ADD [DownloadId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='UnlimitedDownloads') BEGIN ALTER TABLE [Product] ADD [UnlimitedDownloads] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='MaxNumberOfDownloads') BEGIN ALTER TABLE [Product] ADD [MaxNumberOfDownloads] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DownloadExpirationDays') BEGIN ALTER TABLE [Product] ADD [DownloadExpirationDays] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DownloadActivationTypeId') BEGIN ALTER TABLE [Product] ADD [DownloadActivationTypeId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='HasSampleDownload') BEGIN ALTER TABLE [Product] ADD [HasSampleDownload] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='SampleDownloadId') BEGIN ALTER TABLE [Product] ADD [SampleDownloadId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='HasUserAgreement') BEGIN ALTER TABLE [Product] ADD [HasUserAgreement] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='UserAgreementText') BEGIN ALTER TABLE [Product] ADD [UserAgreementText] nvarchar(MAX) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsRecurring') BEGIN ALTER TABLE [Product] ADD [IsRecurring] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='RecurringCycleLength') BEGIN ALTER TABLE [Product] ADD [RecurringCycleLength] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='RecurringCyclePeriodId') BEGIN ALTER TABLE [Product] ADD [RecurringCyclePeriodId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='RecurringTotalCycles') BEGIN ALTER TABLE [Product] ADD [RecurringTotalCycles] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsShipEnabled') BEGIN ALTER TABLE [Product] ADD [IsShipEnabled] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsFreeShipping') BEGIN ALTER TABLE [Product] ADD [IsFreeShipping] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AdditionalShippingCharge') BEGIN ALTER TABLE [Product] ADD [AdditionalShippingCharge] decimal(18,4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='IsTaxExempt') BEGIN ALTER TABLE [Product] ADD [IsTaxExempt] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='TaxCategoryId') BEGIN ALTER TABLE [Product] ADD [TaxCategoryId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ManageInventoryMethodId') BEGIN ALTER TABLE [Product] ADD [ManageInventoryMethodId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='StockQuantity') BEGIN ALTER TABLE [Product] ADD [StockQuantity] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DisplayStockAvailability') BEGIN ALTER TABLE [Product] ADD [DisplayStockAvailability] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DisplayStockQuantity') BEGIN ALTER TABLE [Product] ADD [DisplayStockQuantity] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='MinStockQuantity') BEGIN ALTER TABLE [Product] ADD [MinStockQuantity] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='LowStockActivityId') BEGIN ALTER TABLE [Product] ADD [LowStockActivityId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='NotifyAdminForQuantityBelow') BEGIN ALTER TABLE [Product] ADD [NotifyAdminForQuantityBelow] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='BackorderModeId') BEGIN ALTER TABLE [Product] ADD [BackorderModeId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AllowBackInStockSubscriptions') BEGIN ALTER TABLE [Product] ADD [AllowBackInStockSubscriptions] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='OrderMinimumQuantity') BEGIN ALTER TABLE [Product] ADD [OrderMinimumQuantity] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='OrderMaximumQuantity') BEGIN ALTER TABLE [Product] ADD [OrderMaximumQuantity] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AllowedQuantities') BEGIN ALTER TABLE [Product] ADD [AllowedQuantities] nvarchar(1000) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DisableBuyButton') BEGIN ALTER TABLE [Product] ADD [DisableBuyButton] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DisableWishlistButton') BEGIN ALTER TABLE [Product] ADD [DisableWishlistButton] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AvailableForPreOrder') BEGIN ALTER TABLE [Product] ADD [AvailableForPreOrder] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='CallForPrice') BEGIN ALTER TABLE [Product] ADD [CallForPrice] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Price') BEGIN ALTER TABLE [Product] ADD [Price] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='OldPrice') BEGIN ALTER TABLE [Product] ADD [OldPrice] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ProductCost') BEGIN ALTER TABLE [Product] ADD [ProductCost] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='SpecialPrice') BEGIN ALTER TABLE [Product] ADD [SpecialPrice] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='SpecialPriceStartDateTimeUtc') BEGIN ALTER TABLE [Product] ADD [SpecialPriceStartDateTimeUtc] datetime NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='SpecialPriceEndDateTimeUtc') BEGIN ALTER TABLE [Product] ADD [SpecialPriceEndDateTimeUtc] datetime NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='CustomerEntersPrice') BEGIN ALTER TABLE [Product] ADD [CustomerEntersPrice] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='MinimumCustomerEnteredPrice') BEGIN ALTER TABLE [Product] ADD [MinimumCustomerEnteredPrice] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='MaximumCustomerEnteredPrice') BEGIN ALTER TABLE [Product] ADD [MaximumCustomerEnteredPrice] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='HasTierPrices') BEGIN ALTER TABLE [Product] ADD [HasTierPrices] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='HasDiscountsApplied') BEGIN ALTER TABLE [Product] ADD [HasDiscountsApplied] bit NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Weight') BEGIN ALTER TABLE [Product] ADD [Weight] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Length') BEGIN ALTER TABLE [Product] ADD [Length] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Width') BEGIN ALTER TABLE [Product] ADD [Width] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='Height') BEGIN ALTER TABLE [Product] ADD [Height] decimal(18, 4) NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AvailableStartDateTimeUtc') BEGIN ALTER TABLE [Product] ADD [AvailableStartDateTimeUtc] datetime NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='AvailableEndDateTimeUtc') BEGIN ALTER TABLE [Product] ADD [AvailableEndDateTimeUtc] datetime NULL END GO --remove old product variant references IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'BackInStockSubscription_ProductVariant' AND parent_object_id = Object_id('BackInStockSubscription') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[BackInStockSubscription] DROP CONSTRAINT BackInStockSubscription_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'OrderItem_ProductVariant' AND parent_object_id = Object_id('OrderItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[OrderItem] DROP CONSTRAINT OrderItem_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ProductVariantAttribute_ProductVariant' AND parent_object_id = Object_id('ProductVariant_ProductAttribute_Mapping') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[ProductVariant_ProductAttribute_Mapping] DROP CONSTRAINT ProductVariantAttribute_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ProductVariantAttributeCombination_ProductVariant' AND parent_object_id = Object_id('ProductVariantAttributeCombination') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[ProductVariantAttributeCombination] DROP CONSTRAINT ProductVariantAttributeCombination_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ShoppingCartItem_ProductVariant' AND parent_object_id = Object_id('ShoppingCartItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[ShoppingCartItem] DROP CONSTRAINT ShoppingCartItem_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'TierPrice_ProductVariant' AND parent_object_id = Object_id('TierPrice') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[TierPrice] DROP CONSTRAINT TierPrice_ProductVariant END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE name = 'Discount_AppliedToProductVariants_Target' AND parent_object_id = Object_id('Discount_AppliedToProductVariants') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE dbo.[Discount_AppliedToProductVariants] DROP CONSTRAINT Discount_AppliedToProductVariants_Target END GO --new ProductId columns in references tables IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[BackInStockSubscription]') and NAME='ProductId') BEGIN ALTER TABLE [BackInStockSubscription] ADD [ProductId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='ProductId') BEGIN ALTER TABLE [OrderItem] ADD [ProductId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariant_ProductAttribute_Mapping]') and NAME='ProductId') BEGIN --one more validatation here because we'll rename [ProductVariant_ProductAttribute_Mapping] table a bit later IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[Product_ProductAttribute_Mapping]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN ALTER TABLE [ProductVariant_ProductAttribute_Mapping] ADD [ProductId] int NULL END END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeCombination]') and NAME='ProductId') BEGIN ALTER TABLE [ProductVariantAttributeCombination] ADD [ProductId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ShoppingCartItem]') and NAME='ProductId') BEGIN ALTER TABLE [ShoppingCartItem] ADD [ProductId] int NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[TierPrice]') and NAME='ProductId') BEGIN ALTER TABLE [TierPrice] ADD [ProductId] int NULL END GO --new table for discount <=> product mapping (have some issue with just adding and renaming columns) IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[Discount_AppliedToProducts]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN CREATE TABLE [dbo].[Discount_AppliedToProducts]( [Discount_Id] [int] NOT NULL, [Product_Id] [int] NOT NULL, [ProductVariant_Id] [int] NOT NULL, PRIMARY KEY CLUSTERED ( [Discount_Id] ASC, [Product_Id] ASC ) ) --copy records DECLARE @ExistingDiscountID int DECLARE @ExistingDiscountProductVariantID int DECLARE cur_existingdiscountmapping CURSOR FOR SELECT [Discount_Id], [ProductVariant_Id] FROM [Discount_AppliedToProductVariants] OPEN cur_existingdiscountmapping FETCH NEXT FROM cur_existingdiscountmapping INTO @ExistingDiscountID,@ExistingDiscountProductVariantID WHILE @@FETCH_STATUS = 0 BEGIN EXEC sp_executesql N'INSERT INTO [Discount_AppliedToProducts] ([Discount_Id], [Product_Id], [ProductVariant_Id]) VALUES (@ExistingDiscountID, @ExistingDiscountProductVariantID, @ExistingDiscountProductVariantID)', N'@ExistingDiscountID int, @ExistingDiscountProductVariantID int', @ExistingDiscountID, @ExistingDiscountProductVariantID --fetch next identifier FETCH NEXT FROM cur_existingdiscountmapping INTO @ExistingDiscountID,@ExistingDiscountProductVariantID END CLOSE cur_existingdiscountmapping DEALLOCATE cur_existingdiscountmapping --drop old table DROP TABLE [Discount_AppliedToProductVariants] END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductVariant]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN DECLARE @ExistingProductVariantID int DECLARE cur_existingproductvariant CURSOR FOR SELECT [ID] FROM [ProductVariant] OPEN cur_existingproductvariant FETCH NEXT FROM cur_existingproductvariant INTO @ExistingProductVariantID WHILE @@FETCH_STATUS = 0 BEGIN DECLARE @ProductId int SET @ProductId = null -- clear cache (variable scope) DECLARE @Name nvarchar(400) SET @Name = null -- clear cache (variable scope) DECLARE @Description nvarchar(MAX) SET @Description = null -- clear cache (variable scope) DECLARE @Sku nvarchar(400) SET @Sku = null -- clear cache (variable scope) DECLARE @ManufacturerPartNumber nvarchar(400) SET @ManufacturerPartNumber = null -- clear cache (variable scope) DECLARE @Gtin nvarchar(400) SET @Gtin = null -- clear cache (variable scope) DECLARE @IsGiftCard bit SET @IsGiftCard = null -- clear cache (variable scope) DECLARE @GiftCardTypeId int SET @GiftCardTypeId = null -- clear cache (variable scope) DECLARE @RequireOtherProducts bit SET @RequireOtherProducts = null -- clear cache (variable scope) DECLARE @RequiredProductIds nvarchar(1000) SET @RequiredProductIds = null -- clear cache (variable scope) DECLARE @AutomaticallyAddRequiredProducts bit SET @AutomaticallyAddRequiredProducts = null -- clear cache (variable scope) DECLARE @IsDownload bit SET @IsDownload = null -- clear cache (variable scope) DECLARE @DownloadId int SET @DownloadId = null -- clear cache (variable scope) DECLARE @UnlimitedDownloads bit SET @UnlimitedDownloads = null -- clear cache (variable scope) DECLARE @MaxNumberOfDownloads int SET @MaxNumberOfDownloads = null -- clear cache (variable scope) DECLARE @DownloadExpirationDays int SET @DownloadExpirationDays = null -- clear cache (variable scope) DECLARE @DownloadActivationTypeId int SET @DownloadActivationTypeId = null -- clear cache (variable scope) DECLARE @HasSampleDownload bit SET @HasSampleDownload = null -- clear cache (variable scope) DECLARE @SampleDownloadId int SET @SampleDownloadId = null -- clear cache (variable scope) DECLARE @HasUserAgreement bit SET @HasUserAgreement = null -- clear cache (variable scope) DECLARE @UserAgreementText nvarchar(MAX) SET @UserAgreementText = null -- clear cache (variable scope) DECLARE @IsRecurring bit SET @IsRecurring = null -- clear cache (variable scope) DECLARE @RecurringCycleLength int SET @RecurringCycleLength = null -- clear cache (variable scope) DECLARE @RecurringCyclePeriodId int SET @RecurringCyclePeriodId = null -- clear cache (variable scope) DECLARE @RecurringTotalCycles int SET @RecurringTotalCycles = null -- clear cache (variable scope) DECLARE @IsShipEnabled bit SET @IsShipEnabled = null -- clear cache (variable scope) DECLARE @IsFreeShipping bit SET @IsFreeShipping = null -- clear cache (variable scope) DECLARE @AdditionalShippingCharge decimal(18,4) SET @AdditionalShippingCharge = null -- clear cache (variable scope) DECLARE @IsTaxExempt bit SET @IsTaxExempt = null -- clear cache (variable scope) DECLARE @TaxCategoryId int SET @TaxCategoryId = null -- clear cache (variable scope) DECLARE @ManageInventoryMethodId int SET @ManageInventoryMethodId = null -- clear cache (variable scope) DECLARE @StockQuantity int SET @StockQuantity = null -- clear cache (variable scope) DECLARE @DisplayStockAvailability bit SET @DisplayStockAvailability = null -- clear cache (variable scope) DECLARE @DisplayStockQuantity bit SET @DisplayStockQuantity = null -- clear cache (variable scope) DECLARE @MinStockQuantity int SET @MinStockQuantity = null -- clear cache (variable scope) DECLARE @LowStockActivityId int SET @LowStockActivityId = null -- clear cache (variable scope) DECLARE @NotifyAdminForQuantityBelow int SET @NotifyAdminForQuantityBelow = null -- clear cache (variable scope) DECLARE @BackorderModeId int SET @BackorderModeId = null -- clear cache (variable scope) DECLARE @AllowBackInStockSubscriptions bit SET @AllowBackInStockSubscriptions = null -- clear cache (variable scope) DECLARE @OrderMinimumQuantity int SET @OrderMinimumQuantity = null -- clear cache (variable scope) DECLARE @OrderMaximumQuantity int SET @OrderMaximumQuantity = null -- clear cache (variable scope) DECLARE @AllowedQuantities nvarchar(1000) SET @AllowedQuantities = null -- clear cache (variable scope) DECLARE @DisableBuyButton bit SET @DisableBuyButton = null -- clear cache (variable scope) DECLARE @DisableWishlistButton bit SET @DisableWishlistButton = null -- clear cache (variable scope) DECLARE @AvailableForPreOrder bit SET @AvailableForPreOrder = null -- clear cache (variable scope) DECLARE @CallForPrice bit SET @CallForPrice = null -- clear cache (variable scope) DECLARE @Price decimal(18,4) SET @Price = null -- clear cache (variable scope) DECLARE @OldPrice decimal(18,4) SET @OldPrice = null -- clear cache (variable scope) DECLARE @ProductCost decimal(18,4) SET @ProductCost = null -- clear cache (variable scope) DECLARE @SpecialPrice decimal(18,4) SET @SpecialPrice = null -- clear cache (variable scope) DECLARE @SpecialPriceStartDateTimeUtc datetime SET @SpecialPriceStartDateTimeUtc = null -- clear cache (variable scope) DECLARE @SpecialPriceEndDateTimeUtc datetime SET @SpecialPriceEndDateTimeUtc = null -- clear cache (variable scope) DECLARE @CustomerEntersPrice bit SET @CustomerEntersPrice = null -- clear cache (variable scope) DECLARE @MinimumCustomerEnteredPrice decimal(18,4) SET @MinimumCustomerEnteredPrice = null -- clear cache (variable scope) DECLARE @MaximumCustomerEnteredPrice decimal(18,4) SET @MaximumCustomerEnteredPrice = null -- clear cache (variable scope) DECLARE @HasTierPrices bit SET @HasTierPrices = null -- clear cache (variable scope) DECLARE @HasDiscountsApplied bit SET @HasDiscountsApplied = null -- clear cache (variable scope) DECLARE @Weight decimal(18, 4) SET @Weight = null -- clear cache (variable scope) DECLARE @Length decimal(18, 4) SET @Length = null -- clear cache (variable scope) DECLARE @Width decimal(18, 4) SET @Width = null -- clear cache (variable scope) DECLARE @Height decimal(18, 4) SET @Height = null -- clear cache (variable scope) DECLARE @PictureId int SET @PictureId = null -- clear cache (variable scope) DECLARE @AvailableStartDateTimeUtc datetime SET @AvailableStartDateTimeUtc = null -- clear cache (variable scope) DECLARE @AvailableEndDateTimeUtc datetime SET @AvailableEndDateTimeUtc = null -- clear cache (variable scope) DECLARE @Published bit SET @Published = null -- clear cache (variable scope) DECLARE @Deleted bit SET @Deleted = null -- clear cache (variable scope) DECLARE @DisplayOrder int SET @DisplayOrder = null -- clear cache (variable scope) DECLARE @CreatedOnUtc datetime SET @CreatedOnUtc = null -- clear cache (variable scope) DECLARE @UpdatedOnUtc datetime SET @UpdatedOnUtc = null -- clear cache (variable scope) DECLARE @sql nvarchar(4000) SET @sql = 'SELECT @ProductId = [ProductId], @Name = [Name], @Description = [Description], @Sku = [Sku], @ManufacturerPartNumber = [ManufacturerPartNumber], @Gtin = [Gtin], @IsGiftCard = [IsGiftCard], @GiftCardTypeId = [GiftCardTypeId], @RequireOtherProducts = [RequireOtherProducts], @RequiredProductIds= [RequiredProductVariantIds], @AutomaticallyAddRequiredProducts = [AutomaticallyAddRequiredProductVariants], @IsDownload = [IsDownload], @DownloadId = [DownloadId], @UnlimitedDownloads = [UnlimitedDownloads], @MaxNumberOfDownloads = [MaxNumberOfDownloads], @DownloadExpirationDays = [DownloadExpirationDays], @DownloadActivationTypeId = [DownloadActivationTypeId], @HasSampleDownload = [HasSampleDownload], @SampleDownloadId = [SampleDownloadId], @HasUserAgreement = [HasUserAgreement], @UserAgreementText = [UserAgreementText], @IsRecurring = [IsRecurring], @RecurringCycleLength = [RecurringCycleLength], @RecurringCyclePeriodId = [RecurringCyclePeriodId], @RecurringTotalCycles = [RecurringTotalCycles], @IsShipEnabled = [IsShipEnabled], @IsFreeShipping = [IsFreeShipping], @AdditionalShippingCharge = [AdditionalShippingCharge], @IsTaxExempt = [IsTaxExempt], @TaxCategoryId = [TaxCategoryId], @ManageInventoryMethodId = [ManageInventoryMethodId], @StockQuantity = [StockQuantity], @DisplayStockAvailability = [DisplayStockAvailability], @DisplayStockQuantity = [DisplayStockQuantity], @MinStockQuantity = [MinStockQuantity], @LowStockActivityId = [LowStockActivityId], @NotifyAdminForQuantityBelow = [NotifyAdminForQuantityBelow], @BackorderModeId = [BackorderModeId], @AllowBackInStockSubscriptions = [AllowBackInStockSubscriptions], @OrderMinimumQuantity = [OrderMinimumQuantity], @OrderMaximumQuantity = [OrderMaximumQuantity], @AllowedQuantities = [AllowedQuantities], @DisableBuyButton = [DisableBuyButton], @DisableWishlistButton = [DisableWishlistButton], @AvailableForPreOrder = [AvailableForPreOrder], @CallForPrice = [CallForPrice], @Price = [Price], @OldPrice = [OldPrice], @ProductCost = [ProductCost], @SpecialPrice = [SpecialPrice], @SpecialPriceStartDateTimeUtc = [SpecialPriceStartDateTimeUtc], @SpecialPriceEndDateTimeUtc = [SpecialPriceEndDateTimeUtc], @CustomerEntersPrice = [CustomerEntersPrice], @MinimumCustomerEnteredPrice = [MinimumCustomerEnteredPrice], @MaximumCustomerEnteredPrice = [MaximumCustomerEnteredPrice], @HasTierPrices = [HasTierPrices], @HasDiscountsApplied = [HasDiscountsApplied], @Weight = [Weight], @Length = [Length], @Width = [Width], @Height = [Height], @PictureId = [PictureId], @AvailableStartDateTimeUtc = [AvailableStartDateTimeUtc], @AvailableEndDateTimeUtc = [AvailableEndDateTimeUtc], @Published = [Published], @Deleted = [Deleted], @DisplayOrder = [DisplayOrder], @CreatedOnUtc = [CreatedOnUtc], @UpdatedOnUtc = [UpdatedOnUtc] FROM [ProductVariant] WHERE [Id]=' + ISNULL(CAST(@ExistingProductVariantID AS nvarchar(max)), '0') EXEC sp_executesql @sql, N'@ProductId int OUTPUT, @Name nvarchar(400) OUTPUT, @Description nvarchar(MAX) OUTPUT, @Sku nvarchar(400) OUTPUT, @ManufacturerPartNumber nvarchar(400) OUTPUT, @Gtin nvarchar(400) OUTPUT, @IsGiftCard bit OUTPUT, @GiftCardTypeId int OUTPUT, @RequireOtherProducts bit OUTPUT, @RequiredProductIds nvarchar(1000) OUTPUT, @AutomaticallyAddRequiredProducts bit OUTPUT, @IsDownload bit OUTPUT, @DownloadId int OUTPUT, @UnlimitedDownloads bit OUTPUT, @MaxNumberOfDownloads int OUTPUT, @DownloadExpirationDays int OUTPUT, @DownloadActivationTypeId int OUTPUT, @HasSampleDownload bit OUTPUT, @SampleDownloadId int OUTPUT, @HasUserAgreement bit OUTPUT, @UserAgreementText nvarchar(MAX) OUTPUT, @IsRecurring bit OUTPUT, @RecurringCycleLength int OUTPUT, @RecurringCyclePeriodId int OUTPUT, @RecurringTotalCycles int OUTPUT, @IsShipEnabled bit OUTPUT, @IsFreeShipping bit OUTPUT, @AdditionalShippingCharge decimal(18,4) OUTPUT, @IsTaxExempt bit OUTPUT, @TaxCategoryId int OUTPUT, @ManageInventoryMethodId int OUTPUT, @StockQuantity int OUTPUT, @DisplayStockAvailability bit OUTPUT, @DisplayStockQuantity bit OUTPUT, @MinStockQuantity int OUTPUT, @LowStockActivityId int OUTPUT, @NotifyAdminForQuantityBelow int OUTPUT, @BackorderModeId int OUTPUT, @AllowBackInStockSubscriptions bit OUTPUT, @OrderMinimumQuantity int OUTPUT, @OrderMaximumQuantity int OUTPUT, @AllowedQuantities nvarchar(1000) OUTPUT, @DisableBuyButton bit OUTPUT, @DisableWishlistButton bit OUTPUT, @AvailableForPreOrder bit OUTPUT, @CallForPrice bit OUTPUT, @Price decimal(18,4) OUTPUT, @OldPrice decimal(18,4) OUTPUT, @ProductCost decimal(18,4) OUTPUT, @SpecialPrice decimal(18,4) OUTPUT, @SpecialPriceStartDateTimeUtc datetime OUTPUT, @SpecialPriceEndDateTimeUtc datetime OUTPUT, @CustomerEntersPrice bit OUTPUT, @MinimumCustomerEnteredPrice decimal(18,4) OUTPUT, @MaximumCustomerEnteredPrice bit OUTPUT, @HasTierPrices bit OUTPUT, @HasDiscountsApplied bit OUTPUT, @Weight decimal(18, 4) OUTPUT, @Length decimal(18, 4) OUTPUT, @Width decimal(18, 4) OUTPUT, @Height decimal(18, 4) OUTPUT, @PictureId int OUTPUT, @AvailableStartDateTimeUtc datetime OUTPUT, @AvailableEndDateTimeUtc datetime OUTPUT, @Published bit OUTPUT, @Deleted bit OUTPUT, @DisplayOrder int OUTPUT, @CreatedOnUtc datetime OUTPUT, @UpdatedOnUtc datetime OUTPUT', @ProductId OUTPUT, @Name OUTPUT, @Description OUTPUT, @Sku OUTPUT, @ManufacturerPartNumber OUTPUT, @Gtin OUTPUT, @IsGiftCard OUTPUT, @GiftCardTypeId OUTPUT, @RequireOtherProducts OUTPUT, @RequiredProductIds OUTPUT, @AutomaticallyAddRequiredProducts OUTPUT, @IsDownload OUTPUT, @DownloadId OUTPUT, @UnlimitedDownloads OUTPUT, @MaxNumberOfDownloads OUTPUT, @DownloadExpirationDays OUTPUT, @DownloadActivationTypeId OUTPUT, @HasSampleDownload OUTPUT, @SampleDownloadId OUTPUT, @HasUserAgreement OUTPUT, @UserAgreementText OUTPUT, @IsRecurring OUTPUT, @RecurringCycleLength OUTPUT, @RecurringCyclePeriodId OUTPUT, @RecurringTotalCycles OUTPUT, @IsShipEnabled OUTPUT, @IsFreeShipping OUTPUT, @AdditionalShippingCharge OUTPUT, @IsTaxExempt OUTPUT, @TaxCategoryId OUTPUT, @ManageInventoryMethodId OUTPUT, @StockQuantity OUTPUT, @DisplayStockAvailability OUTPUT, @DisplayStockQuantity OUTPUT, @MinStockQuantity OUTPUT, @LowStockActivityId OUTPUT, @NotifyAdminForQuantityBelow OUTPUT, @BackorderModeId OUTPUT, @AllowBackInStockSubscriptions OUTPUT, @OrderMinimumQuantity OUTPUT, @OrderMaximumQuantity OUTPUT, @AllowedQuantities OUTPUT, @DisableBuyButton OUTPUT, @DisableWishlistButton OUTPUT, @AvailableForPreOrder OUTPUT, @CallForPrice OUTPUT, @Price OUTPUT, @OldPrice OUTPUT, @ProductCost OUTPUT, @SpecialPrice OUTPUT, @SpecialPriceStartDateTimeUtc OUTPUT, @SpecialPriceEndDateTimeUtc OUTPUT, @CustomerEntersPrice OUTPUT, @MinimumCustomerEnteredPrice OUTPUT, @MaximumCustomerEnteredPrice OUTPUT, @HasTierPrices OUTPUT, @HasDiscountsApplied OUTPUT, @Weight OUTPUT, @Length OUTPUT, @Width OUTPUT, @Height OUTPUT, @PictureId OUTPUT, @AvailableStartDateTimeUtc OUTPUT, @AvailableEndDateTimeUtc OUTPUT, @Published OUTPUT, @Deleted OUTPUT, @DisplayOrder OUTPUT, @CreatedOnUtc OUTPUT, @UpdatedOnUtc OUTPUT --how many variants do we have? DECLARE @NumberOfVariants int SELECT @NumberOfVariants = COUNT(1) FROM [ProductVariant] WHERE [ProductId]=@ProductId --product templates DECLARE @SimpleProductTemplateId int SELECT @SimpleProductTemplateId = [Id] FROM [ProductTemplate] WHERE [ViewPath] = N'ProductTemplate.Simple' DECLARE @GroupedProductTemplateId int SELECT @GroupedProductTemplateId = [Id] FROM [ProductTemplate] WHERE [ViewPath] = N'ProductTemplate.Grouped' --new product id: --if we have a simple product it'll be the same --if we have a grouped product, then it'll be the identifier of a new associated product DECLARE @NewProductId int SET @NewProductId = null -- clear cache (variable scope) --process a product (simple or grouped) IF (@NumberOfVariants <= 1) BEGIN --simple product UPDATE [Product] SET [ProductTypeId] = 5, [ParentProductId] = 0, [Sku] = @Sku, [ManufacturerPartNumber] = @ManufacturerPartNumber, [Gtin] = @Gtin, [IsGiftCard] = @IsGiftCard, [GiftCardTypeId] = @GiftCardTypeId, [RequireOtherProducts] = @RequireOtherProducts, --a store owner should manually update [RequiredProductIds] property after upgrade --[RequiredProductIds] = @RequiredProductIds, [AutomaticallyAddRequiredProducts] = @AutomaticallyAddRequiredProducts, [IsDownload] = @IsDownload, [DownloadId] = @DownloadId, [UnlimitedDownloads] = @UnlimitedDownloads, [MaxNumberOfDownloads] = @MaxNumberOfDownloads, [DownloadExpirationDays] = @DownloadExpirationDays, [DownloadActivationTypeId] = @DownloadActivationTypeId, [HasSampleDownload] = @HasSampleDownload, [SampleDownloadId] = @SampleDownloadId, [HasUserAgreement] = @HasUserAgreement, [UserAgreementText] = @UserAgreementText, [IsRecurring] = @IsRecurring, [RecurringCycleLength] = @RecurringCycleLength, [RecurringCyclePeriodId] = @RecurringCyclePeriodId, [RecurringTotalCycles] = @RecurringTotalCycles, [IsShipEnabled] = @IsShipEnabled, [IsFreeShipping] = @IsFreeShipping, [AdditionalShippingCharge] = @AdditionalShippingCharge, [IsTaxExempt] = @IsTaxExempt, [TaxCategoryId] = @TaxCategoryId, [ManageInventoryMethodId] = @ManageInventoryMethodId, [StockQuantity] = @StockQuantity, [DisplayStockAvailability] = @DisplayStockAvailability, [DisplayStockQuantity] = @DisplayStockQuantity, [MinStockQuantity] = @MinStockQuantity, [LowStockActivityId] = @LowStockActivityId, [NotifyAdminForQuantityBelow] = @NotifyAdminForQuantityBelow, [BackorderModeId] = @BackorderModeId, [AllowBackInStockSubscriptions] = @AllowBackInStockSubscriptions, [OrderMinimumQuantity] = @OrderMinimumQuantity, [OrderMaximumQuantity] = @OrderMaximumQuantity, [AllowedQuantities] = @AllowedQuantities, [DisableBuyButton] = @DisableBuyButton, [DisableWishlistButton] = @DisableWishlistButton, [AvailableForPreOrder] = @AvailableForPreOrder, [CallForPrice] = @CallForPrice, [Price] = @Price, [OldPrice] = @OldPrice, [ProductCost] = @ProductCost, [SpecialPrice] = @SpecialPrice, [SpecialPriceStartDateTimeUtc] = @SpecialPriceStartDateTimeUtc, [SpecialPriceEndDateTimeUtc] = @SpecialPriceEndDateTimeUtc, [CustomerEntersPrice] = @CustomerEntersPrice, [MinimumCustomerEnteredPrice] = @MinimumCustomerEnteredPrice, [MaximumCustomerEnteredPrice] = @MaximumCustomerEnteredPrice, [HasTierPrices] = @HasTierPrices, [HasDiscountsApplied] = @HasDiscountsApplied, [Weight] = @Weight, [Length] = @Length, [Width] = @Width, [Height] = @Height, [AvailableStartDateTimeUtc] = @AvailableStartDateTimeUtc, [AvailableEndDateTimeUtc] = @AvailableEndDateTimeUtc WHERE [Id]=@ProductId --product type UPDATE [Product] SET [ProductTypeId]=5 WHERE [Id]=@ProductId --product template UPDATE [Product] SET [ProductTemplateId]=@SimpleProductTemplateId WHERE [Id]=@ProductId --deleted? IF (@Deleted = 1) BEGIN UPDATE [Product] SET [Deleted]=@Deleted WHERE [Id]=@ProductId END --published? IF (@Published = 0) BEGIN UPDATE [Product] SET [Published]=@Published WHERE [Id]=@ProductId END SET @NewProductId = @ProductId END ELSE BEGIN --grouped product UPDATE [Product] SET [ProductTypeId] = 10, [ParentProductId] = 0, [Sku] = null, [ManufacturerPartNumber] = null, [Gtin] = null, [IsGiftCard] = 0, [GiftCardTypeId] = 0, [RequireOtherProducts] = 0, [RequiredProductIds] = null, [AutomaticallyAddRequiredProducts] = 0, [IsDownload] = 0, [DownloadId] = 0, [UnlimitedDownloads] = @UnlimitedDownloads, [MaxNumberOfDownloads] = @MaxNumberOfDownloads, [DownloadExpirationDays] = @DownloadExpirationDays, [DownloadActivationTypeId] = @DownloadActivationTypeId, [HasSampleDownload] = 0, [SampleDownloadId] = 0, [HasUserAgreement] = @HasUserAgreement, [UserAgreementText] = @UserAgreementText, [IsRecurring] = @IsRecurring, [RecurringCycleLength] = @RecurringCycleLength, [RecurringCyclePeriodId] = @RecurringCyclePeriodId, [RecurringTotalCycles] = @RecurringTotalCycles, [IsShipEnabled] = @IsShipEnabled, [IsFreeShipping] = @IsFreeShipping, [AdditionalShippingCharge] = @AdditionalShippingCharge, [IsTaxExempt] = @IsTaxExempt, [TaxCategoryId] = @TaxCategoryId, [ManageInventoryMethodId] = @ManageInventoryMethodId, [StockQuantity] = @StockQuantity, [DisplayStockAvailability] = @DisplayStockAvailability, [DisplayStockQuantity] = @DisplayStockQuantity, [MinStockQuantity] = @MinStockQuantity, [LowStockActivityId] = @LowStockActivityId, [NotifyAdminForQuantityBelow] = @NotifyAdminForQuantityBelow, [BackorderModeId] = @BackorderModeId, [AllowBackInStockSubscriptions] = @AllowBackInStockSubscriptions, [OrderMinimumQuantity] = @OrderMinimumQuantity, [OrderMaximumQuantity] = @OrderMaximumQuantity, [AllowedQuantities] = @AllowedQuantities, [DisableBuyButton] = @DisableBuyButton, [DisableWishlistButton] = @DisableWishlistButton, [AvailableForPreOrder] = @AvailableForPreOrder, [CallForPrice] = @CallForPrice, [Price] = @Price, [OldPrice] = @OldPrice, [ProductCost] = @ProductCost, [SpecialPrice] = @SpecialPrice, [SpecialPriceStartDateTimeUtc] = @SpecialPriceStartDateTimeUtc, [SpecialPriceEndDateTimeUtc] = @SpecialPriceEndDateTimeUtc, [CustomerEntersPrice] = @CustomerEntersPrice, [MinimumCustomerEnteredPrice] = @MinimumCustomerEnteredPrice, [MaximumCustomerEnteredPrice] = @MaximumCustomerEnteredPrice, [HasTierPrices] = 0, [HasDiscountsApplied] = 0, [Weight] = @Weight, [Length] = @Length, [Width] = @Width, [Height] = @Height, [AvailableStartDateTimeUtc] = @AvailableStartDateTimeUtc, [AvailableEndDateTimeUtc] = @AvailableEndDateTimeUtc WHERE [Id]=@ProductId --product type UPDATE [Product] SET [ProductTypeId]=10 WHERE [Id]=@ProductId --product template UPDATE [Product] SET [ProductTemplateId]=@GroupedProductTemplateId WHERE [Id]=@ProductId --insert a product variant (now we name it an associated product) DECLARE @AssociatedProductName nvarchar(1000) SELECT @AssociatedProductName = [Name] FROM [Product] WHERE [Id]=@ProductId --append a product variant name IF (len(@Name) > 0) BEGIN SET @AssociatedProductName = @AssociatedProductName + ' ' + @Name END --vendor DECLARE @AssociatedProductVendorId int SELECT @AssociatedProductVendorId = [VendorId] FROM [Product] WHERE [Id]=@ProductId --published? DECLARE @AssociatedProductPublished bit SELECT @AssociatedProductPublished = [Published] FROM [Product] WHERE [Id]=@ProductId IF (@Published = 0) BEGIN SET @AssociatedProductPublished = @Published END --deleted? DECLARE @AssociatedProductDeleted bit SELECT @AssociatedProductDeleted = [Deleted] FROM [Product] WHERE [Id]=@ProductId IF (@Deleted = 1) BEGIN SET @AssociatedProductDeleted = @Deleted END INSERT INTO [Product] (Name, ShortDescription, ProductTemplateId, VendorId, ShowOnHomePage, AllowCustomerReviews, ApprovedRatingSum, NotApprovedRatingSum, ApprovedTotalReviews, NotApprovedTotalReviews, SubjectToAcl, LimitedToStores, Published, Deleted, CreatedOnUtc, UpdatedOnUtc, Sku, ManufacturerPartNumber, Gtin, IsGiftCard, GiftCardTypeId, RequireOtherProducts, AutomaticallyAddRequiredProducts, IsDownload, DownloadId, UnlimitedDownloads, MaxNumberOfDownloads, DownloadExpirationDays, DownloadActivationTypeId, HasSampleDownload, SampleDownloadId, HasUserAgreement, UserAgreementText, IsRecurring, RecurringCycleLength, RecurringCyclePeriodId, RecurringTotalCycles, IsShipEnabled, IsFreeShipping, AdditionalShippingCharge, IsTaxExempt, TaxCategoryId, ManageInventoryMethodId, StockQuantity, DisplayStockAvailability, DisplayStockQuantity, MinStockQuantity, LowStockActivityId, NotifyAdminForQuantityBelow, BackorderModeId, AllowBackInStockSubscriptions, OrderMinimumQuantity, OrderMaximumQuantity, AllowedQuantities, DisableBuyButton, DisableWishlistButton, AvailableForPreOrder, CallForPrice, Price, OldPrice, ProductCost, SpecialPrice, SpecialPriceStartDateTimeUtc, SpecialPriceEndDateTimeUtc, CustomerEntersPrice, MinimumCustomerEnteredPrice, MaximumCustomerEnteredPrice, HasTierPrices, HasDiscountsApplied, Weight, Length, Width, Height, AvailableStartDateTimeUtc, AvailableEndDateTimeUtc, ProductTypeId, ParentProductId) VALUES (@AssociatedProductName, @Description, @SimpleProductTemplateId, @AssociatedProductVendorId, 0, 0, 0, 0, 0, 0, 0, 0, @AssociatedProductPublished, @AssociatedProductDeleted, @CreatedOnUtc, @UpdatedOnUtc, @Sku,@ManufacturerPartNumber, @Gtin, @IsGiftCard, @GiftCardTypeId, @RequireOtherProducts, --a store owner should manually update [RequiredProductIds] property after upgrade @AutomaticallyAddRequiredProducts, @IsDownload, @DownloadId, @UnlimitedDownloads, @MaxNumberOfDownloads, @DownloadExpirationDays, @DownloadActivationTypeId, @HasSampleDownload, @SampleDownloadId, @HasUserAgreement, @UserAgreementText, @IsRecurring, @RecurringCycleLength, @RecurringCyclePeriodId, @RecurringTotalCycles, @IsShipEnabled, @IsFreeShipping, @AdditionalShippingCharge, @IsTaxExempt, @TaxCategoryId, @ManageInventoryMethodId, @StockQuantity, @DisplayStockAvailability, @DisplayStockQuantity, @MinStockQuantity, @LowStockActivityId, @NotifyAdminForQuantityBelow, @BackorderModeId, @AllowBackInStockSubscriptions, @OrderMinimumQuantity, @OrderMaximumQuantity, @AllowedQuantities, @DisableBuyButton, @DisableWishlistButton, @AvailableForPreOrder, @CallForPrice, @Price, @OldPrice, @ProductCost, @SpecialPrice, @SpecialPriceStartDateTimeUtc, @SpecialPriceEndDateTimeUtc, @CustomerEntersPrice, @MinimumCustomerEnteredPrice, @MaximumCustomerEnteredPrice, @HasTierPrices, @HasDiscountsApplied, @Weight, @Length, @Width, @Height, @AvailableStartDateTimeUtc, @AvailableEndDateTimeUtc, --simple product 5 , @ProductId) SET @NewProductId = @@IDENTITY --product variant picture IF (@PictureId > 0) BEGIN INSERT INTO [Product_Picture_Mapping] ([ProductId], [PictureId], [DisplayOrder]) VALUES (@NewProductId, @PictureId, 1) END END --back in stock subscriptions. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[BackInStockSubscription]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [BackInStockSubscription] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --order items. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [OrderItem] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --product variant attributes. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariant_ProductAttribute_Mapping]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [ProductVariant_ProductAttribute_Mapping] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --attribute combinations. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeCombination]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [ProductVariantAttributeCombination] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --shopping cart items. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ShoppingCartItem]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [ShoppingCartItem] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --tier prices. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[TierPrice]') and NAME='ProductVariantId') BEGIN EXEC sp_executesql N'UPDATE [TierPrice] SET [ProductId] = @NewProductId WHERE [ProductVariantId] = @ExistingProductVariantID', N'@NewProductId int OUTPUT, @ExistingProductVariantID int OUTPUT', @NewProductId OUTPUT, @ExistingProductVariantID OUTPUT END --discounts. move ProductVariantId to the new ProductId column IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Discount_AppliedToProducts]') and NAME='ProductVariant_Id') BEGIN EXEC sp_executesql N'UPDATE [Discount_AppliedToProducts] SET [Product_Id] = @NewProductId WHERE [ProductVariant_Id] = @ExistingProductVariantID', N'@NewProductId int, @ExistingProductVariantID int', @NewProductId, @ExistingProductVariantID END --fetch next product variant identifier FETCH NEXT FROM cur_existingproductvariant INTO @ExistingProductVariantID END CLOSE cur_existingproductvariant DEALLOCATE cur_existingproductvariant END GO --back in stock subscriptions ALTER TABLE [BackInStockSubscription] ALTER COLUMN [ProductId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'BackInStockSubscription_Product' AND parent_object_id = Object_id('BackInStockSubscription') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[BackInStockSubscription] WITH CHECK ADD CONSTRAINT [BackInStockSubscription_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[BackInStockSubscription]') and NAME='ProductVariantId') BEGIN ALTER TABLE [BackInStockSubscription] DROP COLUMN [ProductVariantId] END GO --order items ALTER TABLE [OrderItem] ALTER COLUMN [ProductId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'OrderItem_Product' AND parent_object_id = Object_id('OrderItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[OrderItem] WITH CHECK ADD CONSTRAINT [OrderItem_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='ProductVariantId') BEGIN ALTER TABLE [OrderItem] DROP COLUMN [ProductVariantId] END GO --product variant attributes IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductVariant_ProductAttribute_Mapping]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN ALTER TABLE [ProductVariant_ProductAttribute_Mapping] ALTER COLUMN [ProductId] int NOT NULL END GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ProductVariantAttribute_Product' AND parent_object_id = Object_id('ProductVariant_ProductAttribute_Mapping') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN --one more validatation here because we'll rename [ProductVariant_ProductAttribute_Mapping] table a bit later IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[Product_ProductAttribute_Mapping]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN ALTER TABLE [dbo].[ProductVariant_ProductAttribute_Mapping] WITH CHECK ADD CONSTRAINT [ProductVariantAttribute_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END END GO IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_ProductVariant_ProductAttribute_Mapping_ProductVariantId' and object_id=object_id(N'[ProductVariant_ProductAttribute_Mapping]')) BEGIN DROP INDEX [IX_ProductVariant_ProductAttribute_Mapping_ProductVariantId] ON [ProductVariant_ProductAttribute_Mapping] CREATE NONCLUSTERED INDEX [IX_Product_ProductAttribute_Mapping_ProductId] ON [ProductVariant_ProductAttribute_Mapping] ([ProductId] ASC) END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariant_ProductAttribute_Mapping]') and NAME='ProductVariantId') BEGIN ALTER TABLE [ProductVariant_ProductAttribute_Mapping] DROP COLUMN [ProductVariantId] END GO IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductVariant_ProductAttribute_Mapping]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN EXEC sp_rename 'ProductVariant_ProductAttribute_Mapping', 'Product_ProductAttribute_Mapping'; END GO --attribute combinations ALTER TABLE [ProductVariantAttributeCombination] ALTER COLUMN [ProductId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ProductVariantAttributeCombination_Product' AND parent_object_id = Object_id('ProductVariantAttributeCombination') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[ProductVariantAttributeCombination] WITH CHECK ADD CONSTRAINT [ProductVariantAttributeCombination_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeCombination]') and NAME='ProductVariantId') BEGIN ALTER TABLE [ProductVariantAttributeCombination] DROP COLUMN [ProductVariantId] END GO --shopping cart items ALTER TABLE [ShoppingCartItem] ALTER COLUMN [ProductId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'ShoppingCartItem_Product' AND parent_object_id = Object_id('ShoppingCartItem') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[ShoppingCartItem] WITH CHECK ADD CONSTRAINT [ShoppingCartItem_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ShoppingCartItem]') and NAME='ProductVariantId') BEGIN ALTER TABLE [ShoppingCartItem] DROP COLUMN [ProductVariantId] END GO --tier prices ALTER TABLE [TierPrice] ALTER COLUMN [ProductId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'TierPrice_Product' AND parent_object_id = Object_id('TierPrice') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[TierPrice] WITH CHECK ADD CONSTRAINT [TierPrice_Product] FOREIGN KEY([ProductId]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_TierPrice_ProductVariantId' and object_id=object_id(N'[TierPrice]')) BEGIN DROP INDEX [IX_TierPrice_ProductVariantId] ON [TierPrice] CREATE NONCLUSTERED INDEX [IX_TierPrice_ProductId] ON [TierPrice] ([ProductId] ASC) END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[TierPrice]') and NAME='ProductVariantId') BEGIN ALTER TABLE [TierPrice] DROP COLUMN [ProductVariantId] END GO --discounts IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'Discount_AppliedToProducts_Source' AND parent_object_id = Object_id('Discount_AppliedToProducts') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[Discount_AppliedToProducts] WITH CHECK ADD CONSTRAINT [Discount_AppliedToProducts_Source] FOREIGN KEY([Discount_Id]) REFERENCES [dbo].[Discount] ([Id]) ON DELETE CASCADE END GO IF NOT EXISTS (SELECT 1 FROM sys.objects WHERE name = 'Discount_AppliedToProducts_Target' AND parent_object_id = Object_id('Discount_AppliedToProducts') AND Objectproperty(object_id,N'IsForeignKey') = 1) BEGIN ALTER TABLE [dbo].[Discount_AppliedToProducts] WITH CHECK ADD CONSTRAINT [Discount_AppliedToProducts_Target] FOREIGN KEY([Product_Id]) REFERENCES [dbo].[Product] ([Id]) ON DELETE CASCADE END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Discount_AppliedToProducts]') and NAME='ProductVariant_Id') BEGIN ALTER TABLE [Discount_AppliedToProducts] DROP COLUMN [ProductVariant_Id] END GO --drop product variant table IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductVariant]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN DROP TABLE [ProductVariant] END GO --new Product columns. Set "NOT NULL" where required IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_ParentProductId' and object_id=object_id(N'[Product]')) BEGIN DROP INDEX [IX_Product_ParentProductId] ON [Product] END GO IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_ParentGroupedProductId' and object_id=object_id(N'[Product]')) BEGIN DROP INDEX [IX_Product_ParentGroupedProductId] ON [Product] END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN exec ('UPDATE [Product] SET [ParentProductId] = 0 WHERE [ParentProductId] is null') exec ('ALTER TABLE [Product] ALTER COLUMN [ParentProductId] int NOT NULL') END GO ALTER TABLE [Product] ALTER COLUMN [ProductTypeId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsGiftCard] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [GiftCardTypeId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [RequireOtherProducts] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [AutomaticallyAddRequiredProducts] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsDownload] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DownloadId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [UnlimitedDownloads] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [MaxNumberOfDownloads] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DownloadActivationTypeId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [HasSampleDownload] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [SampleDownloadId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [HasUserAgreement] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsRecurring] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [RecurringCycleLength] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [RecurringCyclePeriodId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [RecurringTotalCycles] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsShipEnabled] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsFreeShipping] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [AdditionalShippingCharge] decimal(18,4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [IsTaxExempt] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [TaxCategoryId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [ManageInventoryMethodId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [StockQuantity] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DisplayStockAvailability] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DisplayStockQuantity] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [MinStockQuantity] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [LowStockActivityId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [NotifyAdminForQuantityBelow] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [BackorderModeId] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [AllowBackInStockSubscriptions] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [OrderMinimumQuantity] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [OrderMaximumQuantity] int NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DisableBuyButton] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [DisableWishlistButton] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [AvailableForPreOrder] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [CallForPrice] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [Price] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [OldPrice] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [ProductCost] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [CustomerEntersPrice] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [MinimumCustomerEnteredPrice] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [MaximumCustomerEnteredPrice] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [HasTierPrices] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [HasDiscountsApplied] bit NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [Weight] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [Length] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [Width] decimal(18, 4) NOT NULL GO ALTER TABLE [Product] ALTER COLUMN [Height] decimal(18, 4) NOT NULL GO -- new indexes IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_PriceDatesEtc' and object_id=object_id(N'[Product]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Product_PriceDatesEtc] ON [Product] ([Price] ASC, [AvailableStartDateTimeUtc] ASC, [AvailableEndDateTimeUtc] ASC, [Published] ASC, [Deleted] ASC) END GO IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_ParentProductId' and object_id=object_id(N'[Product]')) BEGIN IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN CREATE NONCLUSTERED INDEX [IX_Product_ParentProductId] ON [Product] ([ParentProductId] ASC) END END GO IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[FullText_Enable]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [FullText_Enable] GO CREATE PROCEDURE [FullText_Enable] AS BEGIN --create catalog EXEC(' IF NOT EXISTS (SELECT 1 FROM sys.fulltext_catalogs WHERE [name] = ''nopCommerceFullTextCatalog'') CREATE FULLTEXT CATALOG [nopCommerceFullTextCatalog] AS DEFAULT') --create indexes DECLARE @create_index_text nvarchar(4000) SET @create_index_text = ' IF NOT EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[Product]'')) CREATE FULLTEXT INDEX ON [Product]([Name], [ShortDescription], [FullDescription], [Sku]) KEY INDEX [' + dbo.[nop_getprimarykey_indexname] ('Product') + '] ON [nopCommerceFullTextCatalog] WITH CHANGE_TRACKING AUTO' EXEC(@create_index_text) SET @create_index_text = ' IF NOT EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[LocalizedProperty]'')) CREATE FULLTEXT INDEX ON [LocalizedProperty]([LocaleValue]) KEY INDEX [' + dbo.[nop_getprimarykey_indexname] ('LocalizedProperty') + '] ON [nopCommerceFullTextCatalog] WITH CHANGE_TRACKING AUTO' EXEC(@create_index_text) SET @create_index_text = ' IF NOT EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[ProductTag]'')) CREATE FULLTEXT INDEX ON [ProductTag]([Name]) KEY INDEX [' + dbo.[nop_getprimarykey_indexname] ('ProductTag') + '] ON [nopCommerceFullTextCatalog] WITH CHANGE_TRACKING AUTO' EXEC(@create_index_text) END GO IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[FullText_Disable]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [FullText_Disable] GO CREATE PROCEDURE [FullText_Disable] AS BEGIN EXEC(' --drop indexes IF EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[Product]'')) DROP FULLTEXT INDEX ON [Product] ') EXEC(' IF EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[LocalizedProperty]'')) DROP FULLTEXT INDEX ON [LocalizedProperty] ') EXEC(' IF EXISTS (SELECT 1 FROM sys.fulltext_indexes WHERE object_id = object_id(''[ProductTag]'')) DROP FULLTEXT INDEX ON [ProductTag] ') --drop catalog EXEC(' IF EXISTS (SELECT 1 FROM sys.fulltext_catalogs WHERE [name] = ''nopCommerceFullTextCatalog'') DROP FULLTEXT CATALOG [nopCommerceFullTextCatalog] ') END GO --you have to manually re-configure "google products" (froogle) plugin IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[GoogleProduct]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN DELETE FROM [GoogleProduct] IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[GoogleProduct]') and NAME='ProductVariantId') BEGIN EXEC sp_rename 'GoogleProduct.ProductVariantId', 'ProductId', 'COLUMN'; END END GO IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[temp_generate_sename]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [dbo].[temp_generate_sename] GO CREATE PROCEDURE [dbo].[temp_generate_sename] ( @table_name nvarchar(1000), @entity_id int, @result nvarchar(1000) OUTPUT ) AS BEGIN --get current name DECLARE @current_sename nvarchar(1000) DECLARE @sql nvarchar(4000) SET @sql = 'SELECT @current_sename = [Name] FROM [' + @table_name + '] WHERE [Id] = ' + ISNULL(CAST(@entity_id AS nvarchar(max)), '0') EXEC sp_executesql @sql,N'@current_sename nvarchar(1000) OUTPUT',@current_sename OUTPUT --generate se name DECLARE @new_sename nvarchar(1000) SET @new_sename = '' --ensure only allowed chars DECLARE @allowed_se_chars varchar(4000) --Note for store owners: add more chars below if want them to be supported when migrating your data SET @allowed_se_chars = N'abcdefghijklmnopqrstuvwxyz1234567890 _-' DECLARE @l int SET @l = len(@current_sename) DECLARE @p int SET @p = 1 WHILE @p <= @l BEGIN DECLARE @c nvarchar(1) SET @c = substring(@current_sename, @p, 1) IF CHARINDEX(@c,@allowed_se_chars) > 0 BEGIN SET @new_sename = @new_sename + @c END SET @p = @p + 1 END --replace spaces with '-' SELECT @new_sename = REPLACE(@new_sename,' ','-'); WHILE CHARINDEX('--',@new_sename) > 0 SELECT @new_sename = REPLACE(@new_sename,'--','-'); WHILE CHARINDEX('__',@new_sename) > 0 SELECT @new_sename = REPLACE(@new_sename,'__','_'); --ensure not empty IF (@new_sename is null or @new_sename = '') SELECT @new_sename = ISNULL(CAST(@entity_id AS nvarchar(max)), '0'); --lowercase SELECT @new_sename = LOWER(@new_sename) --ensure this sename is not reserved WHILE (1=1) BEGIN DECLARE @sename_is_already_reserved bit SET @sename_is_already_reserved = 0 SET @sql = 'IF EXISTS (SELECT 1 FROM [UrlRecord] WHERE [Slug] = @sename AND [EntityId] <> ' + ISNULL(CAST(@entity_id AS nvarchar(max)), '0') + ') BEGIN SELECT @sename_is_already_reserved = 1 END' EXEC sp_executesql @sql,N'@sename nvarchar(1000), @sename_is_already_reserved nvarchar(4000) OUTPUT',@new_sename,@sename_is_already_reserved OUTPUT IF (@sename_is_already_reserved > 0) BEGIN --add some digit to the end in this case SET @new_sename = @new_sename + '-1' END ELSE BEGIN BREAK END END --return SET @result = @new_sename END GO --set search engine friendly name (UrlRecord) for associated products (new products added before in this upgrade script). [ParentProductId] > 0 IF EXISTS (SELECT 1 FROM sys.objects WHERE object_id = OBJECT_ID(N'[Product]') and OBJECTPROPERTY(object_id, N'IsUserTable') = 1) BEGIN DECLARE @sename_existing_entity_id int DECLARE cur_sename_existing_entity CURSOR FOR SELECT [Id] FROM [Product] WHERE [ParentProductId] > 0 OPEN cur_sename_existing_entity FETCH NEXT FROM cur_sename_existing_entity INTO @sename_existing_entity_id WHILE @@FETCH_STATUS = 0 BEGIN DECLARE @sename nvarchar(1000) SET @sename = null -- clear cache (variable scope) DECLARE @table_name nvarchar(1000) SET @table_name = N'Product' --main sename EXEC [dbo].[temp_generate_sename] @table_name = @table_name, @entity_id = @sename_existing_entity_id, @result = @sename OUTPUT IF EXISTS(SELECT 1 FROM [UrlRecord] WHERE [LanguageId]=0 AND [EntityId]=@sename_existing_entity_id AND [EntityName]=@table_name) BEGIN UPDATE [UrlRecord] SET [Slug] = @sename WHERE [LanguageId]=0 AND [EntityId]=@sename_existing_entity_id AND [EntityName]=@table_name END ELSE BEGIN INSERT INTO [UrlRecord] ([EntityId], [EntityName], [Slug], [LanguageId], [IsActive]) VALUES (@sename_existing_entity_id, @table_name, @sename, 0, 1) END --fetch next identifier FETCH NEXT FROM cur_sename_existing_entity INTO @sename_existing_entity_id END CLOSE cur_sename_existing_entity DEALLOCATE cur_sename_existing_entity END GO --drop temporary procedures & functions IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[temp_generate_sename]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [temp_generate_sename] GO --new Product property IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='VisibleIndividually') BEGIN ALTER TABLE [Product] ADD [VisibleIndividually] bit NULL END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN EXEC('UPDATE [Product] SET [VisibleIndividually] = 0 WHERE [VisibleIndividually] IS NULL AND [ParentProductId] > 0') END GO IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN EXEC('UPDATE [Product] SET [VisibleIndividually] = 1 WHERE [VisibleIndividually] IS NULL AND [ParentProductId] = 0') END GO ALTER TABLE [Product] ALTER COLUMN [VisibleIndividually] bit NOT NULL GO --more indexes IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_VisibleIndividually' and object_id=object_id(N'[Product]')) BEGIN CREATE NONCLUSTERED INDEX [IX_Product_VisibleIndividually] ON [Product] ([VisibleIndividually] ASC) END GO --new [DisplayOrder] property IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='DisplayOrder') BEGIN ALTER TABLE [Product] ADD [DisplayOrder] int NULL END GO UPDATE [Product] SET [DisplayOrder] = 0 GO ALTER TABLE [Product] ALTER COLUMN [DisplayOrder] int NOT NULL GO --rename ParentProductId to ParentGroupedProductId IF EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[Product]') and NAME='ParentProductId') BEGIN EXEC sp_rename 'Product.ParentProductId', 'ParentGroupedProductId', 'COLUMN'; END GO --updated stored procedure IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductLoadAllPaged]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [ProductLoadAllPaged] GO CREATE PROCEDURE [dbo].[ProductLoadAllPaged] ( @CategoryIds nvarchar(MAX) = null, --a list of category IDs (comma-separated list). e.g. 1,2,3 @ManufacturerId int = 0, @StoreId int = 0, @VendorId int = 0, @ParentGroupedProductId int = 0, @ProductTypeId int = null, --product type identifier, null - load all products @VisibleIndividuallyOnly bit = 0, --0 - load all products , 1 - "visible indivially" only @ProductTagId int = 0, @FeaturedProducts bit = null, --0 featured only , 1 not featured only, null - load all products @PriceMin decimal(18, 4) = null, @PriceMax decimal(18, 4) = null, @Keywords nvarchar(4000) = null, @SearchDescriptions bit = 0, --a value indicating whether to search by a specified "keyword" in product descriptions @SearchProductTags bit = 0, --a value indicating whether to search by a specified "keyword" in product tags @UseFullTextSearch bit = 0, @FullTextMode int = 0, --0 - using CONTAINS with , 5 - using CONTAINS and OR with , 10 - using CONTAINS and AND with @FilteredSpecs nvarchar(MAX) = null, --filter by attributes (comma-separated list). e.g. 14,15,16 @LanguageId int = 0, @OrderBy int = 0, --0 - position, 5 - Name: A to Z, 6 - Name: Z to A, 10 - Price: Low to High, 11 - Price: High to Low, 15 - creation date @AllowedCustomerRoleIds nvarchar(MAX) = null, --a list of customer role IDs (comma-separated list) for which a product should be shown (if a subjet to ACL) @PageIndex int = 0, @PageSize int = 2147483644, @ShowHidden bit = 0, @LoadFilterableSpecificationAttributeOptionIds bit = 0, --a value indicating whether we should load the specification attribute option identifiers applied to loaded products (all pages) @FilterableSpecificationAttributeOptionIds nvarchar(MAX) = null OUTPUT, --the specification attribute option identifiers applied to loaded products (all pages). returned as a comma separated list of identifiers @TotalRecords int = null OUTPUT ) AS BEGIN /* Products that filtered by keywords */ CREATE TABLE #KeywordProducts ( [ProductId] int NOT NULL ) DECLARE @SearchKeywords bit, @sql nvarchar(max), @sql_orderby nvarchar(max) SET NOCOUNT ON --filter by keywords SET @Keywords = isnull(@Keywords, '') SET @Keywords = rtrim(ltrim(@Keywords)) IF ISNULL(@Keywords, '') != '' BEGIN SET @SearchKeywords = 1 IF @UseFullTextSearch = 1 BEGIN --remove wrong chars (' ") SET @Keywords = REPLACE(@Keywords, '''', '') SET @Keywords = REPLACE(@Keywords, '"', '') --full-text search IF @FullTextMode = 0 BEGIN --0 - using CONTAINS with SET @Keywords = ' "' + @Keywords + '*" ' END ELSE BEGIN --5 - using CONTAINS and OR with --10 - using CONTAINS and AND with --clean multiple spaces WHILE CHARINDEX(' ', @Keywords) > 0 SET @Keywords = REPLACE(@Keywords, ' ', ' ') DECLARE @concat_term nvarchar(100) IF @FullTextMode = 5 --5 - using CONTAINS and OR with BEGIN SET @concat_term = 'OR' END IF @FullTextMode = 10 --10 - using CONTAINS and AND with BEGIN SET @concat_term = 'AND' END --now let's build search string declare @fulltext_keywords nvarchar(4000) set @fulltext_keywords = N'' declare @index int set @index = CHARINDEX(' ', @Keywords, 0) -- if index = 0, then only one field was passed IF(@index = 0) set @fulltext_keywords = ' "' + @Keywords + '*" ' ELSE BEGIN DECLARE @first BIT SET @first = 1 WHILE @index > 0 BEGIN IF (@first = 0) SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' ELSE SET @first = 0 SET @fulltext_keywords = @fulltext_keywords + '"' + SUBSTRING(@Keywords, 1, @index - 1) + '*"' SET @Keywords = SUBSTRING(@Keywords, @index + 1, LEN(@Keywords) - @index) SET @index = CHARINDEX(' ', @Keywords, 0) end -- add the last field IF LEN(@fulltext_keywords) > 0 SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' + '"' + SUBSTRING(@Keywords, 1, LEN(@Keywords)) + '*"' END SET @Keywords = @fulltext_keywords END END ELSE BEGIN --usual search by PATINDEX SET @Keywords = '%' + @Keywords + '%' END --PRINT @Keywords --product name SET @sql = ' INSERT INTO #KeywordProducts ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Name]) > 0 ' --SKU SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Sku], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Sku]) > 0 ' --localized product name SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' IF @SearchDescriptions = 1 BEGIN --product short description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[ShortDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[ShortDescription]) > 0 ' --product full description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[FullDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[FullDescription]) > 0 ' --localized product short description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''ShortDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' --localized product full description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''FullDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END IF @SearchProductTags = 1 BEGIN --product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM Product_ProductTag_Mapping pptm with(NOLOCK) INNER JOIN ProductTag pt with(NOLOCK) ON pt.Id = pptm.ProductTag_Id WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pt.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pt.[Name]) > 0 ' --localized product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM LocalizedProperty lp with (NOLOCK) INNER JOIN Product_ProductTag_Mapping pptm with(NOLOCK) ON lp.EntityId = pptm.ProductTag_Id WHERE lp.LocaleKeyGroup = N''ProductTag'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END --PRINT (@sql) EXEC sp_executesql @sql, N'@Keywords nvarchar(4000)', @Keywords END ELSE BEGIN SET @SearchKeywords = 0 END --filter by category IDs SET @CategoryIds = isnull(@CategoryIds, '') CREATE TABLE #FilteredCategoryIds ( CategoryId int not null ) INSERT INTO #FilteredCategoryIds (CategoryId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@CategoryIds, ',') DECLARE @CategoryIdsCount int SET @CategoryIdsCount = (SELECT COUNT(1) FROM #FilteredCategoryIds) --filter by attributes SET @FilteredSpecs = isnull(@FilteredSpecs, '') CREATE TABLE #FilteredSpecs ( SpecificationAttributeOptionId int not null ) INSERT INTO #FilteredSpecs (SpecificationAttributeOptionId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@FilteredSpecs, ',') DECLARE @SpecAttributesCount int SET @SpecAttributesCount = (SELECT COUNT(1) FROM #FilteredSpecs) --filter by customer role IDs (access control list) SET @AllowedCustomerRoleIds = isnull(@AllowedCustomerRoleIds, '') CREATE TABLE #FilteredCustomerRoleIds ( CustomerRoleId int not null ) INSERT INTO #FilteredCustomerRoleIds (CustomerRoleId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@AllowedCustomerRoleIds, ',') --paging DECLARE @PageLowerBound int DECLARE @PageUpperBound int DECLARE @RowsToReturn int SET @RowsToReturn = @PageSize * (@PageIndex + 1) SET @PageLowerBound = @PageSize * @PageIndex SET @PageUpperBound = @PageLowerBound + @PageSize + 1 CREATE TABLE #DisplayOrderTmp ( [Id] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) SET @sql = ' INSERT INTO #DisplayOrderTmp ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK)' IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Category_Mapping pcm with (NOLOCK) ON p.Id = pcm.ProductId' END IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Manufacturer_Mapping pmm with (NOLOCK) ON p.Id = pmm.ProductId' END IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_ProductTag_Mapping pptm with (NOLOCK) ON p.Id = pptm.Product_Id' END --searching by keywords IF @SearchKeywords = 1 BEGIN SET @sql = @sql + ' JOIN #KeywordProducts kp ON p.Id = kp.ProductId' END SET @sql = @sql + ' WHERE p.Deleted = 0' --filter by category IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' AND pcm.CategoryId IN (SELECT CategoryId FROM #FilteredCategoryIds)' IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pcm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by manufacturer IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' AND pmm.ManufacturerId = ' + CAST(@ManufacturerId AS nvarchar(max)) IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pmm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by vendor IF @VendorId > 0 BEGIN SET @sql = @sql + ' AND p.VendorId = ' + CAST(@VendorId AS nvarchar(max)) END --filter by parent grouped product identifer IF @ParentGroupedProductId > 0 BEGIN SET @sql = @sql + ' AND p.ParentGroupedProductId = ' + CAST(@ParentGroupedProductId AS nvarchar(max)) END --filter by product type IF @ProductTypeId is not null BEGIN SET @sql = @sql + ' AND p.ProductTypeId = ' + CAST(@ProductTypeId AS nvarchar(max)) END --filter by parent product identifer IF @VisibleIndividuallyOnly = 1 BEGIN SET @sql = @sql + ' AND p.VisibleIndividually = 1' END --filter by product tag IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' AND pptm.ProductTag_Id = ' + CAST(@ProductTagId AS nvarchar(max)) END --show hidden IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND p.Published = 1 AND p.Deleted = 0 AND (getutcdate() BETWEEN ISNULL(p.AvailableStartDateTimeUtc, ''1/1/1900'') and ISNULL(p.AvailableEndDateTimeUtc, ''1/1/2999''))' END --min price IF @PriceMin > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) )' END --max price IF @PriceMax > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) )' END --show hidden and ACL IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND (p.SubjectToAcl = 0 OR EXISTS ( SELECT 1 FROM #FilteredCustomerRoleIds [fcr] WHERE [fcr].CustomerRoleId IN ( SELECT [acl].CustomerRoleId FROM [AclRecord] acl with (NOLOCK) WHERE [acl].EntityId = p.Id AND [acl].EntityName = ''Product'' ) ))' END --show hidden and filter by store IF @StoreId > 0 BEGIN SET @sql = @sql + ' AND (p.LimitedToStores = 0 OR EXISTS ( SELECT 1 FROM [StoreMapping] sm with (NOLOCK) WHERE [sm].EntityId = p.Id AND [sm].EntityName = ''Product'' and [sm].StoreId=' + CAST(@StoreId AS nvarchar(max)) + ' ))' END --filter by specs IF @SpecAttributesCount > 0 BEGIN SET @sql = @sql + ' AND NOT EXISTS ( SELECT 1 FROM #FilteredSpecs [fs] WHERE [fs].SpecificationAttributeOptionId NOT IN ( SELECT psam.SpecificationAttributeOptionId FROM Product_SpecificationAttribute_Mapping psam with (NOLOCK) WHERE psam.AllowFiltering = 1 AND psam.ProductId = p.Id ) )' END --sorting SET @sql_orderby = '' IF @OrderBy = 5 /* Name: A to Z */ SET @sql_orderby = ' p.[Name] ASC' ELSE IF @OrderBy = 6 /* Name: Z to A */ SET @sql_orderby = ' p.[Name] DESC' ELSE IF @OrderBy = 10 /* Price: Low to High */ SET @sql_orderby = ' p.[Price] ASC' ELSE IF @OrderBy = 11 /* Price: High to Low */ SET @sql_orderby = ' p.[Price] DESC' ELSE IF @OrderBy = 15 /* creation date */ SET @sql_orderby = ' p.[CreatedOnUtc] DESC' ELSE /* default sorting, 0 (position) */ BEGIN --category position (display order) IF @CategoryIdsCount > 0 SET @sql_orderby = ' pcm.DisplayOrder ASC' --manufacturer position (display order) IF @ManufacturerId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' pmm.DisplayOrder ASC' END --parent grouped product specified (sort associated products) IF @ParentGroupedProductId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[DisplayOrder] ASC' END --name IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[Name] ASC' END SET @sql = @sql + ' ORDER BY' + @sql_orderby --PRINT (@sql) EXEC sp_executesql @sql DROP TABLE #FilteredCategoryIds DROP TABLE #FilteredSpecs DROP TABLE #FilteredCustomerRoleIds DROP TABLE #KeywordProducts CREATE TABLE #PageIndex ( [IndexId] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) INSERT INTO #PageIndex ([ProductId]) SELECT ProductId FROM #DisplayOrderTmp GROUP BY ProductId ORDER BY min([Id]) --total records SET @TotalRecords = @@rowcount DROP TABLE #DisplayOrderTmp --prepare filterable specification attribute option identifier (if requested) IF @LoadFilterableSpecificationAttributeOptionIds = 1 BEGIN CREATE TABLE #FilterableSpecs ( [SpecificationAttributeOptionId] int NOT NULL ) INSERT INTO #FilterableSpecs ([SpecificationAttributeOptionId]) SELECT DISTINCT [psam].SpecificationAttributeOptionId FROM [Product_SpecificationAttribute_Mapping] [psam] with (NOLOCK) WHERE [psam].[AllowFiltering] = 1 AND [psam].[ProductId] IN (SELECT [pi].ProductId FROM #PageIndex [pi]) --build comma separated list of filterable identifiers SELECT @FilterableSpecificationAttributeOptionIds = COALESCE(@FilterableSpecificationAttributeOptionIds + ',' , '') + CAST(SpecificationAttributeOptionId as nvarchar(4000)) FROM #FilterableSpecs DROP TABLE #FilterableSpecs END --return products SELECT TOP (@RowsToReturn) p.* FROM #PageIndex [pi] INNER JOIN Product p with (NOLOCK) on p.Id = [pi].[ProductId] WHERE [pi].IndexId > @PageLowerBound AND [pi].IndexId < @PageUpperBound ORDER BY [pi].IndexId DROP TABLE #PageIndex END GO --updated product type values UPDATE [Product] SET [ProductTypeId]=5 WHERE [ProductTypeId]=0 GO IF EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_Product_ParentProductId' and object_id=object_id(N'[Product]')) BEGIN EXEC sp_rename 'Product.IX_Product_ParentProductId', 'IX_Product_ParentGroupedProductId', 'INDEX'; END GO --deleted obsolete setting DELETE FROM [Setting] WHERE [name] = N'pdfsettings.enabled' GO --pictures per attribute values IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeValue]') and NAME='PictureId') BEGIN ALTER TABLE [ProductVariantAttributeValue] ADD [PictureId] int NULL END GO UPDATE [ProductVariantAttributeValue] SET [PictureId] = 0 WHERE [PictureId] IS NULL GO ALTER TABLE [ProductVariantAttributeValue] ALTER COLUMN [PictureId] int NOT NULL GO --bundled products IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeValue]') and NAME='AttributeValueTypeId') BEGIN ALTER TABLE [ProductVariantAttributeValue] ADD [AttributeValueTypeId] int NULL END GO UPDATE [ProductVariantAttributeValue] SET [AttributeValueTypeId] = 0 WHERE [AttributeValueTypeId] IS NULL GO ALTER TABLE [ProductVariantAttributeValue] ALTER COLUMN [AttributeValueTypeId] int NOT NULL GO IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[ProductVariantAttributeValue]') and NAME='AssociatedProductId') BEGIN ALTER TABLE [ProductVariantAttributeValue] ADD [AssociatedProductId] int NULL END GO UPDATE [ProductVariantAttributeValue] SET [AssociatedProductId] = 0 WHERE [AssociatedProductId] IS NULL GO ALTER TABLE [ProductVariantAttributeValue] ALTER COLUMN [AssociatedProductId] int NOT NULL GO --more indexes IF NOT EXISTS (SELECT 1 from sys.indexes WHERE [NAME]=N'IX_UrlRecord_Custom_1' and object_id=object_id(N'[UrlRecord]')) BEGIN CREATE NONCLUSTERED INDEX [IX_UrlRecord_Custom_1] ON [UrlRecord] ([EntityId] ASC, [EntityName] ASC, [LanguageId] ASC, [IsActive] ASC) END GO --new column IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='OriginalProductCost') BEGIN ALTER TABLE [OrderItem] ADD [OriginalProductCost] decimal(18,4) NULL --set values based on products EXEC('UPDATE [OrderItem] SET [OrderItem].[OriginalProductCost] = p.[ProductCost] FROM [OrderItem] oi INNER JOIN [Product] p ON oi.[ProductId] = p.[Id]') END GO UPDATE [OrderItem] SET [OriginalProductCost] = 0 WHERE [OriginalProductCost] IS NULL GO ALTER TABLE [OrderItem] ALTER COLUMN [OriginalProductCost] decimal(18,4) NOT NULL GO --new column IF NOT EXISTS (SELECT 1 FROM sys.columns WHERE object_id=object_id('[OrderItem]') and NAME='OriginalProductCost') BEGIN ALTER TABLE [OrderItem] ADD [OriginalProductCost] decimal(18,4) NULL END GO --set values based on products UPDATE [OrderItem] SET [OrderItem].[OriginalProductCost] = p.[ProductCost] FROM [OrderItem] oi INNER JOIN [Product] p ON oi.[ProductId] = p.[Id] GO UPDATE [OrderItem] SET [OriginalProductCost] = 0 WHERE [OriginalProductCost] IS NULL GO ALTER TABLE [OrderItem] ALTER COLUMN [OriginalProductCost] decimal(18,4) NOT NULL GO --rename a country UPDATE [Country] SET [Name] = N'Croatia' WHERE [Name] = N'Croatia (local Name: Hrvatska)' GO --updated stored procedure IF EXISTS ( SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[ProductLoadAllPaged]') AND OBJECTPROPERTY(object_id,N'IsProcedure') = 1) DROP PROCEDURE [ProductLoadAllPaged] GO CREATE PROCEDURE [dbo].[ProductLoadAllPaged] ( @CategoryIds nvarchar(MAX) = null, --a list of category IDs (comma-separated list). e.g. 1,2,3 @ManufacturerId int = 0, @StoreId int = 0, @VendorId int = 0, @ParentGroupedProductId int = 0, @ProductTypeId int = null, --product type identifier, null - load all products @VisibleIndividuallyOnly bit = 0, --0 - load all products , 1 - "visible indivially" only @ProductTagId int = 0, @FeaturedProducts bit = null, --0 featured only , 1 not featured only, null - load all products @PriceMin decimal(18, 4) = null, @PriceMax decimal(18, 4) = null, @Keywords nvarchar(4000) = null, @SearchDescriptions bit = 0, --a value indicating whether to search by a specified "keyword" in product descriptions @SearchSku bit = 0, --a value indicating whether to search by a specified "keyword" in product SKU @SearchProductTags bit = 0, --a value indicating whether to search by a specified "keyword" in product tags @UseFullTextSearch bit = 0, @FullTextMode int = 0, --0 - using CONTAINS with , 5 - using CONTAINS and OR with , 10 - using CONTAINS and AND with @FilteredSpecs nvarchar(MAX) = null, --filter by attributes (comma-separated list). e.g. 14,15,16 @LanguageId int = 0, @OrderBy int = 0, --0 - position, 5 - Name: A to Z, 6 - Name: Z to A, 10 - Price: Low to High, 11 - Price: High to Low, 15 - creation date @AllowedCustomerRoleIds nvarchar(MAX) = null, --a list of customer role IDs (comma-separated list) for which a product should be shown (if a subjet to ACL) @PageIndex int = 0, @PageSize int = 2147483644, @ShowHidden bit = 0, @LoadFilterableSpecificationAttributeOptionIds bit = 0, --a value indicating whether we should load the specification attribute option identifiers applied to loaded products (all pages) @FilterableSpecificationAttributeOptionIds nvarchar(MAX) = null OUTPUT, --the specification attribute option identifiers applied to loaded products (all pages). returned as a comma separated list of identifiers @TotalRecords int = null OUTPUT ) AS BEGIN /* Products that filtered by keywords */ CREATE TABLE #KeywordProducts ( [ProductId] int NOT NULL ) DECLARE @SearchKeywords bit, @sql nvarchar(max), @sql_orderby nvarchar(max) SET NOCOUNT ON --filter by keywords SET @Keywords = isnull(@Keywords, '') SET @Keywords = rtrim(ltrim(@Keywords)) IF ISNULL(@Keywords, '') != '' BEGIN SET @SearchKeywords = 1 IF @UseFullTextSearch = 1 BEGIN --remove wrong chars (' ") SET @Keywords = REPLACE(@Keywords, '''', '') SET @Keywords = REPLACE(@Keywords, '"', '') --full-text search IF @FullTextMode = 0 BEGIN --0 - using CONTAINS with SET @Keywords = ' "' + @Keywords + '*" ' END ELSE BEGIN --5 - using CONTAINS and OR with --10 - using CONTAINS and AND with --clean multiple spaces WHILE CHARINDEX(' ', @Keywords) > 0 SET @Keywords = REPLACE(@Keywords, ' ', ' ') DECLARE @concat_term nvarchar(100) IF @FullTextMode = 5 --5 - using CONTAINS and OR with BEGIN SET @concat_term = 'OR' END IF @FullTextMode = 10 --10 - using CONTAINS and AND with BEGIN SET @concat_term = 'AND' END --now let's build search string declare @fulltext_keywords nvarchar(4000) set @fulltext_keywords = N'' declare @index int set @index = CHARINDEX(' ', @Keywords, 0) -- if index = 0, then only one field was passed IF(@index = 0) set @fulltext_keywords = ' "' + @Keywords + '*" ' ELSE BEGIN DECLARE @first BIT SET @first = 1 WHILE @index > 0 BEGIN IF (@first = 0) SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' ELSE SET @first = 0 SET @fulltext_keywords = @fulltext_keywords + '"' + SUBSTRING(@Keywords, 1, @index - 1) + '*"' SET @Keywords = SUBSTRING(@Keywords, @index + 1, LEN(@Keywords) - @index) SET @index = CHARINDEX(' ', @Keywords, 0) end -- add the last field IF LEN(@fulltext_keywords) > 0 SET @fulltext_keywords = @fulltext_keywords + ' ' + @concat_term + ' ' + '"' + SUBSTRING(@Keywords, 1, LEN(@Keywords)) + '*"' END SET @Keywords = @fulltext_keywords END END ELSE BEGIN --usual search by PATINDEX SET @Keywords = '%' + @Keywords + '%' END --PRINT @Keywords --product name SET @sql = ' INSERT INTO #KeywordProducts ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Name]) > 0 ' --localized product name SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' IF @SearchDescriptions = 1 BEGIN --product short description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[ShortDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[ShortDescription]) > 0 ' --product full description SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[FullDescription], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[FullDescription]) > 0 ' --localized product short description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''ShortDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' --localized product full description SET @sql = @sql + ' UNION SELECT lp.EntityId FROM LocalizedProperty lp with (NOLOCK) WHERE lp.LocaleKeyGroup = N''Product'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''FullDescription''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END --SKU IF @SearchSku = 1 BEGIN SET @sql = @sql + ' UNION SELECT p.Id FROM Product p with (NOLOCK) WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(p.[Sku], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, p.[Sku]) > 0 ' END IF @SearchProductTags = 1 BEGIN --product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM Product_ProductTag_Mapping pptm with(NOLOCK) INNER JOIN ProductTag pt with(NOLOCK) ON pt.Id = pptm.ProductTag_Id WHERE ' IF @UseFullTextSearch = 1 SET @sql = @sql + 'CONTAINS(pt.[Name], @Keywords) ' ELSE SET @sql = @sql + 'PATINDEX(@Keywords, pt.[Name]) > 0 ' --localized product tag SET @sql = @sql + ' UNION SELECT pptm.Product_Id FROM LocalizedProperty lp with (NOLOCK) INNER JOIN Product_ProductTag_Mapping pptm with(NOLOCK) ON lp.EntityId = pptm.ProductTag_Id WHERE lp.LocaleKeyGroup = N''ProductTag'' AND lp.LanguageId = ' + ISNULL(CAST(@LanguageId AS nvarchar(max)), '0') + ' AND lp.LocaleKey = N''Name''' IF @UseFullTextSearch = 1 SET @sql = @sql + ' AND CONTAINS(lp.[LocaleValue], @Keywords) ' ELSE SET @sql = @sql + ' AND PATINDEX(@Keywords, lp.[LocaleValue]) > 0 ' END --PRINT (@sql) EXEC sp_executesql @sql, N'@Keywords nvarchar(4000)', @Keywords END ELSE BEGIN SET @SearchKeywords = 0 END --filter by category IDs SET @CategoryIds = isnull(@CategoryIds, '') CREATE TABLE #FilteredCategoryIds ( CategoryId int not null ) INSERT INTO #FilteredCategoryIds (CategoryId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@CategoryIds, ',') DECLARE @CategoryIdsCount int SET @CategoryIdsCount = (SELECT COUNT(1) FROM #FilteredCategoryIds) --filter by attributes SET @FilteredSpecs = isnull(@FilteredSpecs, '') CREATE TABLE #FilteredSpecs ( SpecificationAttributeOptionId int not null ) INSERT INTO #FilteredSpecs (SpecificationAttributeOptionId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@FilteredSpecs, ',') DECLARE @SpecAttributesCount int SET @SpecAttributesCount = (SELECT COUNT(1) FROM #FilteredSpecs) --filter by customer role IDs (access control list) SET @AllowedCustomerRoleIds = isnull(@AllowedCustomerRoleIds, '') CREATE TABLE #FilteredCustomerRoleIds ( CustomerRoleId int not null ) INSERT INTO #FilteredCustomerRoleIds (CustomerRoleId) SELECT CAST(data as int) FROM [nop_splitstring_to_table](@AllowedCustomerRoleIds, ',') --paging DECLARE @PageLowerBound int DECLARE @PageUpperBound int DECLARE @RowsToReturn int SET @RowsToReturn = @PageSize * (@PageIndex + 1) SET @PageLowerBound = @PageSize * @PageIndex SET @PageUpperBound = @PageLowerBound + @PageSize + 1 CREATE TABLE #DisplayOrderTmp ( [Id] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) SET @sql = ' INSERT INTO #DisplayOrderTmp ([ProductId]) SELECT p.Id FROM Product p with (NOLOCK)' IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Category_Mapping pcm with (NOLOCK) ON p.Id = pcm.ProductId' END IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_Manufacturer_Mapping pmm with (NOLOCK) ON p.Id = pmm.ProductId' END IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' LEFT JOIN Product_ProductTag_Mapping pptm with (NOLOCK) ON p.Id = pptm.Product_Id' END --searching by keywords IF @SearchKeywords = 1 BEGIN SET @sql = @sql + ' JOIN #KeywordProducts kp ON p.Id = kp.ProductId' END SET @sql = @sql + ' WHERE p.Deleted = 0' --filter by category IF @CategoryIdsCount > 0 BEGIN SET @sql = @sql + ' AND pcm.CategoryId IN (SELECT CategoryId FROM #FilteredCategoryIds)' IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pcm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by manufacturer IF @ManufacturerId > 0 BEGIN SET @sql = @sql + ' AND pmm.ManufacturerId = ' + CAST(@ManufacturerId AS nvarchar(max)) IF @FeaturedProducts IS NOT NULL BEGIN SET @sql = @sql + ' AND pmm.IsFeaturedProduct = ' + CAST(@FeaturedProducts AS nvarchar(max)) END END --filter by vendor IF @VendorId > 0 BEGIN SET @sql = @sql + ' AND p.VendorId = ' + CAST(@VendorId AS nvarchar(max)) END --filter by parent grouped product identifer IF @ParentGroupedProductId > 0 BEGIN SET @sql = @sql + ' AND p.ParentGroupedProductId = ' + CAST(@ParentGroupedProductId AS nvarchar(max)) END --filter by product type IF @ProductTypeId is not null BEGIN SET @sql = @sql + ' AND p.ProductTypeId = ' + CAST(@ProductTypeId AS nvarchar(max)) END --filter by parent product identifer IF @VisibleIndividuallyOnly = 1 BEGIN SET @sql = @sql + ' AND p.VisibleIndividually = 1' END --filter by product tag IF ISNULL(@ProductTagId, 0) != 0 BEGIN SET @sql = @sql + ' AND pptm.ProductTag_Id = ' + CAST(@ProductTagId AS nvarchar(max)) END --show hidden IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND p.Published = 1 AND p.Deleted = 0 AND (getutcdate() BETWEEN ISNULL(p.AvailableStartDateTimeUtc, ''1/1/1900'') and ISNULL(p.AvailableEndDateTimeUtc, ''1/1/2999''))' END --min price IF @PriceMin > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price >= ' + CAST(@PriceMin AS nvarchar(max)) + ') ) )' END --max price IF @PriceMax > 0 BEGIN SET @sql = @sql + ' AND ( ( --special price (specified price and valid date range) (p.SpecialPrice IS NOT NULL AND (getutcdate() BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.SpecialPrice <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) OR ( --regular price (price isnt specified or date range isnt valid) (p.SpecialPrice IS NULL OR (getutcdate() NOT BETWEEN isnull(p.SpecialPriceStartDateTimeUtc, ''1/1/1900'') AND isnull(p.SpecialPriceEndDateTimeUtc, ''1/1/2999''))) AND (p.Price <= ' + CAST(@PriceMax AS nvarchar(max)) + ') ) )' END --show hidden and ACL IF @ShowHidden = 0 BEGIN SET @sql = @sql + ' AND (p.SubjectToAcl = 0 OR EXISTS ( SELECT 1 FROM #FilteredCustomerRoleIds [fcr] WHERE [fcr].CustomerRoleId IN ( SELECT [acl].CustomerRoleId FROM [AclRecord] acl with (NOLOCK) WHERE [acl].EntityId = p.Id AND [acl].EntityName = ''Product'' ) ))' END --show hidden and filter by store IF @StoreId > 0 BEGIN SET @sql = @sql + ' AND (p.LimitedToStores = 0 OR EXISTS ( SELECT 1 FROM [StoreMapping] sm with (NOLOCK) WHERE [sm].EntityId = p.Id AND [sm].EntityName = ''Product'' and [sm].StoreId=' + CAST(@StoreId AS nvarchar(max)) + ' ))' END --filter by specs IF @SpecAttributesCount > 0 BEGIN SET @sql = @sql + ' AND NOT EXISTS ( SELECT 1 FROM #FilteredSpecs [fs] WHERE [fs].SpecificationAttributeOptionId NOT IN ( SELECT psam.SpecificationAttributeOptionId FROM Product_SpecificationAttribute_Mapping psam with (NOLOCK) WHERE psam.AllowFiltering = 1 AND psam.ProductId = p.Id ) )' END --sorting SET @sql_orderby = '' IF @OrderBy = 5 /* Name: A to Z */ SET @sql_orderby = ' p.[Name] ASC' ELSE IF @OrderBy = 6 /* Name: Z to A */ SET @sql_orderby = ' p.[Name] DESC' ELSE IF @OrderBy = 10 /* Price: Low to High */ SET @sql_orderby = ' p.[Price] ASC' ELSE IF @OrderBy = 11 /* Price: High to Low */ SET @sql_orderby = ' p.[Price] DESC' ELSE IF @OrderBy = 15 /* creation date */ SET @sql_orderby = ' p.[CreatedOnUtc] DESC' ELSE /* default sorting, 0 (position) */ BEGIN --category position (display order) IF @CategoryIdsCount > 0 SET @sql_orderby = ' pcm.DisplayOrder ASC' --manufacturer position (display order) IF @ManufacturerId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' pmm.DisplayOrder ASC' END --parent grouped product specified (sort associated products) IF @ParentGroupedProductId > 0 BEGIN IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[DisplayOrder] ASC' END --name IF LEN(@sql_orderby) > 0 SET @sql_orderby = @sql_orderby + ', ' SET @sql_orderby = @sql_orderby + ' p.[Name] ASC' END SET @sql = @sql + ' ORDER BY' + @sql_orderby --PRINT (@sql) EXEC sp_executesql @sql DROP TABLE #FilteredCategoryIds DROP TABLE #FilteredSpecs DROP TABLE #FilteredCustomerRoleIds DROP TABLE #KeywordProducts CREATE TABLE #PageIndex ( [IndexId] int IDENTITY (1, 1) NOT NULL, [ProductId] int NOT NULL ) INSERT INTO #PageIndex ([ProductId]) SELECT ProductId FROM #DisplayOrderTmp GROUP BY ProductId ORDER BY min([Id]) --total records SET @TotalRecords = @@rowcount DROP TABLE #DisplayOrderTmp --prepare filterable specification attribute option identifier (if requested) IF @LoadFilterableSpecificationAttributeOptionIds = 1 BEGIN CREATE TABLE #FilterableSpecs ( [SpecificationAttributeOptionId] int NOT NULL ) INSERT INTO #FilterableSpecs ([SpecificationAttributeOptionId]) SELECT DISTINCT [psam].SpecificationAttributeOptionId FROM [Product_SpecificationAttribute_Mapping] [psam] with (NOLOCK) WHERE [psam].[AllowFiltering] = 1 AND [psam].[ProductId] IN (SELECT [pi].ProductId FROM #PageIndex [pi]) --build comma separated list of filterable identifiers SELECT @FilterableSpecificationAttributeOptionIds = COALESCE(@FilterableSpecificationAttributeOptionIds + ',' , '') + CAST(SpecificationAttributeOptionId as nvarchar(4000)) FROM #FilterableSpecs DROP TABLE #FilterableSpecs END --return products SELECT TOP (@RowsToReturn) p.* FROM #PageIndex [pi] INNER JOIN Product p with (NOLOCK) on p.Id = [pi].[ProductId] WHERE [pi].IndexId > @PageLowerBound AND [pi].IndexId < @PageUpperBound ORDER BY [pi].IndexId DROP TABLE #PageIndex END GO