
Frontend JavaScript
Modular JavaScript architecture for the store frontend.
Overview
MiniShop3 uses a modular JavaScript architecture without jQuery. All modules are plain ES6+ and target modern browsers.
Features:
- Modular structure (core, ui, modules)
- REST API client with token handling
- Hook system for extension
- Automatic cart rendering
- Events for integration
Module structure
assets/components/minishop3/js/web/
├── core/ # Core
│ ├── ApiClient.js # HTTP client
│ ├── TokenManager.js # Token management
│ ├── CartAPI.js # Cart API
│ ├── OrderAPI.js # Order API
│ └── CustomerAPI.js # Customer API
├── ui/ # UI handlers
│ ├── CartUI.js # Cart UI
│ ├── OrderUI.js # Order UI
│ └── CustomerUI.js # Customer UI
├── modules/ # Helper modules
│ ├── hooks.js # Hook system
│ ├── message.js # Notifications
│ ├── auth-forms.js # Auth forms
│ └── customer-addresses.js # Address management
├── lib/ # Libraries
│ └── izitoast/ # iziToast notifications
├── ms3.js # Main module
└── order-addresses.js # Order address selectionLoad order
System setting ms3_frontend_assets
Controls the order of CSS and JS assets:
[
"[[+cssUrl]]web/lib/izitoast/iziToast.min.css",
"[[+jsUrl]]web/lib/izitoast/iziToast.js",
"[[+jsUrl]]web/modules/hooks.js",
"[[+jsUrl]]web/modules/message.js",
"[[+jsUrl]]web/core/ApiClient.js",
"[[+jsUrl]]web/core/TokenManager.js",
"[[+jsUrl]]web/core/CartAPI.js",
"[[+jsUrl]]web/core/OrderAPI.js",
"[[+jsUrl]]web/core/CustomerAPI.js",
"[[+jsUrl]]web/ui/CartUI.js",
"[[+jsUrl]]web/ui/OrderUI.js",
"[[+jsUrl]]web/ui/CustomerUI.js",
"[[+jsUrl]]web/ms3.js"
]Placeholders:
| Placeholder | Value |
|---|---|
[[+cssUrl]] | assets/components/minishop3/css/ |
[[+jsUrl]] | assets/components/minishop3/js/ |
Loading:
- CSS in
<head> - JS at end of HTML with
defer
Changing order
Edit system setting ms3_frontend_assets to change or disable files.
Example: disable iziToast (use your own notifications):
[
"[[+jsUrl]]web/modules/hooks.js",
"[[+jsUrl]]web/modules/message.js",
"[[+jsUrl]]web/core/ApiClient.js",
...
]Main object ms3
Initialization
The main ms3 object is initialized automatically:
// assets/components/minishop3/js/web/ms3.js
const ms3 = {
// Config (from server)
config: {},
// Core
tokenManager: null,
apiClient: null,
// API modules
cartAPI: null,
orderAPI: null,
customerAPI: null,
// UI modules
cartUI: null,
orderUI: null,
customerUI: null,
// Utilities
hooks: null,
message: null,
// Init
async init() {
this.config = window.ms3Config || {}
// Hooks and messages
this.hooks = window.ms3Hooks
this.message = window.ms3Message
// Tokens and API client
this.tokenManager = new TokenManager(this.config)
await this.tokenManager.ensureToken()
this.apiClient = new ApiClient(this.config, this.tokenManager)
// API modules
this.cartAPI = new CartAPI(this.apiClient)
this.orderAPI = new OrderAPI(this.apiClient)
this.customerAPI = new CustomerAPI(this.apiClient)
// UI modules
this.cartUI = new CartUI(this.cartAPI, this.hooks, this.message, this.config)
this.orderUI = new OrderUI(this.orderAPI, this.hooks, this.message, this.config)
this.customerUI = new CustomerUI(this.customerAPI, this.hooks, this.message, this.config)
// Init UI
this.cartUI.init()
this.orderUI.init()
this.customerUI.init()
// Form handlers
this.initFormHandler()
}
}ms3Config
Passed from the server via snippets:
window.ms3Config = {
apiUrl: '/assets/components/minishop3/api.php',
token: 'abc123...',
tokenName: 'ms3_token',
// Render config
render: {
cart: [
{ token: 'mini', selector: '#headerMiniCart' },
{ token: 'full', selector: '#cartContent' }
]
}
}Core
TokenManager
Auth token management with localStorage:
class TokenManager {
constructor(config) {
this.config = config
this.tokenName = config.tokenName || 'ms3_token'
}
getToken() {
const data = localStorage.getItem(this.tokenName)
if (!data) return null
const { token, expires } = JSON.parse(data)
if (Date.now() > expires) {
this.removeToken()
return null
}
return token
}
setToken(token, ttl = 86400) {
const data = {
token,
expires: Date.now() + (ttl * 1000)
}
localStorage.setItem(this.tokenName, JSON.stringify(data))
}
removeToken() {
localStorage.removeItem(this.tokenName)
}
async ensureToken() {
let token = this.getToken()
if (!token) {
token = await this.fetchNewToken()
}
return token
}
async fetchNewToken() {
const response = await fetch(`${this.config.apiUrl}?route=/api/v1/token`)
const data = await response.json()
if (data.success && data.object?.token) {
this.setToken(data.object.token, data.object.ttl || 86400)
return data.object.token
}
return null
}
}Usage:
// Get current token
const token = ms3.tokenManager.getToken()
// Force refresh
await ms3.tokenManager.fetchNewToken()ApiClient
HTTP client that adds the token automatically:
class ApiClient {
constructor(config, tokenManager) {
this.baseUrl = config.apiUrl || '/assets/components/minishop3/api.php'
this.tokenManager = tokenManager
}
async request(method, route, data = null) {
const url = new URL(this.baseUrl, window.location.origin)
url.searchParams.set('route', route)
const token = this.tokenManager.getToken()
const options = {
method,
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
}
if (token) {
options.headers['Authorization'] = `Bearer ${token}`
}
if (data && method !== 'GET') {
options.body = JSON.stringify(data)
}
const response = await fetch(url.toString(), options)
return response.json()
}
get(route) { return this.request('GET', route) }
post(route, data) { return this.request('POST', route, data) }
put(route, data) { return this.request('PUT', route, data) }
patch(route, data) { return this.request('PATCH', route, data) }
delete(route) { return this.request('DELETE', route) }
}CartAPI
Cart API:
class CartAPI {
constructor(apiClient) {
this.api = apiClient
}
async get() {
return this.api.get('/api/v1/cart/get')
}
async add(id, count = 1, options = {}, renderTokens = null) {
return this.api.post('/api/v1/cart/add', {
id, count, options, render: renderTokens
})
}
async change(key, count, renderTokens = null) {
return this.api.post('/api/v1/cart/change', {
key, count, render: renderTokens
})
}
async remove(key, renderTokens = null) {
return this.api.post('/api/v1/cart/remove', {
key, render: renderTokens
})
}
async clean(renderTokens = null) {
return this.api.post('/api/v1/cart/clean', {
render: renderTokens
})
}
}Examples:
// Add to cart
await ms3.cartAPI.add(123, 2, { color: 'red', size: 'L' })
// Change quantity
await ms3.cartAPI.change('abc123', 5)
// Remove item
await ms3.cartAPI.remove('abc123')OrderAPI
Order API:
class OrderAPI {
constructor(apiClient) {
this.api = apiClient
}
async add(key, value) {
return this.api.post('/api/v1/order/add', { key, value })
}
async remove(key) {
return this.api.post('/api/v1/order/remove', { key })
}
async clean() {
return this.api.post('/api/v1/order/clean')
}
async submit() {
return this.api.post('/api/v1/order/submit')
}
async get() {
return this.api.get('/api/v1/order/get')
}
}CustomerAPI
Customer API:
class CustomerAPI {
constructor(apiClient) {
this.api = apiClient
}
async add(key, value) {
return this.api.post('/api/v1/customer/add', { key, value })
}
async changeAddress(key, value) {
return this.api.post('/api/v1/customer/changeAddress', { key, value })
}
async updateProfile(data) {
return this.api.put('/api/v1/customer/profile', data)
}
async createAddress(data) {
return this.api.post('/api/v1/customer/addresses', data)
}
async updateAddress(id, data) {
return this.api.put(`/api/v1/customer/addresses/${id}`, data)
}
async deleteAddress(id) {
return this.api.delete(`/api/v1/customer/addresses/${id}`)
}
}UI modules
CartUI
Cart UI handler:
class CartUI {
constructor(cartAPI, hooks, message, config) {
this.cart = cartAPI
this.hooks = hooks
this.message = message
this.config = config
}
init() {
this.initQuantityButtons() // +/- buttons
this.initQuantityInputs() // Quantity inputs
this.initOptionSelects() // Option selects
}
async handleAdd(id, count = 1, options = {}) {
const hookData = { id, count, options }
await this.hooks.runHooks('beforeAddCart', hookData)
if (hookData.cancel) return
const renderTokens = this.getRenderTokens()
const response = await this.cart.add(id, count, options, renderTokens)
await this.hooks.runHooks('afterAddCart', { id, count, options, response })
if (response.success) {
if (response.data?.render) {
this.renderCart(response.data.render)
}
this.dispatchCartUpdated(response.data)
if (response.message) {
this.message.success(response.message)
}
} else {
if (response.message) {
this.message.error(response.message)
}
}
}
renderCart(renderData) {
const cartRenderConfig = this.config.render?.cart
for (const token in renderData) {
const html = renderData[token]
const config = cartRenderConfig.find(item => item.token === token)
if (config?.selector) {
const element = document.querySelector(config.selector)
if (element) {
element.innerHTML = html
}
}
}
// Re-init handlers
setTimeout(() => this.init(), 100)
}
dispatchCartUpdated(data) {
document.dispatchEvent(new CustomEvent('ms3:cart:updated', {
detail: data
}))
}
}OrderUI
Order form handler:
class OrderUI {
init() {
document.querySelectorAll('.ms3_order_form').forEach(form => {
this.initForm(form)
})
}
initRegularInput(input) {
input.addEventListener('change', async () => {
const response = await this.handleAdd(input.name, input.value)
if (response.success) {
if (response.data?.[input.name] !== undefined) {
input.value = response.data[input.name]
}
} else {
input.classList.add('is-invalid')
}
})
}
async handleSubmit() {
const hookData = {}
await this.hooks.runHooks('beforeSubmitOrder', hookData)
if (hookData.cancel) return { success: false }
const response = await this.order.submit()
await this.hooks.runHooks('afterSubmitOrder', { response })
if (response.success && response.data?.redirect) {
window.location.href = response.data.redirect
}
return response
}
}CustomerUI
Customer form handler:
class CustomerUI {
init() {
document.querySelectorAll('.ms3_customer_form').forEach(form => {
this.initForm(form)
})
}
async handleProfileUpdate(formData) {
const data = {}
for (const [key, value] of formData.entries()) {
if (key === 'ms3_action') continue
data[key] = value
}
const response = await this.customer.updateProfile(data)
if (response.success) {
this.message.success('Profile updated')
setTimeout(() => window.location.reload(), 1000)
}
return response
}
async handleAddressCreate(formData) {
// Similar logic
}
}Hook system
Registering hooks
// window.ms3Hooks
const ms3Hooks = {
hooks: {},
addHook(name, callback, priority = 10) {
if (!this.hooks[name]) {
this.hooks[name] = []
}
this.hooks[name].push({ callback, priority })
this.hooks[name].sort((a, b) => a.priority - b.priority)
},
async runHooks(name, data = {}) {
if (!this.hooks[name]) return data
for (const hook of this.hooks[name]) {
await hook.callback(data)
if (data.cancel) break
}
return data
}
}Available hooks
Cart
| Hook | Description |
|---|---|
beforeAddCart | Before adding product |
afterAddCart | After adding product |
beforeChangeCart | Before changing quantity |
afterChangeCart | After changing quantity |
beforeRemoveCart | Before removing product |
afterRemoveCart | After removing product |
beforeCleanCart | Before clearing cart |
afterCleanCart | After clearing cart |
Order
| Hook | Description |
|---|---|
beforeAddOrder | Before saving field |
afterAddOrder | After saving field |
beforeSubmitOrder | Before submit |
afterSubmitOrder | After submit |
beforeCleanOrder | Before clear |
afterCleanOrder | After clear |
Customer
| Hook | Description |
|---|---|
beforeAddCustomer | Before saving data |
afterAddCustomer | After saving data |
beforeUpdateProfile | Before profile update |
afterUpdateProfile | After profile update |
beforeCreateAddress | Before creating address |
afterCreateAddress | After creating address |
beforeChangeAddressCustomer | Before changing address |
afterChangeAddressCustomer | After changing address |
Examples
Validation before add:
ms3Hooks.addHook('beforeAddCart', async (data) => {
if (data.count < 1) {
ms3.message.error('Minimum quantity: 1')
data.cancel = true
return
}
if (!data.options.size) {
ms3.message.warning('Select size')
data.cancel = true
return
}
})Analytics:
ms3Hooks.addHook('afterAddCart', async (data) => {
gtag('event', 'add_to_cart', {
currency: 'RUB',
items: [{
item_id: data.id,
quantity: data.count
}]
})
ym(COUNTER_ID, 'reachGoal', 'add_to_cart')
})Clear cart confirmation:
ms3Hooks.addHook('beforeCleanCart', async (data) => {
if (!confirm('Clear cart?')) {
data.cancel = true
}
})Order error handling:
ms3Hooks.addHook('afterSubmitOrder', async (data) => {
if (!data.response.success) {
data.response.errors?.forEach(field => {
const input = document.querySelector(`[name="${field}"]`)
if (input) {
input.classList.add('is-invalid')
}
})
}
})Notifications
ms3Message
// window.ms3Message
const ms3Message = {
show(type, message) {
if (typeof iziToast !== 'undefined') {
iziToast[type]({
message,
position: 'topRight',
timeout: 3000
})
} else {
alert(message)
}
},
success(message) { this.show('success', message) },
error(message) { this.show('error', message) },
info(message) { this.show('info', message) },
warning(message) { this.show('warning', message) }
}Custom notifications
Replace with your library:
// Override after ms3 loads
window.ms3Message = {
show(type, message) {
Toastify({
text: message,
className: type,
duration: 3000
}).showToast()
},
success(message) { this.show('success', message) },
error(message) { this.show('error', message) },
info(message) { this.show('info', message) },
warning(message) { this.show('warning', message) }
}DOM events
Subscribing
// Cart update
document.addEventListener('ms3:cart:updated', (e) => {
console.log('Cart updated:', e.detail)
const counter = document.querySelector('.cart-counter')
if (counter) {
counter.textContent = e.detail.total_count || 0
}
})Forms and handlers
Automatic form handling
MiniShop3 handles forms with class .ms3_form and attributes data-ms3-entity and data-ms3-method:
<!-- Add to cart -->
<form class="ms3_form" data-ms3-entity="cart" data-ms3-method="add">
<input type="hidden" name="id" value="123">
<input type="number" name="count" value="1">
<button type="submit">Add to cart</button>
</form>
<!-- Submit order -->
<form class="ms3_form" data-ms3-entity="order" data-ms3-method="submit">
<button type="submit">Checkout</button>
</form>Routing
// From ms3.js
initFormHandler() {
document.querySelectorAll('.ms3_form').forEach(form => {
form.addEventListener('submit', async (e) => {
e.preventDefault()
const entity = form.dataset.ms3Entity
const method = form.dataset.ms3Method
const formData = new FormData(form)
switch (`${entity}/${method}`) {
case 'cart/add':
await this.cartUI.handleAdd(
formData.get('id'),
formData.get('count'),
this.extractOptions(formData)
)
break
case 'order/submit':
await this.orderUI.handleSubmit()
break
case 'customer/profile':
await this.customerUI.handleProfileUpdate(formData)
break
// ...
}
})
})
}Extension and customization
Replacing API module
// Custom CartAPI
class CustomCartAPI extends CartAPI {
async add(id, count, options, renderTokens) {
console.log('Adding product:', id)
const result = await super.add(id, count, options, renderTokens)
console.log('Product added:', result)
return result
}
}
// Replace after init
document.addEventListener('DOMContentLoaded', () => {
ms3.cartAPI = new CustomCartAPI(ms3.apiClient)
ms3.cartUI.cart = ms3.cartAPI
})Replacing UI module
// Custom CartUI
class CustomCartUI extends CartUI {
async handleAdd(id, count, options) {
document.body.classList.add('loading')
await super.handleAdd(id, count, options)
document.body.classList.remove('loading')
}
renderCart(renderData) {
super.renderCart(renderData)
this.animateCartUpdate()
}
animateCartUpdate() {
const cart = document.querySelector('.cart-icon')
cart?.classList.add('bounce')
setTimeout(() => cart?.classList.remove('bounce'), 500)
}
}Adding a module
// Favorites module
class FavoritesAPI {
constructor(apiClient) {
this.api = apiClient
}
async add(productId) {
return this.api.post('/api/v1/favorites/add', { id: productId })
}
async remove(productId) {
return this.api.post('/api/v1/favorites/remove', { id: productId })
}
async get() {
return this.api.get('/api/v1/favorites/get')
}
}
// Attach
document.addEventListener('DOMContentLoaded', () => {
ms3.favoritesAPI = new FavoritesAPI(ms3.apiClient)
})Frontend lexicons
ms3Lexicon
window.ms3Lexicon = {
'ms3_frontend_currency': 'USD',
'ms3_frontend_add_to_cart': 'Add to cart',
'ms3_customer_login_success': 'You are logged in',
// ...
}Usage in code
// In AuthForms
getLexicon(key) {
if (window.ms3Lexicon?.[key]) {
return window.ms3Lexicon[key]
}
const fallbacks = {
'ms3_customer_err_login_required': 'Enter email and password',
'ms3_customer_login_success': 'You are logged in'
}
return fallbacks[key] || key
}Additional modules
AuthForms
Auth and registration forms:
const authForms = new AuthForms({
apiUrl: '/assets/components/minishop3/api.php',
loginRoute: '/api/v1/customer/login',
registerRoute: '/api/v1/customer/register',
loginFormId: 'ms3-login-form',
registerFormId: 'ms3-register-form'
})
authForms.init()CustomerAddresses
Address management in account:
const customerAddresses = new CustomerAddresses({
apiUrl: '/assets/components/minishop3/api.php',
setDefaultSelector: '.set-default-address',
deleteSelector: '.delete-address'
})
customerAddresses.init()order-addresses.js
Saved address selection on checkout:
<select id="saved_address_id">
<option value="">New address</option>
<option value="1" data-address='{"city":"New York","street":"5th Ave"}'>
New York, 5th Ave
</option>
</select>The module fills the form when an address is selected.
Compatibility
Requirements
- Modern browsers (ES6+)
- No jQuery
fetch,Promise,async/awaitsupport
Polyfills
For older browsers include polyfills:
<script src="https://polyfill.io/v3/polyfill.min.js?features=fetch,Promise"></script>