mirror of
https://github.com/frappe/erpnext.git
synced 2026-06-06 05:39:12 +00:00
chore: Drive E-commerce via Website Item
- Removed Shopping Cart Settings - Portal fully driven via E Commerce Settings - All Item listing querying will happen via ProductQuery engine only - Product Listing via Website Items - removed redundant code - Moved all website logic from Item to Website Item
This commit is contained in:
@@ -3,160 +3,6 @@ from frappe.utils import cint
|
||||
from erpnext.portal.product_configurator.item_variants_cache import ItemVariantsCacheManager
|
||||
from erpnext.shopping_cart.product_info import get_product_info_for_website
|
||||
|
||||
def get_field_filter_data():
|
||||
e_commerce_settings = get_e_commerce_settings()
|
||||
filter_fields = [row.fieldname for row in e_commerce_settings.filter_fields]
|
||||
|
||||
meta = frappe.get_meta('Item')
|
||||
fields = [df for df in meta.fields if df.fieldname in filter_fields]
|
||||
|
||||
filter_data = []
|
||||
for f in fields:
|
||||
doctype = f.get_link_doctype()
|
||||
|
||||
# apply enable/disable/show_in_website filter
|
||||
meta = frappe.get_meta(doctype)
|
||||
filters = {}
|
||||
if meta.has_field('enabled'):
|
||||
filters['enabled'] = 1
|
||||
if meta.has_field('disabled'):
|
||||
filters['disabled'] = 0
|
||||
if meta.has_field('show_in_website'):
|
||||
filters['show_in_website'] = 1
|
||||
|
||||
values = [d.name for d in frappe.get_all(doctype, filters)]
|
||||
filter_data.append([f, values])
|
||||
|
||||
return filter_data
|
||||
|
||||
|
||||
def get_attribute_filter_data():
|
||||
e_commerce_settings = get_e_commerce_settings()
|
||||
attributes = [row.attribute for row in e_commerce_settings.filter_attributes]
|
||||
attribute_docs = [
|
||||
frappe.get_doc('Item Attribute', attribute) for attribute in attributes
|
||||
]
|
||||
|
||||
# mark attribute values as checked if they are present in the request url
|
||||
if frappe.form_dict:
|
||||
for attr in attribute_docs:
|
||||
if attr.name in frappe.form_dict:
|
||||
value = frappe.form_dict[attr.name]
|
||||
if value:
|
||||
enabled_values = value.split(',')
|
||||
else:
|
||||
enabled_values = []
|
||||
|
||||
for v in enabled_values:
|
||||
for item_attribute_row in attr.item_attribute_values:
|
||||
if v == item_attribute_row.attribute_value:
|
||||
item_attribute_row.checked = True
|
||||
|
||||
return attribute_docs
|
||||
|
||||
|
||||
def get_products_for_website(field_filters=None, attribute_filters=None, search=None):
|
||||
if attribute_filters:
|
||||
item_codes = get_item_codes_by_attributes(attribute_filters)
|
||||
items_by_attributes = get_items([['name', 'in', item_codes]])
|
||||
|
||||
if field_filters:
|
||||
items_by_fields = get_items_by_fields(field_filters)
|
||||
|
||||
if attribute_filters and not field_filters:
|
||||
return items_by_attributes
|
||||
|
||||
if field_filters and not attribute_filters:
|
||||
return items_by_fields
|
||||
|
||||
if field_filters and attribute_filters:
|
||||
items_intersection = []
|
||||
item_codes_in_attribute = [item.name for item in items_by_attributes]
|
||||
|
||||
for item in items_by_fields:
|
||||
if item.name in item_codes_in_attribute:
|
||||
items_intersection.append(item)
|
||||
|
||||
return items_intersection
|
||||
|
||||
if search:
|
||||
return get_items(search=search)
|
||||
|
||||
return get_items()
|
||||
|
||||
|
||||
@frappe.whitelist(allow_guest=True)
|
||||
def get_products_html_for_website(field_filters=None, attribute_filters=None):
|
||||
field_filters = frappe.parse_json(field_filters)
|
||||
attribute_filters = frappe.parse_json(attribute_filters)
|
||||
|
||||
items = get_products_for_website(field_filters, attribute_filters)
|
||||
html = ''.join(get_html_for_items(items))
|
||||
|
||||
if not items:
|
||||
html = frappe.render_template('erpnext/www/all-products/not_found.html', {})
|
||||
|
||||
return html
|
||||
|
||||
|
||||
def get_item_codes_by_attributes(attribute_filters, template_item_code=None):
|
||||
items = []
|
||||
|
||||
for attribute, values in attribute_filters.items():
|
||||
attribute_values = values
|
||||
|
||||
if not isinstance(attribute_values, list):
|
||||
attribute_values = [attribute_values]
|
||||
|
||||
if not attribute_values: continue
|
||||
|
||||
wheres = []
|
||||
query_values = []
|
||||
for attribute_value in attribute_values:
|
||||
wheres.append('( attribute = %s and attribute_value = %s )')
|
||||
query_values += [attribute, attribute_value]
|
||||
|
||||
attribute_query = ' or '.join(wheres)
|
||||
|
||||
if template_item_code:
|
||||
variant_of_query = 'AND t2.variant_of = %s'
|
||||
query_values.append(template_item_code)
|
||||
else:
|
||||
variant_of_query = ''
|
||||
|
||||
query = '''
|
||||
SELECT
|
||||
t1.parent
|
||||
FROM
|
||||
`tabItem Variant Attribute` t1
|
||||
WHERE
|
||||
1 = 1
|
||||
AND (
|
||||
{attribute_query}
|
||||
)
|
||||
AND EXISTS (
|
||||
SELECT
|
||||
1
|
||||
FROM
|
||||
`tabItem` t2
|
||||
WHERE
|
||||
t2.name = t1.parent
|
||||
{variant_of_query}
|
||||
)
|
||||
GROUP BY
|
||||
t1.parent
|
||||
ORDER BY
|
||||
NULL
|
||||
'''.format(attribute_query=attribute_query, variant_of_query=variant_of_query)
|
||||
|
||||
item_codes = set([r[0] for r in frappe.db.sql(query, query_values)])
|
||||
items.append(item_codes)
|
||||
|
||||
res = list(set.intersection(*items))
|
||||
|
||||
return res
|
||||
|
||||
|
||||
@frappe.whitelist(allow_guest=True)
|
||||
def get_attributes_and_values(item_code):
|
||||
'''Build a list of attributes and their possible values.
|
||||
@@ -270,140 +116,6 @@ def get_items_with_selected_attributes(item_code, selected_attributes):
|
||||
|
||||
return set.intersection(*items)
|
||||
|
||||
|
||||
def get_items_by_fields(field_filters):
|
||||
meta = frappe.get_meta('Item')
|
||||
filters = []
|
||||
for fieldname, values in field_filters.items():
|
||||
if not values: continue
|
||||
|
||||
_doctype = 'Item'
|
||||
_fieldname = fieldname
|
||||
|
||||
df = meta.get_field(fieldname)
|
||||
if df.fieldtype == 'Table MultiSelect':
|
||||
child_doctype = df.options
|
||||
child_meta = frappe.get_meta(child_doctype)
|
||||
fields = child_meta.get("fields", { "fieldtype": "Link", "in_list_view": 1 })
|
||||
if fields:
|
||||
_doctype = child_doctype
|
||||
_fieldname = fields[0].fieldname
|
||||
|
||||
if len(values) == 1:
|
||||
filters.append([_doctype, _fieldname, '=', values[0]])
|
||||
else:
|
||||
filters.append([_doctype, _fieldname, 'in', values])
|
||||
|
||||
return get_items(filters)
|
||||
|
||||
|
||||
def get_items(filters=None, search=None):
|
||||
start = frappe.form_dict.start or 0
|
||||
e_commerce_settings = get_e_commerce_settings()
|
||||
page_length = e_commerce_settings.products_per_page
|
||||
|
||||
filters = filters or []
|
||||
# convert to list of filters
|
||||
if isinstance(filters, dict):
|
||||
filters = [['Item', fieldname, '=', value] for fieldname, value in filters.items()]
|
||||
|
||||
enabled_items_filter = get_conditions({ 'disabled': 0 }, 'and')
|
||||
|
||||
show_in_website_condition = ''
|
||||
if e_commerce_settings.hide_variants:
|
||||
show_in_website_condition = get_conditions({'show_in_website': 1 }, 'and')
|
||||
else:
|
||||
show_in_website_condition = get_conditions([
|
||||
['show_in_website', '=', 1],
|
||||
['show_variant_in_website', '=', 1]
|
||||
], 'or')
|
||||
|
||||
search_condition = ''
|
||||
if search:
|
||||
# Default fields to search from
|
||||
default_fields = {'name', 'item_name', 'description', 'item_group'}
|
||||
|
||||
# Get meta search fields
|
||||
meta = frappe.get_meta("Item")
|
||||
meta_fields = set(meta.get_search_fields())
|
||||
|
||||
# Join the meta fields and default fields set
|
||||
search_fields = default_fields.union(meta_fields)
|
||||
try:
|
||||
if frappe.db.count('Item', cache=True) > 50000:
|
||||
search_fields.remove('description')
|
||||
except KeyError:
|
||||
pass
|
||||
|
||||
# Build or filters for query
|
||||
search = '%{}%'.format(search)
|
||||
or_filters = [[field, 'like', search] for field in search_fields]
|
||||
|
||||
search_condition = get_conditions(or_filters, 'or')
|
||||
|
||||
filter_condition = get_conditions(filters, 'and')
|
||||
|
||||
where_conditions = ' and '.join(
|
||||
[condition for condition in [enabled_items_filter, show_in_website_condition, \
|
||||
search_condition, filter_condition] if condition]
|
||||
)
|
||||
|
||||
left_joins = []
|
||||
for f in filters:
|
||||
if len(f) == 4 and f[0] != 'Item':
|
||||
left_joins.append(f[0])
|
||||
|
||||
left_join = ' '.join(['LEFT JOIN `tab{0}` on (`tab{0}`.parent = `tabItem`.name)'.format(l) for l in left_joins])
|
||||
|
||||
results = frappe.db.sql('''
|
||||
SELECT
|
||||
`tabItem`.`name`, `tabItem`.`item_name`, `tabItem`.`item_code`,
|
||||
`tabItem`.`website_image`, `tabItem`.`image`,
|
||||
`tabItem`.`web_long_description`, `tabItem`.`description`,
|
||||
`tabItem`.`route`, `tabItem`.`item_group`
|
||||
FROM
|
||||
`tabItem`
|
||||
{left_join}
|
||||
WHERE
|
||||
{where_conditions}
|
||||
GROUP BY
|
||||
`tabItem`.`name`
|
||||
ORDER BY
|
||||
`tabItem`.`weightage` DESC
|
||||
LIMIT
|
||||
{page_length}
|
||||
OFFSET
|
||||
{start}
|
||||
'''.format(
|
||||
where_conditions=where_conditions,
|
||||
start=start,
|
||||
page_length=page_length,
|
||||
left_join=left_join
|
||||
)
|
||||
, as_dict=1)
|
||||
|
||||
for r in results:
|
||||
r.description = r.web_long_description or r.description
|
||||
r.image = r.website_image or r.image
|
||||
product_info = get_product_info_for_website(r.item_code, skip_quotation_creation=True).get('product_info')
|
||||
if product_info:
|
||||
r.formatted_price = product_info['price'].get('formatted_price') if product_info['price'] else None
|
||||
|
||||
return results
|
||||
|
||||
|
||||
def get_conditions(filter_list, and_or='and'):
|
||||
from frappe.model.db_query import DatabaseQuery
|
||||
|
||||
if not filter_list:
|
||||
return ''
|
||||
|
||||
conditions = []
|
||||
DatabaseQuery('Item').build_filter_conditions(filter_list, conditions, ignore_permissions=True)
|
||||
join_by = ' {0} '.format(and_or)
|
||||
|
||||
return '(' + join_by.join(conditions) + ')'
|
||||
|
||||
# utilities
|
||||
|
||||
def get_item_attributes(item_code):
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
from __future__ import unicode_literals
|
||||
import frappe
|
||||
from erpnext.shopping_cart.doctype.shopping_cart_settings.shopping_cart_settings import get_shopping_cart_settings
|
||||
from erpnext.e_commerce.doctype.e_commerce_settings.e_commerce_settings import get_shopping_cart_settings
|
||||
from erpnext.shopping_cart.cart import get_debtors_account
|
||||
from frappe.utils.nestedset import get_root_of
|
||||
|
||||
|
||||
Reference in New Issue
Block a user