DEV Community

Diógenes Polanco
Diógenes Polanco

Posted on

How to develop a plugin for NopCommerce

The add-ons are specialized modules which maximize the functionality of nopcommerce such as payment processors, shipping provider, etc.

Before launching the code, it is necessary to know the following of nopcommerce:

  • Architecture of nopCommerce
  • Types of plugin
  • Register new routes
  • The plugin structure, required files, and locations
  • Understanding Layout / Design
  • Handling "Install" and "Uninstall" methods
  • Handling requests. Controllers, models and views

Architecture of nopCommerce

It is an open source e-commerce, so you can download from github without any inconvenience and even collaborate. Projects and folders are listed in the order in which they appear in Visual Studio. To build a plugin my recommendation is not to change the source code of the libraries of the solution framework.

\Libraries\Nop.Core

The Nop.Core project contains a set of core classes for nopCommerce, such as caching, events, helpers, and business objects (for example, Order and Customer entities).

\Libraries\Nop.Data

The Nop.Data project contains a set of classes and functions for reading from and writing to a database or other data store. It helps separate data-access logic from your business objects.

\Libraries\Nop.Services

This project contains a set of core services, business logic, validations or calculations related with the data, if needed. Some people call it Business Access Layer (BAL).

Projects into \Plugins\ solution folder

\Plugins is a Visual Studio solution folder that contains plugin projects. Physically it's located in the root of your solution. But plugins DLLs are automatically copied in \Presentation\Nop.Web\Plugins\ directory which is used for already deployed plugins because the build output paths of all plugins are set to ..\..\Presentation\Nop.Web\Plugins\{Group}.{Name}\. This allows plugins to contain some external files, such as static content (CSS or JS files) without having to copy files between projects to be able to run the project.

\Presentation\Nop.Web

Nop.Web is an MVC web application project, a presentation layer for public store which also contains administration panel included as an area. If you haven't used ASP.NET before, please find more info here. This is the application that you actually run. It is the startup project of the application.

\Presentation\Nop.Web.Framework

Nop.Web.Framework is a class library project containing some common presentation things for Nop.Web project.

\Test\Nop.Core.Tests

Nop.Core.Tests is the test project for the Nop.Core project.

\Test\Nop.Services.Tests

Nop.Services.Tests is the test project for the Nop.Services project.

\Test\Nop.Tests

Nop.Tests is a class library project containing some common test classes and helpers for other test projects. It does not have any test.

\Test\Nop.Web.MVC.Tests

Nop.Web.MVC.Tests is the test project for the presentation layer projects.

Types of plugin

The type of its complement is defined by the interface that implements the main complement class.

Discount Rules (IDiscountRequirementRule)For defining requirements that must be met my an order for a discount to be applied

Exchange Rate Prodider (IExchangeRateProvider)For getting the exchange rates for currencies

External Authentication Method (IExternalAuthenticationMethod)For adding new external authentication methods (e.g. “Login with Facebook”)

Widgets (IWidgetsPlugin)For when your plugin doesn’t fit any of the other types

Payment Method (IPaymentMethod)For example, Paypal or Worldpay

Shipping Rate Computation Method (IShippingRateComputationMethod)For calculating shipping rates (many plugins of this time call web services provided by couriers to retrieve rates)

Pickup points is an option providing customers the flexibility to select a point where they can receive parcels.

Tax Provider (ITaxProvider)For calculating tax rates

Widget (IWidgetPlugin)Widgets are blocks of content shown throughout the site. The default theme contains many widget “areas”.
You can assign your new widget to one of these areas, and wherever the area is referenced in the theme, your widget will appear.

Register new routes

ASP.NET Core routing is responsible for assigning incoming browser requests via a get to the MVC controller. NopCommerce has an IRouteProvider interface that is used for route registration during application startup. All main routes are registered in the RouteProvider class located in the Nop.Web project.

public partial class RouteProvider : IRouteProvider
{
public void RegisterRoutes(IRouteBuilder routeBuilder)
{
//home page
routeBuilder.MapLocalizedRoute("HomePage", "", new { controller = "Home", action = "Index" });
}
}

The plugin structure, required files, and locations

First thing you need to do is to create a new "Class Library" project in the solution. It's a good practice to place all plugins into \Plugins directory in the root of your solution (do not mix up with \Plugins subdirectory located in \Nop.Web directory which is used for already deployed plugins). It's a good practice to place all plugins into "Plugins" solution folder (you can find more information about solution folders here).

