Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/bdcrops/module-simplenews
BDC_SimpleNews full-fledged Module Step by Step
https://github.com/bdcrops/module-simplenews
certified-associate-developer-exam certified-professional-developer console-comand declarative-programming declarative-schema declarative-schema-in-magento2 magento-2-certified magento-certification magento2-extension observer php-framework plugins professional-developer-plus simplenews study-notes weapi
Last synced: about 2 months ago
JSON representation
BDC_SimpleNews full-fledged Module Step by Step
- Host: GitHub
- URL: https://github.com/bdcrops/module-simplenews
- Owner: bdcrops
- Created: 2019-09-04T06:27:15.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2020-01-01T07:20:49.000Z (about 5 years ago)
- Last Synced: 2024-04-09T19:22:51.131Z (9 months ago)
- Topics: certified-associate-developer-exam, certified-professional-developer, console-comand, declarative-programming, declarative-schema, declarative-schema-in-magento2, magento-2-certified, magento-certification, magento2-extension, observer, php-framework, plugins, professional-developer-plus, simplenews, study-notes, weapi
- Language: JavaScript
- Homepage: https://www.bdcrops.com
- Size: 3.06 MB
- Stars: 3
- Watchers: 2
- Forks: 4
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Magento 2 SimpleNews module
**Magento 2 Module development** or **Magento 2 SimpleNews Module** Create a full-fledged Module Step by Step. You could just follow my code to create this module from the scratch. Or you can directly download the compressed tar file and install it and play it.
## PREREQUISITES
- No prerequisites
- May be More benefited Who are know Magento 2 Basic frontend , backend & install local/server .
- Fundamentals of Magento 2 Development or Module Development as a first step.## Goal:
- Develop Full-fledged Module Step by Step .
- Magento 2 Certified [Associate](mcad.md)/[Professional](mcpd.md) Developer exam Preparation hands-on practice.## Magento 2 SimpleNews Module Step By Step (BDCrops)
### [PartA: News Module Basic (Architecture & Customization)](#PartA)
- [Step 2A1: Create a directory for the module like above format](#Step2A1)
- [Step 2A2: Declare module by using configuration file module.xml](#Step2A2)
- [Step 2A3: Register module by registration.php & composer.json](#Step2A3)
- [Step 2A4: Configure declarative schema (create table etc/db_schema.xml)](#Step2A4)
- [Step 2A5: Schema whitelist (etc/db_schema_whitelist.json) ](#Step2A15)
- [Step 2A6: Enable the module](#Step2A6)
- [Step 2A7: Develop data & schema patches(Installing & upgrading data)](#Step2A7)
- [Step 2A8: Create Model News for business Logic](#Step2A8)
- [Step 2A9: Create Model's ResourceModel to handle real database transaction](#Step2A9)
- [Step 2A10: Create Model's collection class](#Step2A10)
- [Step 2A11: Setup the frontend route](#Step2A11)
- [Step 2A12: Create IndexController](#Step2A12)### [Part B: News Module for Back End](#PartB)
- [Step 2B1: Setup Module's backend configuration](#Step2B1)
- [Step 2B2: Create a custom source model](#Step2B2)
- [Step 2B3: Create a role for this config section](#Step2B3)
- [Step 2B4: Set some default value for configuration options](#Step2B4)
- [Step 2B5: Create a Helper Data class](#Step2B5)
- [Step 2B6: Create the menu for Magento backend](#Step2B6)
- [Step 2B7: Create backend route file](#Step2B7)
- [Step 2B8: Update the acl.xml to add more roles](#Step2B8)
- [Step 2B9: Create layout for grid](#Step2B9)
- [Step 2B10: Create layout for Grid Container](#Step2B10)
- [Step 2B11: Create layout for ajax load](#Step2B11)
- [Step 2B12: Create news status option file](#Step2B12)
- [Step 2B13: Create News Block for backend](#Step2B13)
- [Step 2B14: Create Grid block file for Ajax load](#Step2B14)
- [Step 2B15: Create backend controller for child action class to extend](#Step2B15)
- [Step 2B16: Create Backend Action file Index.php](#Step2B16)
- [Step 2B17: Create another Action for ajax](#Step2B17)
- [Step 2B18: Create layout file simplenews_news_edit.xml for edit form](#Step2B18)
- [Step 2B19: Create the layout for create form](#Step2B19)
- [Step 2B20: Create a form container block](#Step2B20)
- [Step 2B21: create a block for the left-side tabs](#Step2B21)
- [Step 2B22: Create a block for Form information](#Step2B22)
- [Step 2B23: Create a block to declare the fields for the edit form](#Step2B23)
- [Step 2B24: Create a controller action for create a new News](#Step2B24)
- [Step 2B25: Create Edit Action for the Edit form](#Step2B25)
- [Step 2B26: A Save Action for the edit form](#Step2B26)
- [Step 2B27: Delete Action for the edit Form](#Step2B27)
- [Step 2B28: The mass delete action the grid list](#Step2B28)
- [Step 2B29: Backend Menu and Grid List](#Step2B29)### [Part C : News Module for Front End](#PartC)
- [Step 2C1: Create Layout file for page handle](#Step2C1)
- [Step 2C2: Create another layout file by update the previous layout](#Step2C2)
- [Step 2C3: Create Block NewList file](#Step2C3)
- [Step 2C4: Create frontend template file list.phtml](#Step2C4)
- [Step 2C5: Create an abstract class by extending Magento Core Action class](#Step2C5)
- [Step 2C6: Update Index Controller by extends the abstract class 'New.php'](#Step2C6)
- [Step 2C7: Create a layout file for news detail page](#Step2C7)
- [Step 2C8: Create News view action](#Step2C8)
- [Step 2C9: create view news block](#Step2C9)
- [Step 2C10: Create news view template file](#Step2C10)
- [Step 2C11: Create CSS file for styling the frontend Page](#Step2C11)
- [Step 2C12: Create Latest New Block](#Step2C12)
- [Step 2C13: Create a Block for positioning the latest news: Left or Right](#Step2C13)
- [Step 2C14: Create the template file for Latest News](#Step2C14)
- [Step 2C15: Frontend view for the module](#Step2C15)### [Part D : News Console/Command](#PartD)
- [Step 2D1: Adding a new command Dependency Injection](#Step2D1)
- [Step 2D2: Adding a new command class](#Step2D3)
- [Step 2D3: Adding a new command Helper class](#Step2D3)### [Part E : Create/Set / Configure Custom Cron Jobs](#PartE)
- [Step 2E.1: Create crontab.xml ](#Step2E1)
- [Step 2E.2: defined to run the execute method of class](#Step2E3)
- [Step 2E.3: Run all cron jobs ](#Step2E3)
- [Step 2E.4: Create custom cron group ](#Step2E4)
- [Step 2E.5: Run new cron group cron jobs ](#Step2E5)### [Part F : Create REST WEB API](#PartF)
- [Step2F1: Web API Routes/Configuration](#Step2F1)
- [Step2F2: Define Interface– etc/di.xml](#Step2F2)
- [Step2F3: Declare API Interface](#Step2F3)
- [Step2F4: Declare Data API Interface](#Step2F4)
- [Step2F5: Create Model](#Step2F5)
- [Step2F6: Communicating with new API call](#Step2F6)
- [Step2F7: Adding ACL Web API](#Step2F7)### [PartG: Dependency Injection configuration ](#PartG)
- [Step2G.1: DI Preference,Arguments & Virtual Types Implements](#Step2G1)
- [Step2G.2: DI Observer Implements](#Step2G2)
- [Step2G.3: DI Plugins (Interceptors)](#Step2G3)### [PartH : Customization Layout Configuration & JavaScript ](#PartH)
- [Step2H.1: Layout Configuration](#Step2H1)
- [Step2H.2: Customization JavaScript Map & Mixin](#Step2H2)### [PartI : UI Components Library](#PartI)
- [Step2I1: Rendering Grid(collections & listing component configuration)](#Step2I1)
- [Step2I.2: Rendering Form ()](#Step2I2)### [PartJ : Entity-Attribute-Value (EAV)](#PartJ)
- [Step2J1: Rendering Grid(collections & listing component configuration)](#Step2J1)
- [Step2J2: ](#Step2J2)***
## Part A : News Module for Basic [Go to Top](#top)
#### Explain Magento 2 Basic Directory Structure ?
- app – is used for additional elements; as a rule, app contains the following subdirectories:
- code – contains the installed modules;
- design – contains the installed themes. The frontend themes are located at the frontend folder; themes for admin panel – in the adminhtml folder;
- etc – contains the Magento 2 configuration files;
- i18n – contains the installed language packs.
- bin – contains Magento file responsible for the execution of CLI-commands in Magento 2.
- dev – contains Integration and Functional test files.
- generated – utilized for generated classes in Magento 2.
- lib – contains Magento 2 libraries and non-module based code.
- phpserver – contains Router.php file, implemented to realize the built-in PHP server.
- pub – used for static files storage:
- errors – contains files responsible for displaying errors in the browser (this behavior is by default disabled);
- media – contains all media-files from the website;
- static – contains the generated theme and module files.
- var – contains temporary files, like:
- cache – contains all the cached objects, except for pages;
- composer_home – root directory of the installation wizard;
- log – stores Magento 2 logs;
- page_cache – contains pages cached with Full Page Cache;
- view_preprocessed – contains minified templates and compiled LESS.
- vendor – contains core files of Magento 2. Moreover, this directory can contain the additionally installed modules. You should perform operations with components from this directory via Composer.#### What is Model View ViewModel (MVVM) Architecture ?
![](doc/MVVMPattern.png)
- Model: Holds business logic of application & depends on an associated class—the ResourceModel—for database access. Models rely on service contracts to expose their functionality to other layers of application.
- View: Structure & layout of what a user sees on a screen - the actual HTML. This is achieved in the PHTML files distributed with modules. PHTML files are associated to each ViewModel in the Layout XML files, which would be referred to as binders in the MVVM dialect. The layout files might also assign JavaScript files to be used in the final page.
- ViewModel: Interacts with Model layer, exposing only necessary information to View layer handled by the module’s Block classes. Note that this was usually part of the Controller role of an MVC system. On MVVM, the controller is only responsible for handling the user flow, meaning that it receives requests and either tells the system to render a view or to redirect the user to another route.#### Magento 2 architecture is split into 4 (PDSP)layers?
![](doc/archi_diagrams_layers_alt4.jpg)
- Persistence layer: describes resource model, which is responsible for extracting and modifying data in the database using CRUD requests.Additional business logic capabilities are also implemented here, for example, data validation and database functions implementation.
- Domain layer: responsible for the business logic, which does not contain resource-specific or database-specific information. Domain layer can also include service contracts.Each data model at the level of domain layer depends on the resource model, which is responsible for accessing the database.
- Service layer: interlayer between presentation layer and domain layer. It implements service contracts, which are defined using PHP interfaces. Service contracts allow to add or change business logic resource model using dependency injection file (di.xml). Service layer is also used for granting access to API (REST/SOAP or other modules).Service interface is declared in /Api namespace of the module.
Data (entity) interface is declared in /Api/Data. Data entities are data structures passed to and returned from service interfaces.- Presentation Layer: upper layer. It contains all the View elements (including layouts, blocks, templates, css, js) and controllers.Presentation Layer usually calls service layer using service contracts. But, depending on the implementation, it may overlap with business logic.
#### Magento has 5 areas types?
- Magento Admin (adminhtml): entry point for this area is index.php or pub/index.php. The Admin panel area includes the code needed for store management. The /app/design/adminhtml directory contains all the code for components you’ll see while working in the Admin panel.
- Storefront (frontend): entry point for this area is index.php or pub/index.php. The storefront (or frontend) contains template and layout files that define the appearance of your storefront.
- Basic (base): used as a fallback for files absent in adminhtml and frontend areas.
- Cron (crontab): In cron.php, the \Magento\Framework\App\Cron class always loads the 'crontab' area.
You can also send requests to Magento using the SOAP and REST APIs. These two areas:
- Web API REST (webapi_rest): entry point for this area is index.php or pub/index.php. The REST area has a front controller that understands how to do URL lookups for REST-based URLs.
- Web API SOAP (webapi_soap): entry point for this area is index.php or pub/index.php.#### Module folder holds one part of the architecture, as follows?
- Api or Api/Data: Service contracts, defining service interfaces & data interfaces
- Adapter:Classes follow adapter pattern & wrap around classes from third-party libraries allow to use functionality from third-party libraries in code by converting the third-party class interfaces into an interface that is expected by native code.( module-search/Adapter/)
- Block: ViewModels of our MVVM architecture
- Collector: module-deploy/Collector/Collector.php
- Command: directory is used for storing the PHP files that are responsible for console programs execution. In our case, Console/Command/ImagesResizeCommand.php processes commands for product images resizing.
- Controller: Responsible for handling the user’s flow while interacting with the system
- Config: module-deploy/Config/BundleConfig.php
- Cron: We use the directory to store the files, which are later executed on the Cron launching.
- CustomerData: directory contains PHP files responsible for processing information for sections. Magento 2 has a special functionality, which allows for processing, updating and transferring the information asynchronously.
- etc: Configuration XML files module defines itself & its parts (routes, models, blocks, observers, and cron jobs) within this folder, also be used by non-core modules to override the functionality of core modules.
- [etc/acl.xml](etc/acl.xml)
- [etc/adminhtml/menu.xml](etc/adminhtml/menu.xml)
- [etc/adminhtml/system.xml](etc/adminhtml/system.xml)
- etc/{area}/routes.xml
- etc/{area}/events.xml
- etc/crontab/events.xml
- etc/config.xml
- etc/cron_groups.xml
- etc/crontab.xml
- [etc/db_schema.xml](etc/db_schema.xml)
- [etc/di.xml](etc/di.xml)
- etc/events.xml
- etc/module.xml
- etc/setup/events.xml
- etc/webapi.xml
- etc/webapi_rest/di.xml
- etc/webapi_rest/events.xml
- etc/webapi_soap/events.xml
[ReadDevDoc](https://devdocs.magento.com/guides/v2.3/config-guide/config/config-files.html)- Exception: (module-sales/Exception/)
- Files: Sample file (module-inventory-import-export/Files/)
- fixtures: Sample Data module (module-sales-sample-data/fixtures/orders.csv)
- Gateway: (module-paypal/Gateway)
- Helper: Classes that hold code used in more than one application layer. For example, in the Cms module, helper classes are responsible for preparing HTML for presentation to the browser.
- i18n: Holds internationalization CSV files, used for translation
- Indexer: IndexHandler (module-inventory-indexer/Indexer)
- Model: For Models and ResourceModels
- Observer: Holds Observers, or Models which are “observing” system events. Usually, when such an event is fired, the observer instantiates a Model to handle the necessary business logic for such an event.
- Package: module-deploy/Package
- Pricing: Final price model (module-msrp-grouped-product/Pricing)
- Process: module-deploy/Process
- Plugin: directory comprises plugin files allow us to modify certain module’s functions if necessary described in the configuration file: vendor/magento/module-catalog/etc/di.xml
- SearchAdapter: module-elasticsearch/SearchAdapter
- ReportXml :vendor/magento/module-analytics/ReportXml
- Setup: Migration classes, responsible for schema & data creation
- Service: [exam] (module-media-storage/Service/ImageResize.php,module-deploy/ or module-catalog-url-rewrite/Service/V1/StoreViewService.php )
- src : vendor/magento/magento2-functional-testing-framework/src/Magento/
- Strategy: module-deploy/Strategy
- Source: module-deploy/Source
- Test: Unit tests
- Ui: Elements such as grids & forms used in admin application
- view – Layout (XML) files & template (PHTML) files for front-end & admin application contains template files, CSS and JS files, module media files. These files are located in subfolders depending on the area of use: adminhtml, frontend or base (common files for the administrative and frontal parts of the site). These subdirectories, in turn, including static view files, design templates, email templates, and layout files:
- view/{area}/email – contains emails templates.
- view/{area}/layout – contains files for layout modifications.
- view/{area}/page_layout – contains files for page_layout modifications.
- view/{area}/templates – contains files of the module templates (phtml).
- view/{area}/ui_component – contains XML-files of the UI module components.
- view/{area}/ui_component/templates
- view/{area}/web – contains CSS, JS, static and media module files.
- view/{area}/web/js – contains js
- view/{area}/web/template – contains html
- view/{area}/requirejs-config.js- ViewModel: (module-sales/ViewModel)
### Step 2A1: Create a directory for the module like above format
In this module, we will use `BDCrops` for Vendor name and `SimpleNews` for ModuleName. So we need to make this folder: `app/code/BDC/SimpleNews`
### Step 2A2: Declare module by using configuration file module.xml
Magento 2 looks for configuration information for each module in that module’s etc directory. We need to create folder etc and add module.xml:
- Create [etc/module.xml](/etc/module.xml) And the content for this file:Source
```
```
In this file, we register a module with name `BDC_SimpleNews` and the version is `1.0.0`.
#### Notes[u can skip]:
- Magento 2 need Two Mandatory File to run/activate Module etc/module.xml & registration.php### Step 2A3: Register module by registration.php
All Magento 2 module must be registered in the Magento system through the magento ComponentRegistrar class. This file will be placed in module root directory.
In this step, we need to create this file:
- Create [registration.php](registration.php) and insert this following code into it:
Source```
\Magento\Framework\Component\ComponentRegistrar::register(
\Magento\Framework\Component\ComponentRegistrar::MODULE,
'BDC_SimpleNews', __DIR__
);
```
Modules in vendor folder would update using composer And all the modules in app/code would not be updated through composer That's why when you need to override any module you add it in app/code
- Create [composer.json](composer.json) and insert this following code into it:
Source
```
{
"name": "bdc/module-simplenews",
"description": "BDCrops SimpleNews module for Magento 2 extensions.",
"type": "magento2-module",
"version": "1.0.3",
"license": [
"OSL-3.0",
"AFL-3.0"
],
"authors": [{
"name": "Abdul Matin",
"email": "[email protected]",
"company": "BDCrops Inc"
}
],
"homepage": "https://www.bdcrops.com",
"autoload": {
"files": [
"registration.php"
],
"psr-4": {
"BDC\\SimpleNews\\": ""
}
}
}```
### Step 2A4: Configure declarative schema (create table schema Installation file)
- Create [etc/db_schema.xml](etc/db_schema.xml) & insert this following code into it:
Source
```
```
#### Note:
[Tutarials module-declarative](https://github.com/bdcrops/module-declarative)### Step 2A5: Schema whitelist (etc/db_schema_whitelist.json)
You will not be able to run a declarative mode without creating a schema whitelist.
Note: it is recommended to generate a new whitelist for every release for the double-check purposes.Before running the upgrade command you need to add your schema to db_whitelist_schema.json file by running the following command.
For that, you need a /etc/db_schema_whitelist.json file that will store all the content added with declarative schema. To generate this file, run:![db_schema](https://github.com/bdcrops/BDC_Declarative/blob/master/view/adminhtml/web/images/whitelist.png)
```
php bin/magento setup:db-declaration:generate-whitelist [options]
php bin/magento setup:db-declaration:generate-whitelist --module-name=vendor_module
php bin/magento setup:db-declaration:generate-whitelist --module-name=BDC_SimpleNews
```Now, there are db_whitelist_schema.json file will be create in /vendor/module/etc folder.
![db_whitelist_schema](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/db_schema_whitelist.png)### Step 2A6: Enable the module
By finish above step, you have created an empty module. Now we will enable it in Magento environment.Before enable the module, we must check to make sure Magento has recognize our module or not by enter the following at the command line:
~~~
php bin/magento module:status
~~~If you follow above step, you will see this in the result:
~~~
List of disabled modules:
BDC_SimpleNews
~~~This means the module has recognized by the system but it is still disabled. Run this command to enable it:
~~~
php bin/magento module:enable BDC_SimpleNews
~~~The module has enabled successfully if you saw this result:
~~~
The following modules has been enabled:
- BDC_SimpleNews
~~~This’s the first time you enable this module so Magento require to check and upgrade module database. We need to run this comment:
~~~
php bin/magento setup:upgrade
~~~Now you can check under `Stores -> Configuration -> Advanced -> Advanced` that the module is present.
Also you can check Database Table from PhpMyAdmin or Your Favorite tools:
![Table db_schema](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/dbTableCreatedDeclarativeSchema.png)
### Step 2A7: Develop data and schema patches (Insert data Installing and upgrading data)
Since in the old method, we used to write scripts in Install Schema or Upgrade schema when a table was created, but now in the new version, this will be done through Patch system.A data patch is a class that contains data modification instructions. It is defined in a / /Setup/Patch/Data/.php file and implements \Magento\Setup\Model\Patch\DataPatchInterface.
A schema patch contains custom schema modification instructions. These modifications can be complex.
It is defined in a//Setup/Patch/Schema/.php file and implements \Magento\Setup\Model\Patch\SchemaPatchInterface.
So to add data to the bdc_simplenews table create AddData.php file inside folder BDC/SimpleNews/Setup/Patch/Data and write the following code- Create [Setup/Patch/Data/AddData.php](Setup/Patch/Data/AddData.php)
Source
```
news = $news;
}
public function apply(){
$newsData = [];
$newsData['title'] = "BDC News Head1";
$newsData['summary'] = "BDC News Summary";
$newsData['description'] = "BDCrops Inc description evulation of bangladesh";
//$newsData['status'] = 1;$this->news->addData($newsData);
$this->news->getResource()->save($this->news);}
public static function getDependencies() { return []; }
public static function getVersion() { return '2.0.0'; }
public function getAliases() { return []; }}
```
### Step 2A8: Create Model News for business Logic
We need to create these files to insert, update, delete and get data in the database.
- Create model file: [Model/News.php](Model/News.php):
Source
```
_init('BDC\SimpleNews\Model\Resource\News');
}/**
* Loading news data
*
* @param mixed $key
* @param string $field
* @return $this
*/
public function load($key, $field = null) {
if ($field === null) {
$this->_getResource()->load($this, $key, 'id');
return $this;
}
$this->_getResource()->load($this, $key, $field);
return $this;
}
}```
#### Note: basic concepts of models, resource models, and collections
CRUD Models in Magento 2 can manage data in database easily, you don’t need to write many line of code to create a CRUD. CRUD is stand for Create, Read, Update and Delete.
The Magento ORM is used by the Repository implementations that are part of the Magento 2 service contracts. This is an important variation from Magento 1, as a module should no longer rely on other modules using a specific ORM, and instead of it use only the entity repositories. The service contracts will be covered in more details in the second part of the article.The Magento ORM is built around models, resource models, and resource collections. The Magento ORM elements are following:- Models are data and behavior, representing entities.
- Resource Models are data mappers for the storage structure.
- Collections are encapsulating sets of models and related functionality, such as filtering, sorting, and paging.
- Resources include database connections via adapters.
#### Note:native Magento save/load
The ORM gives you a possibility to create, load, update, and delete data in a database. A collection in Magento is a class that implements both the IteratorAggregate and the Countable PHP5 SPL interfaces. Collections are widely used in Magento to store a set of objects of a specific type.#### Note: Models
Models are like a black box which provides a layer of abstraction on top of the resource models. The fetching, extraction, and manipulation of data occur through models. As a rule of thumb, every entity we create (i.e. every table we create in our database) should have its own model class. Every model extends the Magento\Framework\Model\AbstractModelclass, which inherits the \Magento\Framework\DataObjectclass, hence, we can call the setDataand getData functions on our model, to get or set the data of a model respectively. class only has one method, _ construct(), when we call the _ init()method, and pass the resource model’s name as its paramete### Step 2A9: Create Model's ResourceModel to handle real database transaction
- Create resource model [Model/Resource/News.php](Model/Resource/News.php):
Source
```
_init('bdc_simplenews', 'id'); }
}```
#### Note: Resource Model
All of the actual database operations are executed by the resource model. Every model must have a resource model, since all of the methods of a resource model expects a model as its first parameter. All resource models must extend the Magento\Framework\Model\ResourceModel\Db\AbstractDbclass.
here also has one method, <__ construct>, where we call the <_ initmethod>, and pass two parameters to it. The name of the table in the database, and the name of the primary column in that table.
Resource Model. In Magento 2, the model class defines the methods an end-user-programmer will use to interact with a model’s data. A resource model class contains the methods that will actually fetch the information from the database. Each CRUD model in Magento 2 has a corresponding resource model class.Every CRUD resource model class extends the Magento\Framework\Model\ResourceModel\Db\AbstractDb class. This base class contains the basic logic for fetching information from a single database table.
For a basic model like ours, the only thing a resource model must do is call the _ init method from _ construct. The _ init method for a resource model accepts two arguments. The first is the name of the database table (bdc_simplenews), and the second is the ID column for the model (id). While it’s beyond the scope of this article, Magento 2’s active record implementation contains no method for linking tables via primary keys. How to use multiple database tables is up to each individual module developer, and a resource model will typically contain the SQL generating methods needed to fetch information from related tables.### Step 2A10: Create Model's collection class
- Create collection [Model/Resource/News/Collection.php](Model/Resource/News/Collection.php ):
Source
```
_init('BDC\SimpleNews\Model\News', 'BDC\SimpleNews\Model\Resource\News');
}
}```
#### Note: Collection
Collections are used when we want to fetch multiple rows from our table. Meaning collections
- group of models. Collections can be used when we want to
- Fetch multiple rows from a table
- Join tables with our primary table
- Select specific columns
- Apply a WHERE clause to our query
- Use GROUP BY or ORDER BY in our queryWith a model and resource model, you have everything you need to fetch and save individual models into the database. However, there are times where you’ll want to fetch multiple models of a particular type. To solve this problem, every CRUD model in Magento 2 has a corresponding resource model collection. A collection collects individual models. It’s considered a resource model since it builds the SQL code necessary to pull information from a database table.
All collections in Magento 2 extend the base \Magento\Framework\Model\ResourceModel\Db\Collection\AbstractCollection collection class. Like a model and resource model, a collection resource model must call the _ init method. A collection resource model’s _ init method accepts two arguments. The first is the model that this collection collects. The second is that collected model’s resource model.
We create a new Magento 2 block an inject \Magento\Catalog\Model\ResourceModel\Product\CollectionFactory class. This is needed to get a collection from factory. A getProductCollection returns a new product collection. This method does the following:- create a new collection from collection factory
- filter attributes (collumns)
- This time we want to load all data (* ), you can also add a comma seperated list of attribute names. Only this attributes get loaded – lazy loading optimizes your database select statement.
- filter by attribute values
It is possible to filter your collection by loaded attribute values. For example all products with price small than 1000 $. My example use setPageSize() to only load a given number of products.### Step 2A11: Request Flow Processing- Frontend Route
#### Create custom route on Frontend route?
We will find how to create a frontend route, admin route and how to use route to rewrite controller.
- Routes.xml
To register a frontend route, we must create a routes.xml file:
File: app/code/BDC/SampleNews/etc/frontend/routes.xml
```
```
Please look into the code, you will see it’s very simple to register a route. You must use the standard router for the frontend. This route will have a child which define the module for it and 2 attributes:The id attribute is a unique string which will identify this route. You will use this string to declare the layout handle for the action of this module
The frontName attribute is also a unique string which will be shown on the url request. For example, if you declare a route like this:
The url to this module should be:http://example.com/index.php/samplenews/controller/action
And the layout handle for this action is: samplenews_controller_action.xml So with this example path, you must create the action class in this folder:
~~~
{namespace}/{module}/Controller/{Controller}/{Action}.phphttp://example.com///
~~~The Router is used to assign a URL to a corresponding controller and action. In this module, we need to create a route for frontend area. So we need to add this file:
- Create [etc/frontend/routes.xml](etc/frontend/routes.xml):
Source
~~~
~~~
After define the route, the URL path to our module will be: `http://example.com/news/`
#### Note: Controllers, Routers and Responses
- Routers: define name for a module which we can use in the url to find the module and execute the controller action.
- Controllers:Controllers in Magento 2 differ from typical controllers in MVC applications. Magento 2 controllers are responsible for only one specific URL and contain only one execute method. This method is responsible for returning result object and occasional processing of input POST data. All controllers inherit \Magento\Framework\App\Action\Action class. The required controller is searched in the Base Router, and then it’s called in the Front Controller.
- Responses: The controller in Magento 2 can return several response types depending on the purpose and the necessary result.- Frontend route: Please look into the code, you will see it’s very simple to register a route. You must use the standard router for the frontend. This route will have a child which define the module for it and 2 attributes:
1. The id attribute is a unique string which will identify this route. You will use this string to declare the layout handle for the action of this module
2. The frontName attribute is also a unique string which will be shown on the url request. For example, if you declare a route like this: The url to this module should be:
http://example.com/index.php/nwes/controller/action
And the layout handle for this action is: samplenews_controller_action.xml So with this example path, you must create the action class in this folder: {namespace}/{module}/Controller/{Controller}/{Action}.php### Step 2A12: Create IndexController
- Factory Object
We are done with creating the database table, CRUD model, resource model and collection. So how to use them?In this part, we will talk about Factory Object for model. As you know in OOP, a factory method will be used to instantiate an object. In Magento, the Factory Object do the same thing.
The Factory class name is the name of Model class and append with the ‘Factory’ word. So for our example, we will have NewsFactory class. You must not create this class. Magento will create it for you. Whenever Magento’s object manager encounters a class name that ends in the word ‘Factory’, it will automatically generate the Factory class in the var/generation folder if the class does not already exist. You will see the factory class:
```
use BDC\SimpleNews\Model\NewsFactory;
var/generation///Model/ClassFactory.php
var/generation/BDC/SimpleNew/Model/NewsFactory.php
```
To instantiate a model object we will use automatic constructor dependency injection to inject a factory object, then use factory object to instantiate the model object.#### Magento 2 Registry & Register?
Magento 2 authorizes you to register global variable that supports the static registry method. Magento 1, as well as Magento 2, authorize you to register global variable that supports the static registry method.
To implement that, maybe you used to work with Mage::register() and Mage::registry() in Magento 1, but now in Magento 2 platform, there is a difference in running the registry. You will be required to apply \Magento\Framework\Registry, that accepts the settings and the registry of the restored data. However, first of all, you need to learn how to create or use the own custom registry and also show you how to retrieve global Magento 2 registry objects like current product, category, cms page, cms block, etc.
And that is lucky because all of them will be referred here. The topic today will help you be familiar with Magento 2 registry objects.#### How to get and set custom attribute in registry / register
```
/**
* @var \Magento\Framework\Registry
*/protected $_registry;
/**
* ...
* ...
* @param \Magento\Framework\Registry $registry,
*/
public function __construct(
...,
...,
\Magento\Framework\Registry $registry,
...) {
$this->_registry = $registry;
...
...
}/**
* Setting custom variable in registry to be used
*
*/
public function setCustomVariable() {
$this->registry->register('custom_var', 'Added Value');
}/**
* Retrieving custom variable from registry
* @return string
*/
public function getCustomVariable() {
return $this->registry->registry('custom_var');
}
```- Create controller [Controller/Index/Index.php](Controller/Index/Index.php):
Source
```
_modelNewsFactory = $modelNewsFactory;
}
public function execute(){
/**
* When Magento get your model, it will generate a Factory class
* for your model at var/generaton folder and we can get your
* model by this way
*/
$newsModel = $this->_modelNewsFactory->create();// Load the item with ID is 1
$item = $newsModel->load(1);
var_dump($item->getData());// Get news collection
$newsCollection = $newsModel->getCollection();
// Load all data of collection
var_dump($newsCollection->getData());
}
}```
After define the Controller, the URL path to our module will be: `http://example.com/news/` below data
![NewsDataFrontend](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/newsFrontendData.png)
#### Notes:
- Action class: extensions of the Action class that a router returns on matched requests. The execute() function in these classes contain the logic for dispatching requests.Each Action should implement one or more Magento\Framework\App\Action\HttpHTTP MethodActionInterface to declare which HTTP request methods it can process.Magento has a form key validation in place for all POST non-AJAX requests - if your Action doesn’t need that validation or you want to modify it you can implement CsrfAwareActionInterface.
If you need to forward a request to another action in your class, use the _ forward() function. Example:
```
$this->_forward('action', 'controller', 'Other_Module')
```#### Responses: create a frontend controller with different response types
- Page Result (\Magento\Framework\View\Result\Page) is the most common type of response. By returning this object, the controller starts the standard page rendering based on the corresponding XML layout handle.
```
public function __construct(
$pageFactory Magento\Framework\View\Result\PageFactory
) {
$this->pageResultFactory = $pageFactory
}
public function execute()
{
return $this->pageResultFactory->create();
}
```
- Raw Result (\Magento\Framework\Controller\Result\Raw) is used if you want to return a string to the browser without using Magento layout and view rendering.
```
public function __construct(
Magento\Framework\Controller\Result\Raw $rawResultFactory ,
) {
$this->rawResultFactory = $rawResultFactory;
}
public function execute()
{
$result = $this->rawResultFactory->create();
$result->setHeader('Content-Type', 'text/xml');
$result->setContents(');
return $result;
}
```- Forward Result (\Magento\Framework\Controller\Result\Forward) allows to call another method/controller without changing the URL or redirecting.
```
public function __construct(
Magento\Framework\Controller\Result\Forward\Factory $resultForwardFactory
) {
$this->resultForwardFactory = $resultForwardFactory;
}
public function execute()
{
$result = $this->resultForwardFactory->create();
$result->forward('noroute');
return $result;
}
```
- Redirect Result (\Magento\Framework\Controller\Result\Redirect) is used when a user needs to be redirected to a different URL.
```
public function __construct(
Magento\Framework\Controller\Result\Redirect\Factory $resultRedirectFactory
) {
$this->resultRedirectFactory = $resultRedirectFactory;
}
public function execute()
{
$result = $this->resultRedirectFactory->create();
$result->setPath('*/*/index');
return $result;
}
```***
### Part B: News Module for Back End [Go to Top](#top)
### What is Scope?
If your Magento installation has a hierarchy of websites, stores, or views, you can set the context, or “scope” of a configuration setting to apply to a specific part of the installation. The context of many database entities can also be assigned a specific scope to determine how it is used in the store hierarchy. To learn more, see: Product Scope and Price Scope.
Some configuration settings such as postal code, have a [global] scope because the same value is used throughout the system. The [website] scope applies to any stores below that level in the hierarchy, including all stores and their views. Any item with the scope of [store view] can be set differently for each store view, which is typically used to support multiple languages.
Unless the store is running in Single Store Mode, the scope of each configuration setting appears in small text below the field label. If your installation includes multiple websites, stores or views, you should always choose the Store View where the settings apply before making any changes.
![](https://docs.magento.com/m2/ce/user_guide/images/images/scope-multisite.png)
### What is Scope Settings?
- Global: System-wide settings and resources that are available throughout the Magento installation.
- Website: Settings and resources that are limited to the current website. Each website has a default store.
- Store: Settings and resources that are limited to the current store. Each store has a default root category (main menu) and default store view.
- Store View: Setting and resources that are limited to the current store view.#### How to write & get config values by scope?
An important, but less good documented feature of Magento 2 is how to write and get config values by scope. You will find tons of code samples on how do this globally. Sometime you need to do different settings for different stores programmatically. So here is how this works.Magento saves all adminhtml settings in core_config table in your Magento database. There you can get values by its path, a string which indicates path to and a variable name. With this path, you can get or set values by Magento 2 core methods. For this you need to use:
- \Magento\Framework\App\Config\Storage\WriteInterface
to write config values to database
- \Magento\Framework\App\Config\ScopeConfigInterface
to read config values from database#### How Write store config values by scope?
The following sample code shows how to write store config values by scope:
```
class WriteConfig {
protected $_logger;
protected $_storeManager;
protected $_configWriter;public function __construct(
\Psr\Log\LoggerInterface $logger,
\Magento\Framework\App\Config\Storage\WriterInterface $configWriter,
\Magento\Store\Model\StoreManagerInterface $storeManager ){
$this->_logger = $logger;
$this->_configWriter = $configWriter;
$this->_storeManager = $storeManager;
}public function setConfig($value) {
//for all websites
$websites = $this->_storeManager->getWebsites();
$scope = "websites";
foreach($websites as $website) {
echo $website->getId().":\n";$this->_configWriter->save('my_section/something/configvaluename', $value, $scope, $website->getId());
}return $this;
}
}
```
You just need to call setConfig() method with a given value. This method stores this value into a defined path for all websites. So it generates a new setting (line in core_config table) for each defined website. This is done by using third and fourth param on save method. You use a unique path, a value, a scope and the id of this scope. If you do not use scope, you will wirte the value to default (store id 0). You can store values to scopes “website” or “store“.#### How Read store config values by scope?
Now it is time to read the data by store. You can do this with the following sample code:
```
class ReadConfig
{
protected $_scopeConfig;public function __construct(
\Magento\Framework\App\Helper\Context $context,
\Magento\Framework\App\Config\ScopeConfigInterface $scopeConfig
){
$this->_scopeConfig = $scopeConfig;
parent::__construct($context);
}public function getConfig() {
return $this->_scopeConfig->getValue("my_section/something/configvaluename", "websites");
}
}
```It is quite easy, you only need to use getValue() method and add a second param with scope (here we use website scope). This will return the stored value for the current website.
#### How to Use Configuration Files in Magento 2?
There are 2 main places for storing configuration values In Magento 2: database (the core_config_data table) and XML files. The configurations, stored in the database can be changed via the administrator panel, while the data, located in the XML files are of a technical nature and can be changed only by a developer.It’s easy to use configuration files in Magento 2. Configuration files include:
- app/etc/config.php — contains the declaration of all modules;
- app/etc/env.php — describes the array, which contains the front end name for the back end panel, the data for connection to the database, the table prefixes, the current store mode, the types and statuses of the cache.
These files are generated during the Magento 2 setup. You can change them directly editing the file. However, running terminal commands bin/magento is considered to be best practice.#### Explain Utilize Configuration XML and Variables Scope in Magento 2?
- etc/config.xml — contains default option values from Stores > Configuration in the admin panel menu. This menu can be configured at system.xml;
- di.xml — contains configurations for the dependency injection;
- etc/events.xml — a list of observers and events;
- etc/routes.xml — a list of routers;
- etc/config.xml — contains the default values for the module settings Stores > Configuration;
- etc/acl.xml — adds module resources to a resource tree that allows you to configure access for different users.
- etc/crontab.xml — adds and configures the task for the cronjob;
- etc/module.xml — announces the name and the version of the module, as well as its dependencies on other modules;
- etc/widget.xml — stores the widget settings;
- etc/indexer.xml — announces a new kind of indexing. It specifies the view_id parameter, which points at the views described in - etc/mview.xml;
etc/mview.xml — describes the representations of all the indices described in etc/indexer.xml;
- etc/webapi.xml — defines web API components, which service method to use and which resource to connect for a specific request;
- etc/view.xml — contains the properties of product images;
- etc/product_types.xml — describes types of products in a store;
- etc/product_options.xml — describes the types of options, that can have products and classes to render them;
- etc/extension_attributes.xml — a new ability to add a custom attribute appeared in Magento 2. This file describes the attribute, its type, which can be simple or complex and represent an interface;
- etc/catalog_attributes.xml — groups attributes;
- etc/adminhtml/system.xml — can only apply to the admin area, adds tabs to Stores > Configuration, describes sections and fields of a form;
- etc/adminhtml/menu.xml — can only apply to the admin area, adds an item to the admin panel menu.#### Explain loading of configurations occurs in three stages?
- Loading system-level configurations. Loading the files necessary to run Magento 2, such as config.php;
- Loading global area configurations. Loading the files located in the app/etc/Magento 2 directory, such as di.xml, as well as the files related to the global scope and located directly in the etc/module folders.
- Loading configurations for specific areas. Loading the files located in the folders etc/adminhtml or etc/frontend.
Configuration files are connected according to their complete xPaths. Specific attributes are defined in the $idAttributes array as identifiers. After 2 files are connected, they contain all the nodes and values from the original files. The second XML file either adds or replaces the nodes of the first XML file.#### What interfaces provides Magento/Framework/Config for developers?
Magento/Framework/Config provides the following interfaces for developers:
- \Magento\Framework\Config\DataInterface — returns a value by key within the scope, connects the configuration data to an object;
- \Magento\Framework\Config\ScopeInterface — identifies the current scope;
- \Magento\Framework\Config\FileResolverInterface — identifies a set of files that must be read;
- \Magento\Framework\Config\ReaderInterface — reads configuration data;
- \Magento\Framework\Config\ConverterInterface — converts a DOM object to an array;
- \Magento\Framework\Config\SchemaLocatorInterface — determines the path to validation schemes;
- \Magento\Framework\Config\ValidationStateInterface — determines the current validation status.#### Class groups used to load XML:
- Config is used to access the configuration values;
- Reader is used to read the file;
- SchemaLocator stores the path to the validation scheme.#### How many types of validation for XML configuration ?
Magento 2 provides two types of validation for XML configuration files: validation before a merging and validation after a merging. It can be either the same or different schemes.
To create a custom configuration file, you need to create the following elements:- XSD schema
- Config PHP file
- Config reader
- Schema locator
- Converter#### Explain product_types.xml module of Magento_Catalog as an example ?
product_types.xml module of Magento_Catalog as an example of a custom configuration file. Each module can add its own product type using the product_types.xml file and these files will be validated and merged.
- We will start with creating an XSD file. product_types.xsd validation scheme is used in Magento_Catalog before merging, product_types_merged.xsd is used for a merged XML file.
- Create a PHP configuration file to access the data from the file. In our example it is Config.php. To provide access to data from the product_types.xml file, it implements the Magento\Catalog\Model\ProductType\ConfigInterface interface and all its methods.
- In Config.php of the constructor, we need to get a reader class. In our case, it is Magento\Catalog\Model\ProductType\Config\Reader. This is a small class with the definition of the $_ idAttributes property. In the constructor of the $fileName variable, we define the name of the XML file.
- Magento\Catalog\Model\ProductType\Config\SchemaLocator implements two methods: getSchema and getPerFileSchema and returns the path to merged XSD and ordinary XSD files. In the constructor, we define these paths in the properties of $_ schema and $_ perFileSchema.
- Creating a converter class. In our example: Magento\Catalog\Model\ProductType\Config\Converter implements \Magento\Framework\Config\ConverterInterface and realizes the convert method, which converts the node’s DOM tree to an array.
That is it concerning configuration XML, variables scope and configuration files in Magento 2.### Step 2B1: Setup Module's Backend /System configuration
#### What is System configuration?
The system.xml is a configuration file which is used to create configuration fields in Magento 2 System Configuration. You will need this if your module has some settings which the admin needs to set. You can go to Store -> Setting -> Configuration to check how it look like.The magento 2 system configuration page is divided logically in few parts: Tabs, Sections, Groups, Fields.![](doc/themes14-1.png)
#### How to Set default value?
Each field in system.xml after create will not have any value. When you call them, you will receive ‘null’ result. So for the module, we will need to set the default value for the field and you will call the value without go to config, set value and save it. This default value will be saved in config.xml which is located in etc folder. Let’s create it for this simple configuration: [etc/config.xml](etc/config.xml)```
{value}
```
#### How to Get value from configuration ?
First all of let’s save value and flush cache, then you can get saved value from database.
In the system.xml, we have added 2 fields: enable and display_text. So the path should be:
samplenews/general/enable
samplenews/general/display_text
Simple calling:ex
```
$this->scopeConfig->getValue('samplenews/general/enable', \Magento\Store\Model\ScopeInterface::SCOPE_STORE);
$this->scopeConfig->getValue('samplenews/general/display_text', \Magento\Store\Model\ScopeInterface::SCOPE_STORE);
```- Create file[etc/adminhtml/system.xml](etc/adminhtml/system.xml)
Purpose: This file will declare your configurations in Stores > Settings > Configuration section) and insert this following code into it:Source
```
BDC
Simple News
bdc
BDC_SimpleNews::system_config
General Settings
Enable in frontend
Magento\Config\Model\Config\Source\Yesno
Head title
Fill head title of news list page at here
required-entry
Lastest news block position
BDC\SimpleNews\Model\System\Config\LastestNews\Position
```
#### Note: [System configuration](https://inviqa.com/blog/how-use-system-configuration-and-helpers-magento-2) :
System configuration values in Magento 2 are stored in the core_config_data database table, which is exactly the same as in Magento 1. But the xml config files differ.
The system.xml is a configuration file which is used to create configuration fields in Magento 2 System Configuration. the system config file is at etc/adminhtml/system.xmlThe system.xml is a configuration file which is used to create configuration fields in Magento 2 System Configuration. You will need this if your module has some settings which the admin needs to set. You can go to Store -> Setting -> Configuration to check how it look like.
#### Note:- [MCAD: 5.4 Set up a menu item](https://belvg.com/blog/magento-2-custom-system-configuration.html)
- How do you add a new menu item to a given tab?
- How do you add a new tab to the Admin menu?### Step 2B2:Create Custom Source Model
- Create file [Model/System/Config/LastestNews/Position.php](Model/System/Config/LastestNews/Position.php):
Source```
__('Left'),
self::RIGHT => __('Right'),
self::DISABLED => __('Disabled')
];
}
}```
### Step 2B3: Create ACL Role for Config Section
- Create file [etc/acl.xml](etc/acl.xml)
(Purpose: This file will create a role for your configuration section) and insert this following code into it:
Source```
```
#### Note:- 5.3 Define / identify basic terms and elements of ACL
- How would you add a new ACL resource to a new entity?
- How do you manage the existing ACL hierarchy?### Step 2B4: Set some default value for configuration options
- Create file [etc/config.xml](etc/config.xml) and insert this following code into it:
Source```
1
BDC - Simple News
1
```
### Step 2B5: Create a Helper Data class
#### Why need to Create Helper?
Magento 2, the Helper can be called in controllers, models, views and even in other helpers.
Helpers can be considered as global and always available elements. They can even be created as single objects’ instances. Besides, they can be called everywhere once you inject them in the class. Helpers are mainly created to offer methods for the most common functionalities. For instance, you can use helpers to build logs in the application of Magento.
Magento 2 Helper Class includes various functions and methods which are used commonly throughout the application. All the methods which have been declared as Helpers can be called anywhere including file, model, block, controller class or from another helper in Magento 2.#### What is Helper?
In the early version of Magento 2, a Helper Factory is available, which enables developers to instantiate helper methods. Besides, you can use the below code to use ObjectManager to instantiate the Helper Factory.
```
$object_manager = \Magento\Core\Model\ObjectManager::getInstance();
$helper_factory = $object_manager->get('\Magento\Core\Model\Factory\Helper');
$helper = $helper_factory->get('\Magento\Core\Helper\Data');
```
However, this code still exist some problems. Luckly, a better concept has been introducted which is Dependency Injection in Magento 2.Using this concept, the environment will create and provide you an object instead of instantiating it. For instance, if a class is written like the following:
```
class Helper{
public function __contruct(Helper $xyz){
$this->xyz= $xyz;
}
}
```
In the Helper class constructor, an object of Helper class is auto-created and assigned the reference, $xyz. This is Dependency Injection.Via this concept, high-value loose coupling modules together concept is provided by Magento 2. If you want to inject it into a specific class, just add an object to the constructor of it. However, you need to remember that you cannot inject one dependency twice.
- Create file: [Helper/Data.php](Helper/Data.php) and insert this following code into it:
Source```
_scopeConfig = $scopeConfig;
}/**
* Check for module is enabled in frontend
*
* @return bool
*/
public function isEnabledInFrontend($store = null){
return $this->_scopeConfig->getValue(
self::XML_PATH_ENABLED,
ScopeInterface::SCOPE_STORE
);
}/**
* Get head title for news list page
*
* @return string
*/
public function getHeadTitle() {
return $this->_scopeConfig->getValue(
self::XML_PATH_HEAD_TITLE,
ScopeInterface::SCOPE_STORE
);
}/**
* Get lastest news block position (Left, Right, Disabled)
*
* @return int
*/
public function getLastestNewsBlockPosition() {
return $this->_scopeConfig->getValue(
self::XML_PATH_LASTEST_NEWS,
ScopeInterface::SCOPE_STORE
);
}
}```
### Step 2B6: Create the menu for Magento backend
#### Expalin some attributes:id , title, module, parent ,action, resource ? asBelow Code:
```
```- id attribute is the identifier for this note. It’s a unique string and should follow the format: {Vendor_ModuleName}::{menu_description}.
- title attribute is the text which will be shown on the menu bar.
- module attribute is defined the module which this menu is belong to.
- sortOrder attribute is defined the position of the menu. Lower value will display on top of menu.
- parent attribute is an Id of other menu node. It will tell Magento that this menu is a child of another menu. In this example, we have parent=”BDC_SampleNews::samplenews”, so we - know this menu “Manage News” is a child of “Hello World” menu and it will show inside of Hello World menu.
- action attribute will define the url of the page which this menu link to. As we talk above, the url will be followed this format {router_name}{controller_folder}{action_name}. - In this example, this menu will link to the module SampleNews, controller News and action Index
- resource attribute is used to defined the ACL rule which the admin user must have in order to see and access this menu. We will find more detail about ACL in other topic.- Create file: [etc/adminhtml/menu.xml](etc/adminhtml/menu.xml) (Purpose: The menu item of your module will be declared here) and insert this following code into it:
Source
```
```
![MenuLinkAdmin](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/adminhtmlMenu.png)
### Step 2B7: Request Flow Processing / Create backend route file
#### How to Create Admin route ?
This route will be same as the frontend route but you must declare it in adminhtml folder with router id is admin.
File: app/code/BDC/SampleNews/etc/adminhtml/routes.xmlThe url of the admin page is the same structure with frontend page, but the admin_area name will be added before route_frontName to recognize this is a admin router. For example, the url of admin cms page:
http://example.com/index.php/admin/bdc_samplenews/controller/action
The controller action for admin page will be added inside of the folder Controller/Adminhtml. For example for above url:
```
{namespace}/{module}/Controller/Adminhtml/{Controller}/{Action}.php
```- Create file [etc/adminhtml/routes.xml](etc/adminhtml/routes.xml)
Purpose: The router of your module for backend will be declared here insert this following code into it:Source
```
```
#### Note:
- Admin Route: This route will be same as the frontend route but you must declare it in adminhtml folder with router id is admin. /etc/adminhtml/routes.xml
The url of the admin page is the same structure with frontend page, but the admin_area name will be added before route_frontName to recognize this is a admin router. For example, the url of admin cms page:
```
http://example.com/index.php/admin/simplenews/controller/action {namespace}/{module}/Controller/Adminhtml/{Controller}/{Action}.php
```### Step 2B8: Update the acl.xml to add more roles
#### How to add/Create our module to ACL role?
As in the Admin Menu and System Configuration article, you saw that we alway have a resource attribute when create it. Now we will register that resources to the system, so Magento can realize and let us set a role for them. To register the resource, we use the acl.xml file which located in
```
app/code/{namespace}/{module}/etc/acl.xml
```
#### ACL Rules for Developers?
As a module developer, ACL rules present a few interesting challenges. First, there are several places that you, as a module developer, are expected to add ACL rule checks to your module. A few examples- Every URL endpoint/controller in the admin application must implement an _ isAllowed method that determines if a user can access the URL endpoint.
- Every Menu Item in the left hand navigation also has a specific ACL rule that controls whether or not the menu displays for the logged in user. This is often the same rule from _ isAllowed)- Every configuration field in System -> Configuration has a specific ACL rule that controls whether or not the menu displays
Despite being required fields, there are no hard and fast rules as to how a module developer should setup and structure their own rules. Also, a module developer will likely want additional rules that are specific to their module. This article can’t answer these hard questions for you, but we will show you how to check the current user against a specific ACL rule, look up ID values for existing rules, and how to create your own tree of ACL rules.
#### Explain Magento_Backend::admin,Id Title, sortOrder?
- Magento_Backend::admin: Our resource will be placed as child of Magento_Backend::admin. Each resource will have an Id, title and sortOrder attribute:
- Id: attribute is the identify of this resource. You can use this when define resource in Admin menu, configuration and limit access to your module controller. This is a unique string and should be in this format: Vendor_ModuleName::resource_name.
- Title: attribute is the label of this resource when showing in resource tree.
- sortOrder: attribute define the position of this resource in tree.- Open this file [etc/acl.xml](etc/acl.xml) and modify the source code into here like this:
Source
```
```
### Step 2B9: Create layout for grid
- Create file: [view/adminhtml/layout/simplenews_news_index.xml](view/adminhtml/layout/simplenews_news_index.xml) (Purpose: This file is used to declare grid container block) and insert this following code into it:
Source
```
```
### Step 2B10: Create layout for Grid Container
- Create file: app/code/BDC/SimpleNews/view/adminhtml/layout/simplenews_news_grid_block.xml (Purpose: This file is used to declare the content of grid block) and insert this following code into it:
Source```
newsGrid
BDC\SimpleNews\Model\Resource\News\Collection
id
desc
true
true
1
id
news
Delete
*/*/massDelete
Are you sure you want to delete?
*/*/edit
getId
ID
number
id
id
Title
title
Summary
summary
Status
status
options
action
Action
action
getId
false
false
stores
true
Edit
*/*/edit
id
col-actions
col-actions
```
### Step 2B11: Create layout for ajax load
- Create file: app/code/BDC/SimpleNews/view/adminhtml/layout/simplenews_news_grid.xml (Purpose: This file is used to declare the content of grid when you use ajax to reload the grid) and insert this following code into it:
Source
```
```
### Step 2B12: Create news status option file
- Create file: app/code/BDC/SimpleNews/Model/System/Config/Status.php (Purpose: This file is used to get News status options) and insert this following code into it:
Source
```
__('Enabled'),
self::DISABLED => __('Disabled')
];return $options;
}
}```
### Step 2B13: Create News Block for backend
- Create file: app/code/BDC/SimpleNews/Block/Adminhtml/News.php (Purpose: This is the block file of grid container) and insert this following code into it:Source
```
__('Enabled'),
self::DISABLED => __('Disabled')
];return $options;
}
}```
### Step 2B14: Create Status
- Create file: app/code/BDC/SimpleNews/Model/System/Config/Status.php (Purpose: check) and insert this following code into it:Source
```
__('Enabled'),
self::DISABLED => __('Disabled')
];return $options;
}
}```
### Step 2B15: Create a backend controller file for child action class to extend
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News.php (Purpose: I use this file as a root controller and the action classes will be extended this controller) and insert this following code into it:
Source
```
_controller = 'adminhtml_news';
$this->_blockGroup = 'BDC_SimpleNews';
$this->_headerText = __('Manage News');
$this->_addButtonLabel = __('Add News');
parent::_construct();
}
}```
### Step 2B16: Create Backend Action file Index.php
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/Index.php (Purpose: This is the index action) and insert this following code into it:
Source
```
getRequest()->getQuery('ajax')) {
$this->_forward('grid');
return;
}
/** @var \Magento\Backend\Model\View\Result\Page $resultPage */
$resultPage = $this->_resultPageFactory->create();
$resultPage->setActiveMenu('BDC_SimpleNews::main_menu');
$resultPage->getConfig()->getTitle()->prepend(__('Simple News'));return $resultPage;
}
}```
### Step 2B17: Create another Action for ajax
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/Grid.php (Purpose: This is the grid action which is used for loading grid by ajax) and insert this following code into it:
Source
```
_resultPageFactory->create();
}
}```
![allNews](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/allNews.png)
### Step 2B18: Create layout file simplenews_news_edit.xml for edit form
- Create file: app/code/BDC/SimpleNews/view/adminhtml/layout/simplenews_news_edit.xml (Purpose: This file is used to declare blocks which used on editing page) and insert this following code into it:
Source```
```
### Step 2B19: Create the layout for create form
- Create file: app/code/BDC/SimpleNews/view/adminhtml/layout/simplenews_news_create.xml and insert this following code into it:
Source```
```
### Step 2B20: Create a form container block
- Create file: app/code/BDC/SimpleNews/Block/Adminhtml/News/Edit.php (Purpose: This is the block file of form container) and insert this following code into it:
Source
```
_coreRegistry = $registry;
parent::__construct($context, $data);
}/**
* Class constructor
*
* @return void
*/
protected function _construct()
{
$this->_objectId = 'id';
$this->_controller = 'adminhtml_news';
$this->_blockGroup = 'BDC_SimpleNews';parent::_construct();
$this->buttonList->update('save', 'label', __('Save'));
$this->buttonList->add(
'saveandcontinue',
[
'label' => __('Save and Continue Edit'),
'class' => 'save',
'data_attribute' => [
'mage-init' => [
'button' => [
'event' => 'saveAndContinueEdit',
'target' => '#edit_form'
]
]
]
],
-100
);
$this->buttonList->update('delete', 'label', __('Delete'));
}/**
* Retrieve text for header element depending on loaded news
*
* @return string
*/
public function getHeaderText()
{
$newsRegistry = $this->_coreRegistry->registry('simplenews_news');
if ($newsRegistry->getId()) {
$newsTitle = $this->escapeHtml($newsRegistry->getTitle());
return __("Edit News '%1'", $newsTitle);
} else {
return __('Add News');
}
}/**
* Prepare layout
*
* @return \Magento\Framework\View\Element\AbstractBlock
*/
protected function _prepareLayout()
{
$this->_formScripts[] = "
function toggleEditor() {
if (tinyMCE.getInstanceById('news_content') == null) {
tinyMCE.execCommand('mceAddControl', false, 'news_content');
} else {
tinyMCE.execCommand('mceRemoveControl', false, 'news_content');
}
};
";return parent::_prepareLayout();
}
}```
### Step 2B21: create a block for the left-side tabs
- Create file: app/code/BDC/SimpleNews/Block/Adminhtml/News/Edit/Tabs.php (Purpose: This file will declare tabs at left column of the editing page) and insert this following code into it:
Source
```
_coreRegistry = $registry;
parent::__construct($context, $data);
}/**
* Class constructor
*
* @return void
*/
protected function _construct()
{
$this->_objectId = 'id';
$this->_controller = 'adminhtml_news';
$this->_blockGroup = 'BDC_SimpleNews';parent::_construct();
$this->buttonList->update('save', 'label', __('Save'));
$this->buttonList->add(
'saveandcontinue',
[
'label' => __('Save and Continue Edit'),
'class' => 'save',
'data_attribute' => [
'mage-init' => [
'button' => [
'event' => 'saveAndContinueEdit',
'target' => '#edit_form'
]
]
]
],
-100
);
$this->buttonList->update('delete', 'label', __('Delete'));
}/**
* Retrieve text for header element depending on loaded news
*
* @return string
*/
public function getHeaderText()
{
$newsRegistry = $this->_coreRegistry->registry('simplenews_news');
if ($newsRegistry->getId()) {
$newsTitle = $this->escapeHtml($newsRegistry->getTitle());
return __("Edit News '%1'", $newsTitle);
} else {
return __('Add News');
}
}/**
* Prepare layout
*
* @return \Magento\Framework\View\Element\AbstractBlock
*/
protected function _prepareLayout()
{
$this->_formScripts[] = "
function toggleEditor() {
if (tinyMCE.getInstanceById('news_content') == null) {
tinyMCE.execCommand('mceAddControl', false, 'news_content');
} else {
tinyMCE.execCommand('mceRemoveControl', false, 'news_content');
}
};
";return parent::_prepareLayout();
}
}```
### Step 2B22: Create a block for Form information
- Create file: app/code/BDC/SimpleNews/Block/Adminhtml/News/Edit/Form.php (Purpose: This file will declare form information) and insert this following code into it:
Source
```
_coreRegistry = $registry;
parent::__construct($context, $data);
}/**
* Class constructor
*
* @return void
*/
protected function _construct()
{
$this->_objectId = 'id';
$this->_controller = 'adminhtml_news';
$this->_blockGroup = 'BDC_SimpleNews';parent::_construct();
$this->buttonList->update('save', 'label', __('Save'));
$this->buttonList->add(
'saveandcontinue',
[
'label' => __('Save and Continue Edit'),
'class' => 'save',
'data_attribute' => [
'mage-init' => [
'button' => [
'event' => 'saveAndContinueEdit',
'target' => '#edit_form'
]
]
]
],
-100
);
$this->buttonList->update('delete', 'label', __('Delete'));
}/**
* Retrieve text for header element depending on loaded news
*
* @return string
*/
public function getHeaderText()
{
$newsRegistry = $this->_coreRegistry->registry('simplenews_news');
if ($newsRegistry->getId()) {
$newsTitle = $this->escapeHtml($newsRegistry->getTitle());
return __("Edit News '%1'", $newsTitle);
} else {
return __('Add News');
}
}/**
* Prepare layout
*
* @return \Magento\Framework\View\Element\AbstractBlock
*/
protected function _prepareLayout()
{
$this->_formScripts[] = "
function toggleEditor() {
if (tinyMCE.getInstanceById('news_content') == null) {
tinyMCE.execCommand('mceAddControl', false, 'news_content');
} else {
tinyMCE.execCommand('mceRemoveControl', false, 'news_content');
}
};
";return parent::_prepareLayout();
}
}```
### Step 2B23: Create a block to declare the fields for the edit form
- Create file: app/code/BDC/SimpleNews/Block/Adminhtml/News/Edit/Tab/Info.php (Purpose: This file will declare fields in form) and insert this following code into it:Source
```
_coreRegistry = $registry;
parent::__construct($context, $data);
}/**
* Class constructor
*
* @return void
*/
protected function _construct()
{
$this->_objectId = 'id';
$this->_controller = 'adminhtml_news';
$this->_blockGroup = 'BDC_SimpleNews';parent::_construct();
$this->buttonList->update('save', 'label', __('Save'));
$this->buttonList->add(
'saveandcontinue',
[
'label' => __('Save and Continue Edit'),
'class' => 'save',
'data_attribute' => [
'mage-init' => [
'button' => [
'event' => 'saveAndContinueEdit',
'target' => '#edit_form'
]
]
]
],
-100
);
$this->buttonList->update('delete', 'label', __('Delete'));
}/**
* Retrieve text for header element depending on loaded news
*
* @return string
*/
public function getHeaderText()
{
$newsRegistry = $this->_coreRegistry->registry('simplenews_news');
if ($newsRegistry->getId()) {
$newsTitle = $this->escapeHtml($newsRegistry->getTitle());
return __("Edit News '%1'", $newsTitle);
} else {
return __('Add News');
}
}/**
* Prepare layout
*
* @return \Magento\Framework\View\Element\AbstractBlock
*/
protected function _prepareLayout()
{
$this->_formScripts[] = "
function toggleEditor() {
if (tinyMCE.getInstanceById('news_content') == null) {
tinyMCE.execCommand('mceAddControl', false, 'news_content');
} else {
tinyMCE.execCommand('mceRemoveControl', false, 'news_content');
}
};
";return parent::_prepareLayout();
}
}```
### Step 2B24: Create a controller action for create a new News
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/NewAction.php (Purpose: This is the new action) and insert this following code into it:Source
```
_forward('edit');
}
}```
### Step 2B25: Create Edit Action for the Edit form
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/Edit.php (Purpose: This is the edit action for editing news page) and insert this following code into it:
Source
```
getRequest()->getParam('id');
/** @var \BDC\SimpleNews\Model\News $model */
$model = $this->_newsFactory->create();if ($newsId) {
$model->load($newsId);
if (!$model->getId()) {
$this->messageManager->addError(__('This news no longer exists.'));
$this->_redirect('*/*/');
return;
}
}// Restore previously entered form data from session
$data = $this->_session->getNewsData(true);
if (!empty($data)) {
$model->setData($data);
}
$this->_coreRegistry->register('simplenews_news', $model);/** @var \Magento\Backend\Model\View\Result\Page $resultPage */
$resultPage = $this->_resultPageFactory->create();
$resultPage->setActiveMenu('BDC_SimpleNews::main_menu');
$resultPage->getConfig()->getTitle()->prepend(__('Simple News'));return $resultPage;
}
}```
### Step 2B26: A Save Action for the edit form
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/Save.php (Purpose: This is the save action) and insert this following code into it:
Source
```
getRequest()->getParam('id');
/** @var \BDC\SimpleNews\Model\News $model */
$model = $this->_newsFactory->create();if ($newsId) {
$model->load($newsId);
if (!$model->getId()) {
$this->messageManager->addError(__('This news no longer exists.'));
$this->_redirect('*/*/');
return;
}
}// Restore previously entered form data from session
$data = $this->_session->getNewsData(true);
if (!empty($data)) {
$model->setData($data);
}
$this->_coreRegistry->register('simplenews_news', $model);/** @var \Magento\Backend\Model\View\Result\Page $resultPage */
$resultPage = $this->_resultPageFactory->create();
$resultPage->setActiveMenu('BDC_SimpleNews::main_menu');
$resultPage->getConfig()->getTitle()->prepend(__('Simple News'));return $resultPage;
}
}```
### Step 2B27: Delete Action for the edit Form
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/Delete.php (Purpose: This is the delete action) and insert this following code into it:
Source
```
getRequest()->getParam('id');if ($newsId) {
/** @var $newsModel \Mageworld\SimpleNews\Model\News */
$newsModel = $this->_newsFactory->create();
$newsModel->load($newsId);// Check this news exists or not
if (!$newsModel->getId()) {
$this->messageManager->addError(__('This news no longer exists.'));
} else {
try {
// Delete news
$newsModel->delete();
$this->messageManager->addSuccess(__('The news has been deleted.'));// Redirect to grid page
$this->_redirect('*/*/');
return;
} catch (\Exception $e) {
$this->messageManager->addError($e->getMessage());
$this->_redirect('*/*/edit', ['id' => $newsModel->getId()]);
}
}
}
}
}```
### Step 2B28: The mass delete action the grid list
- Create file: app/code/BDC/SimpleNews/Controller/Adminhtml/News/MassDelete.php (Purpose: This file is used for deleting multi items on grid) and insert this following code into it:
Source
```
getRequest()->getParam('news');foreach ($newsIds as $newsId) {
try {
/** @var $newsModel \Mageworld\SimpleNews\Model\News */
$newsModel = $this->_newsFactory->create();
$newsModel->load($newsId)->delete();
} catch (\Exception $e) {
$this->messageManager->addError($e->getMessage());
}
}if (count($newsIds)) {
$this->messageManager->addSuccess(
__('A total of %1 record(s) were deleted.', count($newsIds))
);
}$this->_redirect('*/*/index');
}
}```
![EditNews](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/EditNews.png)
### Step 2B29: Backend Menu and Grid List
![allNews](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/allNews.png)
![addNews](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/addNews.png)
***
## Part C : News Module for Front End [Go to Top](#top)
***
### Step 2C1: Create Layout file for page handle
- Create file: app/code/BDC/SimpleNews/view/frontend/layout/news_news.xml (we will use this layout file as default in our module) and insert this following code into it:
```
```
### Step 2C2: Create another layout file by update the previous layout
- Create file: app/code/BDC/SimpleNews/view/frontend/layout/news_index_index.xml (this file will declare blocks for using in the news list page) and insert this following code into it:
```
```
### Step 2C3: Create Block News List file
- Create file: app/code/BDC/SimpleNews/Block/NewsList.php (this file will set the news data collection and declare pagination for them) and insert this following code into it:
```
_newsFactory = $newsFactory;
parent::__construct($context, $data);
}/**
* Set news collection
*/
protected function _construct()
{
parent::_construct();
$collection = $this->_newsFactory->create()->getCollection()
->setOrder('id', 'DESC');
$this->setCollection($collection);
}/**
* @return $this
*/
protected function _prepareLayout()
{
parent::_prepareLayout();
/** @var \Magento\Theme\Block\Html\Pager */
$pager = $this->getLayout()->createBlock(
'Magento\Theme\Block\Html\Pager','simplenews.news.list.pager'
);
$pager->setLimit(5)
->setShowAmounts(false)
->setCollection($this->getCollection());
$this->setChild('pager', $pager);
$this->getCollection()->load();return $this;
}/**
* @return string
*/
public function getPagerHtml()
{
return $this->getChildHtml('pager');
}
}```
### Step 2C4: Create frontend template file list.phtml
- Create file: app/code/BDC/SimpleNews/view/frontend/templates/list.phtml (this file will set the news data collection and declare pagination for them) and insert this following code into it:
``````
### Step 2C5: Create an abstract class by extending Magento Core Action class
- Create file: app/code/BDC/SimpleNews/Controller/News.php and insert this following code into it:
``````
### Step 2C6: Update Index Controller by extends the abstract class 'New.php'
- update/Edit file: app/code/BDC/SimpleNews/Controller/Index/Index.php and insert this following code into it:
```
_pageFactory->create();$pageFactory->getConfig()->getTitle()->set($this->_dataHelper->getHeadTitle());
//Add breadcrumb
$breadcrumbs = $pageFactory->getLayout()->getBlock('breadcrumbs');
$breadcrumbs->addCrumb('home', ['label'=>__('Home'), 'title'=>__('Home'), 'link'=>$this->_url->getUrl('')]);
$breadcrumbs->addCrumb('simplenews', ['label'=>__('Simple News'), 'title'=>__('Simple News')]);return $pageFactory;
}
}
```### Step 2C7: Create a layout file for news detail page
- Create file: app/code/BDC/SimpleNews/view/frontend/web/css/style.css and insert this following code into it:
```
.simplenews > ul {
list-style: none;
padding: 0;
}
.simplenews > ul li {
padding: 10px 5px;
margin: 0;
background-color: #fff;
border-bottom: 1px #c4c1bc solid;
display: inline-block;
width: 100%;
}
.simplenews > ul li:last-child {
border-bottom: none;
}
.simplenews-list {
float: left;
position: relative;
margin-left: 10px;
width: 100%;
}
.simplenews-list a.news-title {
font-weight: bold;
}
.simplenews-list a.news-title:hover {
text-decoration: none;
}
.block-simplenews .block-title {
margin: 0px 0px 20px;
}
.block-simplenews-heading {
font-size: 18px;
font-weight: 300;
}```
![frontEndNews](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/frontEndNews.png)
### Step 2C8: Create News view action
- Create file: app/code/BDC/SimpleNews/view/frontend/layout/news_index_view.xml and insert this following code into it:
```
```
### Step 2C9: create view news block
- Create file: app/code/BDC/SimpleNews/Controller/Index/View.php and insert this following code into it:
```
getRequest()->getParam('id');
// Get news data
$news = $this->_newsFactory->create()->load($newsId);
// Save news data into the registry
$this->_objectManager->get('Magento\Framework\Registry')
->register('newsData', $news);$pageFactory = $this->_pageFactory->create();
// Add title
$pageFactory->getConfig()->getTitle()->set($news->getTitle());// Add breadcrumb
/** @var \Magento\Theme\Block\Html\Breadcrumbs */
$breadcrumbs = $pageFactory->getLayout()->getBlock('breadcrumbs');
$breadcrumbs->addCrumb('home',
[
'label' => __('Home'),
'title' => __('Home'),
'link' => $this->_url->getUrl('')
]
);
$breadcrumbs->addCrumb('simplenews',
[
'label' => __('Simple News'),
'title' => __('Simple News'),
'link' => $this->_url->getUrl('news')
]
);
$breadcrumbs->addCrumb('news',
[
'label' => $news->getTitle(),
'title' => $news->getTitle()
]
);return $pageFactory;
}
}```
### Step 2C10: Create news view template file
- Create file: app/code/BDC/SimpleNews/Block/View.php (this file will get the news data) and insert this following code into it:
```
getRequest()->getParam('id');
// Get news data
$news = $this->_newsFactory->create()->load($newsId);
// Save news data into the registry
$this->_objectManager->get('Magento\Framework\Registry')
->register('newsData', $news);$pageFactory = $this->_pageFactory->create();
// Add title
$pageFactory->getConfig()->getTitle()->set($news->getTitle());// Add breadcrumb
/** @var \Magento\Theme\Block\Html\Breadcrumbs */
$breadcrumbs = $pageFactory->getLayout()->getBlock('breadcrumbs');
$breadcrumbs->addCrumb('home',
[
'label' => __('Home'),
'title' => __('Home'),
'link' => $this->_url->getUrl('')
]
);
$breadcrumbs->addCrumb('simplenews',
[
'label' => __('Simple News'),
'title' => __('Simple News'),
'link' => $this->_url->getUrl('news')
]
);
$breadcrumbs->addCrumb('news',
[
'label' => $news->getTitle(),
'title' => $news->getTitle()
]
);return $pageFactory;
}
}```
### Step 2C11: Create View file for News details the frontend Page
- Create file: app/code/BDC/SimpleNews/view/frontend/templates/view.phtml (this file will set the news data collection and declare pagination for them) and insert this following code into it
```
getNewsInformation();
?>
getDescription() ?>```
![FrontendNewsDetails](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/FrontendNewsDetails.png)
### Step 2C12: Create Latest New Block
- Open file: app/code/BDC/SimpleNews/view/frontend/layout/news_news.xml (we will add 2 blocks to the page body) and insert this following code into it:
```
```
### Step 2C13: Create a Block for positioning the latest news: Left or Right
- Create file: app/code/BDC/SimpleNews/Block/Lastest.php (this file will get the news data) and insert this following code into it:```
_dataHelper = $dataHelper;
$this->_newsFactory = $newsFactory;
parent::__construct($context);
}/**
* Get five latest news
*
* @return \BDC\SimpleNews\Model\Resource\News\Collection
*/
public function getLatestNews()
{
// Get news collection
$collection = $this->_newsFactory->create()->getCollection();
$collection->addFieldToFilter(
'status',
['eq' => Status::ENABLED]
);
$collection->getSelect()
->order('id DESC')
->limit(5);return $collection;
}
}```
### Step 2C14: Create the template file for Latest News
- Create file: app/code/BDC/SimpleNews/Block/Lastest/Left.php (This file will check the left position and set template file) and insert this following code into it:
```
_dataHelper->getLastestNewsBlockPosition();
// Check this position is applied or not
if ($position == Position::LEFT) {
$this->setTemplate('BDC_SimpleNews::lastest.phtml');
}
}
}```
- Create file: app/code/BDC/SimpleNews/Block/Lastest/Right.php (This file will check the right position and set template file) and insert this following code into it:
```
_dataHelper->getLastestNewsBlockPosition();
// Check this position is applied or not
if ($position == Position::RIGHT) {
$this->setTemplate('BDC_SimpleNews::lastest.phtml');
}
}
}```
### Step 2C15: Frontend view for the module
- Create file: app/code/BDC/SimpleNews/view/frontend/templates/lastest.phtml (This file will display 5 lastest news on the page) and insert this following code into it:
```
getLatestNews();
if ($latestNews->getSize() > 0) :
?>
```
## Part D : News Console Command [Go to Top](#top)
#### Write of command line a interface features?
- Installing Magento (and related tasks such as creating or updating the database schema, creating the deployment configuration, and so on)
- Clearing the cache
- Managing indexes, including reindexing
- Creating translation dictionaries and translation packages
- Generating non-existent classes such as factories and interceptors for plug-ins, generating the dependency injection configuration for the object manager
- Deploying static view files
- Creating CSS from LESS
- Before we start, please take some minutes to know about the naming in Magento 2 CLI.### Step 2D1: Adding a new command Dependency Injection
Adding a new command to CLI is based on passing on the argument from the XML level to the class Magento\Framework\Console\CommandList. Dependency Injection comes in handy here. Let’s
- Edit/Create [etc/di.xml](etc/di.xml):
Source```
BDC\SimpleNews\Console\Command\NewsCreate
```
### Step 2D2: Adding a new command class
We add the object responsible for executing the script to the class Magento\Framework\Console\CommandList. The constructor of this class is simply an array where class objects are passed on in a similar manner as in the above example.Let’s proceed to the next step – creating a class for our new command and a helper responsible for adding a new user:
- Create [Console/Command/NewsCreate.php](Console/Command/NewsCreate.php):
Source```
newsHelper = $newsHelper;
parent::__construct();
}protected function configure()
{
$this->setName('bdcrops:news:create')
->setDescription('Create New News')
->setDefinition($this->getOptionsList());
}protected function execute(InputInterface $input, OutputInterface $output)
{
$output->writeln('Creating new news...');
$this->newsHelper->setData($input);
$this->newsHelper->execute();$output->writeln('');
$output->writeln('News created with the following data:');
$output->writeln('News ID: ' . $this->newsHelper->getNewsId());
$output->writeln('Title: ' . $input->getOption(News::KEY_TITLE));
$output->writeln('Summary: ' . $input->getOption(News::KEY_SUMMARY));
$output->writeln('Description: ' . $input->getOption(News::KEY_DESC));}
protected function getOptionsList(){
return [
new InputOption(News::KEY_TITLE, null, InputOption::VALUE_REQUIRED, '(Required) News Title'),
new InputOption(News::KEY_SUMMARY, null, InputOption::VALUE_REQUIRED, '(Required) News Summary'),
new InputOption(News::KEY_DESC, null, InputOption::VALUE_REQUIRED, '(Required) News Description'),];
}
}
```
### Step 2D3: Helper
- Create [Helper/News.php](Helper/News.php):
Source
```
storeManager = $storeManager;
$this->state = $state;
$this->newsFactory = $newsFactory;parent::__construct($context);
}public function setData(Input $input) {
$this->data = $input;
return $this;
}public function execute() {
$this->state->setAreaCode('frontend');
$news = $this->newsFactory->create();
$news
->setTitle($this->data->getOption(self::KEY_TITLE))
->setSummary($this->data->getOption(self::KEY_SUMMARY))
->setDescription($this->data->getOption(self::KEY_DESC));
$news->save();$this->newsId = $news->getId();
// if($this->data->getOption(self::KEY_SENDEMAIL)) {
// $news->sendNewAccountEmail();
// }
}public function getNewsId() {
return (int)$this->newsId;
}
}
```
The execute() method adds a new user. If any data is incorrect at this stage (i.e. too short password), the script will stop and the console will show an Exception.
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/checkCliList.png)
```
php bin/magento bdcrops:news:create --news-title="Matin Cli News" --news-summary="summary 1" --news-description="News Description 1"
```
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/CliImgAddNews.png)![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/CliAddNewsDB.png)
## Part E : Set / Configure Custom Cron Jobs [Go to Top](#top)
### Step 2E1:Create crontab.xml
Create a crontab.xml file in the following file path and set a time schedule to run the custom cron code which is defined default.
- Create [etc/crontab.xml](etc/crontab.xml):
Source
```
* * * * *
```
Here when defining the crontab for the module we need to define the group name too. Here group_name is the name of the cron group. The group name doesn’t have to be unique and we can run the cron for one group at a time.
Here,
group id: is a cron group name.
job name: is a Unique ID for this cron job.
instance: is a class to be instantiated (classpath).
method: is a method in job instance to call.
schedule: is a schedule in cron format.
```
* * * * * command to be executed
| | | | |
| | | | +----- Day of week (0 - 7) (Sunday=0 or 7)
| | | +------- Month (1 - 12)
| | +--------- Day of month (1 - 31)
| +----------- Hour (0 - 23)
+------------- Minute (0 - 59)
```### Step 2E.2:Defined to run the execute method of class
This file contains the custom cron code and which will be executed while the cron runs in Magento 2.
- Create [Cron/AddNews.php](Cron/AddNews.php):Source
```
newsFactory = $newsFactory;
}
public function execute(){
$this->newsFactory->create()
->setTitle('Scheduled News')
->setSummary('Scheduled News setSummary ' . date('Ymd'))
->setDescription('Scheduled News setDescription ' . date('Ymd'))
->save();
}
}```
### Step 2E.3: Run all cron jobs
After completing the above steps run the below SSH command in your Magento 2 installed root directory to run the Magento 2 cron jobs```
php bin/magento cache:flush
php bin/magento cron:run
```
To check whether the cron is working properly, go to db
```
SELECT * FROM `cron_schedule`
SELECT * FROM `cron_schedule` where `job_code` LIKE "%bdc%"
```![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/cron_cli.png)
### Step 2E.4:Create custom cron group
Declare a new group and specify its configuration options (all of which run in the store’s view scope) through the cron_groups.xml file, located at:
//module-/etc/cron_groups.xml- Create [etc/cron_groups.xml](etc/cron_groups.xml):
Source```
1
4
2
10
60
600
```
Where:- group_name – Name of the custom group.
- schedule_generate_every – Frequency (in minutes) that schedules are written to the cron_schedule table.
- schedule_ahead_for – Time (in minutes) in advance that schedules are written to the cron_schedule table.
- schedule_lifetime – Window of time (in minutes) that cron job must start or will be considered missed (“too late” to run).
- history_cleanup_every – Time (in minutes) that cron history is kept in the database.
- history_success_lifetime – Time (in minutes) that the record of successfully completed cron jobs is kept in the database.
- history_failure_lifetime – Time (in minutes) that the record of failed cron jobs is kept in the database.
- use_separate_process – This feature is available only for Magento 2.1 and later.You can check your new cron group in admin panel at:
Stores -> Configuration -> Advanced -> System -> Cron (Scheduled Tasks)![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/cronGroupAdmin.png)
### Step 2E.5: Run new cron group cron jobs
After completing the above steps run the below SSH command in your Magento 2 installed root directory to run the Magento 2 specific group of cron jobs only.
```
php bin/magento cron:run --group="bdc_crongroup"
```
To check whether the cron is working properly, go to Database & run below query.
```
SELECT * FROM `cron_schedule` where `job_code` LIKE "%bdc%"SELECT * FROM `bdc_simplenews` ORDER BY `id` DESC
```![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/cron_cli.png)
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/cron_sampleNews.png)
#### What is cron Job & Where use in magento 2x?
Cron job is a great feature which is used to do the specific task automatically in exact time and date without manual working. The cron job is the perfect choice to do the repeated action every date or every week.Magento 2 uses cron jobs for,
- Catalog price rules
- Newsletters
- Generating Google sitemaps
- Reindexing
- Automatic updating of currency rates
- All Magento emails (including order confirmation and transactional)
- Customer Alerts and Notifications.
- Private sales (Magento Enterprise Edition only) & more..
In Magento 2, we can create crons easily and it will be listed in the database table (table name: cron_schedule) to process our tasks in scheduled time.
#### Explain ``* * * * * `` ?Schedule is the time the cron will run. In this example, it run in each minute.
```
* * * * * *
| | | | | |
| | | | | +-- Year (range: 1900-3000)
| | | | +---- Day of the Week (range: 1-7, 1 standing for Monday)
| | | +------ Month of the Year (range: 1-12)
| | +-------- Day of the Month (range: 1-31)
| +---------- Hour (range: 0-23)
+------------ Minute (range: 0-59)
```
- * * * * * : it is the time the script autorun (each 1 minute)
- www-data: it is the user will run this cript
- php /home/eden/public_html/magento2/bin/magento cron:run : the command## Part F : Create REST WEB API [Go to Top](#top)
Magento 2 API framework allows developers to create new services for communicating with Magento 2 stores. It supports REST and SOAP web services and is based on CRUD operations (Create, Read, Update, Delete) and a Search Model.
#### What is Magento 2 REST API?
At the moment, Magento 2 uses the following three authentication methods as is described in Magento 2 REST API documentation.
- OAuth 1.0a authentication for third-party applications.
- Tokens to authenticate mobile applications.
- Admins & customers authentication with login credentials.According to the Magento 2 API documentation, these authentication methods can only access the resources assigned to them. Magento 2 API framework first checks whether the call has appropriate authorization to perform the request. The API framework also supports field filtering of API responses to preserve cellular bandwidth.
Developers use Magento 2 APIs for a wide range of tasks. For instance, you can create a shopping app and integrate it with your Magento 2 store. You can also build a web app which your employee could use to help customers make purchases. With the help of APIs, you can integrate your Magento 2 store with CRMs, ERPs or POS systems.
#### How Using Magento 2 REST API?
Using REST API in Magento 2 is a piece of cake. But for that, you need to understand the flow to call APIs in PHP.
If you want to use token-based Magento 2 REST API, first you will need to authenticate and get the token from Magento 2. Then, you will have to pass it in the header of every request you perform.
To get started with the REST API in Magento 2 using token-based authentication, you will need to create a web service User Role and register that role to a new Magento 2 Admin User. Keep in mind that creating a new role and user is necessary because it’s not a good practice to use Magento Owner User in a web service.
#### Why SOAP for eCommerce?
When it comes to e-Commerce websites, APIs play the big role of reading and writing information from and to the server. Be it a customer’s name or his already saved credit card details, every piece of information shown to the end user has to either read from or written to the web server. This is taken care by REST and SOAP APIs. REST and SOAP are models for web services, however, one that’s most recommend for eCommerce websites. Though REST is fast, efficient and simple, SOAP is standardized, secure and apt for payments.
#### How Create Web Service Role in Magento 2?
To create a web service role in Magento 2, follow these steps:
- Login to the Magento 2 Admin Panel.
- Go to System >> User Roles and tap the Add New Role
- Enter the Role Name.
- In Your Password field, enter the current password of your Magento 2 Admin.
- Now, on the left side, click Role Resources.
- In the Resource Access, select only those that are required for your web service.
- Once done, hit the Save Role
#### How Create Web Service User in Magento 2?
Now, create a new user for the newly created role through these steps:
- Go to System >> All Users and hit the Add New User
- Enter the required information including User Name, First and Last Name, Email, Password, etc.
- Now, on the left side, click User Role and select the newly created role.
- Once done, click the Save User#### How Magento 2 REST API Authentication?
As I mentioned earlier, I will authenticate REST API through Token authentication. This means that I will pass a username and password in the initial connection and receive the token . This token will be saved in a variable, which will be passed in the header for further calls.#### How Get Modules Using REST API in Magento 2?
You can fetch almost everything using Magento 2 REST API. The List of REST APIs for Magento EE and CE is a good guide on this topic.To demonstrate the API, I am going to get all the installed modules on a Magento 2 store. Here is the script:
Source```
"apiaccess", "password" => "api@123");
$data_string = json_encode($data);
$ch = curl_init($apiURL);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "POST");
curl_setopt($ch, CURLOPT_POSTFIELDS, $data_string);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_HTTPHEADER, array("Content-Type: application/json","Content-Length: ".strlen($data_string)));
$token = curl_exec($ch);
//decoding generated token and saving it in a variable
$token= json_decode($token);
//******************************************//
//Using above token into header
$headers = array("Authorization: Bearer ".$token);
//API URL to get all Magento 2 modules
$requestUrl='http://www.magento.lan/rest/V1/news';
$ch = curl_init($requestUrl);
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$result = curl_exec($ch);
//decoding result
$result= json_decode($result);
//printing result
print_r($result);```
### Step2F1: Web API Routes/Configuration – etc/webapi.xml
#### What is Web API?
A Web API is an application programming interface for either a web server or a web browser. It is a web development concept, usually limited to a web application's client-side (including any web frameworks being used), and thus usually does not include web server or browser implementation details such as SAPIs or APIs unless publicly accessible by a remote web application.#### What are the Magento 2 web APIs Framework?
Supports developers to use web services that communicate with the Magento system. For instance, a developer can create a customer account, product record through web service. Key features include:
- M2 Support Both REST & SOAP coverage same.
- 3 types of authentication:
* 3rd-party applications authenticate with OAuth 1.0a.
* Mobile applications authenticate using tokens.
* Administrators & customers authenticated with login credentials.
- All accounts & integrations assigned resources access API framework checks any call authorization to perform request.
- Any Magento or third-party service can be configured as a web API with a few lines of xml. To configure a web API define XML elements & attributes in a webapi.xml.
- Based on CRUD & search model
- Supports field filtering of web API responses to conserve mobile bandwidth
- Integration style web APIs enable a single web API call to run multiple services at once for a more efficient integration.#### What can I do with the Magento web APIs?
APIs can be used to perform a wide array of tasks ex:
- Create shopping app as traditional user downloads on a mobile device, also employee uses on a showroom floor to help customers make purchases.
- Integrate with CRM/ERP backend systems ex. Salesforce/Xero/NetSuite.
- Integrate with CMS
- Create JavaScript widgets in storefront/ Admin panel makes AJAX calls to access services.#### How do I get started?
Register web service on Magento Admin following general steps to set up to enable web services.
- If using token-based authentication, create a web services user on Admin by selecting System > Permission > All Users > Add New User. (If session-based or OAuth authentication do not need to create new user in Admin.)
- To create an integration, click System > Extensions > Integration > Add New Integration**. Be sure to restrict which resources the integration can access.
- Use a REST or SOAP client to configure authentication.#### Routes & Configuration?
routes are defined in etc/webapi.xml within a module, and although the structure of the definition xml is directed by the requirements of the REST API, the SOAP API uses the same definitions.The following shows the route configuration for fetching a CMS block, as defined in BDC_SimpleNews::etc/webapi.xml:
```
```
#### Implementation News webapi.xml
- Create [app/code/BDC/SimpleNews/etc/webapi.xml](/etc/webapi.xml)
Source```
```
In the route tag the url attribute defines the route as /V1/cmsBlock/:blockId where the :blockId part represents an id parameter to be supplied. The method attribute defines the HTTP verb the route uses as ‘GET’ (other available verbs are PUT, POST and DELETE).
In the service tag the class attribute associates the service contract Magento\Cms\Api\BlockRepositoryInterface with the route, and the method attribute defines the method to call upon the object provided by the service contract.
### Step2F2: Define Interface – etc/di.xml
#### What is repository?
Repositories give service requestors the ability to perform create, read, update, and delete (CRUD) operations on entities or a list of entities. A repository is an example of a service contract, and its implementation is part of the domain layer.Repositories are service contracts which are interface classes & helps to hide business logic from controller,model & helper, defined repository file which is an interface class & model file in which define methods declared in repository class.To create module’s repository, firstly have to define it in di.xml file at path:
app/code/BDC/SimpleNews/etc/di.xml#### Implementation
- Edit/Create [app/code/BDC/SimpleNews/etc/di.xml](etc/di.xml) and insert this following code into it:```
```
- Final file look like as below:Source
```
BDC\SimpleNews\Console\Command\NewsCreate
```
### Step2F3:Declare API Interface – Api/NewsRepositoryInterface.php
#### What is Repository Interface?
An interface defines the repository with all logical read and write operations for a specific entity. You can see an example of such a repository interface in the diagram. The interface gets implemented by one or more classes that provide data store specific implementations of each interface method
Now, we need to create an interface and model, please note that you need to take care of the comments as well.#### What is Repository Interface magento 2?
Repositories are service contracts which are interface classes & helps to hide your business logic from controller,model and helper.
#### How many Interface types and locations?
A service contract must define data interfaces, which preserve data integrity, and service interfaces, which hide business logic from service requestors.
- Data interfaces: define functions that return information about data entities, return search results, and set validation rules and return validation results. You must define the data interfaces for a service contract in the Api/Data subdirectory for a module.
- Service interfaces: include management, repository, and metadata interfaces. You must define the service interfaces for a service contract in the Api subdirectory for a module.
#### Implementation
- Create [Api/NewsRepositoryInterface.php](Api/NewsRepositoryInterface.php)
Source```
### Step2F4:Data Interface – Api/Data/NewsInterface.php
#### What is Interface class PHP?
An interface allows unrelated classes to implement the same set of methods, regardless of their positions in the class inheritance hierarchy. An interface enables you to model multiple inheritance because a class can implement more than one interface whereas it can extend only one class.#### What is Data interfaces in Magento 2 ?
Define data interfaces in the Api/Data subdirectory for a module.Ex. data interfaces for the Customer module are in the /app/code/Magento/Customer/Api/Data subdirectory.
Now, we need to create an interface and model, please note that you need to take care of the comments as well.
- Create [app/code/BDC/SimpleNews/Api/Data/NewsInterface.php](Api/Data/NewsInterface.php) & insert this following code into it:Source
```
### Step2F5: Create Model – Model/NewsRepository.php
#### What task done by Model NewsRepository?
Get Collection in means showing the items in your store when run the command. With the code snippet in this topic, request the specific number of the news as you need. Let’s start calling the news in Magento 2 now!
#### Implementation
- Create [app/code/BDC/SimpleNews/Model/NewsRepository.php](Model/NewsRepository.php) & insert this following code into it:Source
```
collectionFactory = $collectionFactory;
}
public function getList() {
return $this->collectionFactory->create()->getItems();
}
}```
### Step2F6: Communicating with new API call
#### Accessing routes
As mentioned above, the configuration is conveniently used by both the REST and SOAP APIs. However, the means of accessing resources differs quite a lot.The full REST resource URL is the easiest to determine as it just needs prefixing with ‘http://www.yourdomain.com/rest/’, so in the example above, assuming the news needed has an entity id of 1, the resource url would be ‘http://www.yourdomain.com/rest/V1/news/1’.Testing as guest:
To test REST you can go to http://{domain_name}/rest/V1/{method}/{attribute}/{value}.Example: http://magento2.loc/rest/V1/hello/name/Matin.
This is how response should look like for this example:
http://www.magento.lan/rest/V1/news
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/webapi_restV1News.png)
Here is small code that will test same API call but with SOAP(Not implements):
```
bdcSimpleNewsV1();
var_dump($result);
```
Response for SOAP
```
object(stdClass)#2 (1) {
["result"]=>
string(10) "..."
}
```### Step2F7: Adding ACL Web API
If we don’t set anonymous in resource of webapi.xml, we need to set existing Magento resource or create our own. We can do that by adding acl.xml to etc.
ACL – etc/acl.xml
```
```
In this case we need to add BDC_SimpleNews::news to webapi.xml resource instead anonymous.## PartG:Dependency Injection configuration [Go to Top](#top)
#### What is Dependency injection Magento 2?
Magento 2 Dependency injection is used to replace the Magento 1.x Mage class when you convert to work with Magento 2. The Dependency injection design pattern creates an external environment where you can inject dependencies into an object. Thanks to that, there is no longer to create the objects manually. Namely, as when object A calls object or value B, this means B is a dependency of A
#### What is Dependency Inversion Principle Magento 2?
If you are working with Magento 2 Dependency Injection, you should take look at Magento 2 Dependency Inversion Principle because this principle will restrict the direct working between the high level and low level classes. At that time, the interaction will implement via an interface of the low level classes as an abstract layer.Specifically, the di.xml file takes responsibility for mapping an interface dependency to a preferred implementation class. It can be said that with Magento 2 Dependency Inversion Principle, the dependency of the coding will be reduced significantly due to the abstract layer.
- Object manager - Dependency Injection Container
Object Manager is called as Dependency Injection Container, Magento 2 service class which contains and handle the dependencies between the objects. During the class construction, the object manager injects the appropriate dependency as defined in the di.xml file.- Constructor signature dependencies
In Magento 2, the class definition use constructor signature to get information (type and number of dependencies).- Compiling dependencies
All information related to Magento 2 Dependency Injection are collected in a class and saved in files by a code complier tool. And then the ObjectManager will get this information to generate concrete objects in the application.#### Explain Injection types used in Magento 2?
Magento 2 Dependency Injection includes two types: Constructor Injection and Method Injection. You can see the following code snippet to learn more about both of them.
- Constructor injection
As the above example, $menuItemFactory and $menu are the dependencies that will be added to an object’s class through the constructor injection. Besides, remember that the constructor injection is required to declare all optional and required of an object.- Method injection
About Method Injection, you will use it when an object makes clear a dependency in one of its methods. As if tracking in the referred instance, $command is the dependency passed into the class through the processCommand method.- Groups of Object
In Magento 2, the object is divided into two groups: injectable and non-injectable (newable) objects. What are they?- Injectable Objects
About the injectable Objects, you can call as services or objects which will show the dependencies in their constructors and are created by the object manager via the configuration in the di.xml file. And you can use these injectable objects to request other injectable services in the constructors.- Non-injectable Objects
Non-injectable (Newable) Objects are a bit similar to the injectable objects when they also expose the dependencies in their constructors, however, the newables are allowed to request other newables objects like Entities, Value Objects. In addition, you cannot demand the newable objects for keeping a reference to an injectable object.This is the detialed information related to Magento 2 Dependency Injection design pattern. Wish you have a great time with it!
#### How to Override core code in Magento 2?
By default, there are three different ways to override core functionalities.
1. Using preference
2. Using event\observer
3. Using a plugin#### Define Preference,Arguments, Virtual Types, Plugins ,Observers?
- Preference: One class over another, which allows you to specify which class/type is selected by Magento’s object manager.This means that you can override which method you want from the class, along with the methods that this class extends.
- Arguments:
- Virtual Types: way to inject different dependencies into existing classes without affecting other classes.
- Plugins:
- Observers:### Step2G1:DI Preference,Arguments & Virtual Types Implements
We write log after news item save .
- Edit [Helper/News.php](Helper/News.php) look like:
Source
```
storeManager = $storeManager;
$this->state = $state;
$this->logger = $logger;$this->newsFactory = $newsFactory;
parent::__construct($context);
}public function setData(Input $input){
$this->data = $input;
return $this;
}public function execute() {
$this->state->setAreaCode('frontend');
$news = $this->newsFactory->create();
$news->setTitle($this->data->getOption(self::KEY_TITLE))
->setSummary($this->data->getOption(self::KEY_SUMMARY))
->setDescription($this->data->getOption(self::KEY_DESC));
$news->save();
$this->logger->debug('DI: '.$news->getTitle());}
public function getNewsId(){
return (int)$this->newsId;
}
}```
- Create [Helper/BdcDebug.php](Helper/BdcDebug.php) Insert :
Source
```
#### What is the difference between Preference and Plugin?
Although both of them are used for overriding the core modules, the way to use them is completely different.
With Preference, it must extend a core class. Preference can rewrite function. When you declare a Preference, your new class is expected to be a complete implementation of the class you want to override.
While a plugin allows you to execute your functions before, after or around (before & after) the core function is executed. It's NOT really rewritten function like Preference.
Since your plugin class doesn’t replace the core class, in case there are many plugins hooked onto a target class, Magento 2 just executes them sequentially based on the sortOrder parameter in your file di.xml.
### Step2G1a: Preference: which override all DEBUG log
#### Preference
Class preferences basically do the same thing in Magento 2 that rewrites did in Magento 1. It states a preference for one class over another, which allows you to specify which class/type is selected by Magento’s object manager.
This means that you can override which method you want from the class, along with the methods that this class extends.#### Using class preference I am going to rewrite:
- Model class
- Block Class
- Controller Classapp/code/BDC/SimpleNews/etc/di.xml add below code :
``````
### Step2G1b: Arguments Preference: which override specific class Monolog#### Arguments Preference ?
Argument types:object
Node Formats: {typeName}{typeName}
- Creates an instance of typeName type and passes it in as an argument. You can pass any class name, interface name, or virtual type as typeName.
```
BDC\SimpleNews\Helper\BdcDebug
```
### Step2G1c: VirtualType: which override specific class only work specific module
#### What are Virtual Types?
Within Magento 2, classes can depend on each other using constructor-based Dependency Injection. And instead of only allowing static dependencies (class A injects class B), Magento offers a configuration system that allows one dependency to be replaced with another (class B is swapped out for class C). One of these configurations is Virtual Types.
Virtual Types are defined in a file di.xml which might be located in numerous places - for instance, the etc/ folder of your own module. Virtual Types are in essence new PHP classes (but actually they are not, they are just links), that extend upon their original class while overriding the original class its constructor arguments by adding those in the di.xml file.
#### What are Virtual Types?
Within Magento 2, classes can depend on each other using constructor-based Dependency Injection. And instead of only allowing static dependencies (class A injects class B), Magento offers a configuration system that allows one dependency to be replaced with another (class B is swapped out for class C). One of these configurations is Virtual Types.Virtual Types are defined in a file di.xml which might be located in numerous places - for instance, the etc/ folder of your own module. Virtual Types are in essence new PHP classes (but actually they are not, they are just links), that extend upon their original class while overriding the original class its constructor arguments by adding those in the di.xml file.
#### Create a new PHP child class or a Virtual Type?
Once you realize that a Virtual Types is nothing more than a new PHP child object (as if there was an actual class generating it), it makes you wonder why you should do this through XML. Maybe it is easier to simply create a new PHP class in your module and modify things there? The end result is the same: There is a new object of a new type. (Note that this new class still needs to be used somewhere else to become useful. Typically this is done by using an XML Type to modify the constructor arguments of yet another class and inject this new virtual class in it.)I personally favour new PHP classes over new Virtual Types. However, once the original class has a lengthy constructor, a new PHP class would require you to duplicate all parent dependencies in its own constructor and pass them on to its parent - and perhaps all of that trouble is only needed for replacing one of those dependencies. A Virtual Type is quicker: It requires some XML, yes, but it allows you to single out only that dependency that you actually need to be replaced. The more complex the original constructor, the better it is to use a Virtual Type. (That being said, the more complex the original constructor, the more this original constructor needs to be cleaned up - with references to SOLID.)
#### Virtual Types with namespaces?
Now let's go to the main point of this blog: Virtual Types are identical to PHP classes created on the fly by the Object Manager. And just like all PHP classes, we have specific rules to stick to and namespacing is one of them. So why not use namespaces?
Let's take a dummy example without namespaces:
``````
And now let's see a namespaced version:
``````
To me, the namespaced version looks a lot cleaner. Remember that defining this Virtual Type is only half of the story - if you don't intend to use it elsewhere, it just as well can be removed again. It only becomes useful once it is applied elsewhere, for instance using a Type:
```
BDC\Example\Some\Class\Virtual
```
Once others start debugging the class Magento\Framework\Some\Existing\Class, they might bump into the someDep argument and now, thanks to namespaces, the name of this Virtual Type identifies exactly who put that dependency there. This is why we have namespaces.#### Namespace with Virtual in it?
However, this might also become confusing if the Virtual Type actually looks too similar to a PHP class. I always tend to click through my PhpStorm environment with the generated/ folder excluded from my project. Once in a while, I bump into a class that is not there. And if Magento does not die at that moment, I assume it is something that is generated. Once the class has the word Factory or Proxy in it, this confirms my assumption. Wouldn't it make sense to also include the word Virtual in the namespaced name of a Virtual Type?
This leads to the following classes that would suggest that the PHP class actually is a VirtualType:
```
BDC\Example\Some\Class\Virtual
BDC\Example\Virtual\Some\Class
BDC\Example\Some\ClassVirtual
BDC\Example\VirtualType\Some\Class
BDC\Example\Some\Class\VirtualType
```
Obviously, there are many more variations. But just make sure to add the word Virtual in there.#### In this Module We Implemented as
```
BDC\SimpleNews\Helper\BdcDebug
bdcLogger
```
- Finaly [etc/di.xml](etc/di.xml) look like ;
Source```
BDC\SimpleNews\Console\Command\NewsCreate
BDC\SimpleNews\Helper\BdcDebug
bdcLogger
```
- Run
```
php bin/magento cache:flush
php bin/magento bdcrops:news:create --news-title="News preference" --news-summary="summary preference 1" --news-description="News preference Description 1"```
Now check var/log/bdc_debug.log all log are write there![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/bdc_debug.png)
### Step2G2:DI Events & Observer Implements
[devdocs](https://devdocs.magento.com/guides/v2.3/extension-dev-guide/events-and-observers.html)
#### Magento 2 Events
Events are dispatched by Magento modules on the trigger of a specific action. Not only that, Magento also allows you to also create your own custom event that can be dispatched in your code. When the action is triggered, it will pass data to the relevant observer configured for the dispatched event.
Magento 2 events can be dispatched using Magento\Framework\Event\Manager class and it can be obtained through the dependency injection by defining the dependency in your constructor.#### Magento 2 Observers
Observers are used to catch the action which was triggered from events. In observers, you can set the required functionality or logic that is to be executed in response.Magento 2 observers can be created by lacing your class file under the Module-Root/Observer directory. Your observer class should implement the following;
Magento\Framework\Event\ObserverInterface and define its execution function.
Now let’s start with the execution!
Let’s assume that you want to change the background color of your store if the customer is not logged in.
#### What is observers in Magento 2?
Working with Magento 2 observers is one of many different ways of extending the core functionality of a Magento 2 powered eCommerce store. Thanks to Observers, you can run your custom codes in response to a specific Magento event or even with a custom event.
You can choose other options such as extending and overriding the core modules, copying the core class to the local directory and put it in the same directory path it was in core directory and modify the core class directly. However, creating an observer is the number one choice#### How observers work
Observers are used for catching the action which was triggered before or after events. In observers, you can set the required functionality or logic that is to be executed in response.Magento 2 observers can be created by lacing your class file under the Module-Root/Observer directory. Your observer class should implement the following:
Magento\Framework\Event\ObserverInterface and define its execution function.#### How to use observers
Make sure you have registered the new module to test it before, we will practice on this module.
I will use my module SampleEvent. And then, I will use an observer to customize the product name on the product view page.
#### Some practices when using observer Magento 2:- Make your observer efficient: You should try to keep your observer small and efficient by avoiding complex computations if you can. Because having complex computations in your observer can slow down application processes.
- Don’t include business logic: Your observer should not contain logic other than what is needed for it to run. Business logic should be encapsulated in other classes that your observer uses.
- Declare observer in the appropriate scope:
+ For the frontend events, declare observers in etc/frontend/events.xml, this event will be only used in the frontend. You can't use this event in the backend.
+ For the backend events, declare observers in etc/adminhtml/events.xml, this event will be only used in the backend. This event can't be used in the frontend.
+ Use the global etc/events.xml file only when an event can occur on both the frontend and the backend.
+ You can put events.xml in etc > webapi_rest > events.xml while handling Rest API request.
+ You can put events.xml in etc > webapi_soap > events.xml while handling Soap API request.
+ You can put events.xml in etc > crontab > events.xml while handling scheduled jobs only.
+ You can put events.xml in etc > setup > events.xml while Magento or extensions are being installed or upgraded.
#### Some perfect events in Magento 2:
- Events in Magento controller:
+ controller_action_predispatch - executes before each controller dispatching.
+ controller_action_newsdispatch_{full_action_name} - executes after a controller with specific {full_action_name}.
+ controller_action_newsdispatch_{route_name} - executes after each controller with specific {route_name}.
+ controller_action_newsdispatch - executes after each controller dispatching.
- Edit [Helper/News.php](Helper/News.php):
Source```
storeManager = $storeManager;
$this->state = $state;
$this->logger = $logger;
$this->eventManager = $eventManager;
$this->newsFactory = $newsFactory;parent::__construct($context);
}public function setData(Input $input){
$this->data = $input;
return $this;
}public function execute() {
$this->state->setAreaCode('frontend');
$news = $this->newsFactory->create();
$news->setTitle($this->data->getOption(self::KEY_TITLE))
->setSummary($this->data->getOption(self::KEY_SUMMARY))
->setDescription($this->data->getOption(self::KEY_DESC));
$news->save();
$this->logger->debug('DI: '.$news->getTitle());
// EventCode...
$this->eventManager->dispatch('bdc_simplenews_save_after', ['object' => $news]);
$this->newsId = $news->getId();// if($this->data->getOption(self::KEY_SENDEMAIL)) {
// $news->sendNewAccountEmail();
// }
}public function getNewsId(){
return (int)$this->newsId;
}
}
```
#### AbstractModel Event:
Usually, models extend the Magento\Framework\Model\AbstractModel class. It gives an ability to observe a predefined set of model events. And the model should have AbstractModel::_ eventPrefix attribute specified for observing events of a specific model. The attribute's value equals to "core_abstract" by default.
Also, in models we have AbstractModel::_ eventObject attribute that gives an ability to specify a name of the current model’s instance for different model-specific events.
A list of the global models events:
- model_load_before - executes before each model is loader. Here we can get an access to the following event’s data.
- $observer->getField() - gets currently processed model’s field name.
- $observer->getValue() - gets currently processed model’s field value.
- model_load_after - executes after each model loading.
- model_save_after - executes after each model saving.
- model_save_before - executes before each model saving.
- clean_cache_by_tags - executes after model related cache tags are cleaned.
- model_delete_before - executes before model is deleted.
- model_delete_after - executes after model is deleted.
- model_save_commit_after - executes after the models saving transaction is committed.
- model_delete_commit_after - executes after the models saving transaction commit is deleted.
In this mentioned events, we can get an access to the following data:
$observer->getObject()
List model-specific events:
- {event_prefix}_load_before – executes before model with {event_prefix} is loaded.
- {event_prefix}_load_after – executes after model with {event_prefix} is loaded.
- {event_prefix}_save_before – executes before model with {event_prefix} is saved.
- {event_prefix}_save_after – executes after model with {event_prefix} is saved.
- {event_prefix}_delete_before – executes before model with {event_prefix} is deleted.
- {event_prefix}_delete_after – executes after model with {event_prefix} is deleted.
- {event_prefix}_save_commit_after – executes after model’s data with {event_prefix} is committed.
- {event_prefix}_delete_commit_after – executes after model’s data commit with {event_prefix} is deleted.
- {event_prefix}_clear – executes when a model object is being prepared for correct deleting by the garbage collector.
Furthermore, we can get an access to the following event data from each of them:
- $observer->getDataObject() – gets the current model reference.
- $observer->get{event_object} – gets an event object for the current model.
If you want to find an event in code, you can do this.
Example: You need an event save_before or save after.
- Create an event observer to hook in the event [model prefix]_ save_before. In here we will have observer variable, this variable could get the Model of model which we need to save data on it.
- And then we can use setData(‘column_name’,[new value]) to adjust the data of a column before saving to the database.
So why can we do that?
What is Model Prefix: in object Model, we have a property, this is protected $_ eventPrefix; (You can see in the model, if you don’t have it, you can create it). It’s is a string type. Getting the value and join it with _ save_before, we will have an event name.
EG: protected $ _ eventPrefix = 'abc'; => Event Observer = 'abc_save_before'.
You can declare another event:
- [prefix]_ load_before
- [prefix]_ save_after
- [prefix]_ load_after
These events are default and always available with a model, If you want to use a custom event, you can use eventManager->dispatch(‘event_name’,$data);
- OR [Model/News.php](Model/News.php) just add protected $_ eventPrefix = 'bdc_simplenews';
This event eventPrefix is used by abstract model to generate events automatically.Finaly script look like below:
Source```
_init('BDC\SimpleNews\Model\Resource\News');
}/**
* Loading news data
*
* @param mixed $key
* @param string $field
* @return $this
*/
public function load($key, $field = null) {
if ($field === null) {
$this->_getResource()->load($this, $key, 'id');
return $this;
}
$this->_getResource()->load($this, $key, $field);
return $this;
}
}
```
- Create [Observer/Logger.php](Observer/Logger.php)
Source```
logger = $logger;
}
public function execute(Observer $observer){
$this->logger->debug("Observer:".
$observer->getEvent()->getObject()->getTitle()
);
}
}```
- create [etc/events.xml](etc/events.xml)
Source```
```
#### Note: Explain observer node defines the observer itself and its attributes?
- name – the observer registration name (it is important that the names do not coincide);
- instance – the class, which method will be executed when a specific even occurs;
- method – the method being executed.- add [etc/di.xml](etc/di.xml)
Source```
bdcLogger
```
- Finally [etc/di.xml](etc/di.xml) look like:
Source```
BDC\SimpleNews\Console\Command\NewsCreate
BDC\SimpleNews\Helper\BdcDebug
bdcLogger
bdcLogger
```
- Run
```
php bin/magento cache:flush
php bin/magento bdcrops:news:create --news-title="News Observer" --news-summary="summary Observer 1" --news-description="News Observer Description 1"```
Now check var/log/bdc_debug.log all log are write there![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/bdc_debug.png)
### Step2G3: DI Plugins (Interceptors)
#### Whate is Plugins?
A plugin is a great way to expand or edit a public method’s behavior by using code before, after or around method. First of all, please get an object that provides permission to all public methods of the observed method’s class.
Interception is a software design pattern that is used when we want to insert code dynamically without necessarily changing the original class behavior. The interception pattern in Magento 2 is implemented via plugins.
Plugins are an amazing tool in Magento 2. They allow you to change the behavior of methods for classes without having to rewrite the classes as we did above.
There are 3 different ways to use a plugin to change method behavior. You may have heard them on Sesame Street:
- Before
- After
- Around
#### What is Benefits of Magento 2 Plugin?
For a module developer as you, Magento 2 Interception plugin allows:- Forwarding any method call that is used on an object manager controlled object and taken programmatic action
- Modifying the return value of any method call that is used on an object manager controlled object
- Modifying the arguments of any method call that is used on an object manager controlled object
Proceeding similarly when other modules are in progress of the same method in the same or predictable way.#### What is Limitations of Plugins can not be used ?
- Final methods
- Final classes
- Non-public methods
- Class methods (such as static methods)
- __ construct
- Virtual types
- Objects that are instantiated before Magento\Framework\Interception is bootstrapped#### Declaring a plugin
di.xml file in your module declares a plugin for a class object:
```
```
#### Explain Plugin element attributes assigned in declaration ?
- type name: A class or interface which the plugin observes.
- plugin name: An arbitrary plugin name that identifies a plugin. Also used to merge the configurations for the plugin.
- plugin type: name of a plugin’s class or its virtual type. Use the following naming convention when you specify this element: \Vendor\Module\Plugin\.
Under the type element, we define one or more plugins using the plugin element.More Details :
- name – Using this attribute, you can provide a unique and recognizable name value that is specific to the plugin.- sortOrder – This attribute determines the order of execution when multiple plugins are observing the same method.
- disabled – The default value of this attribute is set to false, but if it is set to true, it will disable the current plugin, and it will not get executed.
- type – This attribute points to the class that we will be using to implement the before, after or around the listener.
Assuming we are writing a plugin for a specific method, let’s choose a random method under Customer.php class, the getName() method.
We define the before, after and around listeners for the getName() method by writing the naming conventions as follows.
```
Before + getName() => beforeGetName();
After + getName() => afterGetName();
Around + getName() => aroundGetName();
```#### Write The Optional Elements Plugins?
- plugin sortOrder: Plugins that call the same method run them using this order.
- plugin disabled: To disable a plugin, set this element to true. The default value is false.#### 3 methods in Plugin:
- 1. before - beforeDispatch()
- 2. around - aroundDispatch()
- 3. after - afterDispatch()#### Plugin Implementation:
- create [Plugin/Logger.php](Plugin/Logger.php)
Source```
writeln('beforeExecute');
}
public function aroundRun(
NewsCreate $command,
\Closure $proceed,
InputInterface $input,
OutputInterface $output) {
$output->writeln('aroundExecute before call');
$proceed->call($command, $input, $output);
$output->writeln('aroundExecute after call');
$this->output = $output;
}//public function afterRun(NewsCreate $command){
//$this->output->writeln('afterExecute');
//}
}
```
- add code app/code/BDC/SimpleNews/etc/di.xml
```
```
- Finally [etc/di.xml](etc/di.xml) look like:Source
```
BDC\SimpleNews\Console\Command\NewsCreate
BDC\SimpleNews\Helper\BdcDebug
bdcLogger
bdcLogger
```
- Run
```
php bin/magento cache:flush
php bin/magento bdcrops:news:create --news-title="News Plugin" --news-summary="summary Plugin 1" --news-description="News Plugin Description 1"```
Now check var/log/bdc_debug.log all log are write there![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/plugin_cli.png)
## PartH: Customization Layout Configuration & JavaScript [Go to Top](#top)
### Step2H1: Layout Configuration
- edit [view/frontend/layout/news_index_index.xml](view/frontend/layout/news_index_index.xml)```
```
Change as below:
```
Head Line:
```
- Edit [view/frontend/templates/list.phtml](view/frontend/templates/list.phtml)
```
getLabel(); ?>
```
- Output![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/custLayout.png)
### Step2H2:Customization JavaScript
#### What is RequireJS?
RequireJS is a javascript module system. It implements the Asynchronous Module Definition (AMD) standard for javascript modules. In the terms of AMD, a javascript module provides a way to
Run a javascript program that doesn’t default to the global namespace
Share javascript code and data between named modules and programs
That’s all RequireJS does. You may use a RequireJS module that implements some special bit of functionality, but its not RequireJS that provides that functionality. RequireJS is the pneumatic tube that ensures the functionality is delivered to you.
#### Explain RequireJS term "map","shim","deps" ?
- Map: Replace a default JS component To use a custom implementation of an existing Magento JS component: Place the custom component source file in one of the following locations:
Your theme JS files: /web/js or /_ /web/js Your module view JS files: /view/frontend/web/js Create a RequireJS configuration file requirejs-config.js, having specified the following:
```
var config = {
"map": {
"*": { "": "" }
}
};
```
- Shim:To build a dependency on the third-party plugin, specify a [shim] in the following configuration files: requirejs-config.js
```
var config = {
"shim": {
"3-rd-party-plugin": ["jquery"]
}
};
```- Deps:Is used when your require js configurations depends upon some dependencies, i.e. you want to load some dependencies before your requires js define()’d is called. example:
```
var config = {
"deps": [
"jquery"
]
};
```
Here, It loads the [jquery] as soon as the require define()’d is called.
- baseUrl: Is used when we want to define the url to load the js files for all the modules used in require js.
baseUrl applied on all the files which defined with name starting with a slash”/”, have any url like “http://” or having .js extension.example:
```
var config = {
"baseUrl": "bdcrops/test"
};require( ["sample/sample1", "https://code.jquery.com/jquery-3.1.1.min.js", "sample2.js"],
function(samplemodule) {
}
);
```
Here, samplemodule is reffered to bdcrops/test/sample/sample1.js,
“https://code.jquery.com/jquery-3.1.1.min.js” is loaded from the url which is specified
and sample2.js is loaded from the same directory.- paths: Paths is used when you want to relate your base url path the your module path.
Path have the same properties as the baseUrl.If the path is started from “/” or any url “http://” then it will not relate to the base url.example:```
var config = {
"baseUrl": "bdcrops/test",
"paths": {
"sample": "web/js"
},
};require( ["sample/sample1"],
function(samplemodule) {
}
);
```
Now, samplemodule is reffered to the file at path “bdcrops/test/web/js/sample1.js”- config: Is used when you want to pass some configurations to all the modules.
These values are the common for all the modules.example:
```
var config = {
"map": {
'*': {
'sample': 'sample1.js'
}
}
config: {
"testData":{
"color":'red'
}
}
};
```
Now in your js file you can access this value by using :console.log(require.s.contexts._ .config.testData.color);It will gives you the “red” in output.#### What is RequireJS “mixin”?
A Magento 2 RequireJS “mixin” allows you to programmatically listen for the initial instantiation of any RequireJS module and manipulate that module before returning it.### How to overwriting JS component using map?
- requireJS configuration
create app/code/BDC/SimpleNews/view/frontend/requirejs-config.js- copy validation lib to mododule
lib/web/mage/validation.js ==>[app/code/BDC/SimpleNews/view/frontend/web/js/validation.js](view/frontend/web/js/validation.js)
Change massages about 1684 line as
```
$.validator.messages = $.extend($.validator.messages, {
required: $.mage.__('This is a required field Custome.'),
```
- overwriting JS component using map
[app/code/BDC/SimpleNews/view/frontend/requirejs-config.js](view/frontend/requirejs-config.js)```
var config = {
'map': {
'*': {
'mage/validation': 'BDC_SimpleNews/js/validation'
}
}
};
```- Clean cache try to login button without fill filed
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/RequireJsValidationJS.png)
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/RequireJsLoginsms.png)#### How to overwriting JS component using mixin?
- app/code/BDC/SimpleNews/view/frontend/requirejs-config.js
```
var config = {
'map': {
'*': {
'mage/validation': 'BDC_SimpleNews/js/validation'
}
},
config: {
mixins: {
'BDC_SimpleNews/js/validation': {
'BDC_SimpleNews/js/validation-mixin': true
}
}
}
};
```
- app/code/BDC/SimpleNews/view/frontend/web/js/validation-mixin.js```
define(function () {
'use strict';var extension = {
isValid: function () {
return true;
}
};return function (target) {
return target.extend(extension);
};
});
```
- Check again login without fill value all validation gone## PartI: Components Library/Customizing UI 15% [Go to Top](#top)
![](doc/GridViewNewsListAdmin.png)
#### how to create an Admin Grid in Magento 2 backend?
As you know, Magento 2 Grid is a kind of table which listing the items in your database table and provide you some features like: sort, filter, delete, update item, etc. The samplenews for this is the grid of products, grid of customer.Magento 2 provide two ways to create Admin Grid:
- Using Layout
- Using Component.
We will find out the detail for both of them. Before we continue please follow this articles to create a simple module with admin menu, router which we will use to learn about grid.#### How to Create Create Admin Grid using Component?
- Declare resource in dependency injection file Now we will create di.xml file which will connect to the Model to get the data for our grid. File: app/code/BDC/SampleNews/etc/di.xml- Create layout file For the action bdc_simplenews/news/index, we will create a layout file name bdc_samplenews_news_index.xml
- Create component layout file As declaration in layout file, we will create a component file bdc_samplenews_news_listing.xml
#### How to Create Admin Grid using Layout?
You have just find how to add a Magento 2 Grid by using Component. Now we will see how to do it by using normal layout/block file.
- Create block for this grid File: app/code/BDC/SampleNews/Block/Adminhtml/News.php
- Create layout file Now we will need a layout file to connect with Grid Block and render the grid. Let’s create this file:app/code/BDC/SampleNews/view/adminhtml/layout/bdc_samplenews_news_index.xml- Create layout file Now we will need a layout file to connect with Grid Block and render the grid. Let’s create this file: app/code/BDC/SampleNews/view/adminhtml/layout/bdc_samplenews_news_index.xml
#### Explain UI component contains child tags argument ,dataSource ,columns ?
-Argument: data_sources to use (which makes the links between your grid and the database) with the tag js_config.
We also declare the spinner, that is the name of the tag "columns" that will be used in our grid. We then declare our buttons in the buttons tag with a name, a label, a class and a target url.- dataSource: dataProvider (the object that will fetch our data in database). With a "class" tag to define the name of the object to be used. This object will be defined later in the di.xml (dependency node file).
We give a name to our dataSource via the "name" attribute and then we give it the field to use as the id for the grid in the database ("primaryFieldName") and for the request ("requestFieldName"). We then define in "config" the component to use (here "Magento_Ui/js/grid/provider") and the identifier in our bdd "indexField" which here has the value "pfay_contacts_id".- columns: It was defined above in the "spinner" section of the "argument" section, here it is named listing_columns.
This area will allow us to define our columns with the identifier to be used to find oneself, the type of fields and filters to use for the grid, the type of sorting that will be used and a label.#### Bookmarks?
The bookmarks allows you to save the state of the listing which you modified with the element "columns_control" previously created. Here is how to integrate the "bookmark" in the "container" :
```
Magento_Ui/js/grid/controls/bookmarks/bookmarks
dataGridActions
contact_test_listing
```
#### Pagination?
The pagination of the grid under magento2 is super well done and very easy to integrate, it is enough just to pay attention to the 2 paths "provider" and "selectProvider".
Here is the code to insert:
```
contacts_test_listing.contacts_test_listing.listing_top.bookmarks
current.paging
contacts_test_listing.contacts_test_listing.contacts_test_columns.pfay_contacts_id
bottom
```
#### Magento2 grid filters?
To be able to filter the table can sometimes be practical, for that a "filter" element can be added to the magento grid. Here's how to do it:
```
contacts_test_listing.contacts_test_listing.listing_top.bookmarks
curren.filters
contacts_test_listing.contacts_test_listing.listing_top.listing_filters
contacts_test_listing.contacts_test_listing.listing_top.bookmarks:current.columns.${ $.index }.visible
```
By default, it takes all the fields available on the grid, it knows how to filter with the "filter" item of your "columns" like these:
ici type text :
textici type textRange :
textRange#### Mass Actions under magento2?
You want to be able to select several lines of your grid to delete them all at once or do another specific processing on all the lines selected at the same time? The Mass Actions are made for this. First of all it will be necessary to add the inputs on the edge of our grid to be able to select the lines, so in "columns" add this before the "column":
```
pfay_contacts_id
```
You now see the checkboxes on the side that allow you to select multiple lines. Here is how to integrate the selectbox which allows to select the action to be performed once we have selected our lines:
```
contacts_test_listing.contacts_test_listing.contacts_test_columns.ids
bottom
pfay_contacts_id
delete
Delete Selected
Delete all selected contacts
Do you want to delete all the selected contacts?
```
Here it is the same, we have to be careful on what we enter as a path for the "selectProvider" and we add the actions following each other. In order to prepare the next tutorial, we will create the MassDelete controller. This is where we will be redirected when we select our action (*/*/massDelete).
#### Create a search field in admin magento2?
To create a search field on the magento admin, you must add an optional element in the container that will be called "filterSearch" like this: In
```
contacts_test_listing.contacts_test_listing_data_source
contacts_test_listing.contacts_test_listing.listing_top.listing_filters_chips
contacts_test_listing.contacts_test_listing.listing_top.bookmarks
current.search
```
For the searchbar to work you have to update your table to add the index.
### Step2I1: Rendering Grid(collections & listing component configuration)
- 1.grid collections
- 2.listing component configuration- Create [Controller/Adminhtml/Index/Index.php](Controller/Adminhtml/Index/Index.php)
Source
```
resultFactory->create(ResultFactory::TYPE_PAGE);
}
}```
- Create [Model/Resource/News/Grid/Collection.php](Model/Resource/News/Grid/Collection.php)
Source
```
- Edit [etc/di.xml](etc/di.xml)
Source
```
BDC\SimpleNews\Model\Resource\News\Grid\Collection
```
- Create [view/adminhtml/layout/simplenews_index_index.xml](view/adminhtml/layout/simplenews_index_index.xml)
Source
```
```
- Create [view/adminhtml/ui_component/bdc_news_grid.xml](view/adminhtml/ui_component/bdc_news_grid.xml)
Source
```
bdc_news_grid.bdc_news_grid_data_source
bdc_news_grid.bdc_news_grid_data_source
bdc_news_columns
add
Add News
primary
*/news/new
Magento\Framework\View\Element\UiComponent\DataProvider\DataProvider
bdc_news_grid_data_source
id
id
Magento_Ui/js/grid/provider
bdc_news_grid.bdc_news_grid.bdc_news_columns.actions
applyAction
view
${ $.$data.rowIndex }
id
text
Title
text
Summary
```
- Clean Cache & Run
http://www.magento.lan/cadmin/simplenews/
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/GridViewNewsListAdmin.png)
#### Goal
- New & save controllers- UI data provider form
- Form ui component configuration
#### Implement Procedure:
- Create [Ui/DataProvider.php](Ui/DataProvider.php)
Source
```
collection = $collectionFactory->create();
}public function getData() {
$result = [];
foreach ($this->collection->getItems() as $item) {
$result[$item->getId()]['general'] = $item->getData();
}
return $result;
}
}```
- Create [Controller/Adminhtml/Index/Index.php](Controller/Adminhtml/Index/Index.php)
Source
```
resultFactory->create(ResultFactory::TYPE_PAGE);
}
}```
- Create [Controller/Adminhtml/Index/NewAction.php](Controller/Adminhtml/Index/NewAction.php)
Source
```
resultFactory->create(ResultFactory::TYPE_PAGE);
}
}```
- Create [Controller/Adminhtml/Index/Save.php](Controller/Adminhtml/Index/NewAction.php)
Source
```
newsFactory = $newsFactory;
parent::__construct($context);
}public function execute(){
$this->newsFactory->create()
->setData($this->getRequest()->getNewsValue()['general'])->save();
return $this->resultRedirectFactory->create()->setPath('simplenews/index/index');
}
}```
- create [view/adminhtml/layout/simplenews_index_index.xml](view/adminhtml/layout/simplenews_index_index.xml)
Source
```
```
- create [view/adminhtml/ui_component/bdc_news_grid.xml](view/adminhtml/ui_component/bdc_news_grid.xml)
Source
```
bdc_news_grid.bdc_news_grid_data_source
bdc_news_grid.bdc_news_grid_data_source
bdc_news_columns
add
Add News
primary
*/index/new
Magento\Framework\View\Element\UiComponent\DataProvider\DataProvider
bdc_news_grid_data_source
id
id
Magento_Ui/js/grid/provider
bdc_news_grid.bdc_news_grid.bdc_news_columns.actions
applyAction
view
${ $.$data.rowIndex }
id
text
Title
text
Summary
```
- check admin panel as
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/GridNewsList.png)- create [view/adminhtml/layout/simplenews_index_new.xml](view/adminhtml/layout/simplenews_index_new.xml)
Source
```
```
- create [view/adminhtml/ui_component/bdc_news_form.xml](view/adminhtml/ui_component/bdc_news_form.xml)
Source
```
bdc_news_form.bdc_news_form_data_source
bdc_news_form.bdc_news_form_data_source
General
tabs
left
save
Save
primary
*/*/save
BDC\SimpleNews\Ui\DataProvider
bdc_news_form_data_source
id
id
BDC\SimpleNews\Model\Resource\News\CollectionFactory
Magento_Ui/js/form/provider
General
Title
text
input
true
Summary
text
input
```
- edit [etc/adminhtml/menu.xml](etc/adminhtml/menu.xml) add below code
```
```
- Check Output on Admin as below
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/GridCollectionNew.png)
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/gridCollectionList.png)### Step2I3: Extending UI Component
#### Goal
- Extending XML Configuration [Magento_Sales]
- Adding New Column to Grid [Magento_Sales]
- Customizing Existing Column [Magento_Sales]#### Implementation
- Copy To app/code/BDC/SimpleNews/view/adminhtml/layout/sales_order_index.xml <==vendor/magento/module-sales/view/adminhtml/layout/sales_order_grid.xml
- Create [view/adminhtml/ui_component/sales_order_grid.xml](view/adminhtml/ui_component/sales_order_grid.xml)
Source
```
MMM dd, YYYY
```
- Edit [etc/di.xml](etc/di.xml)
Source
```
sales_order.base_tax_amount
```
- Add new field 'base_tax_amount' on table sales_order_grid(add filed PhpMyAdmin)
![](https://github.com/bdcrops/BDC_SimpleNews/blob/master/doc/ExtendingXML.png)
## PartJ: Customer/Others [Go to Top](#top)
### Step2J1:Create Indexing & Reindex
#### What is Indexing?
Magento transforms data such as products, categories, and so on, to improve the performance of your storefront. As data changes, the transformed data must be updated—or reindexed. Magento has a very sophisticated architecture that stores lots of merchant data (including catalog data, prices, users, stores, and so on) in many database tables. To optimize storefront performance, Magento accumulates data into special tables using indexers.For example, suppose you change the price of an item from $8.99 to $6.99. Magento must reindex the price change to display it on your storefront.
Without indexing, Magento would have to calculate the price of every product on the fly—taking into account shopping cart price rules, bundle pricing, discounts, tier pricing, and so on. Loading the price for a product would take a long time, possibly resulting in cart abandonment.
#### implementation
- Create [etc/indexer.xml](etc/indexer.xml)
- Create [etc/mview.xml](etc/mview.xml)
- Create [Model/Indexer/Test.php](Model/Indexer/Test.php)
- Run Reindex by command
Run reindex by command linephp bin/magento indexer:reindex
### Step2J2:
### Step2J3
### Step2J4:
### Step2J5:
### Step2J6:## PartK: Products Collection [Go to Top](#top)
### Step2K1:Get New Products Collection
### Step2K1:Get Best Sellers Collection
### Step2K1:Get Most Viewed Product Collection
### Step2K1:Get Featured Product Collection
### Step2K1:Get On Sale Products Collection
### Step2K1:Get Get Recent Viewed Products Collection
### Step2K1:Get Wishlist Products Collection
### Step2K1:Get Rating Collection
### Step2K1:Get Review Collection***
## Ref [Go to Top](#top)
***
https://devdocs.magento.com/guides/v2.3/extension-dev-guide/declarative-schema/https://onilab.com/blog/declarative-schema-magento-2-3-and-higherProducts
https://www.mage-world.com/blog/create-a-module-with-custom-database-table-in-magento-2.html
http://techjeffyu.com/blog/magento-2-a-full-magento-2-module
https://github.com/codingarrow/M2/tree/master/BDC/SimpleNews