8264d6b0bc
- Use absolute route even 3-4 sub-category levels down - Remove scroll from category pills due to accessibility issues - Arrange sub-category pills alphabetically
532 lines
15 KiB
JavaScript
532 lines
15 KiB
JavaScript
erpnext.ProductView = class {
|
|
/* Options:
|
|
- View Type
|
|
- Products Section Wrapper,
|
|
- Item Group: If its an Item Group page
|
|
*/
|
|
constructor(options) {
|
|
Object.assign(this, options);
|
|
this.preference = this.view_type;
|
|
this.make();
|
|
}
|
|
|
|
make(from_filters=false) {
|
|
this.products_section.empty();
|
|
this.prepare_toolbar();
|
|
this.get_item_filter_data(from_filters);
|
|
}
|
|
|
|
prepare_toolbar() {
|
|
this.products_section.append(`
|
|
<div class="toolbar d-flex">
|
|
</div>
|
|
`);
|
|
this.prepare_search();
|
|
this.prepare_view_toggler();
|
|
|
|
new erpnext.ProductSearch();
|
|
}
|
|
|
|
prepare_view_toggler() {
|
|
|
|
if (!$("#list").length || !$("#image-view").length) {
|
|
this.render_view_toggler();
|
|
this.bind_view_toggler_actions();
|
|
this.set_view_state();
|
|
}
|
|
}
|
|
|
|
get_item_filter_data(from_filters=false) {
|
|
// Get and render all Product related views
|
|
let me = this;
|
|
this.from_filters = from_filters;
|
|
let args = this.get_query_filters();
|
|
|
|
this.disable_view_toggler(true);
|
|
|
|
frappe.call({
|
|
method: "erpnext.e_commerce.api.get_product_filter_data",
|
|
args: {
|
|
query_args: args
|
|
},
|
|
callback: function(result) {
|
|
if (!result || result.exc || !result.message || result.message.exc) {
|
|
me.render_no_products_section(true);
|
|
} else {
|
|
// Sub Category results are independent of Items
|
|
if (me.item_group && result.message["sub_categories"].length) {
|
|
me.render_item_sub_categories(result.message["sub_categories"]);
|
|
}
|
|
|
|
if (!result.message["items"].length) {
|
|
// if result has no items or result is empty
|
|
me.render_no_products_section();
|
|
} else {
|
|
// Add discount filters
|
|
me.re_render_discount_filters(result.message["filters"].discount_filters);
|
|
|
|
// Render views
|
|
me.render_list_view(result.message["items"], result.message["settings"]);
|
|
me.render_grid_view(result.message["items"], result.message["settings"]);
|
|
|
|
me.products = result.message["items"];
|
|
me.product_count = result.message["items_count"];
|
|
}
|
|
|
|
// Bind filter actions
|
|
if (!from_filters) {
|
|
// If `get_product_filter_data` was triggered after checking a filter,
|
|
// don't touch filters unnecessarily, only data must change
|
|
// filter persistence is handle on filter change event
|
|
me.bind_filters();
|
|
me.restore_filters_state();
|
|
}
|
|
|
|
// Bottom paging
|
|
me.add_paging_section(result.message["settings"]);
|
|
}
|
|
|
|
me.disable_view_toggler(false);
|
|
}
|
|
});
|
|
}
|
|
|
|
disable_view_toggler(disable=false) {
|
|
$('#list').prop('disabled', disable);
|
|
$('#image-view').prop('disabled', disable);
|
|
}
|
|
|
|
render_grid_view(items, settings) {
|
|
// loop over data and add grid html to it
|
|
let me = this;
|
|
this.prepare_product_area_wrapper("grid");
|
|
|
|
new erpnext.ProductGrid({
|
|
items: items,
|
|
products_section: $("#products-grid-area"),
|
|
settings: settings,
|
|
preference: me.preference
|
|
});
|
|
}
|
|
|
|
render_list_view(items, settings) {
|
|
let me = this;
|
|
this.prepare_product_area_wrapper("list");
|
|
|
|
new erpnext.ProductList({
|
|
items: items,
|
|
products_section: $("#products-list-area"),
|
|
settings: settings,
|
|
preference: me.preference
|
|
});
|
|
}
|
|
|
|
prepare_product_area_wrapper(view) {
|
|
let left_margin = view == "list" ? "ml-2" : "";
|
|
let top_margin = view == "list" ? "mt-6" : "mt-minus-1";
|
|
return this.products_section.append(`
|
|
<br>
|
|
<div id="products-${view}-area" class="row products-list ${ top_margin } ${ left_margin }"></div>
|
|
`);
|
|
}
|
|
|
|
get_query_filters() {
|
|
const filters = frappe.utils.get_query_params();
|
|
let {field_filters, attribute_filters} = filters;
|
|
|
|
field_filters = field_filters ? JSON.parse(field_filters) : {};
|
|
attribute_filters = attribute_filters ? JSON.parse(attribute_filters) : {};
|
|
|
|
return {
|
|
field_filters: field_filters,
|
|
attribute_filters: attribute_filters,
|
|
item_group: this.item_group,
|
|
start: filters.start || null,
|
|
from_filters: this.from_filters || false
|
|
};
|
|
}
|
|
|
|
add_paging_section(settings) {
|
|
$(".product-paging-area").remove();
|
|
|
|
if (this.products) {
|
|
let paging_html = `
|
|
<div class="row product-paging-area mt-5">
|
|
<div class="col-3">
|
|
</div>
|
|
<div class="col-9 text-right">
|
|
`;
|
|
let query_params = frappe.utils.get_query_params();
|
|
let start = query_params.start ? cint(JSON.parse(query_params.start)) : 0;
|
|
let page_length = settings.products_per_page || 0;
|
|
|
|
let prev_disable = start > 0 ? "" : "disabled";
|
|
let next_disable = (this.product_count > page_length) ? "" : "disabled";
|
|
|
|
paging_html += `
|
|
<button class="btn btn-default btn-prev" data-start="${ start - page_length }"
|
|
style="float: left" ${prev_disable}>
|
|
${ __("Prev") }
|
|
</button>`;
|
|
|
|
paging_html += `
|
|
<button class="btn btn-default btn-next" data-start="${ start + page_length }"
|
|
${next_disable}>
|
|
${ __("Next") }
|
|
</button>
|
|
`;
|
|
|
|
paging_html += `</div></div>`;
|
|
|
|
$(".page_content").append(paging_html);
|
|
this.bind_paging_action();
|
|
}
|
|
}
|
|
|
|
prepare_search() {
|
|
$(".toolbar").append(`
|
|
<div class="input-group col-8 p-0">
|
|
<div class="dropdown w-100" id="dropdownMenuSearch">
|
|
<input type="search" name="query" id="search-box" class="form-control font-md"
|
|
placeholder="Search for Products"
|
|
aria-label="Product" aria-describedby="button-addon2">
|
|
<div class="search-icon">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor" stroke-width="2" stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
class="feather feather-search">
|
|
<circle cx="11" cy="11" r="8"></circle>
|
|
<line x1="21" y1="21" x2="16.65" y2="16.65"></line>
|
|
</svg>
|
|
</div>
|
|
<!-- Results dropdown rendered in product_search.js -->
|
|
</div>
|
|
</div>
|
|
`);
|
|
}
|
|
|
|
render_view_toggler() {
|
|
$(".toolbar").append(`<div class="toggle-container col-4 p-0"></div>`);
|
|
|
|
["btn-list-view", "btn-grid-view"].forEach(view => {
|
|
let icon = view === "btn-list-view" ? "list" : "image-view";
|
|
$(".toggle-container").append(`
|
|
<div class="form-group mb-0" id="toggle-view">
|
|
<button id="${ icon }" class="btn ${ view } mr-2">
|
|
<span>
|
|
<svg class="icon icon-md">
|
|
<use href="#icon-${ icon }"></use>
|
|
</svg>
|
|
</span>
|
|
</button>
|
|
</div>
|
|
`);
|
|
});
|
|
}
|
|
|
|
bind_view_toggler_actions() {
|
|
$("#list").click(function() {
|
|
let $btn = $(this);
|
|
$btn.removeClass('btn-primary');
|
|
$btn.addClass('btn-primary');
|
|
$(".btn-grid-view").removeClass('btn-primary');
|
|
|
|
$("#products-grid-area").addClass("hidden");
|
|
$("#products-list-area").removeClass("hidden");
|
|
localStorage.setItem("product_view", "List View");
|
|
});
|
|
|
|
$("#image-view").click(function() {
|
|
let $btn = $(this);
|
|
$btn.removeClass('btn-primary');
|
|
$btn.addClass('btn-primary');
|
|
$(".btn-list-view").removeClass('btn-primary');
|
|
|
|
$("#products-list-area").addClass("hidden");
|
|
$("#products-grid-area").removeClass("hidden");
|
|
localStorage.setItem("product_view", "Grid View");
|
|
});
|
|
}
|
|
|
|
set_view_state() {
|
|
if (this.preference === "List View") {
|
|
$("#list").addClass('btn-primary');
|
|
$("#image-view").removeClass('btn-primary');
|
|
} else {
|
|
$("#image-view").addClass('btn-primary');
|
|
$("#list").removeClass('btn-primary');
|
|
}
|
|
}
|
|
|
|
bind_paging_action() {
|
|
let me = this;
|
|
$('.btn-prev, .btn-next').click((e) => {
|
|
const $btn = $(e.target);
|
|
me.from_filters = false;
|
|
|
|
$btn.prop('disabled', true);
|
|
const start = $btn.data('start');
|
|
|
|
let query_params = frappe.utils.get_query_params();
|
|
query_params.start = start;
|
|
let path = window.location.pathname + '?' + frappe.utils.get_url_from_dict(query_params);
|
|
window.location.href = path;
|
|
});
|
|
}
|
|
|
|
re_render_discount_filters(filter_data) {
|
|
this.get_discount_filter_html(filter_data);
|
|
if (this.from_filters) {
|
|
// Bind filter action if triggered via filters
|
|
// if not from filter action, page load will bind actions
|
|
this.bind_discount_filter_action();
|
|
}
|
|
// discount filters are rendered with Items (later)
|
|
// unlike the other filters
|
|
this.restore_discount_filter();
|
|
}
|
|
|
|
get_discount_filter_html(filter_data) {
|
|
$("#discount-filters").remove();
|
|
if (filter_data) {
|
|
$("#product-filters").append(`
|
|
<div id="discount-filters" class="mb-4 filter-block pb-5">
|
|
<div class="filter-label mb-3">${ __("Discounts") }</div>
|
|
</div>
|
|
`);
|
|
|
|
let html = `<div class="filter-options">`;
|
|
filter_data.forEach(filter => {
|
|
html += `
|
|
<div class="checkbox">
|
|
<label data-value="${ filter[0] }">
|
|
<input type="radio"
|
|
class="product-filter discount-filter"
|
|
name="discount" id="${ filter[0] }"
|
|
data-filter-name="discount"
|
|
data-filter-value="${ filter[0] }"
|
|
style="width: 14px !important"
|
|
>
|
|
<span class="label-area" for="${ filter[0] }">
|
|
${ filter[1] }
|
|
</span>
|
|
</label>
|
|
</div>
|
|
`;
|
|
});
|
|
html += `</div>`;
|
|
|
|
$("#discount-filters").append(html);
|
|
}
|
|
}
|
|
|
|
restore_discount_filter() {
|
|
const filters = frappe.utils.get_query_params();
|
|
let field_filters = filters.field_filters;
|
|
if (!field_filters) return;
|
|
|
|
field_filters = JSON.parse(field_filters);
|
|
|
|
if (field_filters && field_filters["discount"]) {
|
|
const values = field_filters["discount"];
|
|
const selector = values.map(value => {
|
|
return `input[data-filter-name="discount"][data-filter-value="${value}"]`;
|
|
}).join(',');
|
|
$(selector).prop('checked', true);
|
|
this.field_filters = field_filters;
|
|
}
|
|
}
|
|
|
|
bind_discount_filter_action() {
|
|
let me = this;
|
|
$('.discount-filter').on('change', (e) => {
|
|
const $checkbox = $(e.target);
|
|
const is_checked = $checkbox.is(':checked');
|
|
|
|
const {
|
|
filterValue: filter_value
|
|
} = $checkbox.data();
|
|
|
|
delete this.field_filters["discount"];
|
|
|
|
if (is_checked) {
|
|
this.field_filters["discount"] = [];
|
|
this.field_filters["discount"].push(filter_value);
|
|
}
|
|
|
|
if (this.field_filters["discount"].length === 0) {
|
|
delete this.field_filters["discount"];
|
|
}
|
|
|
|
me.change_route_with_filters();
|
|
});
|
|
}
|
|
|
|
bind_filters() {
|
|
let me = this;
|
|
this.field_filters = {};
|
|
this.attribute_filters = {};
|
|
|
|
$('.product-filter').on('change', (e) => {
|
|
me.from_filters = true;
|
|
|
|
const $checkbox = $(e.target);
|
|
const is_checked = $checkbox.is(':checked');
|
|
|
|
if ($checkbox.is('.attribute-filter')) {
|
|
const {
|
|
attributeName: attribute_name,
|
|
attributeValue: attribute_value
|
|
} = $checkbox.data();
|
|
|
|
if (is_checked) {
|
|
this.attribute_filters[attribute_name] = this.attribute_filters[attribute_name] || [];
|
|
this.attribute_filters[attribute_name].push(attribute_value);
|
|
} else {
|
|
this.attribute_filters[attribute_name] = this.attribute_filters[attribute_name] || [];
|
|
this.attribute_filters[attribute_name] = this.attribute_filters[attribute_name].filter(v => v !== attribute_value);
|
|
}
|
|
|
|
if (this.attribute_filters[attribute_name].length === 0) {
|
|
delete this.attribute_filters[attribute_name];
|
|
}
|
|
} else if ($checkbox.is('.field-filter') || $checkbox.is('.discount-filter')) {
|
|
const {
|
|
filterName: filter_name,
|
|
filterValue: filter_value
|
|
} = $checkbox.data();
|
|
|
|
if ($checkbox.is('.discount-filter')) {
|
|
// clear previous discount filter to accomodate new
|
|
delete this.field_filters["discount"];
|
|
}
|
|
if (is_checked) {
|
|
this.field_filters[filter_name] = this.field_filters[filter_name] || [];
|
|
if (!in_list(this.field_filters[filter_name], filter_value)) {
|
|
this.field_filters[filter_name].push(filter_value);
|
|
}
|
|
} else {
|
|
this.field_filters[filter_name] = this.field_filters[filter_name] || [];
|
|
this.field_filters[filter_name] = this.field_filters[filter_name].filter(v => v !== filter_value);
|
|
}
|
|
|
|
if (this.field_filters[filter_name].length === 0) {
|
|
delete this.field_filters[filter_name];
|
|
}
|
|
}
|
|
|
|
me.change_route_with_filters();
|
|
});
|
|
}
|
|
|
|
change_route_with_filters() {
|
|
let route_params = frappe.utils.get_query_params();
|
|
|
|
let start = this.if_key_exists(route_params.start) || 0;
|
|
if (this.from_filters) {
|
|
start = 0; // show items from first page if new filters are triggered
|
|
}
|
|
|
|
const query_string = this.get_query_string({
|
|
start: start,
|
|
field_filters: JSON.stringify(this.if_key_exists(this.field_filters)),
|
|
attribute_filters: JSON.stringify(this.if_key_exists(this.attribute_filters)),
|
|
});
|
|
window.history.pushState('filters', '', `${location.pathname}?` + query_string);
|
|
|
|
$('.page_content input').prop('disabled', true);
|
|
|
|
this.make(true);
|
|
$('.page_content input').prop('disabled', false);
|
|
}
|
|
|
|
restore_filters_state() {
|
|
const filters = frappe.utils.get_query_params();
|
|
let {field_filters, attribute_filters} = filters;
|
|
|
|
if (field_filters) {
|
|
field_filters = JSON.parse(field_filters);
|
|
for (let fieldname in field_filters) {
|
|
const values = field_filters[fieldname];
|
|
const selector = values.map(value => {
|
|
return `input[data-filter-name="${fieldname}"][data-filter-value="${value}"]`;
|
|
}).join(',');
|
|
$(selector).prop('checked', true);
|
|
}
|
|
this.field_filters = field_filters;
|
|
}
|
|
if (attribute_filters) {
|
|
attribute_filters = JSON.parse(attribute_filters);
|
|
for (let attribute in attribute_filters) {
|
|
const values = attribute_filters[attribute];
|
|
const selector = values.map(value => {
|
|
return `input[data-attribute-name="${attribute}"][data-attribute-value="${value}"]`;
|
|
}).join(',');
|
|
$(selector).prop('checked', true);
|
|
}
|
|
this.attribute_filters = attribute_filters;
|
|
}
|
|
}
|
|
|
|
render_no_products_section(error=false) {
|
|
let error_section = `
|
|
<div class="mt-4 w-100 alert alert-error font-md">
|
|
Something went wrong. Please refresh or contact us.
|
|
</div>
|
|
`;
|
|
let no_results_section = `
|
|
<div class="cart-empty frappe-card mt-4">
|
|
<div class="cart-empty-state">
|
|
<img src="/assets/erpnext/images/ui-states/cart-empty-state.png" alt="Empty Cart">
|
|
</div>
|
|
<div class="cart-empty-message mt-4">${ __('No products found') }</p>
|
|
</div>
|
|
`;
|
|
|
|
this.products_section.append(error ? error_section : no_results_section);
|
|
}
|
|
|
|
render_item_sub_categories(categories) {
|
|
if (categories && categories.length) {
|
|
let sub_group_html = `
|
|
<div class="sub-category-container scroll-categories">
|
|
`;
|
|
|
|
categories.forEach(category => {
|
|
sub_group_html += `
|
|
<a href="/${ category.route || '#' }" style="text-decoration: none;">
|
|
<div class="category-pill">
|
|
${ category.name }
|
|
</div>
|
|
</a>
|
|
`;
|
|
});
|
|
sub_group_html += `</div>`;
|
|
|
|
$("#product-listing").prepend(sub_group_html);
|
|
}
|
|
}
|
|
|
|
get_query_string(object) {
|
|
const url = new URLSearchParams();
|
|
for (let key in object) {
|
|
const value = object[key];
|
|
if (value) {
|
|
url.append(key, value);
|
|
}
|
|
}
|
|
return url.toString();
|
|
}
|
|
|
|
if_key_exists(obj) {
|
|
let exists = false;
|
|
for (let key in obj) {
|
|
if (Object.prototype.hasOwnProperty.call(obj, key) && obj[key]) {
|
|
exists = true;
|
|
break;
|
|
}
|
|
}
|
|
return exists ? obj : undefined;
|
|
}
|
|
}; |