A recommended name for a plugin project is "Nop.Plugin.{Group}.{Name}". {Group} is your plugin group (for example, "Payment" or "Shipping"). {Name} is your plugin name (for example, "PayPalStandard"). For example, PayPal Standard payment plugin has the following name: Nop.Plugin.Payments.PayPalStandard. But please note that it's not a requirement. And you can choose any name for a plugin. For example, "MyCustomPlugin".

Once the plugin project is created you have to open its .csproj file in any text editor and replace its content with the following one:

<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net461</TargetFramework>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'">
<OutputPath>..\..\Presentation\Nop.Web\Plugins\MyCustomPlugin</OutputPath>
<OutDir>$(OutputPath)</OutDir>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
<OutputPath>..\..\Presentation\Nop.Web\Plugins\MyCustomPlugin</OutputPath>
<OutDir>$(OutputPath)</OutDir>
</PropertyGroup>
<!-- This target execute after "Build" target -->
<Target Name="NopTarget" AfterTargets="Build">
<!-- Delete unnecessary libraries from plugins path -->
<MSBuild Projects="$(MSBuildProjectDirectory)\..\..\Build\ClearPluginAssemblies.proj" Properties="PluginPath=$(MSBuildProjectDirectory)\$(OutDir)" Targets="NopClear" />
</Target>
</Project>

The next step is creating a  plugin.json file required for each plugin. This file contains meta information describing your plugin. Just copy this file from any other existing plugin and modify it for your needs. For example, PayPal Standard payment plugin has the following plugin.json file:

{
"Group": "Widgets",
"FriendlyName": "My Custom Plugin",
"SystemName": "Widgets.MyCustomPlugin",
"Version": "1.42",
"SupportedVersions": [ "4.00" ],
"Author": "Diogenes Polanco",
"DisplayOrder": 1,
"FileName": "Nop.Plugin.Widgets.MyCustomPlugin.dll",
"Description": "This plugin is my Custom plugin"
}
view raw plugin.json hosted with ❤ by GitHub

Add a class MyCustomPlugin.cs and use the following code: 

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Web.Routing;
using Nop.Core.Plugins;
using Nop.Services.Common;
namespace Nop.Plugin.Widgets.MyCustomPlugin
{
public class MyCustomPlugin : BasePlugin, IWidgetPlugin, IAdminMenuPlugin
{
private readonly ISettingService _settingService;
private readonly IWebHelper _webHelper;
public MyCustomPlugin(ISettingService settingService, IWebHelper webHelper)
{
this._webHelper = webHelper;
this._settingService = settingService;
}
public override string GetConfigurationPageUrl()
{
return $"{_webHelper.GetStoreLocation()}Admin/MyCustomPlugin/Configure";
}
}
}

Handling "Install" and "Uninstall" methods

Some plugins can require additional logic during plugin installation. For example, a plugin can insert new locale resources. So open your IPlugin implementation (in most case it'll be derived from BasePlugin class) and override the following methods:

  • Install. This method will be invoked during plugin installation. You can initialize any settings here, insert new locale resources, or create some new database tables (if required).
  • Uninstall. This method will be invoked during plugin uninstallation.
public override void Install()
{
var settings = new MyCustomPluginSettings()
{
UseSandbox = true,
Message = "Hello World"
};
_settingService.SaveSetting(settings);
_localizationService.AddOrUpdatePluginLocaleResource("Plugin.Widgets.MyCustomPlugin.UseSandbox", "UseSandbox");
_localizationService.AddOrUpdatePluginLocaleResource("Plugin.Widgets.MyCustomPlugin.Message", "Message");
base.Install();
}
public override void Uninstall()
{ _localizationService.DeletePluginLocaleResource("Plugin.Widgets.MyCustomPlugin.UseSandbox");
_localizationService.DeletePluginLocaleResource("Plugin.Widgets.MyCustomPlugin.Message");
base.Uninstall();
}

Understanding Layout / Design

What are layouts? Every web developer/designer wants to maintain a consistent look and feel across all of the pages within the website. Back in the days, the concept of "Master Pages" was introduced in ASP.NET 2.0 which helps in maintaining a consistent look of the website by mapping it with .aspx pages.

Razor also supports this similar concept with a feature called "Layouts". Basically, it allows you to define a common site template and then inherit its look and feel across all the views/pages on your website.

In nopCommerce, there are 2 different kinds of layouts:

  • _ColumnsOne.cshtml
  • _ColumnsTwo.cshtml

All these 2 layouts are inherited from one main layout called: _Root.cshtml. The _Root.cshtml itself is inherited from _Root.Head.cshtml_Root.Head.cshtml is the file you need to look into if you are linked css stylesheet and jquery files (you can add/link more .css and .js files here). The location of all these layouts in nopCommerce is as follows: nopCommerce root directory/Views/Shared/.... If you are using source code version then: \Presentation\Nop.Web\Views\Shared\...

Handling requests. Controllers, models and views

Add a class MyCustomPluginController.cs (in folder “Controllers” within your plugin) and use the following code:   

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Web.Mvc;
using Nop.Web.Framework.Controllers;
namespace Nop.Plugin.Widgets.MyCustomPlugin.Controllers
{
[AdminAuthorize]
public class MyCustomPluginController : BasePluginController
{
public ActionResult Configure()
{
return View("~/Plugins/Widgets.MyCustomPlugin/Views/MyCustomPlugin/Configure.cshtml");
}
}
}

Add a Configure.cshtml (View) – In this case we are creating a blank plugin.

Now, we will add the menu item by adding this code in “MyCustomPlugin.cs”:

public void ManageSiteMap(SiteMapNode rootNode)
{
var menuItem = new SiteMapNode()
{
SystemName = "MyCustomPlugin",
Title = "MyCustomPlugin Title",
ControllerName = "MyCustomPlugin",
ActionName = "Configure",
Visible = true,
RouteValues = new RouteValueDictionary() { { "area", null } },
};
var pluginNode = rootNode.ChildNodes.FirstOrDefault(x => x.SystemName == "Third party plugins");
if (pluginNode != null)
pluginNode.ChildNodes.Add(menuItem);
else
rootNode.ChildNodes.Add(menuItem);
}

We can see our menu item here:

Let’s create a “WidgetsMyCustomPluginViewComponent” file inside the “Components” folder of your custom plugin and simply paste the default home view (PublicInfo.cshtml) from Nop.Web like this:

using System;
using Microsoft.AspNetCore.Mvc;
using Nop.Core;
using Nop.Core.Caching;
using Nop.Plugin.Widgets.MyCustomPlugin.Models;
using Nop.Services.Configuration;
using Nop.Services.Media;
using Nop.Web.Framework.Components;
namespace Nop.Plugin.Widgets.MyCustomPlugin.Components
{
[ViewComponent(Name = "WidgetsMyCustomPlugin")]
public class WidgetsMyCustomPluginViewComponent : NopViewComponent
{
public static string ViewComponentName => "WidgetsMyCustomPlugin";
private readonly IStoreContext _storeContext;
private readonly ISettingService _settingService;
public WidgetsMyCustomPluginViewComponent(IStoreContext storeContext, ISettingService settingService)
{
this._storeContext = storeContext;
this._settingService = settingService;
}
public IViewComponentResult Invoke(string widgetZone, object additionalData)
{
var MyCustomPluginSettings = _settingService.LoadSetting<MyCustomPluginSettings>(_storeContext.CurrentStore.Id);
var model = new PublicInfoModel
{
Message = MyCustomPluginSettings.Message,
UseSandbox = MyCustomPluginSettings.UseSandbox
};
if (!model.UseSandbox && string.IsNullOrEmpty(model.Message))
//no Use Sandbox
return Content("");
return View("~/Plugins/Widgets.MyCustomPlugin/Views/PublicInfo.cshtml", model);
}
}
}

Note: Make sure the output directory (in properties) for your home view (PublicInfo.cshtml) is defined as “Copy if newer”. Rebuild your plugin and try going to your public store (since your plugin is already installed)

We should see the plugin view overriding the default home view like this:

I let you share the source code link on GitHub

Top comments (3)

Collapse
 
jgilbertcastro profile image
Jesus Gilbert

Excellent, is more easy to develop a pluggin for wordpress!!!!!!!!!

Collapse
 
rahulsapkota profile image
Rahul Sapkota • Edited

nice article, do you also have article which does data access as well?

Collapse
 
im_sivakumar profile image
Sivakumar Vinayagam

nopCommerce supports mysql,sqlserver and postgres out of the box. run the open source code and you will figure it out easily.